【Lombok】@Delegate | 将其他类方法注入到当前类

一、实验性功能说明

@Delegate是在lombok v0.10中作为特性引入的(当时实验包还不存在)。

lombok v1.14中,它被移到了实验包中;lombok主包中的旧版本现在已被废弃。

实验因为:

  • 不常用。
  • 难以支持边缘情况,如递归委托。
  • API相当不友好;如果你能简单地实现一些方法,并让@Delegate为你没有手动实现的东西生成委托,那就更好了,但由于泛型擦除的问题,这也无法做到没有注意事项。

当前状态:否定-目前我们认为该功能不会很快脱离实验状态,如果未来版本的javacecj难以继续维护该功能,则可能会放弃对该功能的支持。

二、简介

任何字段或无参数方法都可以用@Delegate来注解,让lombok生成转发对这个字段的调用(或调用这个方法的结果)的委托方法。

Lombok委托该字段类型(或方法的返回类型)的所有公共方法,以及其父类型的方法,但所有以java.lang.Object声明的方法除外。

你可以在@Delegate注解的类型参数中传递任意数量的类。如果你这样做,那么lombok将委托这些类型(以及它们的父类型,除了java.lang.Object)中的所有公共方法,而不是看字段/方法的类型。

所有属于计算类型的公共非Object方法都被复制,无论你是否也为这些方法写了实现。因此,这将导致重复的方法错误。你可以通过使用@Delegate(excludes=SomeType.class)参数来排除被排除类型中的所有公共方法,以及它们的超类型来避免这些错误。

为了非常精确地控制哪些被委托,哪些不被委托,可以用方法签名编写私有内部接口,然后在@Delegate(types=PrivateInnerInterfaceWithIncludesList.class, excludes=SameForExcludes.class)中指定这些私有内部接口作为类型。

三、示例比较

1. Lombok 写法

import java.util.ArrayList;
import java.util.Collection;

import lombok.experimental.Delegate;

public class DelegationExample {
  private interface SimpleCollection {
    boolean add(String item);
    boolean remove(Object item);
  }
  
  @Delegate(types=SimpleCollection.class)
  private final Collection<String> collection = new ArrayList<String>();
}


class ExcludesDelegateExample {
  long counter = 0L;
  
  private interface Add {
    boolean add(String x);
    boolean addAll(Collection<? extends String> x);
  }
  
  @Delegate(excludes=Add.class)
  private final Collection<String> collection = new ArrayList<String>();
  
  public boolean add(String item) {
    counter++;
    return collection.add(item);
  }
  
  public boolean addAll(Collection<? extends String> col) {
    counter += col.size();
    return collection.addAll(col);
  }
}

2. Java 标准写法

import java.util.ArrayList;
import java.util.Collection;

public class DelegationExample {
  private interface SimpleCollection {
    boolean add(String item);
    boolean remove(Object item);
  }
  
  private final Collection<String> collection = new ArrayList<String>();
  
  @java.lang.SuppressWarnings("all")
  public boolean add(final java.lang.String item) {
    return this.collection.add(item);
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean remove(final java.lang.Object item) {
    return this.collection.remove(item);
  }
}

class ExcludesDelegateExample {
  long counter = 0L;
  
  private interface Add {
    boolean add(String x);
    boolean addAll(Collection<? extends String> x);
  }
  
  private final Collection<String> collection = new ArrayList<String>();
  
  public boolean add(String item) {
    counter++;
    return collection.add(item);
  }
  
  public boolean addAll(Collection<? extends String> col) {
    counter += col.size();
    return collection.addAll(col);
  }
  
