2

我目前正在开发一个 GWT 应用程序,作为未来项目的技术证明。我喜欢用 Java 而不是 JavaScript 构建我的 AJAX 代码的方式。但是当我重复调用 RPC 服务时,我似乎遇到了内存问题。浏览器内存使用量不断增长。

在搜索 Google 时,我一直在阅读有关 GWT 有多棒以及它不可能发生内存泄漏的信息,所以任何人都可以解释为什么我的浏览器(Firefox 和 Chromium)内存飞速增长吗?

提前感谢您帮助我,布拉姆

编码:

...

class ExampleTable extends Composite

  private RPCService rpcService;
  private Timer tableUpdater;

  public ExampleTable(){
   ... Init timer and RPC Service
   ... Add components
   initWidget();
  }

  private void getTableDataFromRPCService() {
  this.rpcService.getData(new AsyncCallback<ArrayList<Data>>() {

      @Override
      public void onSuccess(ArrayList<Data> result) {
          ExampleTable.this.updateTable(result);
      }

      @Override
      public void onFailure(Throwable caught) {
          //Do nothing
      }
  });
  }

  private void updateTable(ArrayList<Data> tableData){
    ... Update the table
  }

  private void startUpdateTask() {
      this.serviceUpdater = new Timer() {
          @Override
          public void run() {
            ExampleTable.this.getTableDataFromRPCService();
          }
      };
      serviceUpdater.scheduleRepeating(2000);
  }
}

编辑:

我花了一些时间编写了一个可以在这里下载的测试应用程序。在 Firefox 占用大约 350MB 内存之后,我在启用表更新的情况下运行了大约半个小时的应用程序。我还运行了测试,在 Firefox 中禁用更新表一小时内存使用量略高于 100MB。

(要运行此示例,您需要 GWT 的 Google 可视化 API,它可以从 Google 下载,但由于新的用户政策,我不允许发布链接)

我刚下班回家,在没有更新表数据的情况下开始另一个测试,以查看内存使用量是否不断增加或是否在某个点停止。

这是客户端实现类(GWTMemoryIssue.java):

public class GWTMemoryIssue implements EntryPoint {
//Run with or without table
private static final boolean WITH_TABLE = false; 

private final TestServiceAsync rpcService = GWT.create(TestService.class);

private Panel panel;
private Timer timer;
private Table table;

public void onModuleLoad() {
    RootPanel rootPanel = RootPanel.get();

    this.panel = new VerticalPanel();
    this.panel.setSize("100%", "100%");

    rootPanel.add(panel);

    if (WITH_TABLE) {
        loadTable();
    }else{
        startUpdateTask();
    }

}

private void startUpdateTask() {
    this.timer = new Timer() {

        @Override
        public void run() {
            GWTMemoryIssue.this.getTableData();

        }
    };
    this.timer.scheduleRepeating(2000);
}

public void loadTable() {
    Runnable onLoadCallback = new Runnable() {
        public void run() {
            GWTMemoryIssue.this.table = new Table(createTableData(), createTableOptions());
            GWTMemoryIssue.this.table.setSize("100%", "100%");
            GWTMemoryIssue.this.panel.add(GWTMemoryIssue.this.table);
            GWTMemoryIssue.this.startUpdateTask();
        }
    };

    VisualizationUtils.loadVisualizationApi(onLoadCallback, Table.PACKAGE);
}

private Options createTableOptions() {
    Options options = Options.create();

    return options;
}

private DataTable createTableData() {
    DataTable data = DataTable.create();

    data.addColumn(ColumnType.STRING, "Name");
    data.addColumn(ColumnType.NUMBER, "Intval 1");
    data.addColumn(ColumnType.NUMBER, "Intval 2");
    data.addColumn(ColumnType.NUMBER, "Intval 3");

    return data;
}

private void getTableData() {
    rpcService.getListOfItems(new AsyncCallback<ArrayList<ListItem>>(){
        public void onFailure(Throwable caught) {
            // Do nothing
        }

        public void onSuccess(ArrayList<ListItem> result) {
            if (WITH_TABLE){
                GWTMemoryIssue.this.updateTableData(result);
            }else{
                //Ignore the data from the server
            }
        }
    });
}

private void updateTableData(ArrayList<ListItem> result) {
    DataTable data = createTableData();

    data.addRows(result.size());

    int row = 0;
    for (ListItem li : result) {
        data.setValue(row, 0, li.getName());
        data.setValue(row, 1, li.getIntVal());
        data.setValue(row, 2, li.getIntSecondVal());
        data.setValue(row, 3, li.getThirdIntVal());
        row++;
    }

    this.table.draw(data, createTableOptions());
}
}
4

3 回答 3

1

您在此处提供的所有其他信息都是一些想法。我猜内存增加是由内存中剩余的列表引起的。有可能内存根本没有释放,或者 JavaScript 垃圾收集器没有时间清理,因为更新之间的时间间隔太短。这里有一些你可以做的测试:

  1. 要测试垃圾收集器是否没有时间调整您的代码,以便更新仅运行有限次数,然后检查内存使用量是否在几分钟内减少。如果内存使用量减少,那么在您的实际示例中它可能不是一个问题。但是通过将延迟设置为 30 秒进行简单的测试。
  2. 您可以通过在使用列表后清除列表来帮助垃圾收集器:我不知道什么最有效,所以这里有一些建议:从列表中删除对象,或循环遍历列表并将值设置为 null。在正常情况下这应该是不必要的,因为垃圾收集器会这样做。

您还可以尝试以下 Firefox 内存分析器插件,看看是否可以找到内存增加:http ://ajaxian.com/archives/enhanced-firefox-memory-profiler-add-on

于 2009-10-10T09:32:44.317 回答
1

我一直在使用 GWT 处理许多表和 RPC,直到现在我发现的大多数内存泄漏都是我自己的错。

据我所知,RPC 层似乎没有泄漏,而且您的示例太简单了,不会造成问题。

您可能需要查看 updateTable 方法实际在做什么,您自己的代码中可能存在泄漏。

使用 GWT 可能导致大量内存泄漏的一件事是 IE 中的 Imagebundle。众所周知,这些在 GWT 中非常泄漏,因为它们使用 DXTransform 来支持 alpha 透明度。每次将小部件放在屏幕上时,内存都会大量增加。但是有一些技巧可以避免这种情况。

于 2009-10-09T07:28:23.480 回答
0

不,在 Javascript 中清理垃圾不需要显式操作。它应该自动运行(尽管浏览器中的 GC 与现代 JVM 中的级别不同)。

GWT 尽力避免常见的会导致 JS 内存泄漏的陷阱(JS 和 DOM 节点之间的循环引用在某些浏览器中处理不当)。

所以问题是:内存使用量总是在上升吗?或者它是否在某个点达到顶峰(或者它只是因内存不足而崩溃?)。您的应用程序似乎在不断增长,这可能是正常的……但 GC 应该在某个时候启动。

在我的应用程序中,内存使用量最高约为 64MB。但我不在 Ubuntu 上工作,Windows 上的 IE 是我们的主要目标,尽管我有时也会在 FireFox 上进行测试(而且我也没有看到泄漏)。

您可能需要做的另一件事是避免像您一样每 2 秒轮询一次。如果请求花费的时间超过 2 秒,则您开始排队请求(浏览器对并发连接数有限制)。所以最好在触发新计时器之前等待响应。

于 2009-10-09T12:11:21.670 回答