欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

java基础学习笔记之反射

程序员文章站 2024-03-08 16:16:34
反射 反射:将类的属性和方法映射成相应的类。 反射基本使用 获取class类的三种方法: 类名.class 对象名.getclass() c...

反射

反射:将类的属性和方法映射成相应的类。

反射基本使用

获取class类的三种方法:

  1. 类名.class
  2. 对象名.getclass()
  3. class.forname("要加载的类名")

根据api写就行了,大致流程就是:

  1. 用上述三种方式之一获取特定类的class类,即该类对应的字节码
  2. 调用class对象的getconstructor(class<?>... parametertypes)获取构造方法对象
  3. 调用是构造方法类constructor的newinstance(object... initargs)方法新建对象
  4. 调用class对象的getmethod(string name, class<?>... parametertypes)获取方法对象
  5. 调用方法对象类method的invoke(object obj, object... args)方法,调用对象上相应方法

用方法的参数类型唯一标识一个方法,依据:方法的重载

数组的反射

下面这个例子主要说明几点:

  1. 对于元素同类型的数组,同维数组,class一样
  2. 不同维,class不同
  3. 不同维的,父类都是object,一样
  4. 基本类型以为数组不能直接转换为object[]
  5. java.util.arrays的aslist方法api看看
public class reflecttest {
  public static void main(string[] args) {
    int [] a1 = new int[]{1,2,3};
    int [] a2 = new int[5];
    int [][] a3 = new int[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());//class java.lang.object

    //下句编译不通过:error:(15, 42) java: 不可比较的类型: java.lang.class<capture#1, 共 ? extends int[]>和java.lang.class<capture#2, 共 ? extends int[][]>
    //system.out.println(a1.getclass() == a3.getclass());

    object []b3 = a3;//通过
    //下句编译不通过  error:(17, 24) java: 不兼容的类型: int[]无法转换为java.lang.object[]
    //object [] b1 = a1;

    string s1 = "abc";
    system.out.println(arrays.aslist(a1));//[[i@1540e19d]
    system.out.println(arrays.aslist(s1));//[abc]
  }
}

输出:

true
class [i
class [[i
true
class java.lang.object
[[i@1540e19d]
[abc]

乱入: hashcode与内存泄露问题 参考java api:

  1. hashcode一旦生成,不要变
  2. 对象equals方法返回true,则hascode要一致
  3. 反之,equals方法返回false,hascode不一定互异

如果参与hascode计算的成员变量中途发生变化,则后面remove时失败,造成内存泄露

配置文件加载

类加载器加载只读配置文件
类名.class.getclassloader().getresourceasstream(str);

类名.class.getresourceasstream(str),实质还是调用类加载器。 源码截取(java.lang包下的class.java):

 public inputstream getresourceasstream(string name) {
  name = resolvename(name);
  classloader cl = getclassloader0();
  if (cl==null) {
    // a system class.
    return classloader.getsystemresourceasstream(name);
  }
  return cl.getresourceasstream(name);
}

关于路径str,写法有点讲究。

  1. 不加斜杠,相对路径: str = "config.properties";
  2. 加斜杠,从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工具包

  1. 字符串和整数转换(对比(propertyutils)
  2. 属性级联操作
  3. 操作map