
本教程详细阐述了如何在嵌入式jetty服务器中正确集成jersey rest服务和weld cdi,以解决常见的依赖注入失败问题。通过优化gradle依赖配置,并采用jetty cdi模块推荐的`cdiservletcontainerinitializer`和`enhancedlistener`进行cdi上下文初始化,确保了`@inject`注解能够正常工作,从而实现一个功能完善、支持cdi的独立rest应用。
1. 引言:Jetty、Jersey与Weld CDI集成挑战
在Java SE环境中构建一个独立的、支持RESTful API并具备依赖注入(CDI)功能的Web服务器时,嵌入式Jetty、Jersey(JAX-RS实现)和Weld(CDI实现)是常见的技术栈组合。然而,如果不正确配置CDI上下文,可能会遇到Unsatisfied dependencies等依赖注入失败的问题,导致应用程序无法正常启动或运行。本教程将提供一个经过验证的解决方案,指导您如何正确地将这三者集成。
2. Gradle依赖配置优化
正确的依赖是成功集成的基石。原始配置可能包含一些冗余或不兼容的依赖。以下是针对Jakarta EE 9+环境,推荐的精简且必要的Gradle依赖配置:
plugins { id 'application' id 'java' id 'eclipse'}repositories { mavenCentral()}dependencies { // 日志 implementation 'org.slf4j:slf4j-api:2.0.4' implementation 'ch.qos.logback:logback-classic:1.4.5' // Jetty 服务器核心与Servlet支持 implementation 'org.eclipse.jetty:jetty-servlet:11.0.12' // Jetty CDI集成模块,关键! implementation 'org.eclipse.jetty:jetty-cdi:11.0.12' // Weld Servlet环境支持 implementation 'org.jboss.weld.servlet:weld-servlet-core:4.0.3.Final' // Jersey RESTful框架核心与Servlet容器集成 implementation 'org.glassfish.jersey.containers:jersey-container-servlet-core:3.0.4' // Jersey CDI集成模块,针对Jakarta EE 9+ (CDI 2.0 SE) implementation 'org.glassfish.jersey.media:jersey-cdi2-se:3.0.4' // Jersey JSON处理支持 implementation 'org.glassfish.jersey.media:jersey-media-json-jackson:3.0.4' // 测试依赖 (可选) testImplementation 'org.junit.jupiter:junit-jupiter:5.7.2'}application { mainClass = 'it.gym.StartApp'}tasks.named('test') { useJUnitPlatform()}
关键点说明:
LanguagePro
LanguagePro是一款强大的AI写作助手,可以帮助你更好、更快、更有效地写作。
120 查看详情
org.eclipse.jetty:jetty-cdi: 这是Jetty官方提供的CDI集成模块,它负责将Jetty的生命周期与CDI容器(如Weld)的生命周期关联起来。org.jboss.weld.servlet:weld-servlet-core: 提供了Weld在Servlet环境下的核心支持。org.glassfish.jersey.media:jersey-cdi2-se: 这是Jersey针对Jakarta EE 9+(CDI 2.0 SE)的CDI集成模块,确保Jersey能够正确发现并使用CDI管理的资源类和Bean。我们利用了Maven/Gradle的传递性依赖特性,精简了列表,避免了手动添加所有子依赖。
3. Jetty服务器与Weld CDI集成设置
正确初始化CDI上下文是解决依赖注入问题的核心。传统的通过Weld实例直接设置监听器和BeanManager的方式在嵌入式Jetty中可能无法与Jersey的CDI集成模块协同工作。应采用Jetty CDI模块推荐的ServletContainerInitializer机制。
修改后的StartApp.java主类如下:
package it.gym;import org.eclipse.jetty.cdi.CdiDecoratingListener;import org.eclipse.jetty.cdi.CdiServletContainerInitializer;import org.eclipse.jetty.server.Server;import org.eclipse.jetty.servlet.ServletContextHandler;import org.eclipse.jetty.servlet.ServletHolder;import org.glassfish.jersey.servlet.ServletContainer;import org.jboss.weld.environment.servlet.EnhancedListener;public class StartApp { public static void main(String[] args) { // Weld的初始化不再直接在main方法中进行,而是通过ServletContainerInitializer // Weld weld = new Weld(); // WeldContainer container = weld.initialize(); final Server server = new Server(9000); final ServletContextHandler context = new ServletContextHandler(ServletContextHandler.SESSIONS); context.setContextPath("/"); // 关键的CDI集成配置 // 1. 设置CDI集成模式为CdiDecoratingListener context.setInitParameter( CdiServletContainerInitializer.CDI_INTEGRATION_ATTRIBUTE, CdiDecoratingListener.MODE); // 2. 添加Jetty CDI的ServletContainerInitializer context.addServletContainerInitializer(new CdiServletContainerInitializer()); // 3. 添加Weld的EnhancedListener,确保Weld在Servlet环境中正确启动和关闭 context.addServletContainerInitializer(new EnhancedListener()); // Jersey Servlet配置 final ServletHolder servletHolder = new ServletHolder(ServletContainer.class); servletHolder.setInitOrder(1); servletHolder.setInitParameter( "jersey.config.server.provider.packages", "it.gym.rest"); // 确保指向包含REST资源的包 context.addServlet(servletHolder, "/rest/*"); server.setHandler(context); try { server.start(); server.join(); } catch (Exception e) { e.printStackTrace(); } finally { // 确保Weld容器在应用关闭时也正确关闭 (如果手动初始化,这里需要weld.shutdown()) // 但通过ServletContainerInitializer方式,通常由容器自动管理 } }}
关键点说明:
CdiServletContainerInitializer.CDI_INTEGRATION_ATTRIBUTE: 这个初始化参数告诉Jetty的CDI模块如何处理CDI集成,CdiDecoratingListener.MODE指示使用装饰器监听器模式。context.addServletContainerInitializer(new CdiServletContainerInitializer()): 注册Jetty CDI的初始化器。它会在Servlet容器启动时被调用,负责发现并启动CDI容器。context.addServletContainerInitializer(new EnhancedListener()): 注册Weld提供的增强型监听器。这个监听器是Weld在Servlet环境中正确启动和管理其生命周期的推荐方式。jersey.config.server.provider.packages: 确保此参数指向您的JAX-RS资源类所在的包,以便Jersey能够发现它们。
4. 示例REST资源与CDI Bean
以下是使用@Inject进行依赖注入的REST资源类和CDI Bean的示例,它们将在上述配置下正常工作。
GymEndpoint.java (REST资源类):
package it.gym.rest;import java.util.List;import it.gym.dao.GymDAO;import jakarta.enterprise.context.RequestScoped;import jakarta.inject.Inject;import jakarta.ws.rs.GET;import jakarta.ws.rs.Path;import jakarta.ws.rs.core.Response;import jakarta.ws.rs.core.Response.Status;@Path("/")@RequestScoped // 将REST资源本身也声明为CDI管理的Beanpublic class GymEndpoint { @Inject // 通过CDI注入GymDAO实例 private GymDAO gymDAO; @GET @Path("/test") public Response test() { List entity = gymDAO.getDevices(); return Response.status(Status.OK).entity(entity).build(); }}
GymDAO.java (CDI Bean):
package it.gym.dao;import java.util.ArrayList;import java.util.List;import jakarta.enterprise.context.RequestScoped;@RequestScoped // 将此DAO类声明为CDI管理的Beanpublic class GymDAO { public GymDAO() { // 构造函数,CDI容器会负责实例化 } public List getDevices() { // 模拟数据访问 List devices = new ArrayList(); devices.add("Device A"); devices.add("Device B"); return devices; }}
关键点说明:
@RequestScoped: 确保GymEndpoint和GymDAO都被CDI容器发现并管理。当一个HTTP请求到来时,CDI会为@RequestScoped的Bean创建一个新的实例,并在请求结束后销毁。@Inject: 这是CDI的核心注解,用于请求容器注入一个符合条件的Bean实例。
5. 运行与验证
完成上述配置和代码修改后,运行StartApp的main方法。当服务器启动后,您可以通过访问http://localhost:9000/rest/test来验证您的REST服务。如果一切配置正确,您应该会收到一个包含设备列表的JSON响应(例如[“Device A”, “Device B”]),而不是依赖注入失败的错误。
6. 注意事项与总结
版本兼容性:确保所有Jetty、Jersey、Weld和Jakarta EE API依赖的版本兼容。本教程使用的版本是针对Jakarta EE 9+环境的。如果您的项目使用更早的Java EE或Jakarta EE版本,相应的依赖(尤其是Jersey的CDI模块,如jersey-cdi1x)可能需要调整。传递性依赖:Jetty和Jersey的CDI模块会引入大部分必要的Jakarta EE API和Weld核心库。避免手动添加重复的API依赖,以免造成版本冲突。beans.xml:在某些CDI环境中,需要在META-INF或WEB-INF目录下放置一个空的beans.xml文件来显式激活CDI。但在大多数Weld SE和Servlet环境中,如果类带有CDI注解,Weld会自动发现它们。如果遇到Bean未被发现的问题,可以尝试添加此文件。日志配置:为了更好地调试,确保您的logback.xml或log4j2.xml配置正确,以便查看CDI容器和Jersey的详细启动日志。
通过遵循本教程的步骤,您将能够成功地在嵌入式Jetty服务器中集成Jersey REST服务和Weld CDI,构建一个健壮且易于维护的独立Java应用程序。关键在于理解并正确配置Jetty CDI模块和Weld Servlet监听器,让它们协同工作,从而实现无缝的依赖注入。
以上就是在Jetty嵌入式服务器中集成Jersey REST服务与Weld CDI指南的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/721493.html
微信扫一扫
支付宝扫一扫