C#线程:BeginInvoke和EndInvoke方法

2024-01-09 08:18

本文主要是介绍C#线程:BeginInvoke和EndInvoke方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

开发语言:C#3.0

IDE:Visual Studio 2008

一、C#线程概述

在操作系统中一个进程至少要包含一个线程,然后,在某些时候需要在同一个进程中同时执行多项任务,或是为了提供程序的性能,将要执行的任务分解成多个子任务执行。这就需要在同一个进程中开启多个线程。我们使用C#编写一个应用程序(控制台或桌面程序都可以),然后运行这个程序,并打开windows任务管理器,这时我们就会看到这个应用程序中所含有的线程数,如下图所示。

应用程序中所含有的线程数 

如果任务管理器没有“线程数”列,可以【查看】>【选择列】来显示“线程计数”列。从上图可以看出,几乎所有的进程都拥有两个以上的线程。从而可以看出,线程是提供应用程序性能的重要手段之一,尤其在多核CPU的机器上尤为明显。

二、用委托(Delegate)的BeginInvoke和EndInvoke方法操作线程

在C#中使用线程的方法很多,使用委托的BeginInvoke和EndInvoke方法就是其中之一。BeginInvoke方法可以使用线程异步地执行委托所指向的方法。然后通过EndInvoke方法获得方法的返回值(EndInvoke方法的返回值就是被调用方法的返回值),或是确定方法已经被成功调用。我们可以通过四种方法从EndInvoke方法来获得返回值。

三、直接使用EndInvoke方法来获得返回值

当使用BeginInvoke异步调用方法时,如果方法未执行完,EndInvoke方法就会一直阻塞,直到被调用的方法执行完毕。如下面的代码所示:

 
  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.Linq;  
  4. using System.Text;  
  5. using System.Threading;  
  6.  
  7. namespace MyThread  
  8. {  
  9.     class Program  
  10.     {  
  11.         private static int newTask(int ms)  
  12.         {  
  13.             Console.WriteLine("任务开始");  
  14.             Thread.Sleep(ms);  
  15.             Random random = new Random();  
  16.             int n = random.Next(10000);  
  17.             Console.WriteLine("任务完成");  
  18.             return n;  
  19.         }  
  20.  
  21.         private delegate int NewTaskDelegate(int ms);  
  22.                
  23.           
  24.         static void Main(string[] args)  
  25.         {  
  26.             NewTaskDelegate task = newTask;  
  27.             IAsyncResult asyncResult = task.BeginInvoke(2000, nullnull);  
  28.  
  29.             // EndInvoke方法将被阻塞2秒  
  30.             int result = task.EndInvoke(asyncResult);             
  31.             Console.WriteLine(result);  
  32.         }  
  33.     }  
  34. }  
  35.  

在运行上面的程序后,由于newTask方法通过Sleep延迟了2秒,因此,程序直到2秒后才输出最终结果(一个随机整数)。如果不调用EndInvoke方法,程序会立即退出,这是由于使用BeginInvoke创建的线程都是后台线程,这种线程一但所有的前台线程都退出后(其中主线程就是一个前台线程),不管后台线程是否执行完毕,都会结束线程,并退出程序。关于前台和后台线程的详细内容,将在后面的部分讲解。

读者可以使用上面的程序做以下实验。首先在Main方法的开始部分加入如下代码:

 
  1. Thread.Sleep(10000); 

以使Main方法延迟10秒钟再执行下面的代码,然后按Ctrl+F5运行程序,并打开企业管理器,观察当前程序的线程数,假设线程数是4,在10秒后,线程数会增至5,这是因为调用BeginInvoke方法时会建立一个线程来异步执行newTask方法,因此,线程会增加一个。

四、使用IAsyncResult asyncResult属性来判断异步调用是否完成

