0

0

Vaadin Grid异步内容加载优化:实现平滑渐进式渲染

碧海醫心

碧海醫心

发布时间:2025-07-20 14:44:01

|

767人浏览过

|

来源于php中文网

原创

Vaadin Grid异步内容加载优化:实现平滑渐进式渲染

本文探讨了Vaadin Grid在集成异步数据加载时可能遇到的“同步”加载问题,即尽管启用了Push并使用了异步方法,网格内容仍一次性加载,而非逐项渐进显示。核心解决方案是通过在单独的线程中启动每个项目的异步操作,确保UI在数据准备好之前能立即渲染占位符,从而显著提升用户体验和界面响应性。

Vaadin Grid异步加载的挑战

在使用vaadin构建web应用时,grid组件是展示大量数据的常用选择。为了提供流畅的用户体验,特别是当数据需要耗时操作(如网络请求、复杂计算)才能获取时,异步加载变得至关重要。vaadin的push机制允许服务器端主动向客户端推送更新,这为异步数据加载提供了基础。然而,开发者常遇到的一个问题是,即使启用了push并使用了异步方法(例如spring的@async),grid的内容(特别是通过addcomponentcolumn添加的自定义组件)仍然表现出“同步”加载的特性——即整个网格的ui结构会立即显示,但内部的组件内容却需要等待所有异步操作完成后才一次性填充,导致一段明显的空白期。

问题的根源在于,尽管异步方法本身在后台线程执行耗时操作,但触发这些异步操作的调用可能仍然发生在Vaadin的UI线程中。如果这些调用(即使是返回ListenableFuture的调用)在UI渲染周期中阻塞了UI线程,或者其返回的Future在UI线程中被“等待”以进行后续处理,那么用户仍然会感受到延迟。对于像Image这样的组件,如果其src属性在初始渲染时无法立即获得,我们希望它能先以空状态显示,待数据加载完成后再更新。

初始尝试与局限性

考虑以下Vaadin Grid的设置,它尝试异步加载每个单元格的图标:

// Grid设置
private void setupGrid() {
    Grid grid = new Grid<>();
    // 为每个网格项添加一个组件列,该组件将异步加载图标
    grid.addComponentColumn(this::getIconColumn).setHeader("Name");
    grid.setItems("item1", "item2", "item3");

    // 启用Vaadin Push更新,允许服务器异步推送UI变更
    grid.getDataCommunicator().enablePushUpdates(Executors.newCachedThreadPool());
    add(grid);
}

// 获取图标列的组件方法
private Image getIconColumn(String entityName) {
    Image image = new Image("", ""); // 初始为空图片
    image.setHeight("150px");
    image.setWidth("100px");

    UI ui = UI.getCurrent(); // 获取当前UI实例,用于线程安全更新
    // 异步加载图标,并设置回调
    // 即使asyncLoadIcon是@Async方法,此处的调用仍可能在UI线程中触发,
    // 导致UI在Future完成前无法渲染空图片
    asyncLoadIcon(entityName)
        .addCallback(
            result -> ui.access(() -> image.setSrc(result)), // 成功时更新图片源
            err -> ui.access(() -> Notification.show("Failed to load icon for " + entityName)) // 失败时显示通知
        );

    return image; // 返回图片组件
}

// 模拟异步加载图标的方法
@Async // Spring的异步注解
public ListenableFuture asyncLoadIcon(String entityName) {
    try {
        Thread.sleep(3000); // 模拟耗时操作
        return AsyncResult.forValue("path/to/icon/" + entityName + ".png");
    } catch (InterruptedException e) {
        Thread.currentThread().interrupt();
        return AsyncResult.forExecutionException(new RuntimeException("Icon loading interrupted"));
    }
}

尽管asyncLoadIcon方法被标记为@Async,其内部的耗时操作会在单独的线程中执行,但getIconColumn方法在Vaadin UI渲染过程中被调用。在这种情况下,asyncLoadIcon(entityName)的调用本身,以及ListenableFuture的创建和回调的注册,可能仍然在UI线程中发生。如果这个过程相对耗时(即使是微秒级,但对于大量数据项累积起来就明显了),或者Vaadin的渲染机制在等待Future的“设置”完成,那么UI仍然会感觉被阻塞,直到所有这些Future都被创建并注册完毕。结果是,网格的轮廓显示了,但所有的图片内容都延迟了3秒后才一次性出现,而不是逐个填充。

解决方案:在独立线程中启动异步任务

为了实现真正的渐进式加载,关键在于确保在getIconColumn方法返回Image组件时,该组件能够立即被添加到Grid并渲染出来,而无需等待任何异步操作的启动。这意味着,即使是异步操作的启动也应该发生在非UI线程中。

解决方案是在getIconColumn方法内部,为每个网格项的异步图标加载任务创建一个新的Thread并立即启动它。这样,getIconColumn方法可以迅速返回一个空的Image组件,Vaadin可以立即渲染它。当后台线程中的异步任务完成时,通过UI.access()方法安全地更新UI。

