2012-06-29 23:00 1276人阅读 评论(0) 收藏 举报
 分类:
C/C++/C#/dotnet(126) 
 

目录(?)[+]

 

在应届生找工作的时候,多线程操作几乎是所有的公司都会问及的一个基本问题。

这里做了一个多线程操作的总结,这里总结了通过异步委托来实现多线程操作。

定义一个委托,是创建一个线程的最简单的方法,并且异步调用它。委托是方法的类型安全的引用。同时委托还智齿异步调用方法。

委托使用线程池来完成异步任务。

当自己的程序使用异步委托的时候,委托会自动创建ige执行线程的任务。委托使用线程池完成异步任务,所有的异步委托调用,都会通过调用系统线程池中的线程来完成调用异步任务。

在下面的简单例子中,我们定义了一个异步委托,并在每次输出的时候显示该函数运行在哪个线程中。

在异步委托中,可以使用三种技术来异步的调用委托。下面分别介绍三种调用异步委托的方法。

1. 投票判断异步委托是否完成

在委托中调用BeginInvoke()方法,返回IAsyncResult结果。程序的源代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading;
  6. namespace AsyncDelegate
  7. {
  8. class Program
  9. {
  10. public delegate int TakeSomeTimeDelegate(int data,int ms);
  11. static void Main(string[] args)
  12. {
  13. TakeSomeTimeDelegate dl=TakeSomeTime;
  14. IAsyncResult ar=dl.BeginInvoke(1,200,null,null);
  15. while (!ar.IsCompleted)
  16. {
  17. Console.WriteLine(".");
  18. Console.WriteLine("Run in thread:" + Thread.CurrentThread.ManagedThreadId);
  19. Thread.Sleep(50);
  20. }
  21. int result = dl.EndInvoke(ar);
  22. Console.WriteLine("Result:{0}", result);
  23. }
  24. static int TakeSomeTime(int data, int ms)
  25. {
  26. Console.WriteLine("TakeSomeTime started!");
  27. Console.WriteLine("Run in thread:"+Thread.CurrentThread.ManagedThreadId);
  28. Thread.Sleep(ms);
  29. Console.WriteLine("TakeSomeTime Completed!");
  30. return ++data;
  31. }
  32. }
  33. }

该程序的输出结果如图:

可以看到主线程和异步委托线程是同时执行的。

如果在委托结束之前不等待委托完成其他任务就结束主线程,委托线程就会停止。

int result = dl.EndInvoke(ar); 这里的EndInvoke()函数会一直等在异步委托完成并在异步委托完成之前阻断主线程。这样就可以通过这个函数保证异步委托能够正确完成。

2. 等待句柄判断异步委托完成

通过AsyncWatiHandle属性,访问等待句柄。WaitOne()方法阻断当前线程,直到异步调用线程完成返回可以利用的句柄以后再执行当前线程。

程序:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading;
  6. namespace AsyncDelegate
  7. {
  8. class Program
  9. {
  10. public delegate int TakeSomeTimeDelegate(int data,int ms);
  11. static void Main(string[] args)
  12. {
  13. TakeSomeTimeDelegate dl=TakeSomeTime;
  14. IAsyncResult ar=dl.BeginInvoke(1,200,null,null);
  15. while (true)
  16. {
  17. Console.WriteLine(".");
  18. Console.WriteLine("Run in thread:" + Thread.CurrentThread.ManagedThreadId);
  19. if (ar.AsyncWaitHandle.WaitOne(100, false))
  20. {
  21. Console.WriteLine("Can get the result now");
  22. break;
  23. }
  24. }
  25. //while (!ar.IsCompleted)
  26. //{
  27. //    Console.WriteLine(".");
  28. //    Console.WriteLine("Run in thread:" + Thread.CurrentThread.ManagedThreadId);
  29. //    Thread.Sleep(50);
  30. //}
  31. int result = dl.EndInvoke(ar);
  32. Console.WriteLine("Result:{0}", result);
  33. }
  34. static int TakeSomeTime(int data, int ms)
  35. {
  36. Console.WriteLine("TakeSomeTime started!");
  37. Console.WriteLine("Run in thread:"+Thread.CurrentThread.ManagedThreadId);
  38. Thread.Sleep(ms);
  39. Console.WriteLine("TakeSomeTime Completed!");
  40. return ++data;
  41. }
  42. }
  43. }

运行结果:

ar.AsyncWaitHandle.WaitOne()阻断了当前线程, 直到异步调用线程完成获得可以利用的句柄以后再次执行当前线程。

3. 利用异步回调函数判断异步调用线程是否结束

回调函数的操作比较复杂, 而且对于程序的理解和维护造成非常大的困难。所以一般情况下能不用回调函数就不要使用回调函数。

使用回调函数,必须注意这个函数从委托线程中调用,而不是从主线程中调用回调函数。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading;
  6. namespace AsyncDelegate
  7. {
  8. class Program
  9. {
  10. public delegate int TakeSomeTimeDelegate(int data,int ms);
  11. static void Main(string[] args)
  12. {
  13. TakeSomeTimeDelegate dl=TakeSomeTime;
  14. IAsyncResult ar = dl.BeginInvoke(1, 200, TakeSomeTimeCompleted, dl);
  15. for (int i = 0; i < 10;i++ )
  16. {
  17. Console.WriteLine(".");
  18. Console.WriteLine("Run in thread:" + Thread.CurrentThread.ManagedThreadId);
  19. Thread.Sleep(50);
  20. }
  21. //int result = dl.EndInvoke(ar);
  22. //Console.WriteLine("Result:{0}", result);
  23. }
  24. static int TakeSomeTime(int data, int ms)
  25. {
  26. Console.WriteLine("TakeSomeTime started!");
  27. Console.WriteLine("Run in thread:"+Thread.CurrentThread.ManagedThreadId);
  28. Thread.Sleep(ms);
  29. Console.WriteLine("TakeSomeTime Completed!");
  30. return ++data;
  31. }
  32. static void TakeSomeTimeCompleted(IAsyncResult ar)
  33. {
  34. if (ar==null)
  35. {
  36. throw new ArgumentNullException("ar");
  37. }
  38. TakeSomeTimeDelegate dl = ar.AsyncState as TakeSomeTimeDelegate;
  39. int result = dl.EndInvoke(ar);
  40. Console.WriteLine("result : {0}", result);
  41. //Console.WriteLine("Run in thread:" + Thread.CurrentThread.ManagedThreadId);
  42. }
  43. }
  44. }

