二进制反射格雷码_Java最强大的技术之一:反射

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制

类的加载过程

要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法。所以先要获取到每一个字节码文件对应的Class类型的对象。

9d22283f486c0976a79169fa8f1c964f.png

那么一个class文件(.java文件编译后)在我们的硬盘上,是怎么被加载到内存中的呢?

class进入内存分三步走:

  1. Loading

Loading就是把一个class文件装到内存中,它本来是class文件上一个个的二进制,一个个字节,通过Loading把它放到内存。

  1. Linking

Linking的过程又分为:

  • verification:校验装到内存的class文件是否符合class文件的标准,加入装进来的文件不是“CA FE BA BE”这样的,不符合class文件标准,直接被拒。
2f6c250ab2d5399e6a82255c98680c0c.png
class二进制文件
  • preparation:把class文件静态变量赋默认值,不是赋初始值。比如static int i = 8,在该步骤只是把i赋了默认值0。
  • resolution:把class文件常量池里面用到的符号引用,把它转成直接内存地址,可以访问到的内容。
  1. Initializing

Initializing这一步就是将静态变量赋初始值,比如上面的static int i = 8,在这一步才赋初始值8.

c1c2dedb02cd86a2cd545fecf0492a66.png

Class类

public final class Class<T>extends Objectimplements Serializable, GenericDeclaration, Type, AnnotatedElement

类class的实例表示运行中的Java应用程序中的类和接口。

枚举是一种类,注释是一种接口。

每个数组也属于一个类,这个类反映为一个类对象,由具有相同元素类型和维数的所有数组共享。

原始Java类型(布尔型、字节型、char型、short型、int型、long型、float型和double型)和关键字void也被表示为类对象。

Class类没有公共构造函数。相反,类对象由Java虚拟机在装入类时自动构造,并通过调用类装入器中的defineClass方法来构造。

获取Class类的三种方式

  • 对象.getClass()

通过对象的getClass()方法获取Class类,说明对象已经创建好了,其实已经有Class类了。

  • 类.class

这种方式获取Class类,需要提前知道类的名称,也就是项目中已经导入了相应的包,依赖性强。

  • Class.forName()

只需要传入一个类的完全限定名即可。

推荐使用 Class.forName() 的方式获取Class类。

反射常用到的Class类API

  • 获取类的构造方法
public Constructor getConstructor(Class>... parameterTypes)throws NoSuchMethodException,
                                     SecurityExceptionpublic Constructor>[] getConstructors()throws SecurityException
  • 获取类的成员变量
public Field getField(String name)throws NoSuchFieldException,
                      SecurityExceptionpublic Field[] getFields()throws SecurityException
  • 获取类的成员方法
public Method getMethod(String name,
                        Class>... parameterTypes)throws NoSuchMethodException,
                        SecurityExceptionpublic Method getMethod(String name,
                        Class>... parameterTypes)throws NoSuchMethodException,
                        SecurityException

举个栗子

定义一个实体类:

public class User {
    private Integer id;
    private String userName;
    private String phoneNumber;

    public User() {
    }

    public User(Integer id, String userName, String phoneNumber) {
        this.id = id;
        this.userName = userName;
        this.phoneNumber = phoneNumber;
    }
    
    public void myDefine() {
        System.out.println("xxx");
    }
}

通过反射获得User类的相关信息:

public class CreateObjectTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //对象.getClass()
//        User user = new User();
//        Class clazz = user.getClass();
//        System.out.println(clazz.getPackage());
//        System.out.println(clazz.getName());
//        System.out.println(clazz.getCanonicalName());
//        System.out.println(clazz.getSimpleName());
        //类.class
//        Class clazz = User.class;
        //Class.forName
        Class clazz = Class.forName("com.traveler100.javabase.reflect.CreateObjectTest");
        System.out.println("getConstructors:");
        Arrays.stream(clazz.getConstructors()).iterator().forEachRemaining(System.out::println);
        System.out.println("getFields:");
        Arrays.stream(clazz.getFields()).iterator().forEachRemaining(System.out::println);
        System.out.println("getMethods:");
        Arrays.stream(clazz.getMethods()).iterator().forEachRemaining(System.out::println);
    }
}

运行结果:

195a2ddc03384db6f0f102c971e219ac.png
反射常用API

反射在Spring中的应用

反射在众多框架中都有普遍的应用。比如Spring IOC容器帮我们实例化众多的bean:

Spring配置文件

<bean id="pony" class="com.traveler100.dp.proxy.springaop.Pony">bean>

使用的时候直接这样就能拿到定义的类了:

ApplicationContext ctx = new ClassPathXmlApplicationContext("app_aop.xml");
Pony pony = (Pony) ctx.getBean("pony");

那么是怎么做到的呢?就是通过反射。

Spring通过配置文件实例化对象,并将其放到容器的过程大概就是:

//伪代码
//1.解析元素的id属性得到该字符串值为“pony”  
String idStr = "pony";  
//解析元素的class属性得到该字符串值为“com.traveler100.dp.proxy.springaop.Pony”  
String classStr = "com.traveler100.dp.proxy.springaop.Pony";  
//利用反射机制,通过classStr获取Class类对象  
Class> cls = Class.forName(classStr);  
//实例化对象  
Object obj = cls.newInstance();  
//放到Spring容器  
Map container = new HashMap<>();
container.put(idStr, obj); 

小结

多看一下Class类的API,诸多框架都用到了反射机制,而反射离不开调用这些基本的API。

JavaSE 8 API官网:https://docs.oracle.com/javase/8/docs/api/index.html

推荐阅读

  • 从一道面试题进入Java并发新机制---J.U.C
  • synchronized底层实现知多少?synchronized加锁还会升级?
  • 你知道Object o = new Object()在内存中占多少字节吗?
  • Java8新特性Stream还有这种操作?
  • 终于看懂别人的代码了!总结Java 8之Lambda表达式的写法套路

7d6dadef7780c2eb218c4d6ff083c1c1.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值