(可变参数):传参能传0-n个,一个形参只有一个可变参,可变参要放到最后。
类加载器:
类的加载:
当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
加载:
就是指将class文件读入内存,并为之创建一个Class对象。任何类被使用时系统都会建立一个Class对象。
连接:
验证 是否有正确的内部结构,并和其他类协调一致,准备 负责为类的静态成员分配内存,并设置默认初始化值,解析 将类的二进制数据中的符号引用替换为直接引用。
初始化:
类初始化时机:
1. 创建类的实例
2. 类的静态变量,或者为静态变量赋值
3. 类的静态方法
4. 使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
5. 初始化某个类的子类
6. 直接使用java.exe命令来运行某个主类
public class Fu { static{ System.out.println("父类静态代码块"); } static int a=1; public static void a(){ System.out.println("父类方法"); } }
public class Zi extends Fu{ }
public class Demo01 { public static void main(String[] args) { //当调用静态成员,该类进内存 //System.out.println(Fu.a); //调用静态成员方法 //Fu.a(); //创建子类对象 new Zi(); } }
类加载器作用: 负责将.class文件加载到内存中,并为之生成对应的Class对象。
类加载器的组成:
Bootstrap ClassLoader 根类加载器,也被称为引导类加载器,负责Java核心类的加载比如System,String等。在JDK中JRE的lib目录下rt.jar文件中。
Extension ClassLoader 扩展类加载器,负责JRE的扩展目录中jar包的加载。在JDK中JRE的lib目录下ext目录。
System ClassLoader 系统类加载器(多指我们字节写的类),负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类路径。
反射:
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象。
Class类:
获得Class对象的三种方式:
package com.oracle.demo01; public class Person { public String name; private int age; static{ System.out.println("静态代码块"); } public Person(){ System.out.println("公共空参构造方法"); } public Person(String name,int age){ System.out.println("公共有参构造"); } private Person(int age){ System.out.println("私有有参构造"); } public void eat(){ System.out.println("公共空参方法"); } public void sleep(String name){ System.out.println("公共的有参方法"); } private void play(){ System.out.println("私有空参方法"); } @Override public String toString() { return "Person [name=" + name + ", age=" + age + "]"; } }
public class Demo02 { public static void main(String[] args) throws ClassNotFoundException { //获取字节码文件对象 Person p=new Person(); Class c1=p.getClass(); System.out.println(c1); //通过class属性 Class c2=Person.class; System.out.println(c2); //forname方法 Class c3=Class.forName("com.oracle.demo01.Person"); System.out.println(c3); } }
前两种你必须明确Person类型,后面是指定这种类型的字符串就行.这种扩展更强.我不需要知道你的类.我只提供字符串,按照配置文件加载就可以了。
通过反射获取构造方法并使用:
在反射机制中,把类中的成员(构造方法、成员方法、成员变量)都封装成了对应的类进行表示。其中,构造方法使用类Constructor表示。可通过Class类中提供的方法获取构造方法。
返回一个构造方法:
public Constructor<T> getConstructor(Class<?>... parameterTypes) 获取public修饰, 指定参数类型所对应的构造方法
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 获取指定参数类型所对应的构造方法(包含私有的)
返回多个构造方法
public Constructor<?>[] getConstructors() 获取所有的public 修饰的构造方法
public Constructor<?>[] getDeclaredConstructors() 获取所有的构造方法(包含私有的)
通过反射方式,获取构造方法,创建对象:
步骤:
1. 获取到Class对象
2. 获取指定的构造方法
3. 通过构造方法类Constructor中的方法,创建对象。 public T newInstance(Object... initargs)
通过反射方式,获取私有构造方法,创建对象:
AccessibleObject 类是 Field、Method 和 Constructor 对象的父类。它提供了将反射的对象标记为在使用时取消默认 Java 语言访问控制检查的能力。对于公共成员、默认(打包)访问成员、受保护成员和私有成员,在分别使用 Field、Method 或 Constructor 对象来设置或获取字段、调用方法,或者创建和初始化类的新实例的时候,会执行访问检查。
public void setAccessible(boolean flag) throws SecurityException
参数值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。参数值为 false 则指示反射的对象应该实施 Java 语言访问检查。
步骤:
1. 获取到Class对象
2. 获取指定的构造方法
3. 暴力访问, 通过setAccessible(boolean flag)方法
4. 通过构造方法类Constructor中的方法,创建对象
代码如下:
public class Person { public String name; private int age; static{ System.out.println("静态代码块"); } public Person(){ System.out.println("公共空参构造方法"); } public Person(String name,int age){ System.out.println("公共有参构造"); } private Person(int age){ System.out.println("私有有参构造"); } public void eat(){ System.out.println("公共空参构造方法"); } public void sleep(String name){ System.out.println("公共的有参方法"); } private void play(){ System.out.println("私有空参方法"); } @Override public String toString() { return "Person [name=" + name + ", age=" + age + "]"; } }
public class Demo03 { public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException { //获取Perosn类的字节码文件对象 //获取公共构造方法 Class c=Class.forName("com.oracle.demo01.Person"); //获取公共构造方法 Constructor con1=c.getConstructor(); System.out.println(con1); //获取公共有参构造 Constructor con2=c.getDeclaredConstructor(String.class,int.class); System.out.println(con2); //获取私有有参构造 Constructor con3=c.getDeclaredConstructor(int.class); System.out.println(con3); //调用方法创建对象 Person p1=(Person)con2.newInstance("张三",18);//返回的是Object类型,强转 p1.eat(); //取消java语言检查 //用私有构造方法对象创建Person对象/暴力反射 //破坏了封装性,不推荐使用 con3.setAccessible(true); Person p2=(Person)con3.newInstance(18); p2.eat(); } }
通过反射获取成员变量并使用:
在反射机制中,把类中的成员变量使用类Field表示。可通过Class类中提供的方法获取成员变量。
返回一个成员变量
public Field getField(String name) 获取指定的 public修饰的变量
public Field getDeclaredField(String name) 获取指定的任意变量
返回多个成员变量
public Field[] getFields() 获取所有public 修饰的变量
public Field[] getDeclaredFields() 获取所有的 变量 (包含私有)
通过反射,创建对象,获取指定的成员变量,进行赋值与获取值操作:
1. 获取Class对象
2. 获取构造方法
3. 通过构造方法,创建对象
4. 获取指定的成员变量(私有成员变量,通过setAccessible(boolean flag)方法暴力访问)
5. 通过方法,给指定对象的指定成员变量赋值或者获取值
public class Person { public String name; private int age; static{ System.out.println("静态代码块"); } public Person(){ System.out.println("公共空参构造方法"); } public Person(String name,int age){ System.out.println("公共有参构造"); } private Person(int age){ System.out.println("私有有参构造"); } public void eat(){ System.out.println("公共空参构造方法"); } public void sleep(String name){ System.out.println("公共的有参方法"); } private void play(){ System.out.println("私有空参方法"); } @Override public String toString() { return "Person [name=" + name + ", age=" + age + "]"; } }
public class Demo04 { public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException { //1:获取字节码文件对象 Class c=Class.forName("com.oracle.demo01.Person"); //获取成员变量对象 Field f=c.getField("name"); //通过反射创建一个Person对象/快速利用空参构造创建对象方式 Object obj=c.newInstance(); //赋值 f.set(obj,"张三"); System.out.println(obj); System.out.println(f.get(obj)); } }
通过反射获取成员方法并使用:
在反射机制中,把类中的成员方法使用类Method表示。可通过Class类中提供的方法获取成员方法。
返回获取一个方法:public Method getMethod(String name, Class<?>... parameterTypes),获取public 修饰的方法。
public Method getDeclaredMethod(String name, Class<?>... parameterTypes), 获取任意的方法,包含私有的
参数1: name 要查找的方法名称; 参数2: parameterTypes 该方法的参数类型
返回获取多个方法:
public Method[] getMethods() 获取本类与父类中所有public 修饰的方法
public Method[] getDeclaredMethods() 获取本类中所有的方法(包含私有的)
通过反射,创建对象,调用指定的方法:
获取成员方法,步骤如下:
1. 获取Class对象
2. 获取构造方法
3. 通过构造方法,创建对象
4. 获取指定的方法
5. 执行找到的方法
public Object invoke(Object obj, Object... args)
执行指定对象obj中,当前Method对象所代表的方法,方法要传入的参数通过args指定。
public class Demo05 { public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException { //获取字节码文件对象 Class c=Class.forName("com.oracle.demo01.Person"); //获取成员方法对象 Method m=c.getMethod("sleep",String.class); //快速创建一个对象 Object obj=c.newInstance(); //执行方法 m.invoke(obj,"张三"); } }
反射配置文件:
public class Cooker { public void cook(){ System.out.println("厨师炒菜"); } }
public class Worker { public void work(){ System.out.println("工人工作"); } }
public class Student { public void study(){ System.out.println("学生学习"); } }
#ClassName=com.oracle.demo01.Student #MethodName=study #ClassName=com.oracle.demo01.Cooker #MethodName=cook ClassName=com.oracle.demo01.Worker MethodName=work
public class Demo06 { public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException { //创建字符输入流对象,明确数据源 FileReader fr=new FileReader("src/com/oracle/demo01/pro.properties"); //创建properties集合 Properties pro=new Properties(); //将文件中中的键值对读取到集合中 pro.load(fr); //获取类名 String className=pro.getProperty("ClassName"); //获取方法名 String methodName=pro.getProperty("MethodName"); //获取字节码文件对象 Class c=Class.forName(className); //创建成员方法对象 Method m=c.getMethod(methodName); //快速创建对象 Object obj=c.newInstance(); //执行方法 m.invoke(obj); } }