ASP.NET MVC, MVCContrib, Structuremap, заставить его работать как фабрику контроллеров?

Я пытаюсь получить карту структуры для правильного создания моих контроллеров, я использую DI для вставки INewsService в NewsController, и это единственный конструктор, который у меня есть.

public class NewsController : Controller
{
    private readonly INewsService newsService;

    public NewsController(INewsService newsService)
    {
        this.newsService = newsService;
    }

    public ActionResult List()
    {
        var newsArticles = newsService.GetNews();
        return View(newsArticles);
    }
}

и я использую этот код для запуска приложения

public class Application : HttpApplication
{
    protected void Application_Start()
    {
        RegisterIoC();
        RegisterViewEngine(ViewEngines.Engines);
        RegisterRoutes(RouteTable.Routes);
    }

    public static void RegisterIoC()
    {
        ObjectFactory.Initialize(config => {
            config.UseDefaultStructureMapConfigFile = false;
            config.AddRegistry<PersistenceRegistry>();
            config.AddRegistry<DomainRegistry>();
            config.AddRegistry<ControllerRegistry>();
        });
        DependencyResolver.InitializeWith(new StructureMapDependencyResolver());
        ControllerBuilder.Current.SetControllerFactory(typeof(IoCControllerFactory));            
    }
}

Но Structuremap, похоже, не хочет внедрять INewsService, и я получаю сообщение об ошибке Для этого объекта не определен конструктор без параметров.

Что я пропустил?

Ответов (3)

Решение

Я использую механизм «Соглашения по умолчанию», который предоставляет StructureMap, чтобы избежать необходимости индивидуальной настройки каждого интерфейса. Ниже приведен код, который я использую для этой работы:

В моем Global.asax есть эта строка в Application_Start (которая использует фабрику StructureMap из MvcContrib):

protected void Application_Start()
{
    RegisterRoutes(RouteTable.Routes);
    ObjectFactory.Initialize(x =>
    {
        x.AddRegistry(new RepositoryRegistry());
    });
    ControllerBuilder.Current.SetControllerFactory(typeof(StructureMapControllerFactory));
}

А класс RepositoryRegistry выглядит так:

public class RepositoryRegistry : Registry
{

    public RepositoryRegistry()
    {
        Scan(x =>
        {
            x.Assembly("MyAssemblyName");
            x.With<DefaultConventionScanner>();
        });

    }

}

DefaultConventionScanner ищет пары интерфейсов / классов, которые следуют соглашению об именах ISomethingOrOther и SomethingOrOther, и автоматически связывает последний как конкретный тип для первого интерфейса.

Если вы не хотите использовать этот механизм соглашения по умолчанию, вы должны добавить код в класс Registry для явного сопоставления каждого из ваших интерфейсов с конкретными типами с помощью этого синтаксиса:

ForRequestedType<ISomethingOrOther>().TheDefaultIsConcreteType<SomethingOrOther>();

Если я чего-то не упускаю, вы не сообщаете StructureMap, какой конкретный тип использовать для INewsService. Вам нужно добавить что-то вроде:

TheConcreteTypeOf<INewsService>.Is<MyConcreteNewsService>();

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

В настоящее время ASP.NET MVC создает экземпляры контроллеров с помощью конструктора без параметров по умолчанию, что исключает любое внедрение зависимостей на основе конструктора. Для этого вам действительно нужно использовать проект MvcContrib , который имеет встроенную поддержку StructureMap (и Castle / Spring.NET / Unity), хотя текущей документации не существует (буквально, вы получаете вики-страницу-заглушку, нехороший знак). Пример кода Эрва Вальтера в этом потоке показывает, как настроить интеграцию StructureMap.