黑马程序员-反射

反射

1.Class类

  • Class类:Java程序中的各个Java类属于同一类事物,描述这类事物的Java类名就是Class。
  • Class描述了:类的名字,类的访问属性,类所属包名,字段名称的列表,方法名称的列表等。
  • 类的字节码实例对象信息获取方式:Class主要是获取Java类的字节码信息在获取各个Java类详细信息,需要Java类从硬盘加载到内存中,获取方式有:Class class = Dtae.class;Class.forName("java.lang.String");对象.getClass()
  • 在源程序中出现的类型,都有各自的Class实例对象*
        String str = "abc";
        Class cls1 = str.getClass();
        Class cls2 = String.class;
        Class cls3 = Class.forName("java.lang.String");
        System.out.println(cls1.isPrimitive()); //是否是基本数据类型
        System.out.println(int.class == Integer.TYPE); //int包装数据类型类的类型和int对应的字节码相同
        System.out.println(int.class == Integer.class); //int和Integer的字节码不相同
        System.out.println(int[].class.isArray()); //数组类型的Class实例对象

2.反射介绍

  • 反射就是把Java类中的各种成分映射成相应的Java类。
  • 一个类中的每个成员都可以用相应的反射类的一个实例对象来表示,如类中的方法可以用Method对象来表示。

构造方法的反射:

        Constructor[] con = Class.forName("java.lang.String").getConstructors();//获取此类的所有构造方法
        Constructor con1 = String.class.getConstructor(StringBuffer.class);//获取单个指定构造方法
  • 创建构造方法:String str = (String)con1.newInstance(new StringBuffer("bac"));创建new String(new StringBuffer("abc"))此对象,Class.newInstance()用来创建默认的构造方法。

成员变量的反射:

package com.sergio.NewTecl;

import java.io.File;
import java.lang.reflect.Field;

/**
 * 成员反射
 * Created by Sergio on 2015-06-05.
 */
public class ReflectField {
    public static void main(String[] args) throws Exception {
        ReflectTest rt = new ReflectTest(3, 5);
        //获取rt字节码对象上的x变量值.获取私有变量的值
        Field fieldX = rt.getClass().getDeclaredField("x");
        //设置私有变量获取后可以使用了
        fieldX.setAccessible(true);
        System.out.println(fieldX.get(rt));
        //获取的是共有变量
        Field fieldY = rt.getClass().getField("y");
        System.out.println(fieldY.get(rt));

        changeStringValue(rt);
        System.out.println(rt);
    }

    //更改rt对象中变量的某些值
    private static void changeStringValue(Object object) throws IllegalAccessException {
        Field[] fields = object.getClass().getFields();
        for (Field field : fields) {
            if (field.getType() == String.class) {
                String oldValue = (String) field.get(object);
                String newValue = oldValue.replace('b', 'a');
                field.set(object, newValue);
            }
        }
    }
}


class ReflectTest {
    private int x;
    public int y;
    //更改b为a
    public String str1 = "ball";
    public String str2 = "basketball";
    public String str3 = "itcast";


    public ReflectTest(int x, int y) {
        this.x = x;
        this.y = y;
    }

    @Override
    public String toString() {
        return "ReflectTest{ + str1='" + str1 + '\'' + ", str2='" + str2 + '\'' + ", str3='" + str3
            + '\'' +
            '}';
    }
}

成员方法反射

        String str1 = "abc";
        //str1.charAt(1)调用下面方法方式
        Method method = String.class.getMethod("charAt", int.class);
        //打印1位置的字符,method.invoke(null, 1)静态方法调用
        System.out.println(method.invoke(str1, 1));

数组的反射

  • 具有相同维数和元素类型的数组属于同一个类型,即具有相同的Class实例对象。
  • 基本类型的一维数组可以被当作Object类型使用,但不能当作Object[]类型使用;非基本类型的一维数组,既可以当作Object类型使用,又可以当作Object[]类型使用。
package com.sergio.NewTecl;

import java.lang.reflect.Array;

/**
 * 数组反射实例
 * Created by Sergio on 2015-06-06.
 */
public class ReflectArrayTest {
    public static void main(String[] args) {
        int[] a1 = new int[] {1, 2, 3};
        String[] a4 = new String[] {"xya"};

        printObject(a1);
        printObject(a4);

    }
    //打印数组中的元素
    private static void printObject(Object obj) {
        Class clazz = obj.getClass();
        if (clazz.isArray()) {
            int length = Array.getLength(obj);
            for (int i = 0; i < length; i++) {
                System.out.println(Array.get(obj, i));
            }
        } else {
            System.out.println(obj);
        }
    }
}

类加载器与反射

  • 类加载器是用来加载外部配置文件的主要方式之一。
package com.sergio.NewTecl;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Properties;

/**
 * 反射加载外部文件操作
 * Created by Sergio on 2015-06-06.
 */
public class ReflectCollection {
    public static void main(String[] args) throws Exception {
        //        Collection collections = new ArrayList<>();
        InputStream ips = new FileInputStream("config.properties");
        //类加载方法,加载外部文件配置的主要方式
        //InputStream ips2 = ReflectCollection.class.getClassLoader().getResourceAsStream("config.properties");

        Properties props = new Properties();
        props.load(ips);
        ips.close();

        String className = props.getProperty("className");
        //创建ArrayList构造方法实例,加载的外部文件为className=java.util.ArrayList
        Collection collections = (Collection) Class.forName(className).newInstance();
        ReflectTest rt = new ReflectTest(3, 5);
        ReflectTest rt1 = new ReflectTest(3, 5);
        ReflectTest rt2 = new ReflectTest(3, 5);
        collections.add(rt);
        collections.add(rt1);
        collections.add(rt2);

        System.out.println(collections.size());
    }
}


class ReflectTest1 {
    private int x;
    public int y;

    public ReflectTest1(int x, int y) {
        this.x = x;
        this.y = y;
    }
}


配置文件信息:className=java.util.ArrayList

反射与泛型

  • 通过制定对应的Class对象,程序可以获得该类里面所有的Field,不管该Field使用private方法或者public获得Field对象后都可以使用getType()来获取其类型。Class<?> type= f.getType()获得字段的类型,此方法
    只对普通Field有效,若该Field有泛型修饰,则不能准确得到该Field的泛型参数,如Map<String, Integer>;,为了获得指定Fild的泛型类型,可以如下:Type type = f.getGenerciType()得到泛型类型,然后将Type对象强转为ParameterizedType,表示增加泛型后的类型。Type getRawType()返回被泛型限制的类型。Type[] getActualArguments()返回泛型原始参数类型。
  • 反射获取泛型类型(信息)的步骤:
    1. 获取当前类
    2. 获取目标字段
    3. 获取包含泛型类型的类型getGenericType()
    4. 强转至子类ParameterizedType,因为Type没有任何对应的方法
    5. 获得泛型真正的类型getActualTypeArguments()
package com.sergio.NewTecl;

import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.Date;
import java.util.Vector;

/**
 * 通过反射获取泛型方法的参数类型
 * Created by Sergio on 2015-06-12.
 */
public class GenericDao {
    public static void applyVector(Vector<Date> v1) {

    }

    public static void main(String[] args) throws Exception {
        //要调用哪个泛型方法
        Method method = GenericDao.class.getMethod("applyVector()", Vector.class);
        Type[] types = method.getGenericParameterTypes();//获取泛型方法
        ParameterizedType pType = (ParameterizedType) types[0];//转化为参数类型
        System.out.println(pType.getRawType());//返回被限制的泛型类型
        System.out.println(pType.getActualTypeArguments());返回泛型实际参数类型
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值