Java中的反射与代理(2)
在经典的GoF设计模式中,有一种模式叫做代理模式,好比以前洋务运动的时候所说的「买办」,还有现在咱们经常听到的「代理人」战争中的「代理」,都是同一个意思——代替某人打理。
例如,很多北漂都找中介或者二房东租过房子,还有倒卖iPhone的黄牛党,对于租房的房客和房东、买iPhone的消费者和苹果公司来说,这些中介或者黄牛,就是对方的代理人,因为他们(买家或卖家)根本不关心谁在和自己交易,只要实现交易即可,而且也免去了一个个对接的麻烦,就像这样:
这个是最普通的代理,也称为「静态」代理,如果用代码来描述,就是:
/**
* 需要代理的行为
*/
public interface Renting {
// 租房
public void renthouse();
} /**
* 买方行为
*/
public class Consumer implements Renting {
@Override
public void renthouse() {
System.out.println("租客租房");
}
} /**
* 代理实现
*/
public class RentingProxy implements Renting {
private Renting renting; public RentingProxy(Renting renting) {
this.renting = renting;
} @Override
public void renthouse() {
System.out.println("代理方法前置:租房前看房");
renting.renthouse();
System.out.println("代理方法后置:满意签合同");
} public static void main(String[] args) {
Consumer consumer = new Consumer();
consumer.renthouse();
System.out.println("===================");
Renting proxy = new RentingProxy(consumer);
proxy.renthouse();
}
}
这样就完事了。
然鹅,现在国家下了一道文件,要让租房市场更规范,不仅租前要看房,租后要签合同,而且还需要中介公司在看房时提供资质证明。
这下麻烦了。
因为之前代码实现的时候,只考虑到了当前这一家公司的需求,并且把功能写死了(System.out.println("代理方法前置:租房前看房");这段代码代替的是实际需要执行的功能)。
比较直接的想法是,给每个类都写一个封装类,然后引用已有类,在代理对象的方法内调用被代理类的方法。具体来说,就是这么干:
/**
* 代理的代理
*
* @author 湘王
*/
public class Proxy {
private RentingProxy target; public Proxy(RentingProxy target) {
this.target = target;
} public void zizhi() {
System.out.println("展示资质");
target.renthouse();
} public static void main(String[] args) {
Consumer consumer = new Consumer();
RentingProxy rentingProxy = new RentingProxy(consumer);
Proxy proxy = new Proxy(rentingProxy);
proxy.zizhi();
}
}
可以看到,这种方式的问题有两个:
1、实现起来比较别扭,不够优雅,而且非常丑陋,包了一层又一层;
2、即使这么干行得通,当前公司的问题倒是解决了。但是如果市场上有10000家公司,是不是要改10000次代码呢?显然是NO!
这个时候,就需要通过「动态代理」来实现了——这也是反射最有用的地方。
更形象地说,动态代理好比一家万能中介公司,不管是租房。还是买票,它都可以实现,只是流程和服务不同而已。而具体的流程和服务,完全可以在有了具体业务对象后确定,而且服务还能动态增加和减少,是不是特别棒?!
这么的好处是:
1、符合开闭原则:对扩展开放,对修改关闭
2、实现无侵入式代码扩展,比如可以在调用时被代理类的方法时定义一些前置或者后置操作(如增加展示资质环节)
而且也做非常「优雅」:
具体来说,动态代理的思路就是:
1、从Class和ClassLoader着手,通过反射得到被代理类的Class对象
2、然后据此创建实例,从而调用该实例的方法(之前写反射代码的时候就是这么干的)
动态代理的实现也有两种思路:
1、JDK,Java原生支持的动态代理方式,实现起来比较简单
2、CGLIB,这也是Spring框架底层使用的动态代理,效率更高且更神奇
先用Java 8的默认接口改造:
/**
* 需要代理的行为
*/
public interface Renting {
// 租房
default public void renthouse() {
System.out.println("租客租房");
};
}
再来用JDK的动态代理实现之前的功能:
/**
* JDK动态代理实现
*/
public class DynamicProxy {
// JDK动态代理实现
private static Object getProxyObject(final Object target) {
return Proxy.newProxyInstance(target.getClass().getClassLoader(),
// 实现接口InvocationHandler
target.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 这里可以实现前置方法
System.out.println("代理方法前置1:租房前看中介资质");
System.out.println("代理方法前置2:租房前看房");
if (null != args) {
for (Object arg : args) {
System.out.println(" - " + arg);
}
}
// 调用接口方法
Object result = method.invoke(target, args);
// 这里可以实现后置方法
System.out.println("代理方法后置:满意签合同");
return result;
}
});
} public static void main(String[] args) {
Consumer consumer = new Consumer();
consumer.renthouse();
System.out.println("===================");
Renting logger = (Renting) getProxyObject(consumer);
logger.renthouse();
}
}
如果改用CGLIB实现,需要额外引入相关的jar包依赖asm-x.y.jar和cglib-x.y.z.jar(这里的x、y、z代表版本号,我这里用的是asm-2.7.jar和cglib-3.3.0.jar),我就给出代码好了(懒得引了,有兴趣的童鞋可以自己试试):
/**
* CGLIB动态代理实现
*/
public class CglibProxy {
// CGLIB动态代理实现
private static Object getCglibProxy(final Object target) {
// 实现MethodInterceptor接口
MethodInterceptor interceptor = new MethodInterceptor() {
@Override
public Object intercept(Object object, Method method, Object[] args, MethodProxy proxy) throws Throwable {
// 这里可以实现前置方法
System.out.println("代理方法前置1:租房前看中介资质");
System.out.println("代理方法前置2:租房前看房");
if (null != args) {
for (Object arg : args) {
System.out.println(" - " + arg);
}
}
// 调用接口方法
Object result = method.invoke(target, args);
// 这里可以实现后置方法
System.out.println("代理方法后置:满意签合同");
return result;
}
};
// 增强代理对象
Enhancer enhancer = new Enhancer();
// 设置父类,因为CGLIB是针对指定的类生成一个子类,所以需要指定父类
enhancer.setSuperclass(target.getClass());
// 设置回调
enhancer.setCallback(interceptor);
// 创建并返回代理对象
return enhancer.create();
} public static void main(String[] args) {
Consumer consumer = new Consumer();
consumer.renthouse();
System.out.println("===================");
Renting logger = (Renting) getCglibProxy(consumer);
logger.renthouse();
}
}
Java中的反射与代理(2)的更多相关文章
- 【Java基础】java中的反射机制与动态代理
一.java中的反射机制 java反射的官方定义:在运行状态下,可以获取任意一个类的所有属性和方法,并且可通过某类任意一对象实例调用该类的所有方法.这种动态获取类的信息及动态调用类中方法的功能称为ja ...
- Java中的反射和注解
前言 在Java中,反射机制和注解机制一直是一个很重要的概念,那么他们其中的原理是怎么样呢,我们不仅仅需要会使用,更要知其然而之所以然. 目录 反射机制 反射如何使用 注解定义 注解机制原理 注解如何 ...
- java中InvocationHandler 用于实现代理。
以下的内容部分参考了网络上的内容,在此对原作者表示感谢! Java中动态代理的实现,关键就是这两个东西:Proxy.InvocationHandler,下面从InvocationHandler接口中的 ...
- java中动态反射
java中动态反射能达到的效果和python的语法糖很像,能够截获方法的实现,在真实方法调用之前和之后进行修改,甚至能够用自己的实现进行特别的替代,也可以用其实现面向切片的部分功能.动态代理可以方便实 ...
- 第89节:Java中的反射技术
第89节:Java中的反射技术 反射技术是动态的获取指定的类,和动态的调用类中的内容(没有类前就可以创建对象,将对象的动作完成,这就是动态的获取指定的类). 配置文件把具体实现的类名称定义到配置文件中 ...
- java 中利用反射机制获取和设置实体类的属性值
摘要: 在java编程中,我们经常不知道传入自己方法中的实体类中到底有哪些方法,或者,我们需要根据用户传入的不同的属性来给对象设置不同的属性值,那么,java自带的反射机制可以很方便的达到这种目的,同 ...
- java中的反射(三)
目录 一.反射 1.class类 2.访问字段 3.调用方法 4.调用构造方法 5.获取继承对象 6.动态代理 二.sping中的反射 本篇转自:https://depp.wang/2020/05/0 ...
- java中的反射(二)
java中的反射(一):https://www.cnblogs.com/KeleLLXin/p/14060555.html 目录 一.反射 1.class类 2.访问字段 3.调用方法 4.调用构造方 ...
- java中的反射机制在Android开发中的用处
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反 ...
随机推荐
- composer常用命令(部分摘抄)
1. 仅更新单个库 composer update foo/bar 2. 不编辑composer.json的情况下安装库 composer require "foo/bar:1.0.0&qu ...
- xray+awvs的联动
前言:xray是一款强大的漏扫工具,配合awvs的爬站功能可以十分轻松实现全自动挖洞,这里awvs我是使用的是12版本 1.启动xray,监听本地指定端口,这里我监听的是2222端口 命令:xray_ ...
- 离线安装docker
一.安装步骤 1.下载Docker二进制文件(离线安装包) 下载地址:https://download.docker.com/linux/static/stable/x86_64/ 注:本文使用 do ...
- 想学渗透测试,应该考CISP-PTE还是NISP-PT?|网安伴nisp和cisp
其实两者都可,但要看考生的实际需求! 为什么说两者都可以? 两个证书都由中国信息安全测评中心颁发,CISP-PTE全称国家注册渗透测试工程师,NISP-PT全称国家信息安全水平考试-渗透测试工程师专项 ...
- 【美国血统 American Heritage 题解】已知前序中序 求后序
题目: 题目名称:美国血统 American Heritage 题目来源:美国血统 American Heritage ## 题目描述 农夫约翰非常认真地对待他的奶牛们的血统.然而他不是一个真正优秀的 ...
- 【MySQL】从入门到掌握4-主键与Unique
上期:[MySQL]从入门到掌握3-WorkBench 第一章:主键 在实际开发中,我们不会使用用户名字当作主键. 因为当我们用数据库记录学生信息的时候,学生有可能重名! 我们一般会使用是个int ...
- 写给前端的 react-native 入门指南
前言 本文主要介绍 react-native(下称 RN) 的入门, 和前端的异同点 文章不涉及功能的具体实现 选择优势 我们先说说, 为什么很多人会选择使用 RN .他对应的特性和普通 Web 的区 ...
- Windows 系统 PostgreSQL 手工安装配置方法
自从2020年底开始接触 PostgreSQL 以来就喜欢上了这个数据库,个人感觉比 MySQL 好用,多表联合查询性能好很多,同时也不存在 SQLServer 的版权授权费用问题.搭配 .NET 开 ...
- GIN 索引
GIN(Generalized Inverted Index, 通用倒排索引) 是一个存储对(key, posting list)集合的索引结构,其中key是一个键值,而posting list 是一 ...
- Markdown Support
Markdown 支持一览 Markdown 支持一览 身正不怕影子斜 我实在没有说过这样一句话 -- 鲁迅 古代文学史发展脉络 唐诗 宋词 元曲 冯·诺依曼结构 运算器 控制器 存储器 输入输出设备 ...