- 场景
*/
private String scene;
}
这时候我们再使用Builder注解就会发现,在子类中无法通过builder方法构造父类中的成员变量
给BaseDTO上加上Builder注解也不会有任何效果。事实上,Builder注解只管承接注解的这个类,而不会管他的父类或者子类。如果真的是这样的话,遇到有继承的类,只好又打回原形,写一堆的setter方法了。
试试SuperBuilder吧
===================
这个问题在lombokv1.18.2版本之前其实很难办,但是在这个版本官方引入了一个新的注解@SuperBuilder,无法build父类的问题迎刃而解
The @SuperBuilder annotation produces complex builder APIs for your classes. In contrast to @Builder, @SuperBuilder also works with fields from superclasses. However, it only works for types. Most importantly, it requires that all superclasses also have the@SuperBuilder annotation.
按照官方文档的说法,为了能够使用build方法,只需要在子类和父类上都加@SuperBuilder注解,我们试一下
果然现在就可以在子类的实例中build`父类的成员变量了
Lombok的原理
=============
Lombok自动生成代码的实现也是依赖于JVM开放的扩展点,使其可以在编译的时候修改抽象语法树,从而影响最终生成的字节码
图片来源地址:http://notatube.blogspot.com/2010/12/project-lombok-creating-custom.html
为什么Builder不能处理父类的成员变量
=========================
我们可以翻一下Lombok的源码,Lombok对所有的注解都有两套实现,javac和eclipse,由于我们的运行环境是Idea所以我们选择javac的实现,javac版本的实现在lombok.javac.handlers.HandleBuilder#handle这个方法中
JavacNode parent = annotationNode.up();
if (parent.get() instanceof JCClassDecl) {
job.parentType = parent;
JCClassDecl td = (JCClassDecl) parent.get();
ListBuffer allFields = new ListBuffer();
boolean valuePresent = (hasAnnotation(lombok.Value.class, parent) || hasAnnotation(“lombok.experimental.Value”, parent));
// 取出所有的成员变量
for (JavacNode fieldNode : HandleConstructor.findAllFields(parent, true)) {
JCVariableDecl fd = (JCVariableDecl) fieldNode.get();
JavacNode isDefault = findAnnotation(Builder.Default.class, fieldNode, false);
boolean isFinal = (fd.mods.flags & Flags.FINAL) != 0 || (valuePresent && !hasAnnotation(NonFinal.class, fieldNode));
// 巴拉巴拉,省略掉
}
这里的annotationNode就是Builder注解,站在抽象语法树的角度,调用up方法得到的就是被注解修饰的类,也就是需要生成builder方法的类。
通过查看源代码,@Builder注解是可以修饰类,构造函数和方法的,为了简单起见,上面的代码只截取了@Builder修饰类这一种情况,这段代码关键的地方就在于调用HandleConstructor.findAllFields方法获得类中所有的成员变量:
public static List findAllFields(JavacNode typeNode, boolean evenFinalInitialized) {
ListBuffer fields = new ListBuffer();
// 从抽象语法树出发,遍历类的所有的成员变量
for (JavacNode child : typeNode.down()) {
if (child.getKind() != Kind.FIELD) continue;
JCVariableDecl fieldDecl = (JCVariableDecl) child.get();
//Skip fields that start with $
if (fieldDecl.name.toString().startsWith(“$”)) continue;
long fieldFlags = fieldDecl.mods.flags;
//Skip static fields.
if ((fieldFlags & Flags.STATIC) != 0) continue;
//Skip initialized final fields
boolean isFinal = (fieldFlags & Flags.FINAL) != 0;
if (evenFinalInitialized || !isFinal || fieldDecl.init == null) fields.append(child);
}
return fields.toList();
}
这段代码比较简单,就是对类中的成员变量做了过滤,比如说,静态变量就不能被@Builder方法构造。有一个有意思的点,尽管 可以合法的出现在 j a v a 的变量命名中,但是 L o m b o k 对这种变量做了过滤,因此变量名以 可以合法的出现在java的变量命名中,但是Lombok对这种变量做了过滤,因此变量名以 可以合法的出现在java的变量命名中,但是Lombok对这种变量做了过滤,因此变量名以开始的也不能被@Builder构造,经过我们的验证确实是这样的。
如果我们用JDT AstView看一下ItemDTO的抽象语法树结构,发现Java的抽象语法树设计的确是每个类只包含显式声明的变量而不包括父类的成员变量(该插件支持点击语法树节点可以和源文件联动,且数量只有4个和ItemDTO声明的成员变量数量一致)
因为findAllFields方法是从当前类的抽象语法树出发去找所有的成员变量,所以就只能找到当前类的成员变量,而访问不到父类的成员变量
一个镜像的问题就是,既然@Builder注解不能构造父类的成员变量,那@SuperBuilder是怎么做到的呢?翻一下@SuperBuilder的源码,核心逻辑在lombok.javac.handlers.HandleSuperBuilder#handle
// 巴拉巴拉省略
JCClassDecl td = (JCClassDecl) parent.get();
// 获取继承的父类的抽象语法树
JCTree extendsClause = Javac.getExtendsClause(td);
JCExpression superclassBuilderClass = null;
if (extendsClause instanceof JCTypeApply) {
// Remember the type arguments, because we need them for the extends clause of our abstract builder class.
superclassTypeParams = ((JCTypeApply) extendsClause).getTypeArguments();
// A class name with a generics type, e.g., “Superclass”.
extendsClause = ((JCTypeApply) extendsClause).getType();
}
if (extendsClause instanceof JCFieldAccess) {
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
![img](https://img-blog.csdnimg.cn/img_convert/2b2eb6d1d376b1108a7f85f5b188030d.jpeg)
独家面经总结,超级精彩
本人面试腾讯,阿里,百度等企业总结下来的面试经历,都是真实的,分享给大家!
Java面试准备
准确的说这里又分为两部分:
- Java刷题
- 算法刷题
Java刷题:此份文档详细记录了千道面试题与详解;
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》,点击传送门即可获取!
里又分为两部分:
- Java刷题
- 算法刷题
Java刷题:此份文档详细记录了千道面试题与详解;
[外链图片转存中…(img-iiXXjR1c-1712412401105)]
[外链图片转存中…(img-MhjJAWeH-1712412401105)]
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》,点击传送门即可获取!