WPF性能优化之延迟加载(解决页面卡顿问题)

news2025/5/25 5:37:14

文章目录

  • 前言
  • 一. 基础知识回顾
  • 二. 问题分析
  • 三. 解决方案
    • 1. 新建一个名为DeferredContentHost的控件。
    • 2. 在DeferredContentHost控件中定义一个名为Content的object类型的依赖属性,用于承载要加载的子控件。
    • 3. 在DeferredContentHost控件中定义一个名为Skeleton的object类型的依赖属性,用于在子控件加载前显示骨架屏效果(使用加载效果也可以)。
    • 4. 在DeferredContentHost控件Loaded时显示骨架屏。
    • 5. 在DeferredContentHost控件显示骨架屏后执行Dispatcher.BeginInvoke(),将子控件显示的代码添加到Dispatcher消息队列。
  • 四. 运行效果
    • 4.1 未优化的效果
    • 4.2 优化后的效果


前言

长久以来,WPF的性能一直为人所诟病,其中很大一个原因就是因为WPF在开发过程中,稍有不慎就会阻塞UI线程,导致操作卡顿,甚至页面停止响应。它的原因当然是多方面的,我们今天只讨论比较常见的情况,并给出解决方案,让您开发的软件尽量减少卡顿。


一. 基础知识回顾

我们都知道WPF是单线程模型,所有UI元素必须由创建它们的线程直接操作,并且该线程还负责处理用户输入(鼠标、键盘)、渲染界面、执行事件处理程序、管理布局和动画等工作。所以如果UI线程一旦被阻塞,就会导致灾难性后果,反应到界面上就是卡顿,鼠标无法操作,也不响应键盘输入。

二. 问题分析

当加载一个Window时会执行一系列操作,其中最重要的操作就是布局的测量(Measure )与排列(Arrange),布局系统会从Window的根元素开始沿可视化树逐个调用子级控件的Measure与Arrange方法,以确认页面上的每个控件被渲染到正确的位置。在理想的情况下这种模式可以工作得很好,但是在实际项目中我们往往会在页面上嵌套数量庞大的子控件来实现功能,当要渲染的控件数量达到UI线程处理的瓶颈上限或是在布局计算中耗时过多,这时就可能会导致UI线程被阻塞。

三. 解决方案

既然UI线程不能无限制处理所有请求,那我们给它排个队,一个一个处理不就可以解决这个问题了。在WPF中所有控件都继承自DispatcherObject类,DispatcherObject类中有一个名为Dispatcher的属性,Dispatcher就是管理UI线程消息队列的核心。我们只需要将控件的加载任务送入Dispatcher,让它在合适的时机执行就可以了。以下是实现的过程:

1. 新建一个名为DeferredContentHost的控件。

2. 在DeferredContentHost控件中定义一个名为Content的object类型的依赖属性,用于承载要加载的子控件。

3. 在DeferredContentHost控件中定义一个名为Skeleton的object类型的依赖属性,用于在子控件加载前显示骨架屏效果(使用加载效果也可以)。

4. 在DeferredContentHost控件Loaded时显示骨架屏。

5. 在DeferredContentHost控件显示骨架屏后执行Dispatcher.BeginInvoke(),将子控件显示的代码添加到Dispatcher消息队列。

以上代码的核心在于Dispatcher.BeginInvoke(DispatcherPriority priority, Delegate method)方法中的priority参数,该参数用于指定method委托在消息队列中的执行优先级,以下为DispatcherPriority枚举的所有值:
在这里插入图片描述
从上图可以看出,为了不影响数据绑定、界面渲染、用户输入等操作,我们应该选择尽量低的优先级来执行子控件显示的代码。这里我们使用ContextIdle。以下是完整的代码:

