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

方法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. 泛微weaver_oa filebrowser.jsp 任意目录遍历

    url//document/imp/filebrowser.jsp?dir=/etc/

  2. Rust <9>:启用第三方库的备选 features

  3. jwt token校验获取用户数据

    /** * 用户登录成功之后,获取用户信息 * 1.获取用户id * 2.根据id查询用户 * 3.构建返回值对象 * 4.返回数据 * @return */@RequestMapping(value ...

  4. log4j日志记录到数据库

    log4j API提供 org.apache.log4j.jdbc.JDBCAppender 对象,它能够将日志信息在指定的数据库. JDBCAppender 配置: Property 描述 buff ...

  5. HBase优化——读写优化

    Hbase2.0查询优化 1)设置scan缓存 HBase中Scan查询可以设置缓存,方法是setCaching(),这样可以有效的减少服务端与客户端的交互,更有效的提升扫描查询的性能. Scan s ...

  6. 专业解决 MySQL 查询速度慢与性能差!

    Java技术栈 ,一般把连接数设置得大一些). 并发量:同一时刻数据库服务器处理的请求数量 3.超高的 CPU使用率:CPU资源耗尽出现宕机. 4.磁盘 IO:磁盘 IO性能突然下降.大量消耗磁盘性能 ...

  7. mysql架构精选

    ◆主从架构1.安装服务(主从) yum -y install mysql* /etc/init.d/mysqld start2.修改配置文件:/etc/my.conf(主从) vi /etc/my.c ...

  8. iView的page 组件

    //html <div class="pageNation"> <Page :total= totalPages :page-size= pageSize siz ...

  9. aiohttp上报405: Method Not Allowed

    请求方式不对,修改为“POST”或者“GET” 可参考:https://blog.csdn.net/yiifaa/article/details/80928487

  10. Red5文件结构简介

    Red5文件结构简介 Red5 是 支持Windows,Linux等多平台的RTMP流媒体服务器,最早属于谷歌下的开源项目,先已移植到Github,地址为https://github.com/Red5 ...