- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在 TomEE Java 应用程序中使用 Instance 作为惰性/动态注入(inject)器,并且我注意到我的应用程序中存在内存泄漏。这对我来说是第一次,所以看到 Java EE 库中概述的内存泄漏警告实际上令人惊讶:
package javax.enterprise.inject;
public interface Instance<T> extends Iterable<T>, Provider<T>
{
/**
* Destroy the given Contextual Instance.
* This is especially intended for {@link javax.enterprise.context.Dependent} scoped beans
* which might otherwise create mem leaks.
* @param instance
*/
public void destroy(T instance);
}
@ApplicationScoped
的冲突引起的。和
Instance<T>
.我提供了一个示例,说明图层在我的类中是如何存在的。注意嵌套的
Instance<T>
.这是为了提供任务的动态注入(inject)。
@ApplicationScoped
public class MessageListenerImpl implements MessageListener {
@Resource(name="example.mes")
private ManagedExecutorService mes;
@Inject @Any
private Instance<Worker<ExampleObject>> workerInstance;
// ...
@Override
public void onMessage(Message message) {
ExampleObject eo = new ExampleObject();
Worker<ExampleObject> taskWorker = workerInstance.get();
taskWorker.setObject(eo);
mes.submit(taskWorker);
}
// ...
}
public class Worker<T> implements Runnable {
@Inject @Any
private Instance<Task> taskInstance;
@Setter
private T object
// ...
@Override
public void run() {
Task t = taskInstance.get();
t.setObject(object);
t.doTask();
// Instance destruction, manual cleanup tried here.
}
// ...
}
public interface Task<T> {
void doTask();
void setObject(T obj);
}
destroy(T instance)
的类泄漏是
ExampleObject
,
Worker<T>
,以及
Task<T>
的实现.为了保持异步设计,我尝试传递
Worker<T>
的实例在它的实例中(可能是个坏主意,但我还是试过了),调用
destroy(T instance)
和设置
ExampleObject
至
null
.这清理了
Task<T>
实现和
ExampleObject
,但不是
Worker<T>
.
MessageListenerImpl
内进行同步设计。 (即删除
Worker<T>
并使用
Task<T>
)作为后备工作,调用
destroy(T instance)
清理。这仍然留下了泄漏,这让我相信这一定是与
@ApplicationScoped
的冲突。和
Instance<T>
.
最佳答案
确实这是 Instance
的弱点,它可能会泄漏。 This article有一个很好的解释。 (正如下面 Siliarus 的评论中所强调的,这不是 Instance
的固有错误,而是错误的使用/设计。)
您的 Worker
声明没有范围,因此它是 @Dependent
范围。这意味着每次注入(inject)都会重新创建它。 Instance.get()
本质上是一种注入(inject),因此每次调用 get()
都会创建一个新的依赖范围对象。 .
规范说,依赖范围的对象在它们的“父对象”(意味着它们被注入(inject)的对象)被销毁时被销毁;但是应用程序范围的 bean 与应用程序一样长,保持它们创建的所有依赖范围的 bean 活着。这就是内存泄漏。
要减轻链接文章中所写的操作:
workerInstance.destroy(taskWorker)
只要你不需要taskWorker
不再,最好在 finally
内堵塞:@Override
public void onMessage(Message message) {
ExampleObject eo = new ExampleObject();
Worker<ExampleObject> taskWorker;
try {
taskWorker = workerInstance.get();
taskWorker.setObject(eo);
mes.submit(taskWorker);
}
finally {
workerInstance.destroy(taskWorker);
}
}
@Dependent
更改会发生什么?例如@ApplicationScoped
?如果 destroy()
call 没有显式删除,这不是毫无戒心的开发人员在正常重构中会做的事情,您最终会破坏“全局”资源。 CDI 会小心地重新创建它,因此不会对应用程序造成任何功能损害。仍然打算仅实例化一次的资源将不断被销毁/重新创建,这可能会产生非功能(性能)影响。所以,从我的角度来看,这个解决方案会导致客户端和实现之间不必要的耦合,我宁愿不这样做。 Instance
对于延迟加载,并且只有一个实例,您可能需要缓存它:...
private Worker<ExampleObject> worker;
private Worker<ExampleObject> getWorker() {
if( worker == null ) {
// guard against multi-threaded access if environment is relevant - not shown here
worker = workerInstance.get();
}
return worker;
}
...
Worker<ExampleObject> taskWorker = getWorker();
...
Worker
提供范围,这样它的父级不再负责它的生命周期,而是相关的范围。 关于jakarta-ee - CDI |应用程序/从属范围 |内存泄漏 - javax.enterprise.inject.Instance<T> 未收集垃圾,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49388648/
我正在尝试关注 this tutorial .我想我不是从使用可下载项目开始,而是从我之前做过的一个简单的“spring MVC - Maven - eclipse”项目开始。这个项目运行良好。 因此
我觉得 Java EE 6 规范有些困惑。有几组注释。 我们有javax.ejb注释如 @Stateful和 @Stateless用于创建 EJB。 还有一个@javax.annotation.Man
我正在开发一个依赖于“javax.lang”和“javax.annotation”的java项目。我安装了jre/jdk,类路径上还有很多其他javax.*,我每天都用Java进行开发。但是,这两个包
我正在尝试将一个值添加到 JsonValue 列表中。可以这样做吗? 一些背景知识,我正在从 Rest API 检索 Json 响应,在此 Json 中是一个如下名称列表: { “名称”:{名称1,名
我正在开发一个 JSF2、Icefaces 网络应用程序。我有以下看法: 当我保存上面的 时出现以下异常. Application caught instance of
我遇到了一个小问题,我的代码中有一个异常。 我有一个基本框架类: import java.awt.BorderLayout; import java.awt.Color; import java.aw
我正在使用 Apache Derby 并具有以下代码: DBConnectionFactory.java package edu.unsw.comp9321.jdbc; import java.sql
javax.mail 和 javax.mail-api 有什么区别? 我从 maven 存储库中找到了那些。 什么时候应该使用它们? javax.mail-api自带的软件包,但我无法使用,所以我下载
我是 Java 的新手,正在尝试进入 WebServices。我在某处找到了两个示例,但对可用选项感到困惑。 首先,带注解的javax.jws.WebService 似乎工作正常,但javax.xml
在升级了一些 Glassfish/Grizzly 依赖项之后(为了与最新版本的 Azure SDK IOT 设备客户端兼容),我开始出现错误,因为 com.google.common.EventBus
我收到了一个遗留 JSP 系统,其中 Eclipse 在每次出现以下代码时都会标记“导入 javax.event 无法解析”错误: 这是我的java版本: shakir@anduril:~$ jav
我使用以下代码获取连接到系统的 USB 设备的制造商代码。我添加了 jsr80-1.0.1 jar 。我收到以下错误 javax.usb.UsbException: Properties file j
我正在学习 EJB,当尝试使用 junit 测试它时,出现以下错误 cd.espoirmur.Ejb.InterfaceEjbLocal_80488159 Jun 03, 2016 10:33:58
我必须处理一个需要提供数据源作为参数的 API。问题是我从提供 EntityManager 或 PersistentContext 的上下文访问它,它们似乎没有通过其方法公开任何数据源。如何以编程方式
据我所知,Java ServletContext 和上下文对象在 Java EE 程序开发中很常见。但是,我不太确定它们之间有什么区别,尤其是上下文类的用法。 据我了解,ServletContext
我在一个简单的 tomcat jsp 项目中遇到了这个错误。我在谷歌上阅读的文章暗示我应该在我的项目中包含 servlet-api.jar。我就是这样做的,但没有用。有人对此有任何想法吗?我的 JRE
我想对我的实体进行一些 Bean 验证,以便我可以适本地映射错误以发送回客户端。 我知道 javax.validation.constraints 注释用于实现此目的。我的问题是,对于模式生成,我需要
想法:2018.3 jetty :9.4.11 我按照 IDEA 中给出的说明下载了 jrebel 文件夹 在 Debug模式下运行 Jetty 服务器时出现以下错误。 HTTP ERROR 500
我正在开发一个主要使用无状态 session bean (SLSB) 的 EJB3 应用程序。他们使用容器管理事务 (CMT)。 我希望 bean 知道事务(用于日志记录等)。我可以实现 javax.
我正在使用 OSGI 开发 Web 应用程序。我有一个我找不到的 Maven 配置错误。这是完整的源代码: http://uploading.com/files/8e5c9888/SH_27_test
我是一名优秀的程序员,十分优秀!