[ContentProperty("Content")]
public class DeferredContentHost : FrameworkElement
{
    #region Fields
    private ContentControl _container = new ContentControl();
    #endregion
    #region Methods
    public DeferredContentHost()
    {
        this.AddVisualChild(_container);
        this.Loaded += DeferredContentHost_Loaded;
    }
    private void DeferredContentHost_Loaded(object sender, RoutedEventArgs e)
    {
        if (IsInDesignMode)
        {
            this._container.Content = this.Content;
        }
        else
        {
            this._container.Content = this.Skeleton;
            this.Dispatcher.BeginInvoke((Action)(() => this._container.Content = this.Content), System.Windows.Threading.DispatcherPriority.ContextIdle);
        }
    }
    protected override Visual GetVisualChild(int index)
    {
        return _container;
    }
    protected override Size MeasureOverride(Size availableSize)
    {
        _container.Measure(availableSize);
        if (availableSize.Width == double.PositiveInfinity || availableSize.Height == double.PositiveInfinity)
        {
            return _container.DesiredSize;
        }
        return availableSize;
    }
    protected override Size ArrangeOverride(Size finalSize)
    {
        _container.Arrange(new Rect(0, 0, finalSize.Width, finalSize.Height));
        return base.ArrangeOverride(finalSize);
    }
    #endregion
    #region Properties
    protected override int VisualChildrenCount => 1;
    protected bool IsInDesignMode { get => DesignerProperties.GetIsInDesignMode(this); }
    public object Content
    {
        get { return (object)GetValue(ContentProperty); }
        set { SetValue(ContentProperty, value); }
    }
    // Using a DependencyProperty as the backing store for UIElement.  This enables animation, styling, binding, etc...
    public static readonly DependencyProperty ContentProperty =
        DependencyProperty.Register("Content", typeof(object), typeof(DeferredContentHost));
    public object Skeleton
    {
        get { return (object)GetValue(SkeletonProperty); }
        set { SetValue(SkeletonProperty, value); }
    }
    // Using a DependencyProperty as the backing store for Skeleton.  This enables animation, styling, binding, etc...
    public static readonly DependencyProperty SkeletonProperty =
        DependencyProperty.Register("Skeleton", typeof(object), typeof(DeferredContentHost));
    #endregion
}

四. 运行效果

我们用大图片来模拟阻塞UI线程的情况,下面是两种效果对比。

4.1 未优化的效果

在这里插入图片描述

<DataTemplate x:Key="item1">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="*" />
            <RowDefinition Height="30" />
        </Grid.RowDefinitions>
        <Image Margin="5" Source="{Binding FullName}" />
        <TextBlock
            Grid.Row="1"
            Margin="5,0,5,5"
            HorizontalAlignment="Center"
            VerticalAlignment="Center"
            Text="{Binding Name}"
            TextTrimming="WordEllipsis" />
    </Grid>
</DataTemplate>

一次加载所有大图片,界面停止响应,文本框无法输入文字。

4.2 优化后的效果

在这里插入图片描述

<DataTemplate x:Key="item2">
    <controls:DeferredContentHost>
        <controls:DeferredContentHost.Skeleton>
            <controls:Skeleton>
                <controls:SkeletonGroup Orientation="Vertical">
                    <controls:SkeletonItem
                        Height="*"
                        Margin="5"
                        RadiusX="5"
                        RadiusY="5" />
                    <controls:SkeletonItem
                        Width="120"
                        Height="30"
                        Margin="5,0,5,5"
                        HorizontalAlignment="Center"
                        RadiusX="5"
                        RadiusY="5" />
                </controls:SkeletonGroup>
            </controls:Skeleton>
        </controls:DeferredContentHost.Skeleton>
        <Grid>
            <Grid.RowDefinitions>
                <RowDefinition Height="*" />
                <RowDefinition Height="30" />
            </Grid.RowDefinitions>
            <Image Margin="5" Source="{Binding FullName}" />
            <TextBlock
                Grid.Row="1"
                Margin="5,0,5,5"
                HorizontalAlignment="Center"
                VerticalAlignment="Center"
                Text="{Binding Name}"
                TextTrimming="WordEllipsis" />
        </Grid>
    </controls:DeferredContentHost>
</DataTemplate>

使用DeferredContentHost控件的延迟加载效果,加载过程文本框可以输入文字,界面可以正常响应鼠标操作。


技术交流
QQ群:661224882
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2385113.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

移植 FART 到 Android 10 实现自动化脱壳

