使用Unity做类的增强-阿里云开发者社区

开发者社区> 非常老帅> 正文

使用Unity做类的增强

简介: 我们已经实现了用户注册功能,现在想增加日志记录功能。具体来讲就是在用户注册前后,分别输出一条日志。
+关注继续查看

我们已经实现了用户注册功能,现在想增加日志记录功能。具体来讲就是在用户注册前后,分别输出一条日志。我们当然可以修改原有的业务代码。

现在换个角度来问两个问题:
1. 团队开发中,我们很可能根本拿不到源代码,那又怎么去增加这个功能呢?
2. 这次需求是增加日志,以后再增加其他需求(比如异常处理),是不是仍然要改业务类呢?

总结一下:
我们要在不修改原有类业务代码的前提下,去做类的增强。我们的设计要符合面向对象的原则:对扩展开放,对修改封闭

都有哪些办法呢?我们尝试以下几种方法:


原有业务类

业务模型

namespace testAopByDecorator
{
    public class User
    {
        public string Name { get; set; }
        public int Id { get; set; }
    }
}

接口设计

namespace testAopByDecorator
{
    public interface IUserProcessor
    {
        void RegisterUser(User user);
    }
}

业务实现

using System;

namespace testAopByDecorator
{
    public class UserProcessor : IUserProcessor
    {
        public void RegisterUser(User user)
        {
            if (user == null)
            {
                return;
            }

            Console.WriteLine(string.Format("注册了一个用户{0}:{1}", user.Id, user.Name));
        }
    }
}

上层调用

using System;

namespace testAopByDecorator
{
    class Program
    {
        private static User user = new User { Id = 1, Name = "滇红" };

        static void Main(string[] args)
        {
            Register();
            Console.ReadKey();
        }

        private static void Register()
        {
            IUserProcessor processor = new UserProcessor();
            processor.RegisterUser(user);
        }
    }
}

使用Unity做类的增强

我们将使用第三方的Unity来对原有的类做业务增强,首先使用NuGet安装。
这里写图片描述
业务增强实现

using System;
using System.Collections.Generic;
using Microsoft.Practices.Unity;
using Microsoft.Practices.Unity.InterceptionExtension;

namespace testAopByUnityBehavior
{
    public class UserProcessorLogBehavior : IInterceptionBehavior
    {
        public bool WillExecute
        {
            get
            {
                return true;
            }
        }

        public IEnumerable<Type> GetRequiredInterfaces()
        {
            return Type.EmptyTypes;
        }

        public IMethodReturn Invoke(IMethodInvocation input, GetNextInterceptionBehaviorDelegate getNext)
        {
            User user = input.Inputs[0] as User;

            before(user);
            InvokeInterceptionBehaviorDelegate delegateMethod = getNext();
            IMethodReturn returnMessag = delegateMethod(input, getNext);
            after(user);

            return returnMessag;
        }

        private void after(User user)
        {
            Console.WriteLine("日志结束:" + user.Name);
        }

        private void before(User user)
        {
            Console.WriteLine("日志开始:" + user.Name);
        }
    }
}

上层调用

using Microsoft.Practices.Unity;
using Microsoft.Practices.Unity.InterceptionExtension;
using System;

namespace testAopByUnityBehavior
{
    class Program
    {
        private static User user = new User { Id = 1, Name = "admin" };

        static void Main(string[] args)
        {
            RegisterAndLog();
            Console.ReadKey();
        }

        private static void RegisterAndLog()
        {
            //创建容器
            IUnityContainer container = new UnityContainer();

            //注册扩展
            container.AddNewExtension<Interception>();

            container.RegisterType<IUserProcessor, UserProcessor>(
                new InjectionConstructor(), //构造器拦截
                new Interceptor<InterfaceInterceptor>(),//注册拦截器
                new InterceptionBehavior<UserProcessorLogBehavior>() //注册行为拦截
                );

            User user = new User { Id = 1, Name = "李丹丹" };

            //从容器拿到服务(以为加了拦截器,这里生成的是动态对象)
            IUserProcessor processor = container.Resolve<IUserProcessor>();

            processor.RegisterUser(user);
        }
    }
}

对比一下扩展前后的业务展现
这里写图片描述


这里我们使用代码的方式给原有的业务类配置了拦截器(AOP依赖注入),也可以使用配置文件的方式,这样会更加的灵活。

