Как создать общий репозиторий в Symfony 4

Я работаю с Symfony 4, у меня много репозиториев с обычным поведением, поэтому я хочу избежать повторения кода. Я попытался определить родительский класс репозитория следующим образом:

<?php
namespace App\Repository;

use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Symfony\Bridge\Doctrine\RegistryInterface;

class AppRepository extends ServiceEntityRepository {
    public function __construct(RegistryInterface $registry, $entityClass) {
        parent::__construct($registry, $entityClass);
    }

    // Common behaviour
}

Таким образом, я мог бы определить его дочерние классы, например:

<?php
namespace App\Repository;

use App\Entity\Test;
use App\Repository\AppRepository;
use Symfony\Bridge\Doctrine\RegistryInterface;

class TestRepository extends AppRepository {
    public function __construct(RegistryInterface $registry) {
        parent::__construct($registry, Test::class);
    }
}

Но я получаю такую ​​ошибку:

Cannot autowire service "App\Repository\AppRepository": argument "$entityClass" of method "__construct()" must have a type-hint or be given a value explicitly.

Я попытался установить подсказку типа, например string, object, но это не сработало.

Есть ли способ определить общий репозиторий?

заранее спасибо

Просто исключите AppRepository из файла services.yaml. Процесс autowire пытается подключить его, поэтому вам нужно сказать ему, чтобы он игнорировал его.

Cerad 12.12.2018 18:20

Это сработало !!! Большое спасибо. Добавил в services.yaml exclude: '../src/{Entity,Migrations,Tests,Repository/AppRepository.ph‌​p}'.. Если вы хотите добавить Его как ответ, чтобы я мог отметить его как правильный!

Genarito 12.12.2018 18:42

Почему бы вам просто не создать соответствующий репозиторий для каждой из ваших сущностей?

emix 12.12.2018 18:55

Поскольку это общая система, которая может взаимодействовать с разными СУБД, поэтому в ней много изменений синтаксиса, поэтому у меня есть общий запрос, например, для выполнения хранимых процедур, которые «предварительно проверяют» СУБД. Если мне придется выполнять эту логику во всех моих репозиториях, это приведет к многократному повторению кода.

Genarito 12.12.2018 18:58

У меня был тот же вопрос «почему». Совместное использование кода в репозиториях имеет тенденцию к беспорядку. Но если это сработает, тогда ладно. Также можно рассмотреть возможность использования черты характера.

Cerad 12.12.2018 19:02

черта хорошее предложение! Но я не могу его использовать, так как мне нужно подготовить исполняемый SQL. В моем случае у меня CALL <sp_name> (<params>) для MySQL и ORACLE; и EXEC <sp_name> <params> для SQL Server. И оба выдают разные исключения из-за драйверов ODBC. Так что с этим универсальным репозиторием я могу легко справиться с этим, не проверяя везде! : D

Genarito 12.12.2018 19:11

Просто примечание: нет необходим для реализации __construct() в AppRepository: ServiceEntityRepository::__construct() имеет ту же сигнатуру и будет фактически выполнен.

Boolean_Type 10.02.2020 16:41
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Оживление вашего приложения Laravel: Понимание режима обслуживания
Оживление вашего приложения Laravel: Понимание режима обслуживания
Здравствуйте, разработчики! В сегодняшней статье мы рассмотрим важный аспект управления приложениями, который часто упускается из виду в суете...
Установка и настройка Nginx и PHP на Ubuntu-сервере
Установка и настройка Nginx и PHP на Ubuntu-сервере
В этот раз я сделаю руководство по установке и настройке nginx и php на Ubuntu OS.
Коллекции в Laravel более простым способом
Коллекции в Laravel более простым способом
Привет, читатели, сегодня мы узнаем о коллекциях. В Laravel коллекции - это способ манипулировать массивами и играть с массивами данных. Благодаря...
Как установить PHP на Mac
Как установить PHP на Mac
PHP - это популярный язык программирования, который используется для разработки веб-приложений. Если вы используете Mac и хотите разрабатывать...
6
7
2 134
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Ответ принят как подходящий

Одна из "ловушек" autowire заключается в том, что по умолчанию autowire ищет все классы в src и пытается превратить их в сервисы. В некоторых случаях он в конечном итоге выбирает классы, такие как ваш AppRepository, которые не предназначены для обслуживания, а затем терпит неудачу, когда пытается их автоматически подключить.

Наиболее распространенное решение - явно исключить эти классы:

# config/services.yaml
App\:
    resource: '../src/*'
    exclude: '../src/{DependencyInjection,Entity,Migrations,Tests,Kernel.php,Repository/AppRepository.php}'

Другой подход, который должен работать (не тестировался), - сделать AppRepository абстрактным. Autowire игнорирует абстрактные классы. Репозитории немного сложны, и наличие абстрактных классов, расширяющих не абстрактные классы, несколько необычно.

Просто сделайте свой AppRepository абстрактным, например

abstract class AppRepository {}

Другие вопросы по теме