目录
1.5 获取了Class之后,可以调用无参数构造方法来实例化对象
1.6、如果你只想让一个类的“静态代码块”执行的话,你可以怎么做?
1.10 IO + Properties,怎么快速绑定属性资源文件
2.4 java中为了保证类加载的安全,使用了双亲委派机制。
通过java语言中的反射机制可以操作字节码文件。
优点类似于黑客。(可以读和修改字节码文件。)
通过反射机制可以操作代码片段。(class文件。)
反射机制使得对象的创建更加地灵活,可以通过读取配置文件的方式随意改变new的对象,不会将java代码写死
java.lang.reflect.*;
java.lang.Class:代表整个字节码,代表一个类型,代表整个类。
java.lang.reflect.Method:代表字节码中的方法字节码。代表类中的方法。
java.lang.reflect.Constructor:代表字节码中的构造方法字节码。代表类中的构造方法
java.lang.reflect.Field:代表字节码中的属性字节码。代表类中的成员变量(静态变量+实例变量)。
- java.lang.Class:
- public class User{
- // Field
- int no;
-
- // Constructor
- public User(){
-
- }
- public User(int no){
- this.no = no;
- }
-
- // Method
- public void setNo(int no){
- this.no = no;
- }
- public int getNo(){
- return no;
- }
- }
第一种:
- Class c = Class.forName("完整类名");
- //这个完整的类名可以通过读取配置文件的方式获取
第二种:
Class c = 对象.getClass();
第三种:
- Class c = int.class;
- Class c = String.class;
//c代表的就是日期Date类型
Class c = Class.forName(“java.util.Date”);
//实例化一个Date日期类型的对象
Object obj = c.newInstance();
一定要注意:
newInstance()底层调用的是该类型的无参数构造方法。
如果没有这个无参数构造方法会出现"实例化"异常。
Class.forName(“该类的类名”);
这样类就加载,类加载的时候,静态代码块执行!!!!
在这里,对该方法的返回值不感兴趣,主要是为了使用“类加载”这个动作。
- String path = Thread.currentThread().getContextClassLoader()
- .getResource("写相对路径,但是这个相对路径从src出发开始找").getPath();
-
- String path = Thread.currentThread().getContextClassLoader()
- .getResource("abc").getPath(); //必须保证src下有abc文件。
-
- String path = Thread.currentThread().getContextClassLoader()
- .getResource("a/db").getPath(); //必须保证src下有a目录,a目录下有db文件。
-
- String path = Thread.currentThread().getContextClassLoader()
- .getResource("com/bjpowernode/test.properties").getPath();
- //必须保证src下有com目录,com目录下有bjpowernode目录。
- //bjpowernode目录下有test.properties文件。
这种方式是为了获取一个文件的绝对路径。(通用方式,不会受到环境移植的影响。),但是该文件要求放在类路径下,换句话说:也就是放到src下面。
src下是类的根路径。
- //获取一个文件的绝对路径
- String path = Thread.currrntThread().getContextClassLoader()
- .getResource("classinfo2.properties").getPath();
- FileReader reader = new FileReader(path);
- Properties pro = new Properties();
- pro.load(reader);
- reader.close();
- //通过key获取value
- String class Name = pro.getProperty("className");
- //这里就得到了配置文件中的Name信息,从而可以使用反射机制来new对象
- System.out.println(className);
- InputStream reader = Thread.currentThread().getContextClassLoader()
- .getResourceAsStream("com/bjpowernode/test.properties");
- Properties pro = new Properties();
- pro.load(reader);
- reader.close();
- //通过key获取value
- String class Name = pro.getProperty("className");
- System.out.println(className);
//要求:第一这个文件必须在类路径下
//第二这个文件必须是以.properties结尾。
- ResourceBundle bundle = ResourceBundle.getBundle("com/bjpowernode/test");
- String value = bundle.getString(key);
专门负责加载类的命令/工具。ClassLoader
启动类加载器:rt.jar
扩展类加载器:ext/*.jar
应用类加载器:classpath
假设有这样一段代码:
String s = "abc";
代码在开始执行之前,会将所需要类全部加载到JVM当中。
通过类加载器加载,看到以上代码类加载器会找String.class文件,找到就加载,那么是怎么进行加载的呢?
优先从启动类加载器中加载,这个称为“父”。“父”无法加载到,再从扩展类加载器中加载,这个称为“母”。双亲委派。如果都加载不到,才会考虑从应用类加载器中加载。直到加载到为止。