虽然上面的方法可以很好地实现异步调用,但是当调用EndInvoke方法获得调用结果时,整个程序就象死了一样,这样做用户的感觉并不会太好,因此,我们可以使用asyncResult来判断异步调用是否完成,并显示一些提示信息。这样做可以增加用户体验。代码如下:

 
  1. static void Main(string[] args)  
  2. {  
  3.     NewTaskDelegate task = newTask;  
  4.     IAsyncResult asyncResult = task.BeginInvoke(2000, nullnull);  
  5.    
  6.     while (!asyncResult.IsCompleted)  
  7.     {  
  8.         Console.Write("*");  
  9.         Thread.Sleep(100);  
  10.     }  
  11.     // 由于异步调用已经完成,因此, EndInvoke会立刻返回结果  
  12.     int result = task.EndInvoke(asyncResult);             
  13.     Console.WriteLine(result);  
  14. }  
  15.  

上面代码的执行结果如下图所示。

执行结果 

由于是异步,所以“*”可能会在“任务开始”前输出,如上图所示。

五、使用WaitOne方法等待异步方法执行完成

使用WaitOne方法是另外一种判断异步调用是否完成的方法。代码如下:

 
  1. static void Main(string[] args)  
  2. {  
  3.     NewTaskDelegate task = newTask;  
  4.     IAsyncResult asyncResult = task.BeginInvoke(2000, nullnull);  
  5.  
  6.     while (!asyncResult.AsyncWaitHandle.WaitOne(100, false))  
  7.     {  
  8.          Console.Write("*");                
  9.     }  
  10.  
  11.     int result = task.EndInvoke(asyncResult);  
  12.     Console.WriteLine(result);  
  13. }  
  14.  

WaitOne的第一个参数表示要等待的毫秒数,在指定时间之内,WaitOne方法将一直等待,直到异步调用完成,并发出通知,WaitOne方法才返回true。当等待指定时间之后,异步调用仍未完成,WaitOne方法返回false,如果指定时间为0,表示不等待,如果为-1,表示永远等待,直到异步调用完成。

六、使用回调方式返回结果

