首页 > 其他 > 详细

反射机制

时间:2021-04-18 11:03:09      阅读:31      评论:0      收藏:0      [点我收藏+]

反射机制的概述

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

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!