首页 > 编程语言 >老司机带带你,教你学会Java中又骚又暴力的“反射”技术

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

时间:2022-12-03 10:36:42浏览次数:49  
标签:Java 暴力 int age String toString 带带 public name

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

一. 反射概念

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

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

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


老司机带带你,教你学会Java中又骚又暴力的“反射”技术_创建对象_02

Java中的编译类型有两种:

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

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

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

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


老司机带带你,教你学会Java中又骚又暴力的“反射”技术_创建对象_03

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

三. 具体使用

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

1. 常规实现

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

public class Cat {
private String name;
private int age;

public Cat() {
}

public Cat(String name, int age) {
this.name = name;
this.age = age;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

@Override
public String toString() {
return "Cat{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public class Dog{
private String name;
private int age;

public Dog() {
}
public Dog(String name, int age) {
this.name = name;
this.age = age;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

@Override
public String toString() {
return "Dog{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}

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

public class Demo01 {
public static void main(String[] args){
Cat cat = new Cat("招财", 1);
System.out.println(cat.toString());

Dog dog = new Dog("旺财",2);
System.out.println(dog.toString());
}
}

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

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


老司机带带你,教你学会Java中又骚又暴力的“反射”技术_创建对象_05

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

2. 反射实现

2.1 创建父类BaseEntity

首先我们创建一个父类BaseEntity

public class BaseEntity {
@Override
public String toString() {
//1.获取反射对象
Class<? extends BaseEntity> clazz = this.getClass();
//2.创建 StringBuffer 对象拼接字符串
StringBuffer sb = new StringBuffer();
//3.通过 getSimpleName() 获取类名并拼接
sb.append(clazz.getSimpleName());
//拼接大括号
sb.append("{");
//4.获取所有成员变量对象
Field[] fields = clazz.getDeclaredFields();
Object value = null;
for (int i = 0; i < fields.length; i++) {
//5.获取成员变量对象
Field field = fields[i];
//6.打开访问权限
field.setAccessible(true);
//7.通过 getName() 获取属性名并拼接
sb.append(getName());
sb.append("=");
//8.通过 get() 传递 this 获取对象的属性值
try {
value = field.get(this);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
//9.判断是否为 String 类型的属性,是则添加单引号
if(field.getType() == String.class){
sb.append("'");
sb.append(value);
sb.append("'");
}else{
sb.append(value);
}
//10.判断是否为最后一个属性对象
if(i == fields.length-1){
sb.append("}");
}else{
sb.append(", ");
}
}
//11.通过 toString() 转换成字符串并返回
return sb.toString();
}
}

2.2 继承父类

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

public class Cat extends BaseEntity{
private String name;
private int age;

public Cat() {
}

public Cat(String name, int age) {
this.name = name;
this.age = age;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}
}
public class Dog extends BaseEntity{
private String name;
private int age;

public Dog() {
}
public Dog(String name, int age) {
this.name = name;
this.age = age;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}
}

2.3 运行测试

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

public class Demo01 {
public static void main(String[] args){
Cat cat = new Cat("招财", 1);
System.out.println(cat.toString());

Dog dog = new Dog("旺财",2);
System.out.println(dog.toString());
}
}

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

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

四. 小结

使用反射技术,可以让我们进行动态的创建对象和编译,体现出很高的代码灵活性。但反射技术却对性能有一定的影响,它不如静态创建对象那样来得直接高效。所以反射既有好处也有缺点,但好处远大于缺点。

标签:Java,暴力,int,age,String,toString,带带,public,name
From: https://blog.51cto.com/u_15733182/5908470

相关文章

  • Java中的简单题目
    输入输出importjava.util.Scanner;publicclassTestDemo1{publicstaticvoidmain(String[]args){Scannerscan=newScanner(System.in);inta=scan.nextInt(......
  • Java String 类和常量池
    String对象的两种创建方式:Stringstr1="abcd";Stringstr2=newString("abcd");//falseSystem.out.println(str1==str2);这两种不同的创建方法是有差别的,第一种方......
  • Java通过Lambda表达式根据指定字段去除重复数据(集合去重)
    这里博主给大家封装好了一个工具类,里面有两个方法。方法一:可以根据指定字段去除重复数据。方法二:可以获取到重复的数据。大家在使用过程中直接拷贝下方代码在要去重的类中调......
  • 强制解决Java参数乱码问题
    在我们日常开发中,常常复制的url路径都是这样的原路径:https://xxxx.oss-cn-hangzhou.xxxxxx.com/2022/xx/文件名.pdf复制出来的路径:https://xxxx.oss-cn-hangzhou.xxxxxx.com......
  • 【JavaEE进阶系列 | 从小白到工程师】基本类型包装类的使用,装箱以及拆箱与parseInt方
    一、包装类概述Java中的数据类型分为基本类型和引用类型两大类,使用基本类型可以提升效率但是java是面向对象的语言,java的设计思想是一切皆对象,而基本数据类型不是对象,于是J......
  • 看完这篇,还不懂JAVA内存模型(JMM)算我输
    欢迎关注专栏【JAVA并发】更多技术干活尽在个人公众号——JAVA旭阳前言开篇一个例子,我看看都有谁会?如果不会的,或者不知道原理的,还是老老实实看完这篇文章吧。@Slf4j(......
  • java基础面试
    一、clone一、浅拷贝clone()如果对象中的所有数据域都是数值或者基本类型,使用clone()即可满足需求,如:Personp=newPerson();Personp1=p.clone();这样p和p1分别指向不......
  • C# AES CFB加解密模式兼容JAVA
    C#AESCFB加解密模式兼容JAVA代码片段最近在和java做对接的时候老是遇到加密使用java写的,需要我们使用C#来解密相关数据,AES加解密平常也在用,但是这种跨语言的应用还是比较......
  • 【Java并发入门】03 互斥锁(上):解决原子性问题
    原子性问题的源头是线程切换Q:如果禁用CPU线程切换是不是就解决这个问题了?A:单核CPU可行,但到了多核CPU的时候,有可能是不同的核在处理同一个变量,即便不切换线程,也有问......
  • jenkins javax.mail.AuthenticationFailedException: 535 authentication failed
     邮箱服务器端口是587解决方案:启动脚本增加-Dmail.smtp.starttls.enable=true-Dmail.smtp.ssl.trust=esmtp.*.com  ......