gpt4 book ai didi

JavaFX 分页,在 setPageFactory 上泄漏?

转载 作者:搜寻专家 更新时间:2023-11-01 02:25:24 25 4
gpt4 key购买 nike

在下面的示例中,如果继续按“下一组”按钮,最终应用程序堆将被耗尽。将堆设置为较低的数字,例如 -mx50m -ms50m,它将很快达到上限。您可以看到经典的阶梯式内存消耗发生在 JVisualVM 之类的东西上。在使用 YourKit 进行分析时,发现有许多来自 javafx.* 和 java.* 类包的对象实例没有被 CG。 setPageFactory 似乎是罪魁祸首。也许这段代码正在做一些不应该做的事情,但到目前为止,我怀疑 Windows 8 JDK 1.7_60 上的 JavaFX 2.2 分页控件。 Window 8、Java FX 8、JDK 1.8_05 也存在同样的问题。

这是示例或 JavaFX Pagination.setPageFactory 的问题吗?

public class PaginationSample extends Application {
private static final String[] PAGE_TEXTS_0 = {"Time wounds all heals.", "The more I see, the less I know for sure.",
"Reality leaves a lot to the imagination.", "It's weird not to be weird."};
private static final String[] PAGE_TEXTS_1 = {"Fermions", "Quarks", "Leptons", "Bosons", "Gluon", "Graviton"};
private static final String[] PAGE_TEXTS_2 = {"AAAAAAA", "BBBBB", "CCCCCCCC"};
private static final String[][] ALL_GROUPS = {PAGE_TEXTS_0, PAGE_TEXTS_1, PAGE_TEXTS_2};

private int groupIndex;

@Override
public void start(Stage stage) {
final Pagination pagination = new Pagination();
setPagesGroup(pagination);

Button button = new Button("Next group");
button.setOnAction(new EventHandler<ActionEvent>() {
@Override
public void handle(ActionEvent actionEvent) {
nextGroup();
setPagesGroup(pagination);
}
});

BorderPane pane = new BorderPane();
pane.setCenter(pagination);
pane.setBottom(button);

stage.setScene(new Scene(pane, 400, 250));
stage.setTitle("What's leaking?");

stage.show();
}

private void setPagesGroup(Pagination pagination) {
pagination.setPageFactory(createPageFactory());
pagination.setPageCount(ALL_GROUPS[groupIndex].length);
}

private Callback<Integer, Node> createPageFactory() {
return new Callback<Integer, Node>() {
@Override
public Node call(Integer pageIndex) {
return createPage(ALL_GROUPS[groupIndex][pageIndex]);
}

private Node createPage(String text) {
return new Label(text);
}
};
}

private void nextGroup() {
if (++groupIndex == ALL_GROUPS.length) {
groupIndex = 0;
}
}

public static void main(String... args) {
launch(args);
}
}

下面测试期间分析器中的证据。在测试之前和结束时强制进行垃圾收集。

YourKit showing heap being consumption while clicking the button

补充调查:

如果行

//  pagination.setPageFactory(createPageFactory());

被注释掉了示例仍然有效,因为工厂仍在交付页面并且 setPageCount 方法强制 PaginationSkin 类调用其 resetIndexes 方法。 setPageFactory 和 setPageCount 都会调用 PaginationSkin.resetIndexes 并发生内存消耗。 PaginationSkin.java 可能有问题?

最佳答案

请为要解决的问题投票,您必须登录。

https://javafx-jira.kenai.com/browse/RT-38058

Java 7 和 8 代码调用

    pagination.setPageFactory(...);
pagination.setPageCount(...);

将导致此泄漏。


PaginationSkin 类中似乎存在泄漏。验证复制来源

http://hg.openjdk.java.net/openjfx/2.2.2/master/rt/file/98d1e63be240/javafx-ui-controls/src/com/sun/javafx/scene/control/skin/PaginationSkin.java

然后通过排除和分析内存的过程,您会发现内部类 IndicatorButton 中的这两个添加监听器导致泄漏:

@ line 1179

getSkinnable().getStyleClass().addListener(new ListChangeListener<String>() {
@Override
public void onChanged(Change<? extends String> change) {
setIndicatorType();
}
});


@ line 1197

tooltipVisibleProperty().addListener(new ChangeListener<Boolean>() {
@Override
public void changed(ObservableValue<? extends Boolean> ov, Boolean oldValue, Boolean newValue) {
setTooltipVisible(newValue);
}
});

由于内部类反向引用,在 IndicatorButton 对象从节点树 (getChildren.clear()) 中删除后,垃圾收集器无法删除这些对象事件。在 IDE 中,如果您暂时将 IndicatorButton 设置为静态,则可以中断编译并看到突出显示的对外部类的引用。当这两个监听器被注释掉时,泄漏就消失了。如果您保留这两个监听器,并注释掉监听器内的工作代码行,泄漏仍然会发生。

解决方法:

  1. 创建您自己的 PaginationSkin 版本(上面的源链接)并将其应用于您的 Pagination 控制对象(通过 pagination.setSkin 或 CSS)
  2. 在您的类中修改 IndicatorButton 类以具有 release() 方法,该方法将释放添加的监听器
  3. 在调用 getChildren.clear() 以释放 indicatorButtons 之前,在每个 IndicatorButton 对象上调用 release。

关于JavaFX 分页,在 setPageFactory 上泄漏?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24947997/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com