C# 自定义线程修改UI(一)

在Wpf中界面显示数据的修改,都是通过UI线程完成,如果尝试从其他线程中直接修改控件的值回抛出异常,“调用线程无法访问此对象,因为另一个线程拥有该对象”。

例如:http://www.cnblogs.com/tianma3798/p/5762016.html

解决方案1:通过Invoke或者BeginInvoke将操作发送给Dispatcher 对象,委托Dispatcher去执行UI操作,每个UI控件都有Dispatcher 对象

代码实例:

Xaml代码

复制代码
<ProgressBar x:Name="progressOne"  
                Minimum="0" Maximum="1" 
                Height="30"   Margin="10,85,10,0" 
                VerticalAlignment="Top"/>
<!--绑定 进度条的值,并格式化为百分数-->
<Label x:Name="label" Content="{Binding ElementName=progressOne,Path=Value}" 
        ContentStringFormat="{}{0:P2}"
        HorizontalAlignment="Left" 
        Margin="110,120,0,0"
        FontWeight="Bold" Foreground="Red"
        VerticalAlignment="Top"/>
复制代码

C#代码

复制代码
//启动前程修改UI数据
Task.Run(() =>
{
    int i = 0;
    while (true)
    {
        i++;
        if (i == 100)
            i = 0;
        //定义修改UI委托
        Action<int> action1 = (q) =>
        {
            progressOne.Value = q/100.0;
        };
        progressOne.Dispatcher.BeginInvoke(action1, i);
        Thread.Sleep(100);
    }
});
复制代码

显示:

解决方案2:使用Wpf的双向绑定,在自定义线程中修改源数据,自动通知到UI线程修改界面显示

这种方式有个好处,只关心数据的修改,界面多出使用该数据的界面自动修改

更多双向绑定:http://www.cnblogs.com/tianma3798/p/5765464.html

代码实例:

Xaml代码

复制代码
<Grid>
    <Slider x:Name="slider" 
            Height="30"
            Value="{Binding Path=Result,Mode=TwoWay}" Minimum="0" Maximum="100"
            Margin="35,120,25,0" VerticalAlignment="Top"/>
    <Label x:Name="label" Content="{Binding Path=Result,Mode=TwoWay}"
            ContentStringFormat="当前值:{0}"
            HorizontalAlignment="Left" 
            FontWeight="Bold" Foreground="Red"
            Margin="95,165,0,0" 
            VerticalAlignment="Top"/>
</Grid>
复制代码

C#代码

复制代码
//双向绑定数据
NumberData data = new NumberData();
slider.DataContext = data;
label.DataContext = data;
//启动线程修改绑定数据,界面内容会同步修改
Task.Run(() =>
{
    int i = 0;
    while (true)
    {
        i++;
        if (i == 100) i = 0;
        //修改双向绑定属性
        data.Result = i;
        Thread.Sleep(100);
    }
});
复制代码
复制代码
/// <summary>
/// 实现INotifyPropertyChanged接口,通知
/// </summary>
public class NumberData : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    private int _Result;
    public int Result
    {
        get { return _Result; }
        set
        {
            _Result = value;
            if (this.PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs("Result"));
        }
    }
}
复制代码

更多异步修改UI:http://www.cnblogs.com/tianma3798/p/5766691.html

posted @   天马3798  阅读(1936)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
点击右上角即可分享
微信分享提示