代理模式看这一篇就够了~

静态代理


那么我们该如何实现上面的功能呢,我们以 为登录注册方法 添加 时间日志为例,首先 有经验的工程师在写代码的时候,就应该知道,我们要遵循基于接口而非实现的设计原则,所以我们应把login 和 regist 抽取出来,让LoginAndRegist 类 继承盖接口,如下所示:

public interface UserInter {

/**

* 登录

* @param name name

*/

void login(String name);

/**

* 注册

* @param name name

*/

void regist(String name);

}

public class LoginAndRegist implements UserInter {

private static final String TAG = “Login”;

@Override

public void login(String userName) {

System.out.println(“我是登录方法”);

}

@Override

public void regist(String name) {

System.out.println(“我是注册方法”);

}

}

为LoginAndRegist 类 创建代理类 LoginAndRegistProxy,让代理类 实现 和 原始类同样的接口,并调用原始类的方法 ,并在调用前 打印当前时间戳,代码如下所示:

public class LoginAndRegistProxy implements UserInter {

private static final String TAG = “Login”;

private UserInter userInter;

public LoginAndRegistProxy(UserInter userInter) {

this.userInter = userInter;

}

@Override

public void login(String userName) {

System.out.println(“调用登录接口的时间:” + System.currentTimeMillis());

userInter.login(userName);

}

@Override

public void regist(String name) {

System.out.println(“调用注册接口的时间:” + System.currentTimeMillis());

userInter.login(name);

}

}

在Test中修改调用方法为代理类调用:

public static void main(String[] args) {

LoginAndRegist loginAndRegist = new LoginAndRegist();

LoginAndRegistProxy loginAndRegistProxy = new LoginAndRegistProxy(loginAndRegist);

loginAndRegistProxy.regist(“huanglinqing”);

loginAndRegistProxy.login(“huanglinqing”);

}

执行结果 如下所示:

调用注册接口的时间:1596793891141

我是登录方法

调用登录接口的时间:1596793891141

我是登录方法

Process finished with exit code 0

这样呢,我们就实现了不改变原始的情况下,为类方法添加日志的功能,但是呢,这种方法存在的问题 我们上面也提到了

第一 如果原始类 没有实现接口怎么办

第二 如果原始类的源代码 我们获取不到怎么办

对于原始类 并没有实现接口,并且我们无法修改的情况下,这种我们称为对外部类的扩展,外部类的扩展我们一般使用继承的方式去扩展,这种方式我们就不解释了。

第三 如果为每个类都添加代理类,会增加大量的文件

第三个问题才是我们实际开发中需要首要解决的问题,所以 为了解决静态代理文件过多的的问题,我们需要使用动态代理。

动态代理


动态代理简单的说 就是我们不需要事先为某个原始类编写代理类,而是在运行的时候,动态的创建代理类,然后将原始类替换为代理类。动态代理的魅力在Android中真的是非常非常大,如果你还不了解,一定要回头看我前言中提到的两篇文章。而在java中动态代理的基础是反射,如果你还不了解反射技术,请移步至我的这篇文章Java反射技术详解

动态代理,我们主要依赖的是newProxyInstance方法,该方法返回的是指定接口代理类的实例。

public static Object newProxyInstance(ClassLoader loader,

Class<?>[] interfaces,

InvocationHandler h)

第一个参数loader 指的是目标对象class对应的classLoader

第二个参数interfaces是设置为对象class所实现的接口类型,第一个参数和第二个参数其实在业务上都是固定的,在这里就是UserInter对应的的classLoader和接口类型。

我们主要来看第三个参数 InvocationHandler,它是一个实现了InvocationHandler接口的类对象

首先我们来定义一个MyInvocationHandler实现InvocationHandler

public class MyInvocationHandler implements java.lang.reflect.InvocationHandler {

@Override

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

return null;

}

}

这里实现的invoke方法就是动态代理的核心,此外我们需要将代理类传进来,并在invoke方法中执行代理方法

public class MyInvocationHandler implements java.lang.reflect.InvocationHandler {

private Object target;

public MyInvocationHandler(Object target) {

this.target = target;

}

@Override

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

System.out.println(“开始执行前:” + method.getName());

Object object = method.invoke(target, args);

System.out.println(“执行结束:” + method.getName());

return object;

}

}

当外部代理类调用某个方法的时候,其实就是在调用invoke中的method.invoke方法,args就是执行对应方法所需的参数,我们这里 在方法前后分别加上日志。

上面我们已经说了,动态代理是通过newProxyInstance方法创建的,我们来看Test中如何使用

public static void main(String[] args) {

UserInter loginAndRegist = new LoginAndRegist();

UserInter loginAndRegistProxy = (UserInter) Proxy.newProxyInstance(loginAndRegist.getClass().getClassLoader(),

loginAndRegist.getClass().getInterfaces(),new MyInvocationHandler(loginAndRegist));

loginAndRegistProxy.regist(“huang_动态代理”);

loginAndRegistProxy.login(“huang_静态代理”);

}

newProxyInstance中的参数我们在上面已经说明了,运行结果如下所示:

开始执行前:regist

我是注册方法

执行结束:regist

开始执行前:login

我是登录方法

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
4T-1715842915552)]

[外链图片转存中…(img-iBPVkfW8-1715842915554)]

[外链图片转存中…(img-YhKg5E7R-1715842915555)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值