  @java.lang.SuppressWarnings("all")
  public int size() {
    return this.collection.size();
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean isEmpty() {
    return this.collection.isEmpty();
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean contains(final java.lang.Object arg0) {
    return this.collection.contains(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public java.util.Iterator<java.lang.String> iterator() {
    return this.collection.iterator();
  }
  
  @java.lang.SuppressWarnings("all")
  public java.lang.Object[] toArray() {
    return this.collection.toArray();
  }
  
  @java.lang.SuppressWarnings("all")
  public <T extends .java.lang.Object>T[] toArray(final T[] arg0) {
    return this.collection.<T>toArray(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean remove(final java.lang.Object arg0) {
    return this.collection.remove(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean containsAll(final java.util.Collection<?> arg0) {
    return this.collection.containsAll(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean removeAll(final java.util.Collection<?> arg0) {
    return this.collection.removeAll(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public boolean retainAll(final java.util.Collection<?> arg0) {
    return this.collection.retainAll(arg0);
  }
  
  @java.lang.SuppressWarnings("all")
  public void clear() {
    this.collection.clear();
  }
}

四、支持的配置项

lombok.delegate.flagUsage = [warning | error] (默认: not set)
Lombok@Delegate的任何用法标记为警告或错误(如果已配置)。

五、附属说明

当向注解的typestypes参数传递类时,你不能包括泛型。这是java的一个限制。使用私有的内部接口或扩展预定类型的类,包括泛型参数来解决这个问题。

当向注解传递类时,这些类不需要是字段的超类型。请看例子

@Delegate不能用在静态字段或方法上。

当要委托/排除的计算类型本身包含@Delegate注释时,不能使用@Delegate;换句话说,如果你试图递归使用它,@Delegate会出错。

参考文献

【1】@Delegate | Don’t lose your composition.

### Java Lombok @Data 注解编译错误解决方案 当遇到 `@Data` 或其他 Lombok 注解导致的找不到符号 (cannot find symbol) 的编译错误时,通常是因为 IDE 或构建工具未能正确识别和处理这些注解。以下是详细的排查和解决方法: #### 1. 安装并配置 Lombok 插件 确保已安装适用于开发环境的 Lombok 插件。对于 IntelliJ IDEA 用户来说,即使官方宣称自带集成,也建议手动确认插件状态。 - 进入菜单栏中的 **File -> Settings** - 寻找 **Plugins** 部分,并搜索 "Lombok" - 如果未安装,则点击 **Install**, 并按照提示完成操作[^5] #### 2. 启用注解处理器 某些情况下,默认设置可能禁用了必要的注解处理器功能,这会阻止 Lombok 正常工作。 - 对于 Maven 和 Gradle 构建项目,需确保 pom.xml 文件或 build.gradle 中包含了正确的依赖项以及启用了 APT(Annotation Processing Tool) ```xml <!-- For Maven --> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>${最新版本号}</version> <scope>provided</scope> </dependency> <build> <plugins> <!-- Enable annotation processing --> ... </plugins> </build> ``` ```groovy // For Gradle dependencies { compileOnly 'org.projectlombok:lombok:${最新版本号}' annotationProcessor 'org.projectlombok:lombok' } ``` - 在 Eclipse/IntelliJ IDEA 设置里开启 Annotation Processors 功能 - **Eclipse**: Window → Preferences → Java → Compiler → Building → Enable project specific settings...勾选上"Enable annotation processing" - **IDEA**: File → Settings → Build, Execution, Deployment → Compiler → Annotation Processors 勾选 "Enable annotation processing"[^3] #### 3. 清理缓存与重新导入模块 有时旧版文件残留可能导致冲突,清理相关缓存有助于解决问题。 - 使用命令行执行如下指令来清除本地仓库缓存: ```bash mvn clean install -U # 更新所有快照版本依赖 rm -rf ~/.m2/repository/* # 删除Maven全局库下的所有内容(谨慎操作!) ``` - 或者通过 IDE 提供的功能来进行清理重置动作,比如在 IDEA 上可以选择 Invalidate Caches / Restart 来刷新整个工程结构. #### 4. 检查 Lombok 版本兼容性 不同 JDK 及框架组合下可能存在不匹配情况,因此要保证所使用的 Lombok 是最新的稳定发行版之一;同时也要注意与其他第三方库之间的潜在冲突关系[^1]. ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

猫巳

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值