2

おそらく、抽象型のファクトリは Windsor (問題がある場合は 2.5.3) で開始するのは簡単ではありませんが、とにかくそれを行う必要があります。メッセージの種類に応じてプロセッサを返すファクトリを構築しようとしています。これまでのところ、コードに従ってさまざまな場所から掃除してきました:

public class Complicato
{
    public static void Do(string[] args)
    {
        IKernel kernel = new DefaultKernel();
        IWindsorContainer container = new WindsorContainer();

        kernel.AddFacility<TypedFactoryFacility>();

        container.Install();
        container.Register(

            Component.For<HandlerSelector, ITypedFactoryComponentSelector>(),

            AllTypes.FromThisAssembly().BasedOn(typeof(ITrier<>))
                .WithService.Base().Configure(conf => conf.LifeStyle.Is(LifestyleType.Transient)),
            Component.For<Factor>(),
            Component.For<ITryFactory>().AsFactory(c => c.SelectedWith<HandlerSelector>()).LifeStyle.Singleton);

        var factor = container.Resolve<Factor>();
        var factory = container.Resolve<ITryFactory>();
    }
}

public class HandlerSelector : DefaultTypedFactoryComponentSelector
{
    protected override Type GetComponentType(MethodInfo method, object[] arguments)
    {
        return typeof(ITrier<>).MakeGenericType(arguments[0].GetType());
    }
}

public class Factor
{
    private ITryFactory factory;

    public void Try(IWhat onto)
    {
        factory.GetTrier(onto).Try(onto);
    }
}


public interface ITryFactory
{
    ITrier<IWhat> GetTrier(IWhat onto);
    void Release(object elem);
}


public interface IWhat { }


public interface ITrier<in TWhat> where TWhat : IWhat
{
    void Try(TWhat input);
}


public class TrierYes : ITrier<WhatYes>
{
    public void Try(WhatYes input) { Console.WriteLine("Yes? " + input.Aye()); }
}

public class TrierNo : ITrier<WhatNot>
{
    public void Try(WhatNot input) { Console.WriteLine("No? " + input.Naa()); }
}

public class WhatYes : IWhat
{
    public bool Aye() { return true; }
}

public class WhatNot : IWhat
{
    public bool Naa() { return false; }
}

ここでの主な問題は、id が機能しないことです。最初に null の factory で Factor を取得し、その結果、factory を明示的に解決しようとすると、 ComponentActivator: could not proxy Factories.Complex.ITryFactory with internal message of インターセプター Castle.TypedFactory.Interceptor could not be resolve and "Keys (components特定のキーを使用) - コンテナーに登録されていない Castle.TypedFactory.Interceptor」。Handler セレクターが機能するかどうかもわかりません。今のところ問題ありません。

ITrier をジェネリックではないものにすると、突然動作し始めますが、それは私が達成しようとしているものではありません。

では、ウィンザーの構成で愚かな初心者の間違いを犯したり、型付きファクトリの概念を誤解したりすることはありますか?

完全を期すために、例外メッセージを次に示します。

Castle.MicroKernel.ComponentActivator.ComponentActivatorException was unhandled
  Message=ComponentActivator: could not proxy Factories.Complex.ITryFactory
  Source=Castle.Windsor
  StackTrace:
       at Castle.MicroKernel.ComponentActivator.DefaultComponentActivator.CreateInstance(CreationContext context, Object[] arguments, Type[] signature) in e:\OSS.Code\Castle.Windsor\src\Castle.Windsor\MicroKernel\ComponentActivator\DefaultComponentActivator.cs:line 166
  InnerException: Castle.MicroKernel.Resolvers.DependencyResolverException
       Message=The interceptor Castle.TypedFactory.Interceptor could not be resolved
       Source=Castle.Windsor
       StackTrace:
            at Castle.Core.InterceptorReference.Castle.MicroKernel.IReference<Castle.DynamicProxy.IInterceptor>.Resolve(IKernel kernel, CreationContext context) in e:\OSS.Code\Castle.Windsor\src\Castle.Windsor\Core\InterceptorReference.cs:line 142
4

1 に答える 1

2

そして勝者は

container.AddFacility<TypedFactoryFacility>(); // good code

それ以外の

kernel.AddFacility<TypedFactoryFacility>(); // bad code

現在、ファクトリが注入されていないという問題と、HandlerSelectorが不適切であるという問題しかありません。

NullReferenceは、明示的な初期化コンストラクターをFactorに導入することで解決されました。なぜそれがなくても機能すると思ったのか分かりません。

ハンドラーインターフェイスの最終バージョンは次のとおりです。

public interface ITrier<out TWhat> where TWhat: IWhat
{
    void Try(IWhat input);
}

共分散を許可するため。不必要なキャストを必要とし、ハンドラーがタイプを緩めるため、超エレガントではありません。しかし、これは残酷な現実です。あなたは共変か逆変のどちらかです。

于 2012-10-26T08:35:47.780 に答える