老司机带带你,教你学会Java中又骚又暴力的“反射”技术

绝地灬酷狼 2024-04-08 11:07 191阅读 0赞

在Java中有这么一个很骚的技术,几乎贯穿了所有主流的框架,在所有主流框架的底层中你都可以看见它的身影,这个技术就是反射。关于反射,有很多小白会觉得很难,搞不清楚到底是怎么回事,也不知道该怎么用,今天壹哥就来教教你如何理解和使用Java的反射。

一. 反射概念

我们知道,在物理层面上,反射是一种光学现象,是指光在传播到不同物质时,在分界面上改变传播方向后又返回原来物质中的现象。

1ae50ebf64924b3cbbb86a31f3b16fb5.png

而在Java中,反射是一种机制,而不是一种现象。反射机制指的是程序在运行时能够动态获取类对象的属性,和调用类对象的方法。

e0aa54ee54bf4bb1a192ef2b1115f676.png

Java中的编译类型有两种

  • 静态编译:在编译时确定类型,绑定对象即通过。

  • 动态编译:运行时确定类型,绑定对象。动态编译最大限度地发挥了Java的灵活性,体现了多态的应用,可以减低类之间的耦合性。

使用反射可以赋予 JVM 动态编译的能力,否则类的元数据信息只能用静态编译的方式实现。

二. 反射API

在Java中,JDK为我们提供了一些反射相关的API,如下表所示:
























含义

java.lang.Class

代表整个字节码。代表一个类型,代表整个类。

java.lang.reflect.Method

代表字节码中的方法字节码。代表类中的方法。

java.lang.reflect.Constructor

代表字节码中的构造方法字节码。代表类中的构造方法。

java.lang.reflect.Field

代表字节码中的属性字节码。代表类中的成员变量(静态变量+实例变量)。

接下来我们就来看看反射到底怎么用。

三. 具体使用

下面壹哥就用一个案例来让大家感受一下反射的牛逼之处。

1. 常规实现

我们知道,在Java中的实体类总会有一些固定的方法,比如每个属性的 get()、set()方法,还有初始化属性创建对象的构造方法,打印对象信息的toString()等方法。假如我们在没有使用注解的情况下,需要创建2个普通的实体类:Cat、Dog,代码如下所示:

  1. public class Cat {
  2. private String name;
  3. private int age;
  4. public Cat() {
  5. }
  6. public Cat(String name, int age) {
  7. this.name = name;
  8. this.age = age;
  9. }
  10. public String getName() {
  11. return name;
  12. }
  13. public void setName(String name) {
  14. this.name = name;
  15. }
  16. public int getAge() {
  17. return age;
  18. }
  19. public void setAge(int age) {
  20. this.age = age;
  21. }
  22. @Override
  23. public String toString() {
  24. return "Cat{" +
  25. "name='" + name + '\'' +
  26. ", age=" + age +
  27. '}';
  28. }
  29. }
  30. public class Dog{
  31. private String name;
  32. private int age;
  33. public Dog() {
  34. }
  35. public Dog(String name, int age) {
  36. this.name = name;
  37. this.age = age;
  38. }
  39. public String getName() {
  40. return name;
  41. }
  42. public void setName(String name) {
  43. this.name = name;
  44. }
  45. public int getAge() {
  46. return age;
  47. }
  48. public void setAge(int age) {
  49. this.age = age;
  50. }
  51. @Override
  52. public String toString() {
  53. return "Dog{" +
  54. "name='" + name + '\'' +
  55. ", age=" + age +
  56. '}';
  57. }
  58. }

我们想在案例代码中创建对象并打印对象信息**,如下所示:**

  1. public class Demo01 {
  2. public static void main(String[] args){
  3. Cat cat = new Cat("招财", 1);
  4. System.out.println(cat.toString());
  5. Dog dog = new Dog("旺财",2);
  6. System.out.println(dog.toString());
  7. }
  8. }

1ecc0f776561483281e08842034e2269.png

在上面这两个实体类中,都存在着名字相同、但方法体不同的toString()方法。如果我们在实体类中不重写toString() ,直接通过对象调用 toString(),打印的结果不会是对象的信息。

7ac9837582374d5896c948afb0f94621.png

我们可以使用反射给两个实体类,甚至更多的实体类自动加上toString()方法,从而达到减少代码量的目的。

