分享

反射

 印度阿三17 2019-12-12

反射【框架设计的灵魂】:将类的各个组成部分封装为其他对象,这就是反射机制

好处:1.可以在程序运行过程中,操作这些对象

2. 可以解耦,提高程序的可拓展性

获取class对象的3种方式:

1.Class.forname(“全类名”);将字节码文件加载进内存,返回Class对象【多用于配置文件,将类名定义在配置文件中。读取对象,加载类】

2.类名.class:通过类名的属性class获取【多用于参数的传递】

3.对象.getClass:该方法在object中定义【多用于对象获取字节码的方式】

同一字节码文件(*.class)在一次程序运行中,只会被加载一次,不论通过哪一种 方式获取到的Class对象都是同一个

 


 

Class对象功能

获取功能:

1.获取成员变量们

Field getField(String name)

Field[] getFields() 获取所有public修饰的成员变量

Field getDeclaredField(String name)

Field[] getDeclaredFields() 获取所有成员变量

 

Field方法

Object get(Object obj)  【获取值】

void set(Object obj, Object value) 【设置传入对象值】

 

setAccessible(true)【暴力反射】 忽略访问权限修饰符的安全检查

 

2.获取构造方法们

Constructor<T> getConstructor(类<?>... parameterTypes)

Constructor con=personClass. getConstructor(String.class,int.class)

Constructor<?>[] getConstructors()

Constructor<T> getDeclaredConstructor(类<?>... parameterTypes)  

Constructor<?>[] getDeclaredConstructors()  

 

Constructor

【创建对象】

1.T newInstance(Object... initargs)  

con.newInstance(“张三”,”23”)

2.如果使用空参构造方法创建对象,操作可以简化:Class对象的 newInstance方法

personClass.newInstance()

3.获取成员方法们

Method getMethod(String name, 类<?>... parameterTypes)   

Method[] getMethods()  

Method getDeclaredMethod(String name, 类<?>... parameterTypes)

Method[] getDeclaredMethods()

 

Method

执行方法

Object invoke(Object obj, Object... args)  【第一个参数是类对象】

4.获取全类名

String getName()

返回单位名称(类,接口,数组类,原始类型,或无效)的 类对象表示,作为一个 String。

  

创建一个配置文件pro.properties 

className=cn.itcast.domin.Person

methodName=eat

1.加载配置文件

(1) 创建Properties对象   Properties p= new Properties();

(2)  加载配置文件,转换为一个集合  

  获取class目录下的配置文件

ClassLoder classloader=ReflectTest.class.getClassLoader();[获得该字节码文件的类加载器]

IntputSteam is=Classloader.getResourceAsStream(“pro.properties”);[获取资源所对应的字节流]

pro.load(i);

2.获取配置文件中定义的数据

String  className = pro.getProperty(“className”);

String  className = pro.getProperty(“methodName”);

3.加载该类进内存

Class  cls = Class.forName(className);

4.创建对象

Object obj=cls.newInstance();

5.获取方法对象

Method method=cls.getMethod(methodName);

6.执行方法

method.invoke(obj);

 

【jvm记载类顺序测试】

package edu.xupt.reflect;

public class Demo1 {
static {
System.out.println("main方法将被执行");
}

public static void main(String[] args) throws ClassNotFoundException {
System.out.println(Son.z);


}
}
class Father{
static {
System.out.println("父类静态代码块");
}
public Father() {
System.out.println("父类初始化");
}
static int f=100;
}
class Son extends Father{
static {
System.out.println("子类静态代码块");
z=200;
}
static int z=100;
static final int N=1000;
public Son() {
System.out.println("子类初始化");
}
}



package edu.xupt.reflect;