上面介绍的几种方法实际上只相当于一种方法。这些方法虽然可以成功返回结果,也可以给用户一些提示,但在这个过程中,整个程序就象死了一样(如果读者在GUI程序中使用这些方法就会非常明显),要想在调用的过程中,程序仍然可以正常做其它的工作,就必须使用异步调用的方式。下面我们使用GUI程序来编写一个例子,代码如下:

 
  1. private delegate int MyMethod();  
  2. private int method()  
  3. {  
  4.     Thread.Sleep(10000);  
  5.     return 100;  
  6. }  
  7. private void MethodCompleted(IAsyncResult asyncResult)  
  8. {  
  9.     if (asyncResult == nullreturn;  
  10.     textBox1.Text = (asyncResult.AsyncState as   
  11.     MyMethod).EndInvoke(asyncResult).ToString();  
  12. }  
  13.  
  14. private void button1_Click(object sender, EventArgs e)  
  15. {  
  16.  
  17.     MyMethod my = method;  
  18.     IAsyncResult asyncResult = my.BeginInvoke(MethodCompleted, my);  
  19. }  
  20.  

要注意的是,这里使用了BeginInvoke方法的最后两个参数(如果被调用的方法含有参数的话,这些参数将作为BeginInvoke的前面一部分参数,如果没有参数,BeginInvoke就只有两个参数了)。第一个参数是回调方法委托类型,这个委托只有一个参数,就是IAsyncResult,如MethodCompleted方法所示。当method方法执行完后,系统会自动调用MethodCompleted方法。BeginInvoke的第二个参数需要向MethodCompleted方法中传递一些值,一般可以传递被调用方法的委托,如上面代码中的my。这个值可以使用IAsyncResult.AsyncState属性获得。

由于上面的代码通过异步的方式访问的form上的一个textbox,因此,需要按ctrl+f5运行程序(不能直接按F5运行程序,否则无法在其他线程中访问这个textbox,关于如果在其他线程中访问GUI组件,并在后面的部分详细介绍)。并在form上放一些其他的可视控件,然在点击button1后,其它的控件仍然可以使用,就象什么事都没有发生过一样,在10秒后,在textbox1中将输出100。

七、其他组件的BeginXXX和EndXXX方法

在其他的.net组件中也有类似BeginInvoke和EndInvoke的方法,如System.Net.HttpWebRequest类的BeginGetResponse和EndGetResponse方法,下面是使用这两个方法的一个例子:

 
  1. private void requestCompleted(IAsyncResult asyncResult)  
  2. {  
  3.     if (asyncResult == nullreturn;  
  4.     System.Net.HttpWebRequest hwr = asyncResult.AsyncState as System.Net.HttpWebRequest;  
  5.     System.Net.HttpWebResponse response =   
  6. (System.Net.HttpWebResponse)hwr.EndGetResponse(asyncResult);  
  7.     System.IO.StreamReader sr = new   
  8. System.IO.StreamReader(response.GetResponseStream());  
  9.     textBox1.Text = sr.ReadToEnd();  
  10. }  
  11. private delegate System.Net.HttpWebResponse RequestDelegate(System.Net.HttpWebRequest request);  
  12.  
  13. private void button1_Click(object sender, EventArgs e)  
  14. {  
  15.     System.Net.HttpWebRequest request =   
  16.     (System.Net.HttpWebRequest)System.Net.WebRequest.Create("http://www.cnblogs.com");  
  17.     IAsyncResult asyncResult =request.BeginGetResponse(requestCompleted, request);       
  18. }  
  19.  

以上介绍的就是C#线程中BeginInvoke和EndInvoke方法。

【编辑推荐】

  1. C#自定义特性介绍
  2. C#内置特性介绍
  3. 如何进行C#异常类的自定义
  4. C#编程技巧七条
  5. 预测C#与.NET的发展趋势
【责任编辑: 王苑 TEL:(010)68476606】

这篇关于C#线程:BeginInvoke和EndInvoke方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/586478

相关文章

Windows 上如果忘记了 MySQL 密码 重置密码的两种方法

《Windows上如果忘记了MySQL密码重置密码的两种方法》:本文主要介绍Windows上如果忘记了MySQL密码重置密码的两种方法,本文通过两种方法结合实例代码给大家介绍的非常详细,感... 目录方法 1:以跳过权限验证模式启动 mysql 并重置密码方法 2:使用 my.ini 文件的临时配置在 Wi

MySQL重复数据处理的七种高效方法

《MySQL重复数据处理的七种高效方法》你是不是也曾遇到过这样的烦恼:明明系统测试时一切正常,上线后却频频出现重复数据,大批量导数据时,总有那么几条不听话的记录导致整个事务莫名回滚,今天,我就跟大家分... 目录1. 重复数据插入问题分析1.1 问题本质1.2 常见场景图2. 基础解决方案:使用异常捕获3.

最详细安装 PostgreSQL方法及常见问题解决

《最详细安装PostgreSQL方法及常见问题解决》:本文主要介绍最详细安装PostgreSQL方法及常见问题解决,介绍了在Windows系统上安装PostgreSQL及Linux系统上安装Po... 目录一、在 Windows 系统上安装 PostgreSQL1. 下载 PostgreSQL 安装包2.

SQL中redo log 刷⼊磁盘的常见方法

《SQL中redolog刷⼊磁盘的常见方法》本文主要介绍了SQL中redolog刷⼊磁盘的常见方法,将redolog刷入磁盘的方法确保了数据的持久性和一致性,下面就来具体介绍一下,感兴趣的可以了解... 目录Redo Log 刷入磁盘的方法Redo Log 刷入磁盘的过程代码示例(伪代码)在数据库系统中,r

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

Java中Switch Case多个条件处理方法举例

《Java中SwitchCase多个条件处理方法举例》Java中switch语句用于根据变量值执行不同代码块,适用于多个条件的处理,:本文主要介绍Java中SwitchCase多个条件处理的相... 目录前言基本语法处理多个条件示例1:合并相同代码的多个case示例2:通过字符串合并多个case进阶用法使用

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

C# Where 泛型约束的实现

《C#Where泛型约束的实现》本文主要介绍了C#Where泛型约束的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录使用的对象约束分类where T : structwhere T : classwhere T : ne

Python中__init__方法使用的深度解析

《Python中__init__方法使用的深度解析》在Python的面向对象编程(OOP)体系中,__init__方法如同建造房屋时的奠基仪式——它定义了对象诞生时的初始状态,下面我们就来深入了解下_... 目录一、__init__的基因图谱二、初始化过程的魔法时刻继承链中的初始化顺序self参数的奥秘默认