Java 反射和动态代理文档格式.docx
- 文档编号:17552776
- 上传时间:2022-12-07
- 格式:DOCX
- 页数:20
- 大小:702.46KB
Java 反射和动态代理文档格式.docx
《Java 反射和动态代理文档格式.docx》由会员分享,可在线阅读,更多相关《Java 反射和动态代理文档格式.docx(20页珍藏版)》请在冰豆网上搜索。
用来描述一个类的构造方法
2).Field类:
用来描述一个类的成员变量
3).Method类:
用来描述一个类的方法.
4).Modifer类:
用来描述类内各元素的修饰符
5).Array:
用来对数组进行操作.
Constructor,Field,Method这三个类都是JVM(虚拟机)在程序运行时创建的,用来表示加载类中相应的成员。
也就是说可以通过这些类来获取和改变反射类的所有成员。
反射在框架中是最常用的手段。
一个框架是先于调用者而存在的。
当程序员用一个框架的时候,你写的什么类,框架怎么知道,就是通过java反射机制。
Web.xml、Struts2.xml、applicationContext.xml、hibernate.cfg.xml、Peroson.hbm.xml、sqlMapperConfig.xml、PersonMapper.xml等配置文件中的class对应的字节码,都是通过反射来创建对象的。
Param标签中的参数也是通过反射将其值设置到对象中。
一个标准的javaBean的反射叫做内省。
mybatis的resultMap标签中,把结果集的和PO对象的属性一一对应起来,就能把结果集赋值给PO对象,这也是通过反省实现的。
通过mapper.xml中指定的type,得到PO对象。
二、反射机制的作用:
1、反编译:
.class-->
.java
2、通过反射机制访问java对象的属性,方法,构造方法等;
这样好像更容易理解一些,下边我们具体看怎么实现这些功能。
三、SUN提供的有关反射机制中的类
java.lang.Class;
java.lang.reflect.Constructor;
java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;
很多反射中的方法,属性等操作我们可以从这四个类中查询。
还是哪句话要学着不断的查询API,那才是我们最好的老师。
四、具体功能实现
1、反射机制获取类有三种方法,我们来获取Employee类型
1.//第一种方式:
2.Classc1
=
Class.forName("
Employee"
);
3.//第二种方式:
4.//java中每个类型都有class
属性.
5.Classc2
Employee.class;
6.
7.//第三种方式:
8.//java语言中任何一个java对象都有getClass
方法
9.Employeee
new
Employee();
10.Classc3
e.getClass();
//c3是运行时类
(e的运行时类是Employee)
2、创建对象:
获取类以后我们来创建它的对象,利用newInstance:
1.Class
c
=Class.forName("
2.
3.//创建此Class
对象所表示的类的一个新实例
4.Objecto
c.newInstance();
//调用了Employee的无参数构造方法.
3、获取属性:
分为所有的属性和指定的属性:
a,先看获取所有的属性的写法:
//获取整个类
1.
Class
java.lang.Integer"
//获取所有的属性?
3.
Field[]
fs
c.getDeclaredFields();
4.
5.
//定义可变长的字符串,用来存储属性
StringBuffer
sb
StringBuffer();
7.
//通过追加的方法,将每个属性拼接到此字符串中
8.
//最外边的public定义
9.
sb.append(Modifier.toString(c.getModifiers())
+
"
class
c.getSimpleName()
+"
{\n"
10.
//里边的每一个属性
11.
for(Field
field:
fs){
12.
sb.append("
\t"
//空格
13.
sb.append(Modifier.toString(field.getModifiers())+"
//获得属性的修饰符,例如public,static等等
14.
sb.append(field.getType().getSimpleName()
//属性的类型的名字
15.
sb.append(field.getName()+"
;
\n"
//属性的名字+回车
16.
}
17.
18.
}"
19.
20.
System.out.println(sb);
b,获取特定的属性,对比着传统的方法来学习:
1.public
static
void
main(String[]
args)
throws
Exception{
3.<
span
style="
white-space:
pre"
>
<
/span>
//以前的方式:
/*
User
u
User();
u.age
12;
//set
System.out.println(u.age);
//get
*/
//获取类
User"
//获取id属性
Field
idF
c.getDeclaredField("
id"
//实例化这个类赋给o
Object
o
//打破封装
idF.setAccessible(true);
//使用反射机制可以打破封装性,导致了java对象的属性不安全。
//给o对象的id属性赋值"
110"
idF.set(o,
21.
System.out.println(idF.get(o));
22.}
4、获取属性、方法,和构造方法
属性关键字
含义
FiledgetField(String
name)
返回一个Field对象,它反映此Class对象所表示的类或接口的指定公共成员字段。
Filed[]getFiled()
返回一个包含某些Field对象的数组,这些对象反映此Class对象所表示的类或接口的所有可访问公共字段。
Filed
getDeclaredField(Stringname)
返回一个Field对象,该对象反映此Class对象所表示的类或接口的指定已声明字段。
Filed[]
getDeclaredFiled()
返回Field对象的一个数组,这些对象反映此Class对象所表示的类或接口所声明的所有字段。
方法关键字
getDeclaredMethods()
获取所有的方法
getReturnType()
获得方法的返回类型
getParameterTypes()
获得方法的传入参数类型
getDeclaredMethod("
方法名"
参数类型.class,……)
获得特定的方法
构造方法关键字
getDeclaredConstructors()
获取所有的构造方法
getDeclaredConstructor(参数类型.class,……)
获取特定的构造方法
父类和父接口
getSuperclass()
获取某类的父类
getInterfaces()
获取某类实现的接口
这样我们就可以获得类的各种内容,进行了反编译。
对于JAVA这种先编译再运行的语言来说,反射机制可以使代码更加灵活,更加容易实现面向对象。
五、反射加配置文件,使我们的程序更加灵活
在设计模式学习当中,学习抽象工厂的时候就用到了反射来更加方便的读取数据库链接字符串等,当时不是太理解,就照着抄了。
看一下.NET中的反射+配置文件的使用:
当时用的配置文件是app.config文件,内容是XML格式的,里边填写链接数据库的内容:
configuration>
1.lt;
appSettings>
2.<
add
key="
value="
/>
3.lt;
/appSettings>
4.<
/configuration>
反射的写法:
1.assembly.load("
当前程序集的名称"
).CreateInstance("
当前命名空间名称"
.要实例化的类名);
这样的好处是很容易的方便我们变换数据库,例如我们将系统的数据库从SQLServer升级到Oracle,那么我们写两份D层,在配置文件的内容改一下,或者加条件选择一下即可,带来了很大的方便。
当然了,JAVA中其实也是一样,只不过这里的配置文件为
.properties,称作属性文件。
通过反射读取里边的内容。
这样代码是固定的,但是配置文件的内容我们可以改,这样使我们的代码灵活了很多!
综上为,JAVA反射的再次学习,灵活的运用它,能够使我们的代码更加灵活,但是它也有它的缺点,就是运用它会使我们的软件的性能降低,复杂度增加,所以还要我们慎重的使用它。
六、反射实现动态调用方法
1、创建UserDao接口
2、创建UserDaOracleImpl实现类
3、创建UserDaoMySQLImpl实现类
4、创建TestUserDao测试类
代理模式
代理模式是常用的Java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。
代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。
按照代理的创建时期,代理类可以分为两种。
静态代理:
由程序员创建或特定工具自动生成源代码,再对其编译。
在程序运行前,代理类的.class文件就已经存在了。
动态代理:
在程序运行时,运用反射机制动态创建而成。
静态代理
由程序员创建或工具生成代理类的源码,再编译代理类。
所谓静态也就是在程序运行前就已经存在代理类的字节码文件,代理类和委托类的关系在运行前就确定了。
静态代理的实现步骤
1、创建一个UserDao接口
2、创建UserDaoMySQLImpl实现类
3、创建UserDaoMyOracleImpl实现类
4、创建UserService接口
5、创建UserService接口的实现类UserServiceImpl
6、创建测试类
7、执行结果
动态代理
Spring主要有两大思想,一个是IoC,另一个就是AOP,对于IoC,依赖注入就不用多说了,而对于Spring的核心AOP来说,我们不但要知道怎么通过AOP来满足的我们的功能,我们更需要学习的是其底层是怎么样的一个原理,而AOP的原理就是java的动态代理机制。
jdk动态代理和cglib动态代理。
两种方法同时存在,各有优劣。
jdk动态代理是由Java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的。
总的来说,反射机制在生成类的过程中比较高效,而asm在生成类之后的相关执行过程中比较高效(可以通过将asm生成的类进行缓存,这样解决asm生成类过程低效问题)。
还有一点必须注意:
jdk动态代理的应用前提,必须是目标类基于统一的接口。
如果没有上述前提,jdk动态代理不能应用。
由此可以看出,jdk动态代理有一定的局限性,cglib这种第三方类库实现的动态代理应用更加广泛,且在效率上更有优势。
一、JDK代理(接口代理)
1、JDK代理中的接口和类
在java的动态代理机制中,有两个重要的类或接口,一个是InvocationHandler(Interface)、另一个则是Proxy(Class),这一个类和接口是实现我们动态代理所必须用到的。
每一个动态代理类都必须要实现InvocationHandler这个接口,并且每个代理类的实例都关联到了一个handler,当我们通过代理对象调用一个方法的时候,这个方法的调用就会被转发为由InvocationHandler这个接口的invoke方法来进行调用。
我们来看看InvocationHandler这个接口的唯一一个方法
invoke
方法。
Objectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable
我们看到这个方法一共接受三个参数,那么这三个参数分别代表什么呢?
proxy:
指代我们所代理的那个真实对象
method:
指代的是我们所要调用真实对象的某个方法的Method对象
args:
指代的是调用真实对象某个方法时接受的参数
2、JDK代理实现步骤
1)、创建一个UserDao接口
2)、创建UserDaoMySQLImpl实现类
3)、创建UserDaoOracleImpl实现类
4)、创建代理对象的工厂
5)、创建测试类
6)、执行结果
二、cglib代理
JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,比使用Java反射效率要高。
引入两个JAR文件:
cglib.jar和asm.jar
1、cglib代理中的接口和类
定义一个拦截器。
在调用目标方法时,CGLib会回调MethodInterceptor接口方法拦截,来实现你自己的代理逻辑,类似于JDK中的InvocationHandler接口。
参数:
Object为由CGLib动态生成的代理类实例,Method为上文中实体类所调用的被代理的方法引用,Object[]为参数值列表,MethodProxy为生成的代理类对方法的代理引用。
返回:
从代理实例的方法调用返回的值。
其中,methodProxy.invokeSuper(obj,arg):
调用代理类实例上的proxy方法的父类方法
2、cglib代理实现步骤
4)、创建测试类
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Java 反射和动态代理 反射 动态 代理
