java基础学习笔记之反射
反射
反射:将类的属性和方法映射成相应的类。
反射基本使用
获取Class类的三种方法:
- 类名.class
- 对象名.getClass()
- Class.forName("要加载的类名")
根据API写就行了,大致流程就是:
- 用上述三种方式之一获取特定类的Class类,即该类对应的字节码
- 调用Class对象的getConstructor(Class<?>...parameterTypes)获取构造方法对象
- 调用是构造方法类Constructor的newInstance(Object...initargs)方法新建对象
- 调用Class对象的getMethod(Stringname,Class<?>...parameterTypes)获取方法对象
- 调用方法对象类Method的invoke(Objectobj,Object...args)方法,调用对象上相应方法
用方法的参数类型唯一标识一个方法,依据:方法的重载
数组的反射
下面这个例子主要说明几点:
- 对于元素同类型的数组,同维数组,class一样
- 不同维,class不同
- 不同维的,父类都是Object,一样
- 基本类型以为数组不能直接转换为Object[]
- java.util.Arrays的asList方法API看看
publicclassReflectTest{ publicstaticvoidmain(String[]args){ int[]a1=newint[]{1,2,3}; int[]a2=newint[5]; int[][]a3=newint[2][3]; System.out.println(a1.getClass()==a2.getClass());//true System.out.println(a1.getClass());//class[I System.out.println(a3.getClass());//class[[I System.out.println(a1.getClass().getSuperclass()==a3.getClass().getSuperclass());//true System.out.println(a2.getClass().getSuperclass());//classjava.lang.Object //下句编译不通过:Error:(15,42)java:不可比较的类型:java.lang.Class<capture#1,共?extendsint[]>和java.lang.Class<capture#2,共?extendsint[][]> //System.out.println(a1.getClass()==a3.getClass()); Object[]b3=a3;//通过 //下句编译不通过Error:(17,24)java:不兼容的类型:int[]无法转换为java.lang.Object[] //Object[]b1=a1; Strings1="abc"; System.out.println(Arrays.asList(a1));//[[I@1540e19d] System.out.println(Arrays.asList(s1));//[abc] } }
输出:
true class[I class[[I true classjava.lang.Object [[I@1540e19d] [abc]
乱入:hashcode与内存泄露问题参考javaapi:
- hashcode一旦生成,不要变
- 对象equals方法返回true,则hascode要一致
- 反之,equals方法返回false,hascode不一定互异
如果参与hascode计算的成员变量中途发生变化,则后面remove时失败,造成内存泄露
配置文件加载
类加载器加载只读配置文件
类名.class.getClassLoader().getResourceAsStream(str);
类名.class.getResourceAsStream(str),实质还是调用类加载器。源码截取(java.lang包下的Class.java):
publicInputStreamgetResourceAsStream(Stringname){ name=resolveName(name); ClassLoadercl=getClassLoader0(); if(cl==null){ //Asystemclass. returnClassLoader.getSystemResourceAsStream(name); } returncl.getResourceAsStream(name); }
关于路径str,写法有点讲究。
- 不加斜杠,相对路径:str="config.properties";
- 加斜杠,从classpath的根路径找:str="/org/iot/ui/config.properties";
以前编译java代码时,有些conf/文件夹还要添加进依赖或者标记成source文件夹,里面明明都是xml文件,没Java源码。从这里,我现在知道了,是使用反射加载配置文件的缘故
内省(Instropector)&JavaBean
JavaBean读取属性x的值的流程:变大写、补前缀、获取方法。
"x"-->"X"-->"getX"-->"MethodGetX"
自己用内省操作
我目前没用上,所以不贴代码了,只附上核心类
简单实现:使用java.beans.PropertyDescriptor类
麻烦实现:使用java.beans.Introspector类,遍历getBeanInfo方法的返回值
JavaBean必须有一个不带参数的构造函数
使用BeanUtils工具包
- 字符串和整数转换(对比(PropertyUtils)
- 属性级联操作
- 操作map