Java动态性之--反射机制
1. 动态语言
程序运行时,可以改变结构或变量类型。典型的语言:
- Python、ruby、javascript等
- 如下javascript代码
function test(){ var s = "var a=3;var b=5;alert(a+b);"; eval(s); }
- C,C++,java不是动态语言,java可以称之为“准动态语言”。但是java有一定的动态性,我们可以利用反射机制、字节码操作获得类似动态语言的特性
- java的动态性让编程的时候更加灵活!
2. 反射机制reflection
- 指的是可以于运行时加载、探知、使用编译其完全未知的类。
- 程序在运行状态中,可以动态加载一个只有名称的类,对于任意一个已加载的类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;
Class c = Class.forName(cn.stu.test.User);
- 加载完类之后,在堆内存中,就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息,我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所有,我们形象的称之为:反射
例1:
public class Demo01 {
public static void main(String[] args) {
String path = "cn.stu.ref.bean.User"; try {
//方法一
Class clazz = Class.forName(path);
//
/**
* ==================================
* 对象时表示或封装一些数据。一个类被加载后,
* JVM会创建一个对应该类的Class对象
* 类的整个结构信息会会放到对应的Class对象中。
* 这个Class对象就是一面镜子一样,
* 通过这面镜子我们可以看到对应类的全部信息
* ===================================
*/
System.out.println(clazz.hashCode());
//一个类只对应一个Class对象
Class clazz2 = Class.forName(path);
System.out.println(clazz2.hashCode()); //方法二
Class strClass = String.class;
Class strClass2 = path.getClass();
System.out.println(strClass==strClass2); } catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
3. Class类介绍
当一个class被加载,或当加载器(class loader)的defineClass()被JVM调用,JVM便自动产生一个Class对象
Class类似Reflection的根源
针对任何你想动态加载、运行的类,唯有先获得相应的Class对象
Class类的对象如何获取?
- 运用getClass()
- 运用Class.forName() --常用
- 运用.class语法
例2:
public class Demo02 { public static void main(String[] args) {
String path = "cn.stu.ref.bean.User";
try {
Class clazz = Class.forName(path); //获取类的名称
System.out.println(clazz.getName());//类名+包名
System.out.println(clazz.getSimpleName());//类名 //获取属性名
Field[] fields = clazz.getFields(); //只能获得public的属性
Field[] fields2 = clazz.getDeclaredFields();//获得所有属性
Field field = clazz.getDeclaredField("name");//获取指定name属性 //获取方法名称
Method[] method = clazz.getDeclaredMethods();
Method m1 = clazz.getDeclaredMethod("getName", null);
//如果方法有参,则必须传递参数类型对应的class对象
Method m2 = clazz.getDeclaredMethod("setName", String.class); //获取构造器
Constructor[] constructors = clazz.getDeclaredConstructors();
Constructor c = clazz.getDeclaredConstructor(int.class,int.class,String.class); } catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
例3:
public class Demo03 {
public static void main(String[] args) {
String path = "cn.stu.ref.bean.User"; try {
Class clazz = Class.forName(path); //通过反射API调用构造方法,构造对象
User u = (User) clazz.newInstance();//其实是调用了User的无参构造方法 Constructor c = clazz.getDeclaredConstructor(int.class,int.class,String.class);
User u2 = (User) c.newInstance(1001,18,"张三"); //通过反射API调用普通方法
User u3 = (User) clazz.newInstance();
Method method = clazz.getDeclaredMethod("setName", String.class);
method.invoke(u3,"李四"); //u3.setName("李四") //通过反射API操作属性
User u4 = (User) clazz.newInstance();
Field f = clazz.getDeclaredField("name");
f.setAccessible(true);//不做安全检查,直接访问
f.set(u4, "二麻子"); //通过反射直接写属性
String name = u4.getName();
String name2 = (String) f.get(u4); //通过反射直接读属性的值
} catch (Exception e) {
e.printStackTrace();
}
}
}
4. 反射机制性能问题
- setAccessible
- 启动和金庸访问安全检查的开关,值为true 则指示反射的对象在使用时应该取消java语言访问检查,值为false 则指示反射的对象应该实施java语言访问检查。并不是true就能访问,false就不能访问。
- 禁止安全检查,可以提高反射的运行速度
- 可以考虑使用: cglib/javaassist字节码操作
例4:
/**
* 通过跳过安全检查,提高反射效率
* 三种执行方法的效率差异比较
*/
public class Demo04 { public static void test01(){
User u = new User(); long startTime = System.currentTimeMillis(); for (int i = 0; i < 1000000000L; i++) {
u.getName();
} long endTime = System.currentTimeMillis();
System.out.println("普通方法调用,执行10亿次,耗时:"+(endTime-startTime)+"ms");
} public static void test02() throws Exception{
User u = new User();
Class clazz = u.getClass();
Method m = clazz.getDeclaredMethod("getName", null);
// m.setAccessible(true); long startTime = System.currentTimeMillis(); for (int i = 0; i < 1000000000L; i++) {
m.invoke(u, null);
} long endTime = System.currentTimeMillis();
System.out.println("反射动态方法调用,执行10亿次,耗时:"+(endTime-startTime)+"ms");
} public static void test03() throws Exception{
User u = new User();
Class clazz = u.getClass();
Method m = clazz.getDeclaredMethod("getName", null);
m.setAccessible(true); //不需要执行访问安全检查 long startTime = System.currentTimeMillis(); for (int i = 0; i < 1000000000L; i++) {
m.invoke(u, null);
} long endTime = System.currentTimeMillis();
System.out.println("反射动态方法调用,跳过安全检查,执行10亿次,耗时:"+(endTime-startTime)+"ms");
} public static void main(String[] args) throws Exception {
test01();
test02();
test03();
}
}
输出结果:
普通方法调用,执行10亿次,耗时:6665ms
反射动态方法调用,执行10亿次,耗时:62947ms
反射动态方法调用,跳过安全检查,执行10亿次,耗时:11864ms
5. 反射操作泛型
- java采用泛型擦除的机制来引入泛型。java中的泛型仅仅是给编译器javac使用的,确保数据的安全性和免去强制类型转换的麻烦,但是,一旦编译完成,所有的和泛型有关的类型全部擦除。
- 为了通过反射操操作这些类型以迎合实际开发的需要,java就新增了ParameterzedType,GenericArrayType,TypeVariable和WildcardType几种类型来代表不能被诡异到Class类中的类型但是又和原始类型齐名的类型。
- ParmterizedType: 表示一种参数化的类型,比如Collection<String>
- GenericArrayType: 表示一种元素类型是参数化类型或者类型变量的数组类型
- TypeVariable:是各种类型变量的公共父接口
- WildcardType:代表一种通配符类型表达式
Java动态性之--反射机制的更多相关文章
- Java动态性之反射机制(reflection)
说到反射机制,第一次接触的人可能会比较懵,反射?什么反射?怎么反射?反射是干嘛的?下面我将在这篇文章中讲讲Java的反射机制 不过在这之前,还有一个问题需要解决,标题名中的动态性,说起动态性,我先介绍 ...
- 小白都能学会的Java注解与反射机制
前言 Java注解和反射是很基础的Java知识了,为何还要讲它呢?因为我在面试应聘者的过程中,发现不少面试者很少使用过注解和反射,甚至有人只能说出@Override这一个注解.我建议大家还是尽量能在开 ...
- java中的反射机制在Android开发中的用处
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反 ...
- 黑马程序员——【Java高新技术】——反射机制
---------- android培训.java培训.期待与您交流! ---------- 一.概述 1.Java反射机制:是指“在运行状态中”,对于任意一个类,都能够知道这个类中的所有属性和方法: ...
- Java 中的反射机制
JAVA反射机制 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为ja ...
- JAVA类型信息——反射机制
JAVA类型信息——反射机制 一.反射机制概述 1.反射机制:就是java语言在运行时拥有的一项自我观察的能力,java通过这种能力彻底了解程序自身的情况,并为下一步的动作做准备. 2.反射机制的功能 ...
- 浅说Java中的反射机制(二)
写过一篇Java中的反射机制,不算是写,应该是抄了,因为那是别人写的,这一篇也是别人写的,摘抄如下: 引自于Java基础--反射机制的知识点梳理,作者醉眼识朦胧.(()为我手记) 什么是反射? 正常编 ...
- 浅说Java中的反射机制(一)
在学习传智播客李勇老师的JDBC系列时,会出现反射的概念,由于又是第一次见,不免感到陌生.所以再次在博客园找到一篇文章,先记录如下: 引用自java中的反射机制,作者bingoideas.(()为我手 ...
- java学习之反射机制
java语言区别于C,C++等准静态语言的最大特点就是java的反射机制.静态语言的最直接定义就是不能在运行时改变程序结构或变量的类型.按照这样的定义,python,ruby是动态语言,C,C++,J ...
随机推荐
- 【刷题】LOJ 6223 「网络流 24 题」汽车加油行驶问题
题目描述 给定一个 \(\text{N}\times \text{N}\) 的方形网格,设其左上角为起点◎,坐标为 \(\text{(1,1)}\) ,\(\text{X}\) 轴向右为正, \(\t ...
- 【BZOJ1056】[HAOI2008]排名系统(Splay)
[BZOJ1056][HAOI2008]排名系统(Splay) 题面 BZOJ 洛谷 题解 \(Splay\)随便维护一下就好了,至于名字什么的,我懒得手写哈希表了,直接哈希之后拿\(map\)压. ...
- bzoj2134: 单选错位(trie)
预处理前后缀异或和,用trie得到前后缀最大答案,枚举中间点把左右两边加起来就是当前中间点的最大答案了...这个操作没见过,比较有意思,记录一下 #include<iostream> #i ...
- CODE FESTIVAL 2017 qual B 题解
失踪人口回归.撒花\^o^/ 说来真是惭愧,NOI之后就没怎么刷过题,就写了几道集训队作业题,打了几场比赛还烂的不行,atcoder至今是蓝名=.= 以后还是多更一些博客吧,我可不想清华集训的时候就退 ...
- 洛谷P2480 古代猪文
这道题把我坑了好久...... 原因竟是CRT忘了取正数! 题意:求 指数太大了,首先用欧拉定理取模. 由于模数是质数所以不用加上phi(p) 然后发现phi(p)过大,不能lucas,但是它是个sq ...
- Python基础数据类型-字典(dict)
Python基础数据类型-字典(dict) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 本篇博客使用的是Python3.6版本,以及以后分享的每一篇都是Python3.x版本的哟 ...
- Vue单页面应用阻止浏览器记住密码
Vue单页面应用阻止浏览器记住密码 ——IT唐伯虎 摘要: Vue单页面应用阻止浏览器记住密码. 现象1:路由切换时再次提示“是否记住密码” 登录页面有个密码输入框,输入账号密码进行登录: 登录完成后 ...
- Spark记录-spark与storm比对与选型(转载)
大数据实时处理平台市场上产品众多,本文着重讨论spark与storm的比对,最后结合适用场景进行选型. 一.spark与storm的比较 比较点 Storm Spark Streaming 实时计算模 ...
- js实现表单提交submit(),onsubmit
通常表单的提交有两种方式,一是直接通过html的form提交,代码如下: <form action="" method="" id="forms ...
- 2018年9月28日CCPC秦皇岛站参赛总结
day1: 被中间结果超出int范围给叉了,立刻意识到了自己的弱小以及校赛出题的时候是怎么叉别人的 day2: 签到签了40分钟,谨慎一些还是很好的,机子重启耽误了一些时间 一道暴力+LCS的简单题被 ...