public class Demo2 {
public static void main(String[] args) throws ClassNotFoundException {
System.out.println("获取系统类的加载器");
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println(systemClassLoader);
System.out.println("获取系统类加载器的父加载器-->拓展加载器");
ClassLoader parent = systemClassLoader.getParent();
System.out.println(parent);
System.out.println("获取拓展类加载器的父加载器-->根加载器");
ClassLoader parent1 = parent.getParent();
System.out.println(parent1);
System.out.println(Demo2.class.getClassLoader());
Class<?> aClass = Class.forName("edu.xupt.reflect.Demo2");
System.out.println(aClass.getClassLoader());

System.out.println("检测JDK内置类是谁加载的");
System.out.println(Object.class.getClassLoader());
System.out.println(Class.forName("java.lang.ClassLoader" /*Object*/).getClassLoader());

System.out.println("获得系统类加载器可以加载的路径");
System.out.println(System.getProperty("java.class.path"));
/*
D:\Java\jdk1.8.0_212\jre\lib\charsets.jar;
D:\Java\jdk1.8.0_212\jre\lib\deploy.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\access-bridge-64.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\cldrdata.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\dnsns.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\jaccess.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\jfxrt.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\localedata.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\nashorn.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\sunec.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\sunjce_provider.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\sunmscapi.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\sunpkcs11.jar;
D:\Java\jdk1.8.0_212\jre\lib\ext\zipfs.jar;
D:\Java\jdk1.8.0_212\jre\lib\javaws.jar;
D:\Java\jdk1.8.0_212\jre\lib\jce.jar;
D:\Java\jdk1.8.0_212\jre\lib\jfr.jar;
D:\Java\jdk1.8.0_212\jre\lib\jfxswt.jar;
D:\Java\jdk1.8.0_212\jre\lib\jsse.jar;
D:\Java\jdk1.8.0_212\jre\lib\management-agent.jar;
D:\Java\jdk1.8.0_212\jre\lib\plugin.jar;
D:\Java\jdk1.8.0_212\jre\lib\resources.jar;
D:\Java\jdk1.8.0_212\jre\lib\rt.jar;
D:\Java\zhang\2019.11.23—注解\out\production\2019.11.23—注解;
D:\IntelliJ IDEA 2019.2.3\lib\idea_rt.jar

*/

}
}



【反射基本方法测试】
package edu.xupt.reflect;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class Demo3 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
Class<?> c1 = Class.forName("edu.xupt.reflect.Student");

System.out.println("类名");
System.out.println(c1.getName());
System.out.println(c1.getSimpleName());
System.out.println("======================");
System.out.println("获得成员变量");
Field[] fields = c1.getFields();
for (Field field : fields) {
System.out.println(field);
}
fields = c1.getDeclaredFields();
for (Field field : fields) {
System.out.println(field);
}
System.out.println("======================");
Method[] methods = c1.getMethods();
for (Method method : methods) {
System.out.println(method);
}
System.out.println(">>>>>>>>>>>>>>>>>>>>");
methods = c1.getDeclaredMethods();
for (Method method : methods) {
System.out.println(method);
}
System.out.println("======================");
Constructor<?> constructor = c1.getConstructor(int.class, String.class, boolean.class);
System.out.println(constructor);
Class<String> stringClass = String.class;
System.out.println(stringClass);
}
}

class Student{
private int id;
private String name;
private boolean sex;
// public String name1;

public Student() {
}

public Student(int id, String name, boolean sex) {
this.id = id;
this.name = name;
this.sex = sex;
}

public Student(int id) {
this.id = id;
}

public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getName() {
return name;
}

public String setName(String name) {
return this.name = name;
}
private void show(){
System.out.println("show被调用");
}
public void show1(){
System.out.println("show1被调用");
}

public boolean isSex() {
return sex;
}

public void setSex(boolean sex) {
this.sex = sex;
}

@Override
public String toString() {
return "Student{"
"id=" id
", name='" name '\''
", sex=" sex
'}';
}
}


【单例模式的反射破坏】
package edu.xupt.desidn.singleton;

import java.awt.geom.FlatteningPathIterator;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;

public class Demo6 {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
/* test1 test = new test1();
for (int i = 0; i < 100; i ) {
new Thread(test).start();
}*/
Constructor<Singleton6> Constructor = Singleton6.class.getDeclaredConstructor(null);
Constructor.setAccessible(true);
Singleton6 singleton6 = Constructor.newInstance();
Field flag = Singleton6.class.getDeclaredField("flag");
flag.setAccessible(true);
flag.set(singleton6,false);
Singleton6 singleton61 = Constructor.newInstance();
System.out.println(singleton6 == singleton61);
}
}
class Singleton6{
private static boolean flag= false;
private volatile static Singleton6 singleton6;

private Singleton6() {//[可以考虑直接抛异常,不允许进构造]
if (flag==false){
flag=true;
System.out.println("第一次进构造");}
else {throw new RuntimeException("不要想着用反射破坏我的单例");
}
}

public static Singleton6 getSingleton6() {
if (singleton6==null){
synchronized (Singleton6.class){
if (singleton6==null){
System.out.println("实例化");
singleton6 = new Singleton6();
}
}
}
return singleton6;
}
}
class test6 implements Runnable{
@Override
public void run() {
Singleton6.getSingleton6();
}
}

 

来源:https://www./content-4-596151.html

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约