版权归作者所有&#xff0c;如有转发&#xff0c;请注明文章出处&#xff1a;https://cyrus-studio.github.io/blog/ FART 源码 FART 是 ART 环境下基于主动调用的自动化脱壳方案。 关于 FART 详细介绍参考&#xff1a; FART 自动化脱壳框架简介与脱壳点的选择 FART 主动调用…

COMPUTEX 2025 | 广和通创新解决方案共筑AI交互新纪元

5月20日至23日&#xff0c;广和通携多领域创新解决方案亮相2025年台北国际电脑展&#xff08;COMPUTEX 2025&#xff09;&#xff0c;台北南港展览馆#K0727a展位。此次展会&#xff0c;广和通围绕“Advancing Connectivity Intelligent Future”为主题&#xff0c;设置四大核心…

了解Android studio 初学者零基础推荐(3)

kotlin中的数据类及对象 使用泛型创建可重复使用的类 我们将常在线答题考试&#xff0c;有的考试题型包括判断&#xff0c;或者填空&#xff0c;以及数学题&#xff0c;此外试题内容还包括难易程度&#xff1a;"easy”,"medium"&#xff0c;"hard",…

Spring 定时器和异步线程池 实践指南

前言&#xff1a;Spring&#xff1a;异步线程池和定时器 原理篇 一、Spring Scheduler 1. 创建一个 SpringBoot项目&#xff0c;在启动类上添加 EnableScheduling 注解&#xff0c;表示开启定时任务。 2. 创建SchedulerService&#xff0c;在方法上面启用Scheduled 注解 在方…

零基础设计模式——创建型模式 - 生成器模式

第二部分&#xff1a;创建型模式 - 生成器模式 (Builder Pattern) 前面我们学习了单例、工厂方法和抽象工厂模式&#xff0c;它们都关注如何创建对象。生成器模式&#xff08;也常被称为建造者模式&#xff09;是另一种创建型模式&#xff0c;它专注于将一个复杂对象的构建过程…

MD编辑器推荐【Obsidian】含下载安装和实用教程

为什么推荐 Obsidian &#xff1f; 免费 &#xff08;Typora 开始收费了&#xff09;Typora 实现的功能&#xff0c;它都有&#xff01;代码块可一键复制 文件目录支持文件夹 大纲支持折叠、搜索 特色功能 – 白板 特色功能 – 关系图谱 下载 https://pan.baidu.com/s/1I1fSly…

I-CON: A UNIFYING FRAMEWORK FOR REPRESENTATION LEARNING

I-con:表示学习的统一框架 基本信息 ICLR 2025 博客贡献人 田心 作者 Shaden Alshammari, John Hershey, Axel Feldmann, William T. Freeman, Mark Hamilton 关键词 I-Con框架,表征学习&#xff0c;损失函数统一框架 摘要 随着表征学习领域的快速发展&#xff0c;各类…

Missashe线代题型总结

Missashe线性代数考研题型总结 说明&#xff1a;这篇笔记用于博主对"线代"常考题型进行总结&#xff0c;99%为真题&#xff0c;大概可能应该会逐步更新解题思路。有目录可直接检索。 第一章 行列式 1 具体行列式计算 1&#xff09;么字型 2015 数一 2016 数一三…

蓝桥杯13届 卡牌

问题描述 这天, 小明在整理他的卡牌。 他一共有 n 种卡牌, 第 i 种卡牌上印有正整数数 i(i∈[1,n]), 且第 i 种卡牌 现有 ai​ 张。 而如果有 n 张卡牌, 其中每种卡牌各一张, 那么这 n 张卡牌可以被称为一 套牌。小明为了凑出尽可能多套牌, 拿出了 m 张空白牌, 他可以在上面…

安卓开发用到的设计模式(1)创建型模式

安卓开发用到的设计模式&#xff08;1&#xff09;创建型模式 文章目录 安卓开发用到的设计模式&#xff08;1&#xff09;创建型模式1. 单例模式&#xff08;Singleton Pattern&#xff09;2. 工厂模式&#xff08;Factory Pattern&#xff09;3. 抽象工厂模式&#xff08;Abs…

