主要是对一个过程需要的时间很长执行时会出现界面假死的情况

方法1:

Application.DoEvents(),这种方法当你拖动窗体时,界面不会假死。但在你拖动时代码不再执行,也就是阻塞了,当你不再控制窗体时会继续执行,其实这还是一个单线程

  for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{ textBox1.Text = i.ToString() + " " + j.ToString();
Application.DoEvents();
}
}

方法2:多线程

2.1:取消控件跨线程检测(不推荐有时会出现一些莫名奇妙的错误如控件不能加载等问题)

2.1.1取消窗体内控件的跨线程检查(单个控件取消也可以)    

        public Form1()
{
InitializeComponent();
CheckForIllegalCrossThreadCalls = false;//干掉检测 不再检测跨线程
}

2.1.2新建线程实现跨线程访问

        /// <summary>
/// 新建线程并执行
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button1_Click(object sender, EventArgs e)
{
ThreadStart thStart = new ThreadStart(Pro);//threadStart委托
Thread thread = new Thread(thStart);
thread.Priority = ThreadPriority.Highest;
thread.IsBackground = true; //关闭窗体继续执行
thread.Start(); } public void Pro()
{ for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{
textBox1.Text = j.ToString();
}
}
}

2.2:主线程中操作(推荐)

    2.2.1 不用取消跨线程访问

        /// <summary>
/// 新建线程并执行
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button1_Click(object sender, EventArgs e)
{
ThreadStart thStart = new ThreadStart(Pro);//threadStart委托
Thread thread = new Thread(thStart);
thread.Priority = ThreadPriority.Highest;
thread.IsBackground = true; //关闭窗体继续执行
thread.Start();
} public void Pro()
{
for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{
if (textBox1.InvokeRequired)//不同线程访问了
textBox1.Invoke(new Action<TextBox, string>(SetTxtValue), textBox1, j.ToString());//跨线程了
else//同线程直接赋值
textBox1.Text = j.ToString();
}
}
} private void SetTxtValue(TextBox txt, string value)
{
txt.Text = value;
}

注:多个线程同时访问一个方法时 需要锁定

        public static readonly object obj = new object();
public void Pro()
{
//lock(obj){}=Monitor.Enter(obj) Monitor.Exit(obj)
lock (obj)
{
for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{
if (textBox1.InvokeRequired)//不同线程访问了
textBox1.Invoke(new Action<TextBox, string>(SetTxtValue),
                      textBox1, j.ToString());//跨线程了
else//同线程直接赋值
textBox1.Text = j.ToString();
}
}
}
}

3.窗体与自定义类之间的多线程操作(适用数据量大查询速度慢时 让数据在新线程中查询 防主线程卡死)

3.1自定义类中定义事件,并定义各事件执行的步骤方法

3.2窗体中实现类,并生成类各事件,在多线程中执行自定义类的步骤方法

3.3 代码3.1

public class WeiTuo
{
public int count { get; set; }
public event EventHandler StartEvent;
public event EventHandler MidEvent;
public event EventHandler EndEvent;
public event EventHandler EEvent; public void ExecEvent()
{
try
{ using (SqlConnection con = new SqlConnection("server=.;uid=sa;pwd=123;database=oa"))
{
using (SqlDataAdapter adp = new SqlDataAdapter("select * from a", con))
{ DataTable dt = new DataTable();
adp.Fill(dt); StartEvent(dt.Rows.Count, null);
for (int i = ; i < dt.Rows.Count; i++)
{
a ai = new a() {
ID = (int)dt.Rows[i]["id"],
Code = dt.Rows[i]["cCode"].ToString()
} ;
MidEvent(ai, null);
}
EndEvent(dt.Rows.Count, null); }
}
}
catch (Exception e)
{
EEvent(e.Message, null); }
}
} public class a
{
public int ID { get; set; }
public String Code { get; set; } = "";
}

3.4 代码3.2

private void button6_Click(object sender, EventArgs e)
{ WeiTuo wt = new WeiTuo();
wt.StartEvent += Wt_StartEvent;
wt.MidEvent += Wt_MidEvent;
wt.EndEvent += Wt_EndEvent;
wt.EEvent += Wt_EEvent; Thread th = new Thread(wt.ExecEvent);
th.Start(); } //特殊委托 action
private void Wt_StartEvent(object sender, EventArgs e)
{
Action<string> setLVItem = (s) => { listView1.Items.Add(s); };
this.Invoke(setLVItem, sender.ToString()); }
//特殊委托 action
private void Wt_MidEvent(object sender, EventArgs e)
{
Action<string> setLVItem = (s) => { listView1.Items.Add(s); };
this.Invoke(setLVItem, ((a)sender).Code); } #region 委托实现
private void Wt_EndEvent(object sender, EventArgs e)
{
this.Invoke(new setLVItem(setEndValue), "成功");
} delegate void setLVItem(string s); void setEndValue(string s)
{
MessageBox.Show(s);
}
#endregion private void Wt_EEvent(object sender, EventArgs e)
{
Action<string> ShowBox = (s) => { MessageBox.Show(s); };
this.Invoke(ShowBox, sender.ToString()); }

3.5 结果如下图

task任务与thread大同小异

使用时 尽量少的让控件跨线程 可通过ref 或 out 对参数传出 检测线程结束 再给控件赋值

也可用task的wait方法

