java - Vaadin 中的多线程刷新 UI

标签 java multithreading user-interface vaadin

我正在尝试在我的 Vaadin 应用程序中实现多线程刷新 UI。

此 UI 的一部分是基于容器的 dChart。要构建 dChart,我会迭代容器并按元素的属性之一对元素进行计数,如下所示:

        Collection<?> itemIDS = container.getItemIds();

        for (Object itemID : itemIDS) {
            Property property = container.getContainerProperty(itemID, "status");
            String status = (String) property.getValue();
            if (countMap.containsKey(status)) {
                countMap.put(status, countMap.get(status) + 1);
            } else {
                countMap.put(status, 1);
            }
        }

但是,如果容器有数千个元素,则需要 2-3 秒以上。

用户不能等待这么长时间来刷新 UI。

我读到我可以构建我的 UI,然后在 dChart 完全构建后使用 @Push Vaadin 注释刷新它。

所以我构建了这样的东西:

{
//class with @Push
  void refreshPieDChart(GeneratedPropertyContainer container) {
    new InitializerThread(ui, container).start();
  }

  class InitializerThread extends Thread {
    private LogsUI parentUI;
    private GeneratedPropertyContainer container;

    InitializerThread(LogsUI ui, GeneratedPropertyContainer container) {
        parentUI = ui;
        this.container = container;
    }

    @Override
    public void run() {
        //building dChart etc etc... which takes over 2-3 seconds

        // Init done, update the UI after doing locking
        parentUI.access(new Runnable() {
            @Override
            public void run() {
                chart.setDataSeries(dataSeries).show();
            }
        });
    }
  }
}

但是,如果我刷新页面几次,它就会生成有关 SQLContainer 的错误:

java.lang.IllegalStateException: A trasaction is already active!

因为几次刷新后,我的多个线程使用相同的 SQLContainer 并行运行。

为了解决这个问题,我想停止除最后一个线程之外的所有工作刷新线程,以消除并发问题。我怎样才能做到呢?也许还有其他解决方案?

编辑: 我已经尝试过这样的方法,但问题仍然存在,这是防止并发问题的正确方法吗?

{
  private static final Object mutex = new Object();
//class with @Push
  void refreshPieDChart(GeneratedPropertyContainer container) {
    new InitializerThread(ui, container).start();
  }

  class InitializerThread extends Thread {
    private LogsUI parentUI;
    private GeneratedPropertyContainer container;

    InitializerThread(LogsUI ui, GeneratedPropertyContainer container) {
        parentUI = ui;
        this.container = container;
    }

    @Override
    public void run() {
        //is it correct way to prevent concurrent problem? 
        synchronized(mutex){
            //method to refresh/build chart which takes 2-3s.
        }
        // Init done, update the UI after doing locking
        parentUI.access(new Runnable() {
            @Override
            public void run() {
                chart.setDataSeries(dataSeries).show();
            }
        });
    }
  }
}

最佳答案

这就是我所做的:

Henri Kerola给我指出了一个非常明显的想法:在 SQL 中进行计数。正如我所说,我正在考虑这一点,但这意味着我需要为每个可能的过滤器组合准备 SQL。这是一个相当复杂的事情,而且看起来不太好。

但这让我意识到,如果我在表中使用带过滤器的 SQLContainer,我可以对计数执行相同的操作。我只需要使用我自己的 FreeformQueryFreeformStatementDelegate 创建第二个 SQLContainer

如果我要创建上述所有内容,我可以将相同的过滤器添加到两个容器中,但是我现在不需要计算元素,因为第二个容器为我保存了值。听起来很复杂,但看看我的代码:

FreeformQuery myQuery = new MyFreeformQuery(pool);
FreeformQuery countQuery = new CountMyFreeformQuery(pool);

SQLContainer myContainer = new SQLContainer(myQuery);  //here i hold my all records as in a Question
SQLContainer countContainer = new SQLContainer(countQuery);  //here i hold computed count(*) and status

MyFreeformQuery.java 看起来像:

class ProcessFreeformQuery extends FreeformQuery {

private static final String QUERY_STRING = "SELECT request_date, status, id_foo FROM foo";
private static final String COUNT_STRING = "SELECT COUNT(*) FROM foo";
private static final String PK_COLUMN_NAME = "id_foo";

MyFreeformQuery(JDBCConnectionPool connectionPool) {
    super(QUERY_STRING, connectionPool, PK_COLUMN_NAME);
    setDelegate(new AbstractFreeformStatementDelegate() {

        protected String getPkColumnName() {
            return PK_COLUMN_NAME;
        }

        protected String getQueryString() {
            return QUERY_STRING;
        }

        protected String getCountString() {
            return COUNT_STRING;
        }
    });
}

最重要的CountFreeformQuery.java看起来像:

public class CountFreeformQuery extends FreeformQuery {

private static final String QUERY_STRING_GROUP = "SELECT status, count(*) as num FROM foo GROUP BY status";
private static final String QUERY_STRING = "SELECT status, count(*) as num FROM foo";
private static final String GROUP_BY = "foo.status";

public CountFreeformQuery(JDBCConnectionPool connectionPool) {
    super(QUERY_STRING_GROUP, connectionPool);
    setDelegate(new CountAbstractFreeformStatementDelegate() {
        protected String getQueryString() {
            return QUERY_STRING;
        }

        protected String getGroupBy(){
            return GROUP_BY;
        }

    });
}
}

现在,如果我想在这样之后刷新dChart:

myContainer.addContainerFilter(new Between(DATE_PROPERTY, getTimestamp(currentDate), getOneDayAfter(currentDate)));

我只是对 countContainer 执行相同的操作:

countContainer.addContainerFilter(new Between(DATE_PROPERTY, getTimestamp(currentDate), getOneDayAfter(currentDate)));

并将其传递给不需要计算元素的方法,只需将所有容器添加到 map ,然后添加到 dChart,如下所示:

    Map<String, Long> countMap = new HashMap<String, Long>();
    Collection<?> itemIDS = container.getItemIds();
    for (Object itemID : itemIDS) {
        Property statusProperty = container.getContainerProperty(itemID, "status");
        Property numProperty = container.getContainerProperty(itemID, "num");
        countMap.put((String) statusProperty.getValue(), (Long) numProperty.getValue());
    }

现在我已经计算了 myContainer 中元素的状态,无需多线程或编写大量 SQL。

谢谢大家的建议。

关于java - Vaadin 中的多线程刷新 UI,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31423764/

相关文章:

java - IMAP:(JavaMail)UIDVALIDTY 值总是更改某些文件夹

java - 将参数传递给 Lua 中的对象

c# - SingleProducerConstrained 和 MaxDegreeOfParallelism

android - Android 主 UI 线程的 Looper.prepare() 和 Looper.loop() 方法调用

java - 为什么这不是正方形?龙王金格

java - 谁能给我一个显示 SimpleDateFormat 线程不安全的例子?

java - 当输入无效时禁用 JFace-InputDialog 的 OK 按钮

user-interface - 使用多点触控用户界面提高工作效率

html - 我怎样才能使弹出窗口可滚动而没有内容

带有不同消息的 java swing 工具提示