运行结果:

 
 

C# 多线程操作之异步委托的更多相关文章

  1. 6.26学习 异步委托回调函数 VS 多线程 VS 并行处理

    描述: 我现在是轮询着构建实例,然后这个实例去执行一个方法,但是执行方法需要大约10s时间,全部轮询下来需要很长时间.所以我现在要更改,头给了我两个方法,1多线程 2异步委托回调函数. 异步委托回调函 ...

  2. 异步委托(APM)使用Func异步操作,处理耗时操作

    使用委托进行异步操作,处理一些耗时操作,防止主线程阻塞 使用例子: using System; using System.Collections.Generic; using System.Linq; ...

  3. C#实现异步编程的两个简单机制(异步委托&定时器)及Thread实现多线程

    创建线程的常用方法:异步委托.定时器.Thread类 理解程序.进程.线程三者之间的区别:简而言之,一个程序至少有一个进程,一个进程至少有一个线程进程就是在内存中运行的程序(即运行着的程序):一个进程 ...

  4. 异步委托 多线程实现摇奖器 winform版

    using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using Sy ...

  5. c# 委托访问listbox多线程操作

    c# 委托访问listbox多线程操作 using System;using System.Collections.Generic;using System.ComponentModel;using ...

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

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

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

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

  8. 通俗易懂,C#如何安全、高效地玩转任何种类的内存之Span的脾气秉性(二)。 异步委托 微信小程序支付证书及SSL证书使用 SqlServer无备份下误删数据恢复 把list集合的内容写入到Xml中,通过XmlDocument方式写入Xml文件中 通过XDocument方式把List写入Xml文件

    通俗易懂,C#如何安全.高效地玩转任何种类的内存之Span的脾气秉性(二).   前言 读完上篇<通俗易懂,C#如何安全.高效地玩转任何种类的内存之Span的本质(一).>,相信大家对sp ...

  9. c# Winform 多线程操作

    主要是对一个过程需要的时间很长执行时会出现界面假死的情况 方法1: Application.DoEvents(),这种方法当你拖动窗体时,界面不会假死.但在你拖动时代码不再执行,也就是阻塞了,当你不再 ...

随机推荐

  1. 【默默努力】vue-pc-app

    最近在github上面看到了一个团队的项目,真的非常赞.他们进行vue-cli的二次开发,将项目用自己的方式打包. 今天的这个开源项目地址为:https://github.com/tffe-team/ ...

  2. SpringMVC参数绑定(未完待续)

    1. Strut2与SpringMVC接收请求参数的区别 Struts2通过action类的成员变量接收SpringMVC通过controller方法的形参接收 2. SpringMVC参数绑定流程 ...

  3. springBoot_JPA和lombok

    一.JPA 特点:JPA可以根据方法名自动匹配sql语句. JPA是一个标准 Hibernate是JPA的一个实现,宗旨是尽量少写sql语句. 使用JPA 1.application.properti ...

  4. [洛谷P2472] [SCOI2007]蜥蜴

    题目链接: 蜥蜴 题目分析: 一道网络流,先来分析一下问题: 在一个\(r*c\)的图中分布了一些数,其他地方都用\(0\)填充,我们分别从指定的一些数出发,每次可以移动到周围距离为\(d\)以内的数 ...

  5. Hibernate O/R 映射

    O/R 映射 目前为止我们已经通过应用 Hibernate 见识过十分基础的 O/R 映射了,但是还有三个更加重要的有关映射的话题需要我们更详细的探讨.这三个话题是集合的映射,实体类之间的关联映射以及 ...

  6. ArduinoUno和Leonardo的区别

    学习过Arduino的同学对ArduinoUno和Lenardo的不同点会有所了解,但说起具体的区别估计还是很多人答不上来,今天我们就详细解释下Arduino Uno和Leonardo的不同. 我们从 ...

  7. myeclipse启动jboss报ERROR [MainDeployer] Could not create deployment

    今天用myeclipse启动jboss时报了这样的错,花了我将近一天的时间去解决这个问题,开始以为jar包问题,到最后发现是配置错了,所以分享一下.具体报错信息如下图: ERROR [MainDepl ...

  8. 使用Java代码获取Java进程ID的方法

    需要jre/lib下的tools.jar包 public class Test { public static void main(String[] args) throws Exception {  ...

  9. MyBatis - Mapper动态代理开发

    Mapper接口开发方法编写Mapper接口(相当于Dao接口),由Mybatis框架根据接口定义创建接口的动态代理对象. Mapper接口开发方式是基于入门程序的基础上,对 控制程序 进行分层开发, ...

  10. elasticsearch 中文API 获得(三)

    获取API 获取API允许你通过id从索引中获取类型化的JSON文档,如下例: GetResponse response = client.prepareGet("twitter" ...