码迷,mamicode.com
首页 > 其他好文 > 详细

反射之二

时间:2018-01-28 17:20:18      阅读:200      评论:0      收藏:0      [点我收藏+]

标签:alac   获取   函数   sem   数值   invoke   名称   getname   catch   

生成对象

一般可以用new关键字来创建我们想要的对象。但在特殊情况下,可能只有在程序运行时才知道要常见的对象

所对应的类名称,这时就需要java反射了,分两种情况来讨论用反射创建对象的方式。

 

1.用无参构造方法

调用这个类对应的Class对象的newInstance()方法:

Class c=Class.forName(“java.util.ArrayList”);

List list=(List)c.newInstance();

说明:若指定名称的类没有无参构造方法,在调用newInstance()方法是会抛出会抛出一个NoSuchMethodException

         异常。

 

import java.util.Date;
/*使用反射调用无参构造方法创建指定名称类的对象*/
public class NoArgsCreateInstanceTest{
    public static void main(String[] args) {
        Date currentDate=(Date)newInstance("java.util.Date");
        System.out.println(currentDate);
    }

    public static Object newInstance(String className){
        Object obj=null;
        try{
            /*加载指定名称的类并获取对应的Class对象,
              再调用无参构造方法创建出一个对象*/
              obj=Class.forName(className).newInstance();
        }catch (InstantiationException e) {
            e.printStackTrace();
        }catch (IllegalAccessException e) {
            e.printStackTrace();
        }catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        return obj;
    }
}

2.用带参构造方法

获取名称的类对应的Class对象,然后通过反射获取指定的参数类型的要求的构造方法信息类对象,

调用它的newInstance方法来创建对象。

具体分三步:

1)获取指定类的Class对象

2)通过Class对象获取满足指定参数类型要求的构造方法类对象

3)调用指定Constructor对象的newInstance方法传入对应的参数值,创建出对象。

//创建对象的示例:
import java.lang.reflect.Constructor;
import java.lang.reflect.IncocationTargetException;
import java.util.Date;

/**用反射对指定的带参数的构造方法创建指定类的对象*/
import java.lang.reflect.Constructor;
import java.lang.reflect.IncocationTargetException;
import java.util.Date;

/**用反射对指定的带参数的构造方法创建指定类的对象*/
public class ArgsCreateIntanceTest{
    @SuppressWarnings("unchecked")
    public static void main(String[] args) {
        try{
            //1.加载指定名称的类,获取对应的Class对象
            Class clazz=Class.forName("java.util.Date");

            //2.获取具有指定参数的构造方法
            Constructor constructor=clazz.getConstructor(long.class);
            
            //3.给指定的构造方法传入参数值,创建出一个对象
            Date date=(Date)constructor.newInstance(1234567890L);
            System.out.println(date);
        }catch (ClassNoFoundException e) {
            e.printStackTrace();
        }catch(SecurityException e){
            e.printStackTrace();
        }catch (NoSuchMethodException e) {
            e.printStackTrace();
        }catch (IllegalArgumentException e) {
            //此异常表明向方法传递了一个不合法或不正确的参数或JDK版本不兼容。
            e.pringStackTrace();
        }catch (InstantiationException e) {
            //不在类的加载的时候,所以就会报这样的错误。
            e.printStackTrace();
        }catch(IllegalAccessException e){
            e.printStackTrace();
        }catch(InvocationTatgetException e){
            //一种包装由调用方法或构造方法所抛出异常的受查异常
            e.printStackTrace();
        }

    }
}

调用方法

用反射可以取得指定类中指定方法对象代表,方法对象代表就是java.lang.reflect.lang类的实例,通过Method类的

invoke()方法可以动态调用这个方法。

method类的invoke()方法完整签名是:

public Object invoke(Object obj, object…args)

                    throws IllegaAccessException , IllegalArgumentException,InvocationTargetException

//invoke()的返回值代表的是指动态地调用指定方法后实际返回值

说明:若要通过反射调用类的某个私有方法,可以在这个私有方法对应的Method对象上,先调用setAccessible(true)

来取消java对于本方法的访问检查,然后再调用invoke()方法来真正执行这个私有方法。

iimport java.lang.reflect.InvocationTargetException;

import java.lang.reflect.menthod;

/*用反射来动态地调用指定类的指定方法*/
@SuppressWarnings("unchecked")
public class ReflectInvokeMethodTest{
    public static void main(String[] args) {
        try{
        Class clazz=Clazz.forName("com.qiujy.");

        //利用无参构造函数创建一个Product的对象?? ???
        Product prod=(Product)clazz.newInstance();

        //获取名为setName,带一个类型为String的成员方法所对应的对象代表
        Method menthod1=clazz.getDeclaredMethod("setName",String.class);
        //在prod对象上调用setName,并传值给它,返回值为空
        Object returnValue=method1.invoke(prod,"哇");
        System.out.println("返回值:"+returnValue);
        //获取名为displayInfo,不带参数的成员方法所对应的对象代表
        Method method2=clazz.getDeclaredMethod("displayInfo");

        method2.setAccessible(true);
        //取消访问调查??setAccessible不是意思为可接近的,可理解的吗?true不是表示真的意思吗?
        //在prod对象上的勇士以的displayInfo方法
        
        method2.invoke(prod);
        }catch (ClassNoFoundException e) {
            e.printStackTrace();
        }catch(SecurityException e){
            e.printStackTrace();
        }catch (NoSuchMethodException e) {
            e.printStackTrace();
        }catch (IllegalArgumentException e) {
            e.pringStackTrace();
        }catch (InstantiationException e) {
            e.printStackTrace();
        }catch(IllegalAccessException e){
            e.printStackTrace();
        }catch(InvocationTatgetException e){
            e.printStackTrace();
        }
    }

}
class Product{
    private static long count=0;
    private long id;
    private String name="无名氏";
    public Product(){
        System.out.println("默认构造方法");
        id=++count();
    }
    public long getId(){
        return id;
    }
    public void setId(long id){
        this.id=id;
    }
    public String getName(){
        return name;
    }
    private void displayInfo(){//私有方法
        System.out.println(getClass().getName+"[id="+id+",name="+name+"]");
    }
}

 

 

反射之二

标签:alac   获取   函数   sem   数值   invoke   名称   getname   catch   

原文地址:https://www.cnblogs.com/shijinglu2018/p/8371615.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!