配置文件:unity.config

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <configSections>
    <section name="unity"
              type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection,  
                     Microsoft.Practices.Unity.Configuration"/>
  </configSections>

  <unity xmlns="http://schemas.microsoft.com/practices/2010/unity">
    <alias alias="singleton"
           type="Microsoft.Practices.Unity.ContainerControlledLifetimeManager, Microsoft.Practices.Unity" />

    <sectionExtension type="Microsoft.Practices.Unity.InterceptionExtension.Configuration.InterceptionConfigurationExtension,Microsoft.Practices.Unity.Interception.Configuration" />
    <container>      
      <extension type="Interception" />

      <register type="IInterceptionBehavior" 
                mapTo="testAopByUnityConfig.UserProcessorLogBehavior,testAopByUnityConfig" 
                name="UserProcessorLogBehavior">
      </register>

      <register type="testAopByUnityConfig.IUserProcessor,testAopByUnityConfig" 
                mapTo="testAopByUnityConfig.UserProcessor,testAopByUnityConfig">
        <interceptionBehavior name="UserProcessorLogBehavior"/>
        <interceptor type="InterfaceInterceptor"/>
      </register>
    </container>
  </unity>
</configuration>

上层调用

using Microsoft.Practices.Unity;
using Microsoft.Practices.Unity.Configuration;
using Microsoft.Practices.Unity.InterceptionExtension;
using Microsoft.Practices.Unity.InterceptionExtension.Configuration;
using System;
using System.Configuration;

namespace testAopByUnityConfig
{
    class Program
    {
        private static User user = new User { Id = 1, Name = "admin" };

        static void Main(string[] args)
        {
            RegisterAndLog();
            Console.ReadKey();
        }

        private static void RegisterAndLog()
        {
            var map = new ExeConfigurationFileMap { ExeConfigFilename = "unity.config" };//使用此配置文件  
            var config = ConfigurationManager.OpenMappedExeConfiguration(map, ConfigurationUserLevel.None);
            var section = (UnityConfigurationSection)config.GetSection("unity");//读取配置文件节点  

            var container = new UnityContainer();
            container.LoadConfiguration(section);

            //从容器拿到服务(因为加了拦截器,这里生成的是动态对象)
            IUserProcessor processor = container.Resolve<IUserProcessor>();

            processor.RegisterUser(user);
        }
    }
}

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
Unity 之 贝塞尔曲线介绍和实际使用
Unity 中对贝塞尔曲线的实战应用,制作可视化操作曲线工具,文末附工具源码链接~
50 0
使用OpenApi弹性释放和设置云服务器ECS释放
云服务器ECS的一个重要特性就是按需创建资源。您可以在业务高峰期按需弹性的自定义规则进行资源创建,在完成业务计算的时候释放资源。本篇将提供几个Tips帮助您更加容易和自动化的完成云服务器的释放和弹性设置。
12070 0
Unity使用中的错误 ✨使用unity过程中碰到的一些错误及解决
错误目录 一、打不开unity工程 问题一:在UnityHub打开某个工程时,一直显示已有相同版本的工程存在,一闪而过然后就一直打不开这个工程,重启UnityHub并无卵用 二、打包出错 问题:打包安卓包时老是出现SDK找不到或者版本低(明明JDK、SDK都配置好的前体下。我的unity19版本出现过好多次,一样的JDDK、SDK低版本Unity可以打包,19就不可以) 三、运行出错 问题
12 0
Unity5 AssetBundle系列——资源加载卸载以及AssetBundleManifest的使用
  下面代码列出了对于assetbundle资源的常用操作,其中有针对bundle、asset、gameobject三种类型对象的操作,实际使用中尽量保证成对使用。   这一块的操作比较繁琐,但只要使用正确,是可以保证资源完全没有泄露的。
1522 0
【Unity3D 灵巧小知识点】☀️ | 使用宏定义和Application.platform判断运行平台
Unity 小科普 老规矩,先介绍一下 Unity 的科普小知识: Unity是 实时3D互动内容创作和运营平台 。 包括游戏开发、美术、建筑、汽车设计、影视在内的所有创作者,借助 Unity 将创意变成现实。 Unity 平台提供一整套完善的软件解决方案,可用于创作、运营和变现任何实时互动的2D和3D内容,支持平台包括手机、平板电脑、PC、游戏主机、增强现实和虚拟现实设备。 也可以简单把 Unity 理解为一个游戏引擎,可以用来专业制作游戏!
9 0
+关注
非常老帅
IT行业从业10+年
481
文章
1
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载