cglib代理常用接口和api
关于cglib代理的概念和原理,这里就不多说了,详细请参考以往的博客:
- 初步探究cglib动态代理的原理:http://www.jiajiajia.club/blog/artical/yjw520/134
- cglib动态代理底层实现分析java:http://www.jiajiajia.club/official/weblog/yjw520/34
本篇主要介绍cglib的主要(常用)的api。
一、MethodInterceptor
入门的helloword
import net.sf.cglib.proxy.*;
import java.lang.reflect.Method;
class SampleClass{
public void test(){
System.out.println("test");
}
public void test2(){
System.out.println("test2");
}
}
public class test {
public static void main(String[] args) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(SampleClass.class);
enhancer.setCallback(new MethodInterceptor(){
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("before");
Object res=methodProxy.invokeSuper(o, objects);
System.out.println("after");
return res;
}
});
SampleClass proxy = (SampleClass) enhancer.create();
proxy.test();
proxy.test2();
}
}
输出:
before
test
after
before
test2
after
该种方法将代理SampleClass类的非final的所有成员方法。如果只想代理某个指定的方法,可以实现CallbackFilter接口,如下:
import net.sf.cglib.proxy.*;
import java.lang.reflect.Method;
class SampleClass{
public void test(){
System.out.println("test");
}
public void test2(){
System.out.println("test2");
}
}
public class test {
public static void main(String[] args) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(SampleClass.class);
CallbackHelper callbackHelper = new CallbackHelper(SampleClass.class, new Class[0]) {
@Override
protected Object getCallback(Method method) {
// 只代理方法名是test的方法
if(method.getName().equals("test")){
return new MethodInterceptor(){
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.print("代理---");
return methodProxy.invokeSuper(o, objects);
}
};
}else{
return NoOp.INSTANCE;
}
}
};
enhancer.setCallbackFilter(callbackHelper);
enhancer.setCallbacks(callbackHelper.getCallbacks());
SampleClass proxy = (SampleClass) enhancer.create();
proxy.test();
proxy.test2();
}
}
输出:
代理---test
test2
这种方法可以指定具体代理某一个方法。
二、InvocationHandler
InvocationHandler可以实现对接口的代理,类似与mybatis,代码中开发者只需要写一个接口和一个xml,mybatis会根据xml的描述信息,帮你生成对应的方法实现,参数的解析、sql的生成、发送、结果集的封装等都不用开发者去写。而其中的原理就是利用了代理模式。
首先定义一个接口,这个接口里有各种返回值类型的方法。
import java.util.List;
public interface TestInterface {
String testString(Integer id);
int testInt(String value);
List testList(Integer age,Integer g);
SampleClass testSampleClass();
}
用代理模式去实现这些方法。实现类交由cglib去动态创建。
import net.sf.cglib.proxy.*;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
public class test2 {
public static void main(String[] args) {
Class[] classes = {TestInterface.class};
Enhancer enhancer = new Enhancer();
enhancer.setInterfaces(classes);
CallbackHelper callbackHelper = new CallbackHelper(Object.class, classes) {
@Override
protected Object getCallback(Method method) {
// 只代理TestInterface中的方法
if(method.getDeclaringClass() == TestInterface.class){
return new InvocationHandler() {
@Override
public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
System.out.print("参数:");
for (Object object : objects) {
System.out.print(object+"\t");
}
System.out.println();
Class<?> returnType = method.getReturnType();
if (returnType == int.class || returnType == Integer.class) {
// 返回值类型是整数类型
return 1;
}else if(returnType == String.class){
// 返回值类型是字符串
return "string";
}else if(returnType == List.class){
// 返回值类型是集合
ArrayList arrayList = new ArrayList();
arrayList.add("11");
return arrayList;
}
// ... 等等
else{
// 自定义类型
return returnType.newInstance();
}
}
};
}else{
// 什么都不做
return NoOp.INSTANCE;
}
}
};
enhancer.setCallbackFilter(callbackHelper);
enhancer.setCallbacks(callbackHelper.getCallbacks());
TestInterface proxy = (TestInterface) enhancer.create();
System.out.println("返回值:"+proxy.testInt("123"));;
System.out.println("返回值:"+proxy.testList(1,1));;
System.out.println("返回值:"+proxy.testString(1));;
System.out.println("返回值:"+proxy.testSampleClass());;
}
}
三、FixedValue
该方法会对所有拦截的方法返回同样的值。
import net.sf.cglib.proxy.*;
import java.lang.reflect.Method;
class SampleClass{
public String test(){
return "test";
}
public String test2(){
return "test2";
}
}
public class test {
public static void main(String[] args) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(SampleClass.class);
CallbackHelper callbackHelper = new CallbackHelper(SampleClass.class, new Class[0]) {
@Override
protected Object getCallback(Method method) {
// 只代理返回值类型是String类型的方法,如果返回值类型不是String将会抛异常。
if(method.getReturnType()==String.class){
return new FixedValue() {
@Override
public Object loadObject() throws Exception {
return "proxy ...";
}
};
}else{
return NoOp.INSTANCE;
}
}
};
enhancer.setCallbackFilter(callbackHelper);
enhancer.setCallbacks(callbackHelper.getCallbacks());
SampleClass proxy = (SampleClass) enhancer.create();
System.out.println(proxy.test());
System.out.println(proxy.test2());
System.out.println(proxy.toString());
System.out.println(proxy.hashCode());
}
}
输出:
proxy ...
proxy ...
proxy ...
1874154700
返回值是String类型的都返回同样的值。
四、ImmutableBean
不可变的Bean。ImmutableBean允许创建一个原来对象的包装类,这个包装类是不可变的,任何改变底层对象的包装类操作都会抛出IllegalStateException。但是我们可以通过直接操作底层对象来改变包装类对象。
import net.sf.cglib.beans.ImmutableBean;
class SampleBean{
SampleBean(){
}
SampleBean(int id){
this.id = id;
}
private int id;
public int getId(){
return this.id;
}
public void setId(int id){
this.id = id;
}
}
public class test2 {
public static void main(String[] args) {
SampleBean bean = new SampleBean();
bean.setId(1);
SampleBean immutableBean = (SampleBean)ImmutableBean.create(bean);
System.out.println(bean.getId());
System.out.println(immutableBean.getId());
bean.setId(2);
System.out.println(bean.getId());
System.out.println(immutableBean.getId());
immutableBean.setId(3); // 包装类修改将会抛异常
System.out.println(bean.getId());
System.out.println(immutableBean.getId());
}
}
输出:
1
1
2
2
Exception in thread "main" java.lang.IllegalStateException: Bean is immutable
at com.example.SampleBean$$ImmutableBeanByCGLIB$$e23e512b.setId(<generated>)
at com.example.test2.main(test2.java:31)
五、BeanGenerator
cglib提供的一个操作bean的工具,使用它能够在运行时动态的创建一个bean。
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class test2 {
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
BeanGenerator beanGenerator = new BeanGenerator();
beanGenerator.addProperty("id",int.class);
Object myBean = beanGenerator.create();
Method setter = myBean.getClass().getMethod("setId",int.class);
setter.invoke(myBean,1);
Method getter = myBean.getClass().getMethod("getId");
System.out.println(getter.invoke(myBean));
}
}
动态创建一个类,该类包含一个int类型的id属性,并包含get,set方法。
六、BeanMap
BeanMap类实现了Java Map,将一个bean对象中的所有属性转换为一个String-to-Obejct的Java Map
import net.sf.cglib.beans.BeanMap;
import java.lang.reflect.InvocationTargetException;
class Bean{
int id = 1;
String name = "name";
public int getId(){
return this.id;
}
public String getName(){
return this.name;
}
}
public class test2 {
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Bean bean = new Bean();
BeanMap beanMap = BeanMap.create(bean);
Object id = beanMap.get("id");
Object name = beanMap.get("name");
System.out.println(id);
System.out.println(name);
}
}
输出:
1
name
评论区
请写下您的评论...
猜你喜欢
ASM,java基础
1264
关于cglib代理的概念和原理以及常用api,请参考:初步探究cglib动态代理的原理:http://www.jiajiajia.club/blog/artical/yjw520
算法基础
1223
LocalDate1.获取当前时间LocalDatetoday=LocalDate.now();2.获取当前月份LocalDatetoday=LocalDate.now();intmonth=today.getMonthValue();3.获取当前日期LocalDatetoday=LocalDate.now();intday=today.getDayOfMonth();4.获取指定日期LocalD
blog
初步探究cglib动态代理的原理
java基础
3457
初步探究cglib动态代理的原理之前我们说了一下jdk动态代理的原理http://www.jiajiajia.club/weblog/blog/artical/60本章说一下cglib动态代理,做个
blog
cglib代理指定只代理某个方法
ASM,java基础
1140
cglib代理指定只代理某个方法importnet.sf.cglib.proxy.*;importjava.lang.reflect.Method;classSampleClass
weblog
4888
前言
好久就开始关注cglib动态代理了,但是到今天才算是有点搞明白~cglib动态代理底层用到了asm等字节码操作框架。不懂的可以先百度百度asm是干啥的,在这里也提供一些文章,但是可能比较深入
official
2848
BeanPostProcessor是SpringIOC容器给我们提供的一个扩展接口。,他的作用主要是如果我们需要在Spring容器完成Bean的实例化、配置和其他的初始化前后添加一些自己的逻辑处
工具
2915
/dependency2.java代码调用SoapClientclient=SoapClient.create("http://cwzf.zzcsjr.edu.cn/xysf/OrderView.asmx")//设
weblog
1215
报错信息javax.net.ssl.SSLHandshakeException:sun.security.validator.ValidatorException:PKIXpathbuildingfailed:sun.security.provider.certpath.SunCertPathBuilderException:unabletofindvalidcertificationpathto
最新发表
归档
2018-11
12
2018-12
33
2019-01
28
2019-02
28
2019-03
32
2019-04
27
2019-05
33
2019-06
6
2019-07
12
2019-08
12
2019-09
21
2019-10
8
2019-11
15
2019-12
25
2020-01
9
2020-02
5
2020-03
16
2020-04
4
2020-06
1
2020-07
7
2020-08
13
2020-09
9
2020-10
5
2020-12
3
2021-01
1
2021-02
5
2021-03
7
2021-04
4
2021-05
4
2021-06
1
2021-07
7
2021-08
2
2021-09
8
2021-10
9
2021-11
16
2021-12
14
2022-01
7
2022-05
1
2022-08
3
2022-09
2
2022-10
2
2022-12
5
2023-01
3
2023-02
1
2023-03
4
2023-04
2
2023-06
3
2023-07
4
2023-08
1
2023-10
1
2024-02
1
2024-03
1
2024-04
1
2024-08
1
标签
算法基础
linux
前端
c++
数据结构
框架
数据库
计算机基础
储备知识
java基础
ASM
其他
深入理解java虚拟机
nginx
git
消息中间件
搜索
maven
redis
docker
dubbo
vue
导入导出
软件使用
idea插件
协议
无聊的知识
jenkins
springboot
mqtt协议
keepalived
minio
mysql
ensp
网络基础
xxl-job
rabbitmq
haproxy
srs
音视频
webrtc
javascript
加密算法
目录
没有一个冬天不可逾越,没有一个春天不会来临。最慢的步伐不是跬步,而是徘徊,最快的脚步不是冲刺,而是坚持。