Java反射笔记

原创
2017/11/28 19:27
阅读数 99

Java反射机制对类中的的方法、属性、构造函数等都定义了相应的类

常用类:
java.lang.Class
java.lang.reflect.Constructor
java.lang.reflect.Field
java.lang.reflect.Method
java.lang.reflect.Modifier

获取一个类的Class对象的三种方式

  • Class.forName()
  • 通过每个类的class属性,例如Class c = Student.class;
  • 通过继承自Object类的getClass()方法,此方法是非静态的,通过创建对象调用

获取了某个类的Class对象之后,可以调用newInstance()方法创建实例

还可以通过Class对象获取一个类的属性、方法等、修饰符

例:使用反射获取类A的方法,并调用

package methodInvoke.demo;

public class A {
    public void hello(String name,int age){
        System.out.println("my name is" + name);
        System.out.println("my age is" + age);
    }
}


/*****************************************************/

package methodInvoke.demo;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class Main {
    public static void main(String[] args) throws ClassNotFoundException, 
IllegalAccessException, InstantiationException, 
NoSuchMethodException, InvocationTargetException {
        Class clz = Class.forName("methodInvoke.demo.A");//必须加上包名,不然找不到类
        //Class<?> clz = Class.forName("A");
        //这两种写法基本相同,但是Class<?>更规范,可以在某些类型转换时避免报错
        //泛型的作用时期是在编译时,运行时泛型被擦除
        Object o = clz.newInstance();//非静态方法,必须创建实例才能调用
        //根据方法名和参数列表来获取确定的方法
        //第一个参数是要获取的方法的名称
        //后面依次是目标方法的参数类型类别
        Method m = clz.getMethod("hello",String.class,int.class);
        //指明调用某个实例的该方法,后面依次是要传入的参数
        m.invoke(o,"kevin",20);
    }
}

============================
java反射

Class.forname("className").newInstance();和new的区别

平时开发中我们经常会发现:用到Class.forName()方法。为什么要用呢? 下面分析一下:

 

 主要功能Class.forName(xxx.xx.xx)返回的是一个类Class.forName(xxx.xx.xx)的作用是要求JVM查找并加载指定的类,也就是说JVM会执行该类的静态代码段

 

在初始化一个类,生成一个实例的时候,newInstance()方法和new关键字除了一个是方法,一个是关键字外,最主要有什么区别?它们的区别在于创建对象的方式不一样,前者是使用类加载机制,后者是创建一个新类。那么为什么会有两种创建对象方式?这主要考虑到软件的可伸缩、可扩展和可重用等软件设计思想。

Java中工厂模式经常使用newInstance()方法来创建对象,因此从为什么要使用工厂模式上可以找到具体答案。 例如:class c = Class.forName(“Example”);factory = (ExampleInterface)c.newInstance();

其中ExampleInterface是Example的接口,可以写成如下形式:String className = “Example”;class c = Class.forName(className); factory = (ExampleInterface)c.newInstance();

进一步可以写成如下形式:String className = readfromXMlConfig;//从xml 配置文件中获得字符串class c = Class.forName(className); factory = (ExampleInterface)c.newInstance();

上面代码已经不存在Example的类名称,它的优点是,无论Example类怎么变化,上述代码不变,甚至可以更换Example的兄弟类Example2 , Example3 , Example4……,只要他们继承ExampleInterface就可以。

从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用newInstance()方法的时候,就必须保证:1、这个类已经加载;2、这个类已经连接了。 而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。

现在可以看出,newInstance()实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。

二.new 和Class.forName()有什么区别? 其实上面已经说到一些了,这里来做个总结: 首先,newInstance( )是一个方法,而new是一个关键字; 其次,Class下的newInstance()的使用有局限,因为它生成对象只能调用无参的构造函数,而使用 new关键字生成对象没有这个限制。 简言之:newInstance(): 弱类型,低效率,只能调用无参构造。new: 强类型,相对高效,能调用任何public构造。Class.forName(“”)返回的是类。Class.forName(“”).newInstance()返回的是object

展开阅读全文
加载中
点击引领话题📣 发布并加入讨论🔥
打赏
0 评论
1 收藏
0
分享
返回顶部
顶部