0

0

与 @Named 一起揭开挑战

王林

王林

发布时间:2024-08-29 09:09:32

|

493人浏览过

|

来源于dev.to

转载

在上下文和依赖注入 (cdi) 不断发展的环境中,开发人员经常遇到与 bean 命名、默认实现和潜在冲突相关的障碍。本文详细探讨了 cdi 中与 @named 注释相关的潜在陷阱。我们将深入研究其复杂性,阐明有问题的场景,并讨论替代方法,包括使用 smallrye 中的 @identifier。此外,我们将提供有关构建健壮且可维护的jakarta ee
最佳实践的见解 应用程序。

理解@default

@default 注释是 cdi 中的一个有价值的工具,用于将特定实现显式标记为给定接口或 bean 类型的默认实现。它在处理同一接口的多个实现时发挥作用,允许开发人员指定在不使用其他限定符时默认应注入哪个实现。

考虑存在 greetingservice 接口的多个实现的场景:

@default
public class defaultgreetingservice implements greetingservice {

  @override
  public string greet(string name) {
    return "hello, " + name;
  }
}
public class specialgreetingservice implements greetingservice {

  @override
  public string greet(string name) {
    return "greetings, " + name + "!";
  }
}

在不指定任何限定符的情况下注入 bean 时,cdi 使用 @default 标记的 bean 作为默认值。这在具有多种实现的场景中非常有用,提供了明确的默认选择。

@inject
private greetingservice greetingservice; // injects the @default implementation

虽然 @default 的使用是可选的,但强烈建议使用它,特别是在处理具有多个实现的接口时。它提供了清晰一致的默认选项,防止 bean 注入期间出现歧义和意外行为。

探索@named——一把双刃剑

@named 限定符在 cdi 中发挥着基础作用,为 bean 分配一个人类可读的名称或标识符。开发人员在将 bean 注入其他组件时经常使用它来通过名称引用 bean。

但是,@named 也有其自身的一系列挑战,特别是在没有额外限定符的情况下使用时。默认情况下,cdi 将非限定类名关联为 bean 名称。这可能会导致与 @default 限定符发生冲突,从而导致 bean 注入期间出现意外行为。

@named
public class mybean {
  // implementation
}

在没有显式限定符的情况下注入 mybean 时,cdi 将仅添加 @named 限定符,而不是 @default 限定符。仅当在 bean 或其限定符上显式指定 @default 限定符时,才会应用 @default 限定符。

@inject
private mybean mybean;

在这种情况下,如果有其他具有相同类型名称的 bean,可能会出现歧义。例如,如果有另一个名为 mybean 的 bean,则注入将导致歧义。

为了解决这个问题,开发人员应该明确限定他们打算注入的 bean。

@inject
@named("mybean")
private mybean mybean;

或者,开发人员可以为每个 bean 使用自定义限定符来消除歧义。

问题案例:含糊不清和意外违约

当在没有附加限定符的情况下使用 @named 时,会出现歧义,并且存在相同类型的多个实现。考虑以下场景:

@named
public class servicea implements service {
  // implementation
}
@named
public class serviceb implements service {
  // implementation
}

在没有显式限定符的情况下注入 service 可能会导致歧义,因为两个 bean 按类型匹配,并且没有名称或限定符区分它们。

@inject
private service service;

在这种情况下,cdi 不会隐式添加 @default 或尝试解决歧义,从而导致由于不明确的依赖关系而导致注入失败。

PHP 网络编程技术与实例(曹衍龙)
PHP 网络编程技术与实例(曹衍龙)

PHP网络编程技术详解由浅入深,全面、系统地介绍了PHP开发技术,并提供了大量实例,供读者实战演练。另外,笔者专门为本书录制了相应的配套教学视频,以帮助读者更好地学习本书内容。这些视频和书中的实例源代码一起收录于配书光盘中。本书共分4篇。第1篇是PHP准备篇,介绍了PHP的优势、开发环境及安装;第2篇是PHP基础篇,介绍了PHP中的常量与变量、运算符与表达式、流程控制以及函数;第3篇是进阶篇,介绍

下载

替代方案:从 smallrye common 引入 @identifier

认识到 @named 带来的挑战,开发人员经常寻求替代方案来更明确地控制 bean 识别。其中一种替代方案是来自
的 @identifier 注释 小黑麦常见。此注释提供了一种更清晰、更可控的 bean 命名方法,减少了冲突和意外默认的风险。与 @named 不同,@named 要求每个应用程序都有唯一的值,@identifier 允许多个 bean 具有相同的标识符值,只要它们的类型不同。在处理相同接口或相关类型的不同实现时,这种灵活性特别有用。

要使用@identifier,只需用该注解注释bean类并指定标识符值即可:

@identifier("payment")
public class defaultpaymentprocessor implements paymentprocessor {
  // implementation
}
@identifier("payment")
public class legacypaymentgateway implements paymentgateway {
  // implementation
}

使用@identifier注入bean很简单:

public class client {
  @inject
  @identifier("payment")
  paymentprocessor processor;

  @inject
  @identifier("payment")
  paymentgateway gateway;

}

这里,“付款”@identifier 值被多个 bean 重用,因为 paymentprocessor 和 paymentgateway 的类型不同。 @named 不允许这种灵活性,其中
值在应用程序范围内必须是唯一的。

@named 的另一种替代方法是创建自定义限定符。自定义限定符是用户定义的注释,可用于识别和限定 bean。它们提供对 bean 选择的最精细控制,并且可以根据应用程序的特定需求进行定制。

要创建自定义限定符,请按照以下步骤操作:

  1. 定义一个新的注释类。
  2. 使用@qualifier对注解类进行注解。
  3. (可选)为限定符提供默认值。

例如,以下名为 defaultpaymentgateway 的自定义限定符表示默认支付网关实现:

@qualifier
@retention(runtime)
@target({method, field, parameter, type})
public @interface defaultpaymentgateway {

}

要使用自定义限定符,请用它注释 bean 类:

@defaultpaymentgateway
public class standardpaymentgateway implements paymentgateway {
  // implementation
}
public class expresspaymentgateway implements paymentgateway {
  // implementation
}

然后,使用限定符注入 bean:

@Inject
@DefaultPaymentGateway
private PaymentGateway paymentGateway;

选择正确的方法

bean 识别的最佳方法取决于应用程序的具体需求。对于简单的应用程序,@named 可能就足够了。对于更复杂的应用程序,@identifier 或
自定义限定符提供更多控制和灵活性。

下表总结了每种方法的优缺点:

approach pros cons
@named simple, widely supported can be ambiguous, conflicts with @default
@identifier clearer identification, no conflicts with @default requires additional annotations
custom qualifiers maximum flexibility, fine-grained control requires upfront effort to define and maintain

进一步确认,可以参考官方cdi规范

与 @Named 一起揭开挑战

结论:bean 命名和默认值的策略选择

总之,与 @named 相关的潜在陷阱强调了在 cdi 中使用此注释时需要仔细考虑。当依赖隐式命名时,尤其是在存在多个实现的情况下,可能会出现歧义和意外的默认值。鼓励开发人员探索替代方案,例如来自smallrye common的@identifier,以获得更受控制和更明确的bean识别方法。采用显式限定、自定义限定符和替代方法可确保更流畅、更可控的 cdi 体验,从而实现健壮且可维护的 java。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

834

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

739

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

735

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

26

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号