2. 反射实现

2.1 创建父类BaseEntity

首先我们创建一个父类BaseEntity

  1. public class BaseEntity {
  2. @Override
  3. public String toString() {
  4. //1.获取反射对象
  5. Class<? extends BaseEntity> clazz = this.getClass();
  6. //2.创建 StringBuffer 对象拼接字符串
  7. StringBuffer sb = new StringBuffer();
  8. //3.通过 getSimpleName() 获取类名并拼接
  9. sb.append(clazz.getSimpleName());
  10. //拼接大括号
  11. sb.append("{");
  12. //4.获取所有成员变量对象
  13. Field[] fields = clazz.getDeclaredFields();
  14. Object value = null;
  15. for (int i = 0; i < fields.length; i++) {
  16. //5.获取成员变量对象
  17. Field field = fields[i];
  18. //6.打开访问权限
  19. field.setAccessible(true);
  20. //7.通过 getName() 获取属性名并拼接
  21. sb.append(getName());
  22. sb.append("=");
  23. //8.通过 get() 传递 this 获取对象的属性值
  24. try {
  25. value = field.get(this);
  26. } catch (IllegalAccessException e) {
  27. e.printStackTrace();
  28. }
  29. //9.判断是否为 String 类型的属性,是则添加单引号
  30. if(field.getType() == String.class){
  31. sb.append("'");
  32. sb.append(value);
  33. sb.append("'");
  34. }else{
  35. sb.append(value);
  36. }
  37. //10.判断是否为最后一个属性对象
  38. if(i == fields.length-1){
  39. sb.append("}");
  40. }else{
  41. sb.append(", ");
  42. }
  43. }
  44. //11.通过 toString() 转换成字符串并返回
  45. return sb.toString();
  46. }
  47. }

2.2 继承父类

然后将两个实体类 Cat 和 Dog,都继承 BaseEntity且不重写 toString()方法。

  1. public class Cat extends BaseEntity{
  2. private String name;
  3. private int age;
  4. public Cat() {
  5. }
  6. public Cat(String name, int age) {
  7. this.name = name;
  8. this.age = age;
  9. }
  10. public String getName() {
  11. return name;
  12. }
  13. public void setName(String name) {
  14. this.name = name;
  15. }
  16. public int getAge() {
  17. return age;
  18. }
  19. public void setAge(int age) {
  20. this.age = age;
  21. }
  22. }
  23. public class Dog extends BaseEntity{
  24. private String name;
  25. private int age;
  26. public Dog() {
  27. }
  28. public Dog(String name, int age) {
  29. this.name = name;
  30. this.age = age;
  31. }
  32. public String getName() {
  33. return name;
  34. }
  35. public void setName(String name) {
  36. this.name = name;
  37. }
  38. public int getAge() {
  39. return age;
  40. }
  41. public void setAge(int age) {
  42. this.age = age;
  43. }
  44. }

2.3 运行测试

接下来在案例类中创建对象并打印对象信息,**这里**会发现打印结果不再是之前没有 toString() 的情况,而是会分别打印出各自对象的信息。

  1. public class Demo01 {
  2. public static void main(String[] args){
  3. Cat cat = new Cat("招财", 1);
  4. System.out.println(cat.toString());
  5. Dog dog = new Dog("旺财",2);
  6. System.out.println(dog.toString());
  7. }
  8. }

1af66e45184b4ad7aa27a0fd3e7a27e9.png

我们可以在以上案例中发现,toString()方法在运行状态时,通过反射获取了运行对象的类属性,进行了信息的拼接,从而达到了减少实体类代码量的目的,提高了代码的复用性。

四. 小结

使用反射技术,可以让我们进行动态的创建对象和编译,体现出很高的代码灵活性。但反射技术却对性能有一定的影响,它不如静态创建对象那样来得直接高效。所以反射既有好处也有缺点,但好处远大于缺点。当然,今天壹哥只是通过这样的一个小案例,给大家展现了反射技术强大功能的冰山一角,它还有更多的神奇之处有待于我们继续挖掘,请大家继续关注我哦。

463e23ce5ed14dedb2d94e84bfdec9d2.png

发表评论

表情:
评论列表 (有 0 条评论,191人围观)

还没有评论,来说两句吧...

相关阅读