• Castle 多继承选择


    Castle 多继承选择

    很多时候,我们定义了一个接口,但是这个接口会有多种不同的,这时IOC构造函数注入的时候,就需要自动选择对应的实现。

    public interface ITestService {}
    
    public class TestService : IService
    {
         public string test()
         {
             return "test"
         }
    }
    
    public class TestService_New : ICalculatorService
    {
         public string test()
         {
             return "test new";
         }
    }
    

    此时,我们大多数情况下,需要用TestService来实现ITestService,但是少部分情况下,需要用TestService_New实现ITestService。

    解决方案一

    IOC中只实现需要用到的类:

    Component.For<ITestService>()
        .ImplementedBy<TestService>() 
    

    如果在某些地方部署时,修改为
    Component.For()
    .ImplementedBy<TestService_New>()

    这样每次部署的时候都需要修改代码,而且需要手写的规则会越来越多。

    解决方案二

    参考了Castle的WIKI,发现了可以ServiceOverrides重写实现:

     Component.For<ITestService>()
        .ImplementedBy<TestService>() 
        .Named("myservice.default"),
    Component.For<ITestService>()
        .ImplementedBy<TestService_New>()
        .Named("myservice.new"),
    
    Component.For<TestController>()
        .ServiceOverrides(ServiceOverride.ForKey("myService").Eq("myservice.new"))
    

    尝试了一下,VS提示改方法已过世,建议使用Dependency.OnComponent代替。
    Component.For().
    .DependsOn(Dependency.OnComponent<ITestService, TestService_New>())

    修改代码之后,怎么都不起作用。重新翻看官方WIKI,发现了IsDefault方法,当多个类实现同一个接口时,可以通过IsDefault来置为默认实现。

    解决方案三

    			 //先通过DefaultInterfaces加载正常实现 
                 Classes.FromAssembly(assembly)
                 .IncludeNonPublicTypes()
                 .BasedOn<IApplicationService>()
                 .WithService.DefaultInterfaces()
                 .LifestyleTransient()
                 .Configure(c =>
                 {
                     c.Interceptors<ExceptionInterceptor, TransactionInterceptor>();
                     c.Named(c.Implementation.Name);
                 }),
    			 //根据WEBCONFIG加载特殊实现,并设置为默认实现
                 Classes.FromAssembly(assembly)
                .IncludeNonPublicTypes()
                .BasedOn<IApplicationService>()
                .WithService.Select((type, @base) =>
                          type.GetAllInterfaces()
                              .Where(i => type.Name.Contains(GetInterfaceNameFromConf(i))))
                .LifestyleTransient()
                .Configure(c =>
                {
                    c.Interceptors<ExceptionInterceptor, TransactionInterceptor>().IsDefault()
    				.Named(c.Implementation.FullName +  "_" + ConfigurationManager.AppSettings["City"]);
                })
    
    
    			private string GetInterfaceNameFromConf(Type @interface)
    	        {
    	            var name = @interface.Name;
    	            if ((name.Length > 1 && name[0] == 'I') && char.IsUpper(name[1]))
    	            {
    	                return name.Substring(1) + "_" + ConfigurationManager.AppSettings["City"];
    	            }
    	            return name;
    	        }
    

    OK,大功告成。

  • 相关阅读:
    Spark函数详解系列之RDD基本转换
    Spark算子之aggregateByKey详解
    基于Apache Spark机器学习的客户流失预测
    Spark partitionBy
    Spark中repartition和partitionBy的区别
    Spark快速获得CrossValidator的最佳模型参数
    Spark MLlib之水塘抽样算法(Reservoir Sampling)
    机器学习-加权采样算法简介
    基于Spark的大数据精准营销中搜狗搜索引擎的用户画像挖掘(转)
    Mark 装修建材 清单
  • 原文地址:https://www.cnblogs.com/lazio10000/p/5266876.html
Copyright © 2020-2023  润新知