通过反射实例化对象
- 获取Class,能干什么?
通过Class的newInstance()方法来实例化对象
注意:newInstance()内部实际上调用了无参数构造方法,必须保证无参数构造的存在
代码示例
public class ReflectTest01 {
public static void main(String[] args) {
//不适用反射机制创建对象
User user = new User();
System.out.println(user);
//下面这段代码是以反射机制的方式创建对象
try {
//通过反射机制,获取Class,通过Class来实例化对象
Class c = Class.forName("bean.User");//c代表User类型
//newInstance()这个方法会调用User这个类的无参数构造方法,完成对对象的创建
//重点是:newInstance()调用的是无参构造,必须保证无参构造是存在的
Object obj = c.newInstance();
System.out.println(obj);//bean.User@16b98e56
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
User类:
public class User {
public User(){System.out.println("无参数构造方法");}
public User(String s){}
}
运行结果:
无参数构造方法
bean.User@16b98e56
无参数构造方法
bean.User@7ef20235
一个对象没有构造方法时,会自动创建无参构造方法
【注意:当定义了有参构造方法时,必须创建无参构造方法,否则会报错java.lang.InstantiationException实例化异常】两种创建方式的区别:
通过反射机制创建对象更加灵活
测试反射机制的灵活性
- 验证反射机制的灵活性:
Java代码写一遍,再不改变Java源代码的基础之上,可以做到不同对象的实例化
非常之灵活(符合OCP开闭原则:对扩展开放,对修改关闭) - 为什么要学反射机制:
后期要学高级框架,而在工作过程中,也都是使用高级框架
包括:ssh , ssm , spring , springmvc 。。。
这些高级框架底层实现原理,都采用了反射机制,所以反射机制还是重要的
【学会了反射机制,有利于理解剖析框架底层的源代码】
代码示例:
public class ReflectTest02 {
public static void main(String[] args) {
//这种方式代码就写死了,只能创建一个User类型的对象
// User user = new User();
//一下代码是灵活的,代码不需要改动,可以修改配置文件,配置文件修改之后,可以创建出不同的实例化对象
FileReader reader = null ;
try {
//通过IO流读取classinfo.properties文件
reader = new FileReader("classinfo.properties");
//创建属性类对象Map
Properties pro = new Properties();
//加载
pro.load(reader);
//通过key获取value
String className = pro.getProperty("className");
// System.out.println(className);
//通过反射机制实例化对象
Class c = Class.forName(className);
Object obj = c.newInstance();
System.out.println(obj);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} finally{
if (reader!=null) {
try {
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
classinfo.properties配置文件:
className=bean.User
运行结果:
无参数构造方法
bean.User@15aeb7ab
classinfo.properties配置文件:
className=java.util.Date
运行结果:
Tue Sep 27 09:27:53 CST 2022
只让静态代码块执行
研究一下:Class.forName()发生了什么?
代码示例:
public class TestReflect04 {
public static void main(String[] args) {
try {
Class.forName("Day01.Myclass");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
class Myclass{
//静态代码块在类加载时执行,并且只执行一次
static{
System.out.println("Myclass类的静态代码块执行了");
}
}
运行结果:
Myclass类的静态代码块执行了
重点:如果你只希望一个类的静态代码块执行,其他代码一律不执行
你可以使用:**Class.forName("完整类名");**
这个方法的执行会导致类加载,类加载时,静态代码块执行
提示:后面JDBC技术的时候我们还需要这段代码
获取类路径下文件的绝对路径
FileReader reader = new FileReader("bean.User");
这种方式的路径缺点是:移植性差,在IDEA中默认的当前路径是project的根
这个代码假设离开IDEA,换到其他位置,可能当前路径就不是project的根了,这时这个路径就无效了
通用获取路径的方式
通用路径,即使代码换位置了,这样编写仍然是通用的
【注意】:使用以下通用方式的前提是:这个文件不许在类路径下
【记住】什么是类路径下?方式在src下的都是类路径下
src是类的根路径
String path =Thread.currentThread().getContextClassLoader().getResource("classinfo02.properties").getPath();
System.out.println(path);
采用以上的代码可以拿到一个文件的绝对路径,这种方式获取文件绝对路径是通用的
解释:
**Thread.currentThread()😗*当前线程对象
getContextClassLoader(): 是线程对象的方法,可以获取当前线程的类加载器对象
getResource(): 【获取资源】这是类加载器对象的方法,当前线程的类加载器默认从类的根路径下加载资源
**getPtah() 😗*获取路径【注意!!】以上讲解的方式是通用的,但前提需要在类路径下,才能用这种方式
以流的形式直接返回
代码示例:
public class IoPropertiesTest {
public static void main(String[] args) {
// FileReader reader = null ;
try {
// String path = Thread.currentThread().getContextClassLoader().getResource("classinfo02.properties").getPath();
//
// reader = new FileReader(path);
//直接以流的形式返回
InputStream reader = Thread.currentThread().getContextClassLoader().getResourceAsStream("classinfo02.properties");
Properties pro = new Properties();
pro.load(reader);
String className = pro.getProperty("className");
System.out.println(className);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
// finally {
if (reader!=null) {
try {
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
classinfo02.properties
className=bean.User
运行结果:
bean.User