一个编译器所支持的异步委托的例子
发表于:2007-07-01来源:作者:点击数:
标签:
刚写的例子所以发发。。到得到代码。运行程序using System;using System.Drawing;using System.Collections;using System.ComponentModel;using System.Windows.Forms;using System.Data;using System.Threading;namespace AsyncExample{/// summary/// Form1
刚写的例子所以发发。。到得到代码。运行程序using System;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.Windows.Forms;
using System.Data;
using System.Threading;
namespace AsyncExample
{
/// <summary>
/// Form1 的摘要说明。
/// </summary>
public class Form1 : System.Windows.Forms.Form
{
private System.Windows.Forms.GroupBox groupBox1;
private System.Windows.Forms.TextBox textBox1;
private System.Windows.Forms.TextBox textBox2;
private System.Windows.Forms.Label label1;
private System.Windows.Forms.Label label2;
private System.Windows.Forms.Button button1;
private System.Windows.Forms.Button button2;
private System.ComponentModel.IContainer components;
private System.Windows.Forms.Button button3;
private Thread theThread;
public Form1()
{
//
// Windows 窗体设计器支持所必需的
//
InitializeComponent();
//
// TODO: 在 InitializeComponent 调用后添加任何构造函数代码
//
//把当前线程记录起来,以后比较一下就知道一个函数是否在其他线程中执行了。
theThread=Thread.CurrentThread;
}
/// <summary>
/// 清理所有正在使用的资源。
/// </summary>
protected override void Dispose( bool disposing )
{
if( disposing )
{
if (components != null)
{
components.Dispose();
}
}
base.Dispose( disposing );
}
#region Windows Form Designer generated code
/// <summary>
/// 设计器支持所需的方法 - 不要使用代码编辑器修改
/// 此方法的内容。
/// </summary>
private void InitializeComponent()
{
this.groupBox1 = new System.Windows.Forms.GroupBox();
this.textBox1 = new System.Windows.Forms.TextBox();
this.textBox2 = new System.Windows.Forms.TextBox();
this.label1 = new System.Windows.Forms.Label();
this.label2 = new System.Windows.Forms.Label();
this.button1 = new System.Windows.Forms.Button();
this.button2 = new System.Windows.Forms.Button();
this.button3 = new System.Windows.Forms.Button();
this.groupBox1.SuspendLayout();
this.SuspendLayout();
//
// groupBox1
//
this.groupBox1.Controls.AddRange(new System.Windows.Forms.Control[] {
this.button3,
this.button2,
this.label2,
this.label1,
this.textBox2,
this.textBox1,
this.button1});
this.groupBox1.Name = "groupBox1";
this.groupBox1.Size = new System.Drawing.Size(360, 128);
this.groupBox1.TabIndex = 0;
this.groupBox1.TabStop = false;
this.groupBox1.Text = "输入两个数字";
//
// textBox1
//
this.textBox1.Location = new System.Drawing.Point(56, 24);
this.textBox1.Name = "textBox1";
this.textBox1.TabIndex = 0;
this.textBox1.Text = "111";
//
// textBox2
//
this.textBox2.Location = new System.Drawing.Point(56, 56);
this.textBox2.Name = "textBox2";
this.textBox2.TabIndex = 1;
this.textBox2.Text = "345";
//
// label1
//
this.label1.Location = new System.Drawing.Point(8, 24);
this.label1.Name = "label1";
this.label1.Size = new System.Drawing.Size(40, 23);
this.label1.TabIndex = 2;
this.label1.Text = "数字1";
//
// label2
//
this.label2.Location = new System.Drawing.Point(8, 56);
this.label2.Name = "label2";
this.label2.Size = new System.Drawing.Size(40, 23);
this.label2.TabIndex = 3;
this.label2.Text = "数字2";
//
// button1
//
this.button1.Location = new System.Drawing.Point(192, 24);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(144, 23);
this.button1.TabIndex = 1;
this.button1.Text = "使用Callback";
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// button2
//
this.button2.Location = new System.Drawing.Point(192, 56);
this.button2.Name = "button2";
this.button2.Size = new System.Drawing.Size(144, 23);
this.button2.TabIndex = 4;
this.button2.Text = "使用EndInvoke";
this.button2.Click += new System.EventHandler(this.button2_Click);
//
// button3
//
this.button3.Location = new System.Drawing.Point(192, 88);
this.button3.Name = "button3";
this.button3.Size = new System.Drawing.Size(144, 23);
this.button3.TabIndex = 5;
this.button3.Text = "检测EndInvoke";
this.button3.Click += new System.EventHandler(this.button3_Click);
//
// Form1
//
this.AutoScaleBaseSize = new System.Drawing.Size(6, 14);
this.ClientSize = new System.Drawing.Size(360, 133);
this.Controls.AddRange(new System.Windows.Forms.Control[] {
this.groupBox1});
this.Name = "Form1";
this.Text = "Form1";
this.groupBox1.ResumeLayout(false);
this.ResumeLayout(false);
}
#endregion
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
Application.Run(new Form1());
}
private bool GetNumbers(out double x,out double y)
{
try
{
x=double.Parse(textBox1.Text);
y=double.Parse(textBox2.Text);
return true;
}
catch(Exception)
{
x=0;
y=0;
return false;
}
}
private delegate double 除的委托(double x,double y);
/// <summary>
/// 这个方法是今次要异步执行的方法
/// 因为它是作为From1的一个方法,所以能访问Form1的属性
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
/// <returns>a/b</returns>
private double 除(double a,double b)
{
//取得方法执行所在的线程
Thread thisThread=Thread.CurrentThread;
//在这个例子中,永远不成立。方法总是在其他线程中执行
if(thisThread==this.theThread)
throw(new Exception("if(thisThread==this.theThread)"));
MessageBox.Show(DateTime.Now.ToLongTimeString()+"\n按确定计算"+a+"/"+b+"的结果");
//如果b为0,那么就会抛出异常
//后面会有处理异步调用的异常的说明
return a/b;
}
#region CallBack调
用例子
/// <summary>
/// 使用CallBack来处理异步调用
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button1_Click(object sender, System.EventArgs e)
{
double a,b;
if(!GetNumbers(out a,out b))
{
MessageBox.Show("请输入正确的数字");
return;
}
//可以在这里重新"."一下,看看有没有BeginInvoke的方法显示在列表中?
//定义是没有的,所以IDE没有显示出来。
//不过编译器帮我们弄了这个方法,所以能编译通过
IAsyncResult iar=new 除的委托(除).BeginInvoke(a,b,new AsyncCallback(当除完成调用),"使用CallBack");
//关于BeginInvoke
//new 除的委托(除) 这句不用多说了。
//new 除的委托(除).BeginInvoke 是编译器自动生成的函数。
//(a,b,new AsyncCallback(当除完成调用),"hello:)") 中
//a,b 是和 除的委托 的定义是一致的
//然后后面要加 AsyncCallback callback,object AsyncContext
//看MSDN上对BeginInvoke的签名介绍。。
}
private void 当除完成调用(IAsyncResult iar)
{
//这个是对的。不过我找不到文档说明是不是始终是对的。
System.Runtime.Remoting.Messaging.AsyncResult ar=(System.Runtime.Remoting.Messaging.AsyncResult)iar;
//得到刚才的 new 除的委托(除)
除的委托 d=(除的委托)ar.AsyncDelegate;
double res;
Thread thisThread=Thread.CurrentThread;
string msg=thisThread==this.theThread?"\n这个线程和主线程是一致的":"\n这个线程和主线程是不同的";
try
{
//和刚才的BeginInvoke像对应
res=d.EndInvoke(iar);
//EndInvoke只需要放个IAsyncResult来做结束调用标志就够了。
//因为 除的委托 的定义,只有 In 参数,所以没有其他参数
//看MSDN上对BeginInvoke的签名介绍。。
MessageBox.Show("好了,,在 “除完成调用()”这个方法中,已经得到结果:\n"+res+msg);
}
catch(Exception x)
{
MessageBox.Show("好了,,在 “除完成调用()” 这个方法中检测到异常:\n"+x.Message+msg);
}
}
#endregion
#region 使用EndInvoke
/// <summary>
/// 这个是用来存放IAsyncResult的容器
/// </summary>
ArrayList al=new ArrayList();
/// <summary>
/// 使用EndInvoke来处理异步调用
/// 执行完这个,按确定来处理新弹出来的MessageBox
/// 然后按"检测EndInvoke"来看看结果。。
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button2_Click(object sender, System.EventArgs e)
{
double a,b;
if(!GetNumbers(out a,out b))
{
MessageBox.Show("请输入正确的数字");
return;
}
//这个和上面的调用的最大区别就在参数AsyncCallback为null
IAsyncResult iar=new 除的委托(除).BeginInvoke(a,b,null,"不使用Callback,直接使用iar.IsCompleted");
al.Add(iar);
}
//
测试看看有没有什么东东是已经完成的。。
private void button3_Click(object sender, System.EventArgs e)
{
string msgs="好了,,在 “button3_Click()检测EndInvoke”这个方法中:";
Thread thisThread=Thread.CurrentThread;
string msg=thisThread==this.theThread?"\n这个线程和主线程是一致的":"\n这个线程和主线程是不同的";
msgs+=msg+"\n\n";
int countMax=al.Count;
int count=0;
ArrayList alDelete=new ArrayList();
foreach(IAsyncResult iar in al)
{
//如果没有完成,看看下一个
if(!iar.IsCompleted)continue;
//好了。。这个iar代表的调用是完成了的。
count++;
//暂时不能修改al,所以只好这样
alDelete.Add(iar);
//和上面的用法一样。
System.Runtime.Remoting.Messaging.AsyncResult ar=(System.Runtime.Remoting.Messaging.AsyncResult)iar;
除的委托 d=(除的委托)ar.AsyncDelegate;
double res;
try
{
//同上
res=d.EndInvoke(iar);
msgs+=((string)iar.AsyncState)+"\n已经得到结果:\n"+res+"\n\n";
}
catch(Exception x)
{
msgs+=((string)iar.AsyncState)+"\n检测到异常:\n"+x.Message+"\n\n";
}
}
//从al中删除已经执行完毕的调用的IAsyncResult
foreach(IAsyncResult iar in alDelete)
{
al.Remove(iar);
}
if(countMax>0&&count==0)
msgs+="倒,,还没有几个完成的。。";
if(countMax==0)
msgs="没有任何的调用啊。。";
MessageBox.Show(msgs);
}
#endregion
}
}
原文转自:http://www.ltesting.net