4 异步回调

 private void button1_Click(object sender, EventArgs e)
{ Func<int, int, int> Sum = (i, j) =>
{
Thread.Sleep();
return i + j;
}; listView1.Items.Add("开始"); IAsyncResult iar = Sum.BeginInvoke(, , CallbackWhenDone, "我是测试"); listView1.Items.Add("over"); } private void CallbackWhenDone(IAsyncResult iar)
{
AsyncResult ar = (AsyncResult)iar;
Func<int, int, int> f = (Func<int, int, int>)ar.AsyncDelegate; Action<ListView> a = (lv) =>
{
lv.Items.Add(ar.AsyncState.ToString());
lv.Items.Add(f.EndInvoke(iar).ToString());
}; this.Invoke( a,listView1);
}

c# Winform 多线程操作的更多相关文章

  1. C# winform编程中多线程操作控件方法

    private void Form1_Load(object sender, EventArgs e) { Thread newthread = new Thread(new ThreadStart( ...

  2. 浅述WinForm多线程编程与Control.Invoke的应用

    VS2008.C#3.0在WinForm开发中,我们通常不希望当窗体上点了某个按钮执行某个业务的时候,窗体就被卡死了,直到该业务执行完毕后才缓过来.一个最直接的方法便是使用多线程.多线程编程的方式在W ...

  3. C#多线程操作界面控件的解决方案(转)

    C#中利用委托实现多线程跨线程操作 - 张小鱼 2010-10-22 08:38 在使用VS2005的时候,如果你从非创建这个控件的线程中访问这个控件或者操作这个控件的话就会抛出这个异常.这是微软为了 ...

  4. WinForm多线程学习文档

    基础篇 怎样创建一个线程 受托管的线程与 Windows线程 前台线程与后台线程 名为BeginXXX和EndXXX的方法是做什么用的 异步和多线程有什么关联 WinForm多线程编程篇 我的多线程W ...

  5. WinForm多线程实现HTTP网络检测工具

    一.背景描述与课程介绍 明人不说暗话,跟着阿笨一起玩WinForm.本次分享课程属于<C#高级编程实战技能开发宝典课程系列>中的一部分,阿笨后续会计划将实际项目中的一些比较实用的关于C#高 ...

  6. WinForm多线程编程与Control.Invoke的应用浅谈

    在WinForm开发中,我们通常不希望当窗体上点了某个按钮执行某个业务的时候,窗体就被卡死了,直到该业务执行完毕后才缓过来.一个最直接的方法便是使用多线程.多线程编程的方式在WinForm开发中必不可 ...

  7. 【转】浅述WinForm多线程编程与Control.Invoke的应用

    环境:VS2008.C#3.0 在WinForm开发中,我们通常不希望当窗体上点了某个按钮执行某个业务的时候,窗体就被卡死了,直到该业务执行完毕后才缓过来.一个最直接的方法便是使用多线程.多线程编程的 ...

  8. C#中级-常用多线程操作(持续更新)

    一.前言       多线程操作一直是编程的常用操作,掌握好基本的操作可以让程序运行的更加有效.本文不求大而全,只是将我自己工作中常常用到的多线程操作做个分类和总结.平时记性不好的时候还能看看.本文参 ...

  9. java多线程操作

    进程是程序的一次动态的执行过程,它经历了从代码加载.执行完毕的一个完整过程,这个过程也是进程本身从产生.发展到最终消亡的过程. 多线程是实现并发机制的一种有效的手段.进程和线程一样,都是实现并发的一个 ...

随机推荐

  1. ELK问题处理

    1.Logstash收集tomcat日志时报错warn: log4j:WARN No appenders could be found for logger (org.apache.http.clie ...

  2. jQuery.inArray()方法

    $.inArray() 函数用于在数组中查找指定值,并返回它的索引值(如果没有找到,则返回-1) 提示:源数组不会受到影响,过滤结果只反映在返回的结果数组中. 语法 $.inArray( value, ...

  3. Rust <2>:函数、方法与注释的格式

    rust 函数定义格式如下: fn function_name(a: i64, b: u32, c: bool) -> (d: f64, e: &str) { ... (1, " ...

  4. Java.io包

    Java.io.BufferedInputStream 类添加功能到另一个输入流,缓冲输入以及支持mark和reset methods.Following是关于缓冲输入流的要点: 当创建缓冲输入,创建 ...

  5. js基本函数和基本方法

    日期时间函数(需要用变量调用): var b = new Date(); //获取当前时间 b.getTime() //获取时间戳 b.getFullYear() //获取年份 b.getMonth( ...

  6. 从上一个页面跳入新页面时,如何拿URL中的参数

    var url = document.URL; //获取当前页面的url var urlA = url.split('?');//以url中的问号进行分割; var goodscode = urlA[ ...

  7. XML DTD详解

    XML DTD详解 一个有效的XML文档必然是结构正规的,结构正规的XML文档不一定是有效的,即有效的是格式正规的一个子集. 本文详细介绍DTD,包括其对元素的定义,属性的定义,以及实体的定义. 元素 ...

  8. CentOS 7 配置SFTP

    目前越来越多的FTP客户端软件开始支持SSH协议上传和下载文件,这种协议方式就是SFTP. SFTP的优势主要有两点,一是不需要再配置个FTP服务端:二是SSH协议是安全传输,上传和下载是经过加密的. ...

  9. 二、hibernate的常用API

    hibernate的调用过程 public class demo01 { @Test public void test(){ // 1.加载hibernate核心配置文件 Configuration ...

  10. BiLSTM介绍

    https://www.jiqizhixin.com/articles/2018-10-24-13 (机器之心)