反射机制的概述
1:反射作用:
通过java语言的反射机制操作字节码文件;
优点类似于黑客(可以读和修改字节码文件);
通过反射机制可以操作代码片段。
2:反射机制的相关类在哪个包下?
java.lang.rflect.*;
3:反射机制相关的重要的类有哪些?
java.lang.Class; (代表字节码,代码一个类型,表示整个类)
java.lang.reflect.Method;(代表字节码中方法字节码)
java.lang.reflect.Constrcuctor;(代表字节码中构造方法字节码)
java.lang.reflect.Fild;(代表字节码中属性字节码)
代码实现!!
packagecom.bjpowernode.java.threadsafe;
//整个class01为Class
?
publicclassUser01 {
//Field
intno;
?
//Constructor
publicUser01() {
}
?
publicUser01(intno) {
this.no=no;
}
?
//Method
publicintgetNo() {
returnno;
}
?
publicvoidsetNo(intno) {
this.no=no;
}
}
2:获取Class的三种方式(Class.forName(完整类名包名))
2.1:Class.forName
静态方法,方法是参数是一个完整类名,字符串需要完整类名,完整类名必带java.lang不能省略。
Class c = Class.forName("java.lang.String"); c为String.Class文件
Class c1 = Class.forName("java.lang.Date") c1为Date
2.2:第二种方式 (引用.getClass)
java中任何一个对象都有一个方法:get.class();
String s = "abc";
Class x = s.getClass(); x代表String.Class字节码文件,x是String类型
Date time = new Date();
Class y = time.getClass; 内存地址一样,指向方法区的字节码
2.3:第三种方法(数剧类型.class)
java语言任何一种类型,包括基本数据类型都有.classs属性
Class z = String.class; z代表String类型
Class k = Date.class; k代表Date类型
3:通过反射实例化对象
重点:通过Class的newInstance()方法实例化对象
newInstance()方法内部实际上调用了无参构造方法,必须保证无参构造存在才可能。
代码实现:
//创建对象
User01user01=newUser01();
System.out.println(user01);
//使用反射机制方式创建对象
try {
Classc=Class.forName("com.bjpowernode.java.threadsafe.User01"); //c为User类型
?
//newInstance()这个方法会调用User这个类的无参构造方法,完成对象创建
//重点:newInstance()调用的是无参构造,必须保证无参是存在的
Objectobj=c.newInstance(); //obj创建的是User类的对象
System.out.println(obj);
} catch (InstantiationExceptione) {
e.printStackTrace();
} catch (IllegalAccessExceptione) {
e.printStackTrace();
?
} catch (ClassNotFoundExceptione) {
e.printStackTrace();
}
4:通过属性配置文件实例化对象.
重点:代码灵活,代码不需要改动,可以修改配置文件,配置文件修改之后可以创建不同的实例对象
代码实现!!!(通过io的properties连用)
publicclassRrflectTest03 {
publicstaticvoidmain(String[] args) throwsIOException, ClassNotFoundException, InstantiationException, IllegalAccessException {
//通过io读取classinfo.properties文件
FileReaderreader=newFileReader("DuoXianCheng/classinfo.properties");
//创建属性类对象Map
Propertiesproperties=newProperties();
//加载,调用Properties中load方法将文件加载到Map文件中
properties.load(reader); //文件中的数据会通过管道加载到Map中
//关闭流
reader.close();
?
//通过key获取value
StringclassName=properties.getProperty("className");
?
//通过反射机制实例化对象
Classc=Class.forName(className);
Objectobj =c.newInstance();
System.out.println(obj);
}
}
5:只让静态代码块执行可用forName
重点:只希望静态代码块执行可使用Class.forName("完整类名")
这个方法执行会导致类加载,类加载是静态代码块执行!!
代码实现!!!
publicclassReflectTest04 {
publicstaticvoidmain(String[] args) {
?
?
try {
Class.forName("com.bjpowernode.reflect.MyClass");
} catch (ClassNotFoundExceptione) {
e.printStackTrace();
}
}
}
?
classMyClass {
//静态代码块在类加载是操作,并且只执行一次
static {
System.out.println("努力学习!");
}
}
6:获取类路径下文件的绝对路径!!
怎么获取一个文件的绝对路径,以下讲解那个方式通用的,但必须在src类路径下。
//什么是类路径?方式在src下都是类路径
//src是类的根路径
?
/*解释: Thread.currentThread() 当前线程对象
getContextClassLoader() 线程对象方法,可以获取当前线程的类加载对象
getResource(); 这是类加载对象的方法,当前线程的类加载默认从类的根路径下加载资源
关于代码实现!!
Stringpath=Thread.currentThread().getContextClassLoader().getResource("classinfo3.properties").getPath();
?
///D:/Users/Administrator/IdeaProjects/untitled1/out/production/DuoXianCheng/classinfo3.properties
System.out.println(path); //获取绝对路径
7:以流形式直接返回
代码实现!!
publicclassIoProertiesTest {
publicstaticvoidmain(String[] args) throwsIOException {
//获取一个文件的绝对路径
/*String path = Thread.currentThread().getContextClassLoader()
.getResource("classinfo3.properties").getPath();
FileReader reader = new FileReader(path);*/
?
//直接以流的形式返回
InputStreamreader= Thread.currentThread().getContextClassLoader()
.getResourceAsStream("classinfo3.properties");
?
?
Propertiesproperties=newProperties();
properties.load(reader);
reader.close();
//通过key获取value
StringclassName=properties.getProperty("className");
System.out.println(className);
}
}
8:资源绑定器(给属性文件专属)
只能绑定xxx.properties文件,并且这个文件必须在类路径下,文件扩展名为properties并写路径时,路径后面扩展名不能写。
代码实现:
publicclassZiYuanBangDingQ {
publicstaticvoidmain(String[] args) {
ResourceBundlebundle=ResourceBundle.getBundle("classinfo3"); //.properties扩展名不能写
?
StringclassName=bundle.getString("className");
?
System.out.println(className);
}
}
原文:https://www.cnblogs.com/020626zy/p/14672787.html