线程的一些基本知识

前言 最近在学习线程&#xff0c;线程与进程是面试中可能常考的问题&#xff0c;我总结了线程的一些知识。分享给大家&#xff0c;希望可以帮组到大家。 线程知识总结(包含与进程的区别) 结语 希望可以帮助到有需要的人&#xff0c;bye~~

【Python打卡Day30】模块与包的导入@浙大疏锦行

#一、导入官方库 我们复盘下学习python的逻辑&#xff0c;所谓学习python就是学习python常见的基础语法学习你所处理任务需要用到的第三方库 所以你用到什么学什么库即可。学习python本身就是个伪命题&#xff0c;就像你说学习科目一样&#xff0c;你没说清晰你学习的具体科目…

26考研|高等代数:λ-矩阵

前言 本章知识点较为简单&#xff0c;是作为工具性的一章&#xff0c;在学习过程中&#xff0c;要注意区分行列式因子、不变因子以及初等因子&#xff0c;同时还要对若尔当标准型的计算应该足够熟悉&#xff0c;尤其是复矩阵的若尔当标准型计算是十分重要的。 课本重点回顾 …

我店模式系统开发打造本地生活生态商圈

在当今快节奏的商业环境中&#xff0c;商家们面临着越来越多的挑战&#xff0c;包括市场竞争加剧、消费者需求多样化以及运营效率的提高等。为了应对这些挑战&#xff0c;越来越多的商家开始寻求信息化解决方案&#xff0c;以提升运营效率和客户体验。我的店模式系统平台应运而…

数据库练习(3)

简单选择题要点: 1.锁协议: 数据库原理及应用&#xff08;高级篇)01——封锁协议(图文并解&#xff0c;超详细&#xff0c;一看就会)_数据库锁协议-CSDN博客https://blog.csdn.net/qq_44236958/article/details/105790970 2.tablespace和datafile 一个tablespace可以有一个或多…

OpenGL ES 基本基本使用、绘制基本2D图形

OpenGL ES 绘制基础图形 OpenGL ES基本概念 OpenGL ES (Embedded-System) 是专为嵌入式设备&#xff08;如手机、平板、VR 设备&#xff09;设计的图形 API&#xff0c;是 OpenGL 的轻量级版本。 &#xff5c;下面是一个Android使用 OpenGL ES的基本框架 MainActivity 设置一…

BU9792驱动段式LCD

1、C文件,需要自己添加软件iic或硬件iic驱动&#xff0c;该驱动在我的别的文章内有。亲测bu9792是正常驱动的&#xff08;只用到了前14个SEG&#xff09;&#xff0c;说实话有点懵了。后面的ICSET有个P2根据不同的SEG地址要置1或0&#xff0c;读的时候最高位也是0?读命令寄存器…

力扣-将x减到0的最小操作数

1.题目描述 2.题目链接 1658. 将 x 减到 0 的最小操作数 - 力扣&#xff08;LeetCode&#xff09; 3.题目分析 1&#xff09;正面求解困难 题目要求我们每次都从最左边或者最右边取一个数&#xff0c;使x-元素的值&#xff0c;并在数组中移除该元素。最后返回的最小操作数…

三、【数据建模篇】:用 Django Models 构建测试平台核心数据

【数据建模篇】&#xff1a;用 Django Models 构建测试平台核心数据 前言我们要设计哪些核心数据&#xff1f;准备工作&#xff1a;创建 Django App开始设计数据模型 (Models)1. 通用基础模型 (可选但推荐)2. 项目模型 (Project)3. 模块模型 (Module)4. 测试用例模型 (TestCase…

shp2pgsql 导入 Shp 到 PostGIS 空间数据库

前言 ❝ shp2pgsql是PostGIS自带的命令行工具&#xff0c;用于将Shapefile文件声称SQL脚本导入到PostGIS空间数据库。 1. 安装 PostGIS 通过Application Stack Builder或者下载单独的PostGIS包进行安装。而shp2pgsql则是与PostGIS工具集成在一起&#xff0c;无需单独下载。该命…