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

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

使用Autofac做类的增强

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

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

现在换个角度来问两个问题:
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);
        }
    }
}

使用Autofac做类的增强

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

using Castle.DynamicProxy;
using System;

namespace testAopByAutofac
{
    public class UserProcessorLog : IInterceptor
    {
        public void Intercept(IInvocation invocation)
        {
            User user = invocation.Arguments[0] as User;

            before(user);
            invocation.Proceed();
            after(user);
        }

        private void after(User user)
        {
            Console.WriteLine("注册用户后:" + user.Name);
        }

        private void before(User user)
        {
            Console.WriteLine("注册用户前:" + user.Name);
        }
    }
}

上层调用

using Autofac;
using Autofac.Extras.DynamicProxy2;
using System;

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

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

        private static void RegisterAndLog()
        {
            ContainerBuilder builder = new ContainerBuilder();
            builder
                .RegisterType<UserProcessor>()
                .As<IUserProcessor>()
                .EnableInterfaceInterceptors()
                .InterceptedBy(typeof(UserProcessorLog));
            builder.RegisterType<UserProcessorLog>();

            var container =  builder.Build();

            using (var scope = container.BeginLifetimeScope())
            {
                IUserProcessor processor = (IUserProcessor)scope.Resolve<IUserProcessor>();

                processor.RegisterUser(user);
            }
        }
    }
}

对比一下扩展前后的业务展现

这里写图片描述

这里我们使用代码的方式给原有的业务类配置了拦截器(AOP依赖注入),也可以使用特性的方式。
使用特性加强业务类

using Autofac.Extras.DynamicProxy2;
using System;

namespace testAopByAutofac
{
    [Intercept(typeof(UserProcessorLog))]
    public class UserProcessor : IUserProcessor
    {
        public void RegisterUser(User user)
        {
            if (user == null)
            {
                return;
            }

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

上层调用

using Autofac;
using Autofac.Extras.DynamicProxy2;
using System;

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

        static void Main(string[] args)
        {
            Register();
            Console.WriteLine("--------------");
            RegisterAndLog();

            Console.ReadKey();
        }

        private static void RegisterAndLog()
        {
            ContainerBuilder builder = new ContainerBuilder();
            builder
                .RegisterType<UserProcessor>()
                .As<IUserProcessor>()
                .EnableInterfaceInterceptors();
                //.InterceptedBy(typeof(UserProcessorLog));//可使用特性代替
            builder.RegisterType<UserProcessorLog>();

            var container =  builder.Build();

            using (var scope = container.BeginLifetimeScope())
            {
                IUserProcessor processor = (IUserProcessor)scope.Resolve<IUserProcessor>();

                processor.RegisterUser(user);
            }
        }

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

上面我们使用的是接口注入(EnableInterfaceInterceptors),也可以使用类注入(EnableClassInterceptors),此时原有业务类的方法需要显示声明为vitual虚方法。在上层调用的时候,我们就可以直接注册和解析一个类,不依赖于接口了。

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

相关文章
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
10052 0
Auto Scaling 通过启动模板来使用伸缩服务
功能说明 AutoScaling伸缩组长期依赖的伸缩配置信息本质上就是Ecs实例配置,考虑到Ecs推出了持久化Ecs实例配置的启动模板,因此AutoScaling 将伸缩组与启动模板结合,用户可以通过直接在创建伸缩组时选择启动模板来使用弹性伸缩 伸缩组使用启动模板的默认版本与最新版本而非指定版本时.
3110 0
Asp.net MVC 使用Autofac的简单使用 IOC
Ioc(Inversion of Control)或者叫依赖注入DI(Dependency Injection) 如果一个接口有两个实现类,但是在实现过程中,用到了这两个具体的实现类。 如果采用IOC,则只能是注册一个接口类型,那么如何确保IOC在合适的时候传入不同类的实例?这是我突然间想到的一个问题,希望园友们可以帮忙解答一下!     所谓IOC(控制反转)或者说是依赖注入,就是将你设计好的类交给系统去控制,而不是在你的类内部控制,控制权发生了变化,就称为控制反转。
913 0
阿里云服务器如何登录?阿里云服务器的三种登录方法
购买阿里云ECS云服务器后如何登录?场景不同,阿里云优惠总结大概有三种登录方式: 登录到ECS云服务器控制台 在ECS云服务器控制台用户可以更改密码、更换系.
13865 0
java安全编码指南之:ThreadPool的使用
在java中,除了单个使用Thread之外,我们还会使用到ThreadPool来构建线程池,那么在使用线程池的过程中需要注意哪些事情呢? 一起来看看吧。
2299 0
+关注
非常老帅
IT行业从业10+年
481
文章
1
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载