《Java工程师必读手册》——Java经验之谈系列——那些年,我们踩过的那些Java坑(7)

简介: 《Java工程师必读手册》——Java经验之谈系列——那些年,我们踩过的那些Java坑(7)

接上篇:https://developer.aliyun.com/article/1228285?groupCode=java


七、 公有字段代理

 

在fastjson强制升级到1.2.60时踩过一个坑,作者为了开发快速,在ParseConfig中定义了:

 

image.png 

 

在我们的项目中继承了该类,同时又被AOP动态代理了,于是一行代码引起了一场“血案”。

 

1. 问题现象

 

仍然使用上章的例子,但是把获取、设置方法删除,定义了一个公有字段。例子代码如下:

 

1) UserService.java

 

image.png

 

2) CompanyService.java

 

image.png

 

当我们调用CompanyService的deleteCompany方法时,居然抛出空指针异常(NullPointerException)。经过调试打印,发现是UserService的superUser变量为null。如果把代理删除,就不会出现空指针异常,说明这个问题是由AOP代理导致的。

 

2. 问题分析

 

使用SpringCGLIB代理类时,Spring会创建一个名为“UserService

EnhancerBySpringCGLIB

EnhancerBySpringCGLIB????????”的代理类。这个代理类继承了UserService类,并覆盖了UserService类中的所有非final的public的方法。但是,这个代理类并不调用super基类的方法 相反,它会创建的一个成员userService并指向原始的UserService类对象实例。

 

现在,内存中存在两个对象实例:一个是原始的UserService对象实例,另一个指向UserService的代理对象实例。这个代理类只是一个虚拟代理,它继承了UserService类,并且具有与UserService相同的字段,但是它从来不会去初始化和使用它们。所以,一但通过这个代理类对象实例获取公有成员变量时,将返回一个默认值null。

 

image.png 

 

3. 避坑方法

 

1) 当确定字段不可变时,可以定义为公有静态常量

 

当确定字段不可变时,可以定义为公有静态常量,并用类名称+字段名称访问。类名称+字段名称访问公有静态常量,与类实例的动态代理无关。

 

2) 当确定字段不可变时,可以定义为私有成员变量

 

当确定字段不可变时,可以定义为私有成员变量,提供一个公有Getter方法获取该变量值。当该类实例被动态代理时,代理方法会调用被代理的Getter方法,从而返回被代理类的成员变量值。

 

3) 遵循JavaBean编码规范,不要定义公有成员变量

 

遵循JavaBean编码规范,不要定义公有成员变量。JavaBean规范如下:

 

JavaBean类必须是一个公共类,并将其访问属性设置为public,如public class User{......}

JavaBean类必须有一个空的构造函数:类中必须有一个不带参数的公用构造器

一个JavaBean类不应有公共实例变量,类变量都为private,如:private Integer id;

属性应该通过一组getter/setter方法来访问。

 




后记

 

最后,推荐大家阅读一下《阿里巴巴Java开发手册》,这本书让我受益匪浅。只要学习理解了《阿里巴巴Java开发手册》,就能在日常的Java开发工作中,避免踩到很多常识性的Java坑。

 

这里,作一首五言绝句赠与离开阿里的孤尽、夕华等前辈:

 

《赠别》

技术已封神,

声名四海闻。

功成身退去,

只为揽星辰。

相关文章
|
2月前
|
架构师 前端开发 Java
Java开发工程师的职业规划应该是什么样的?
Java开发工程师的职业规划涵盖多个阶段,包括初入行业(0-1年)、技能提升(1-3年)、技术专家(3-5年)及管理或专家路线选择(5年以上)。各阶段设定了明确的技能要求与职业目标,从掌握Java基础、常用框架到深入研究高级技术、微服务架构乃至担任管理职务或成为技术专家。通过持续学习与实践,结合个人兴趣,Java工程师可在技术或管理领域找到合适的发展方向,最终实现职业成功。
428 83
|
2月前
|
设计模式 架构师 Java
Java开发工程师转架构师需要学习什么
Java开发工程师转型为架构师需掌握多项技能:精通Java及框架、数据库与分布式系统;熟悉设计模式与架构模式;积累项目经验;提升沟通与领导力;持续学习新技术;培养系统设计与抽象能力;了解中间件及开发工具;并注重个人特质与职业发展。具体路径应结合个人目标与实际情况制定。
68 18
|
2月前
|
人工智能 前端开发 Java
Java开发工程师转哪个行业比较好?
Java开发工程师转哪个行业比较好?
303 2
|
3月前
|
人工智能 自然语言处理 Java
Spring AI,Spring团队开发的新组件,Java工程师快来一起体验吧
文章介绍了Spring AI,这是Spring团队开发的新组件,旨在为Java开发者提供易于集成的人工智能API,包括机器学习、自然语言处理和图像识别等功能,并通过实际代码示例展示了如何快速集成和使用这些AI技术。
Spring AI,Spring团队开发的新组件,Java工程师快来一起体验吧
|
2月前
|
小程序 前端开发 JavaScript
Java开发工程师转小程序开发的前景如何?
Java开发工程师转小程序开发的前景如何?
51 0
|
5月前
|
Java 开发者
【技术成长日记】Java 线程的自我修养:从新手到大师的生命周期修炼手册!
【6月更文挑战第19天】Java线程之旅,从新手到大师的进阶之路:始于创建线程的懵懂,理解就绪与运行状态的成长,克服同步难题的进阶,至洞悉生命周期的精通。通过实例,展示线程的创建、运行与同步,展现技能的不断提升与升华。
39 2
|
4月前
|
Java
图解java工程师学习路线
图解java工程师学习路线
244 0
|
5月前
|
Java 数据安全/隐私保护
Java基础手册二(类和对象 对象创建和使用 面向对象封装性 构造方法与参数传递 this关键字 static关键字 继承 多态 方法覆盖 final关键字 访问控制权限修饰符)
Java基础手册二(类和对象 对象创建和使用 面向对象封装性 构造方法与参数传递 this关键字 static关键字 继承 多态 方法覆盖 final关键字 访问控制权限修饰符)
35 0
|
5月前
|
存储 Java
Java基础手册(标识符 关键字 字面值 变量 数据类型 字符编码 运算符 控制语句 方法及方法重载和递归 面向对象与面向过程)
Java基础手册(标识符 关键字 字面值 变量 数据类型 字符编码 运算符 控制语句 方法及方法重载和递归 面向对象与面向过程)
39 0
|
5月前
|
存储 Java 编译器
技术经验解读:一文带你搞懂java中的变量的定义是什么意思
技术经验解读:一文带你搞懂java中的变量的定义是什么意思
45 0
下一篇
无影云桌面