private Image getIconColumn(MangaEntity entity) {
    Image image = new Image("", ""); // 初始为空图片,立即返回
    image.setHeight("150px");
    image.setWidth("100px");

    UI ui = UI.getCurrent(); // 获取当前UI实例
    // 在一个新的线程中启动异步加载任务
    // 这样,getIconColumn方法可以立即返回,不会阻塞UI渲染
    new Thread(() -> {
        // 调用异步加载图标的方法
        // 注意:这里的rsCrawler.asyncLoadIcon假设是一个Spring Bean,
        // 它的@Async注解会确保实际的耗时操作在Spring的线程池中执行。
        // 而new Thread()确保了对rsCrawler.asyncLoadIcon的调用本身不阻塞UI线程。
        rsCrawler.asyncLoadIcon(entity.getUrlName())
            .addCallback(
                // 成功回调:使用ui.access()安全地更新UI
                result -> ui.access(() -> image.setSrc(result)),
                // 失败回调:使用ui.access()安全地显示通知
                err -> ui.access(() -> Notification.show("Failed to parse icon for " + entity.getName()))
            );
    }).start(); // 启动新线程

    return image; // 立即返回Image组件,允许Vaadin立即渲染占位符
}

通过这种方式,当Grid渲染时,getIconColumn方法会非常迅速地返回一个Image实例。Vaadin能够立即将这个空的Image组件添加到Grid中并显示出来。与此同时,一个新的线程被启动,它负责调用asyncLoadIcon并处理回调。当asyncLoadIcon完成其耗时操作后,它会通过addCallback触发UI.access()来更新对应的Image组件的src属性。这样,用户会看到网格结构和空的图片占位符立即出现,然后图片会逐个、渐进地填充进来,极大地提升了用户体验。

TTSMaker
TTSMaker

TTSMaker是一个免费的文本转语音工具,提供语音生成服务,支持多种语言。

下载

注意事项与最佳实践

  1. 线程管理: 尽管new Thread(() -> ...).start()能解决问题,但在生产环境中频繁创建新线程并不是最佳实践,因为它会带来额外的开销和资源管理问题。更推荐的做法是使用ExecutorService(如ThreadPoolExecutor或Executors.newCachedThreadPool())来管理和复用线程。Spring的@Async注解背后通常就是由一个ThreadPoolTaskExecutor支持的,所以上述解决方案中,new Thread()的作用是确保rsCrawler.asyncLoadIcon的调用不阻塞UI线程,而@Async确保了asyncLoadIcon内部的实际耗时操作在线程池中执行。

  2. 错误处理: 确保在异步回调中妥善处理异常。addCallback的第二个参数用于处理错误,并通过UI.access()安全地向用户展示错误信息。

  3. 用户体验: 这种渐进式加载方式显著提升了用户对应用响应速度的感知。对于图片等媒体内容,可以考虑在加载期间显示加载指示器(例如Spinner)或占位符图像,以进一步优化用户体验。

  4. 取消机制: 对于长时间运行的异步任务,如果用户在任务完成前离开了视图,考虑实现任务取消机制,以避免不必要的资源消耗和潜在的内存泄漏。

总结

在Vaadin Grid中实现真正的异步渐进式内容加载,关键在于将耗时操作的启动也从UI线程中分离出来。虽然Vaadin Push和UI.access()确保了UI更新的线程安全,但如果异步任务的初始化本身阻塞了UI渲染,用户体验仍然会受损。通过在单独的线程中启动每个项目的异步任务,我们可以确保Grid能够立即渲染占位符,随后再逐项填充内容,从而提供一个更加流畅和响应迅速的用户界面。在实际项目中,应结合线程池等更高级的线程管理策略,以确保应用的性能和稳定性。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

98

2025.08.06

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

471

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

54

2025.12.01

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

319

2023.10.09

vb连接access数据库的方法
vb连接access数据库的方法

vb连接access数据库方法:1、使用ADO连接,首先导入System.Data.OleDb模块,然后定义一个连接字符串,接着创建一个OleDbConnection对象并使用Open() 方法打开连接;2、使用DAO连接,首先导入 Microsoft.Jet.OLEDB模块,然后定义一个连接字符串,接着创建一个JetConnection对象并使用Open()方法打开连接即可。

371

2023.10.16

asp连接access数据库的方法
asp连接access数据库的方法

连接的方法:1、使用ADO连接数据库;2、使用DSN连接数据库;3、使用连接字符串连接数据库。想了解更详细的asp连接access数据库的方法,可以阅读本专题下面的文章。

119

2023.10.18

access和trunk端口的区别
access和trunk端口的区别

access和trunk端口的区别是Access端口用于连接终端设备,提供单个VLAN的接入,而Trunk端口用于连接交换机之间,提供多个VLAN的传输;Access端口只传输属于指定VLAN的数据,而Trunk端口可以传输多个VLAN的数据,并使用VLAN标签进行区分。想了解更多access和trunk端口相关内容,可以阅读本专题下面的文章。

314

2023.10.31

access怎么导入数据
access怎么导入数据

access导入数据步骤:1. 选择数据源 2. 选择要导入的文件 3. 指定导入选项 4. 选择导入目标 5. 预览数据 6. 导入数据即可。想了解更多access的相关内容,可以阅读本专题下面的文章。

410

2024.04.10

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 3.2万人学习

10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号