视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
关于DDD:管理"工作单元实例"的两种模式的使用方法
2020-11-27 22:41:05 责编:小采
文档


图如下:

在常见的用例场景下,类图的对象图如下:

问题在一个用例执行过程中,如何保证同一个界限上下文内的所有仓储实例可以共享同一个工作单元实例?解决方案1 
仓储采用依赖注入模式 + 使用IOC管理工作单元的生命周期(PerRequest或其它)。

代码示例
代码如下:
using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;

 using Autofac;

 namespace AutoFacStudy
 {
     class Program
     {
         static void Main(string[] args)
         {
             var buider = new ContainerBuilder();
             buider.RegisterType<服务>();
             buider.RegisterType<仓储A>();
             buider.RegisterType<仓储B>();
             buider.RegisterType<工作单元>().InstancePerLifetimeScope();

             var container = buider.Build();

             dynamic 服务 = container.Resolve<服务>();

             //下边两行代码输出一样
             Console.WriteLine(服务.仓储A.工作单元.GetHashCode());
             Console.WriteLine(服务.仓储B.工作单元.GetHashCode());
         }
     }

     public class 服务
     {
         private readonly 仓储A _仓储A;
         private readonly 仓储B _仓储B;

         public 服务(仓储A 仓储A, 仓储B 仓储B)
         {
             _仓储A = 仓储A;
             _仓储B = 仓储B;
         }

         public 仓储A 仓储A
         {
             get { return _仓储A; }
         }

         public 仓储B 仓储B
         {
             get { return _仓储B; }
         }
     }

     public class 工作单元 { }

     public class 仓储A
     {
         private readonly 工作单元 _工作单元;

         public 仓储A(工作单元 工作单元)
         {
             _工作单元 = 工作单元;
         }

         public 工作单元 工作单元
         {
             get { return _工作单元; }
         }
     }

     public class 仓储B
     {
         private readonly 工作单元 _工作单元;

         public 仓储B(工作单元 工作单元)
         {
             _工作单元 = 工作单元;
         }

         public 工作单元 工作单元
         {
             get { return _工作单元; }
         }
     }
 }

解决方案2
仓储采用服务定位器模式 + 使用服务定位器或简单工厂管理工作单元的生命周期(PerRequest或其它)。
代码示例
代码如下:
using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;

 using Autofac;

 namespace AutoFacStudy
 {
     class Program
     {
         public static IContainer 服务定位器;

         static void Main(string[] args)
         {
             var buider = new ContainerBuilder();
             buider.RegisterType<服务>();
             buider.RegisterType<仓储A>();
             buider.RegisterType<仓储B>();
             buider.RegisterType<工作单元>().InstancePerLifetimeScope();

             服务定位器 = buider.Build();

             dynamic 服务 = 服务定位器.Resolve<服务>();

             //下边两行代码输出一样
             Console.WriteLine(服务.仓储A.工作单元.GetHashCode());
             Console.WriteLine(服务.仓储B.工作单元.GetHashCode());
         }
     }

     public class 服务
     {
         private readonly 仓储A _仓储A;
         private readonly 仓储B _仓储B;

         public 服务(仓储A 仓储A, 仓储B 仓储B)
         {
             _仓储A = 仓储A;
             _仓储B = 仓储B;
         }

         public 仓储A 仓储A
         {
             get { return _仓储A; }
         }

         public 仓储B 仓储B
         {
             get { return _仓储B; }
         }
     }

     public class 工作单元 { }

     public class 仓储A
     {
         private readonly 工作单元 _工作单元;

         public 仓储A()
         {
             _工作单元 = Program.服务定位器.Resolve<工作单元>();
         }

         public 工作单元 工作单元
         {
             get { return _工作单元; }
         }
     }

     public class 仓储B
     {
         private readonly 工作单元 _工作单元;

         public 仓储B()
         {
             _工作单元 = Program.服务定位器.Resolve<工作单元>();
         }

         public 工作单元 工作单元
         {
             get { return _工作单元; }
         }
     }
 }

由此示例可以看出,服务定位器和依赖注入可以混合在一起使用。这个例子我为了简单,服务定位器和IOC容器是同一个实例。

有些系统将服务定位器的实现换成简单工厂模式,他们本质上是一样的(服务定位器是一个万能工厂)。

代码示例
代码如下:
public class 工作单元工厂
 {
     public static 工作单元 创建()
     {
         var 工作单元 = (工作单元)CallContext.GetData("工作单元");

         if (工作单元 == null)
         {
             工作单元 = new 工作单元();
             CallContext.SetData("工作单元", 工作单元);
         }

         return 工作单元;
     }
 }

下载本文
显示全文
专题