日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

跨线程访问控件

發(fā)布時間:2023/12/13 编程问答 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 跨线程访问控件 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

【轉(zhuǎn)帖】?

我們在做winform應用的時候,大部分情況下都會碰到使用多線程控制界面上控件信息的問題。然而我們并不能用傳統(tǒng)方法來做這個問題,下面我將詳細的介紹。

????? 首先來看傳統(tǒng)方法:

???? ?public partial class Form1 : Form
??? {
??????? public Form1()
??????? {
??????????? InitializeComponent();
??????? }
??????? private void Form1_Load(object sender, EventArgs e)
??????? {
??????????? Thread thread = new Thread(ThreadFuntion);
??????????? thread.IsBackground = true;
??????????? thread.Start();
??????? }
??????? private void ThreadFuntion()
??????? {
??????????? while (true)
??????????? {
??????????????? this.textBox1.Text = DateTime.Now.ToString();
??????????????? Thread.Sleep(1000);
??????????? }
??????? }
??? }

?????? 運行這段代碼,我們會看到系統(tǒng)拋出一個異常:Cross-thread operation not valid:Control 'textBox1' accessed from a thread other than the thread it was created on . 這是因為.net 2.0以后加強了安全機制,不允許在winform中直接跨線程訪問控件的屬性。那么怎么解決這個問題呢,下面提供幾種方案。

????? 第一種方案,我們在Form1_Load()方法中加一句代碼:

????? private void Form1_Load(object sender, EventArgs e)
??????{
??????????? Control.CheckForIllegalCrossThreadCalls = false;
??????????? Thread thread = new Thread(ThreadFuntion);
??????????? thread.IsBackground = true;
??????????? thread.Start();
??????? }
????? 加入這句代碼以后發(fā)現(xiàn)程序可以正常運行了。這句代碼就是說在這個類中我們不檢查跨線程的調(diào)用是否合法(如果沒有加這句話運行也沒有異常,那么說明系統(tǒng)以及默認的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發(fā)現(xiàn)它是一個static的,也就是說無論我們在項目的什么地方修改了這個值,他就會在全局起作用。而且像這種跨線程訪問是否存在異常,我們通常都會去檢查。如果項目中其他人修改了這個屬性,那么我們的方案就失敗了,我們要采取另外的方案。

????? 下面來看第二種方案,就是使用delegate和invoke來從其他線程中控制控件信息。網(wǎng)上有很多人寫了這種控制方式,然而我看了很多這種帖子,表明上看來是沒有什么問題的,但是實際上并沒有解決這個問題,首先來看網(wǎng)絡上的那種不完善的方式:

public partial class Form1 : Form
??? {
??????? private delegate void FlushClient();//代理
??????? public Form1()
??????? {
??????????? InitializeComponent();
??????? }
??????? private void Form1_Load(object sender, EventArgs e)
??????? {
??????????? Thread thread = new Thread(CrossThreadFlush);

????????????thread.IsBackground=true;
??????????? thread.Start();
??????? }

??????? private void CrossThreadFlush()
??????? {
??????????? //將代理綁定到方法
??????????? FlushClient fc = new FlushClient(ThreadFuntion);
??????????? this.BeginInvoke(fc);//調(diào)用代理
??????? }
??????? private void ThreadFuntion()
??????? {
??????????? while (true)
??????????? {
??????????????? this.textBox1.Text = DateTime.Now.ToString();
??????????????? Thread.Sleep(1000);
??????????? }
??????? }
??? }

?????? 使用這種方式我們可以看到跨線程訪問的異常沒有了。但是新問題出現(xiàn)了,界面沒有響應了。為什么會出現(xiàn)這個問題,我們只是讓新開的線程無限循環(huán)刷新,理論上應該不會對主線程產(chǎn)生影響的。其實不然,這種方式其實相當于把這個新開的線程“注入”到了主控制線程中,它取得了主線程的控制。只要這個線程不返回,那么主線程將永遠都無法響應。就算新開的線程中不使用無限循環(huán),使可以返回了。這種方式的使用多線程也失去了它本來的意義。

?????? 現(xiàn)在來讓我們看看推薦的解決方案:

public partial class Form1 : Form
??? {
??????? private delegate void FlushClient();//代理
??????? public Form1()
??????? {
??????????? InitializeComponent();
??????? }
??????? private void Form1_Load(object sender, EventArgs e)
??????? {
??????????? Thread thread = new Thread(CrossThreadFlush);
??????????? thread.IsBackground = true;
??????????? thread.Start();
??????? }

??????? private void CrossThreadFlush()
??????? {
??????????? while (true)
??????????? {
??????????????? //將sleep和無限循環(huán)放在等待異步的外面
??????????????? Thread.Sleep(1000);
??????????????? ThreadFunction();
??????????? }
??????? }
??????? private void ThreadFunction()
??????? {
??????????? if (this.textBox1.InvokeRequired)//等待異步
??????????? {
??????????????? FlushClient fc = new FlushClient(ThreadFunction);
??????????????? this.Invoke(fc);//通過代理調(diào)用刷新方法
??????????? }
??????????? else
??????????? {
??????????????? this.textBox1.Text = DateTime.Now.ToString();
??????????? }
??????? }
??? }

?????? 運行上述代碼,我們可以看到問題已經(jīng)被解決了,通過等待異步,我們就不會總是持有主線程的控制,這樣就可以在不發(fā)生跨線程調(diào)用異常的情況下完成多線程對

winform多線程控件的控制了。

?

通過主線程的委托 去調(diào)用控件就不算是跨線程訪問控件了。


Code
using?System;
using?System.Collections.Generic;
using?System.ComponentModel;
using?System.Data;
using?System.Drawing;
using?System.Text;
using?System.Windows.Forms;
using?System.Threading;

namespace?WindowsApplication1
{
????
public?partial?class?Form1?:?Form
????{
????????
private?delegate?void?FlushClient();//代理
????????public?Form1()
????????{
????????????InitializeComponent();
????????}
????????
private?void?Form1_Load(object?sender,?EventArgs?e)
????????{
????????????Thread?thread?
=?new?Thread(CrossThreadFlush);
????????????thread.IsBackground?
=?true;
????????????thread.Start();
????????}

????????
private?void?CrossThreadFlush()
????????{
????????????
while?(true)
????????????{
????????????????
//將sleep和無限循環(huán)放在等待異步的外面
????????????????Thread.Sleep(1000);
????????????????ThreadFunction();
????????????}
????????}
????????
private?void?ThreadFunction()
????????{
????????????
if?(this.label1.InvokeRequired)//等待異步
????????????{
????????????????FlushClient?fc?
=?new?FlushClient(ThreadFunction);
????????????????
this.label1.Invoke(fc);//通過代理調(diào)用刷新方法
????????????}
????????????
else//這里在什么情況下執(zhí)行呢,在this.label1.Invoke(fc)時執(zhí)行,以為fc是主線程的委托FlushClient的實例,所以InvokeRequired為false
????????????{
????????????????
this.label1.Text?=?DateTime.Now.ToString();
????????????????
????????????}
????????}
????}
}

前臺線程和后臺線程:一旦所有前臺線程在托管進程(其中 .exe 文件是托管程序集)中被停止,系統(tǒng)將停止所有后臺線程并關閉,主線程就是后臺線程。所有的后臺線程在應用程序退出時都會自動結(jié)束。應用程序必須運行完所有的前臺線程才可以退出。

轉(zhuǎn)載于:https://www.cnblogs.com/shineqiujuan/archive/2009/11/12/1601503.html

總結(jié)

以上是生活随笔為你收集整理的跨线程访问控件的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。