C# 用委托BeginInvoke做异步线程

原创 2015年11月18日 17:48:03

一个应用场景,浏览器上传一个文件,此文件后台调用文件转换,需要耗费相当长的时间,这样,如果是一个线程同步式的做下去,那么用户在浏览器上感觉就是卡住了,卡卡卡卡,这里我们利用委托的BeginInvoke和EndInvoke方法操作线程,BeginInvoke方法可以使用线程异步地执行委托所指向的方法。然后通过EndInvoke方法获得方法的返回值(EndInvoke方法的返回值就是被调用方法的返回值),或是确定方法已经被成功调用,说白了就是相当于开个多线程,你用户文件保存了之后,响应返回,这个BeginInvoke异步去执行委托方法,完了之后呢,再执行你的异步回调函数;

 

大概步骤

1:先把你要异步执行的方法抽离出来;

2:定义一个该异步方法的委托;

3:在调用地方实例化这个委托;

4:调用此委托实例的BeginInvoke方法,在此方法里,前面填委托的参数,接着是委托方法结束后的回调函数;

5:写委托的回调函数,回调函数是固定的参数(IAsyncResultIR

在这个里面,可以获取用户定义的对象,它限定或包含关于异步操作的信息(AsyncState

然后调用EndInvoke,获取到委托方法结束的返回值。

6:调用自定义的回调函数;

 还是看代码吧;

   public class anysFileChange
    {

        /// <summary>
        /// 文件change后,其对应的业务逻辑所需要做的变动锁调用的委托
        /// 转换完之后 会调用所执行的函数,用户需要干什么,这个函数则交由用户自己的逻辑完成
        /// </summary>
        public static anysChangingHandlerCallBack _handler = null;


        /// <summary>
        /// 转换文件
        /// </summary>
        /// <param name="filepath">文件路径</param>  
        /// <returns></returns>
        public static void  ChangingFile(string filepath, string attachId)
        {
            //开启异步转换
            DEGAsyncChangingFile acf = new DEGAsyncChangingFile(anysFileChange.AsyncChangeFileToswf);
            acf.BeginInvoke(filepath, attachId, anysFileChange.CallBackAsync, acf);            
        }
        
        /// <summary>
        /// 异步函数执行完后的的回调函数
        /// </summary>
        /// <param name="IR">异步结果 </param>
        /// <returns></returns>
        private static void CallBackAsync(IAsyncResult IR)
        {
            DEGAsyncChangingFile acf = (DEGAsyncChangingFile)IR.AsyncState;
            ResultObj result = (ResultObj)acf.EndInvoke(IR);
            //如果调用 文件转换结束方法有定义 转换结束需要做的委托那么执行用户定义的委托函数
            if (_handler != null && result != null)
            {
                ///成功才执行客户自定义的回调函数
                if (result.changestaus == EnumChangeStatus.SUCCESS)
                {
                    _handler(result.filepath, result.attachId);
                }
            }
        }

        /// <summary>
        /// 转换文件
        /// </summary>
        /// <param name="filepath">文件路径</param>  
        /// <returns>HSUFResultObj 返回结果对象</returns>
        private static ResultObj AsyncChangeFileToswf(string filepath,string attachId)
        {
            ResultObj res = new ResultObj();
            //转换动作
            ConvertFile cf = new ConvertFile();
            //cf这个类提供了一个写日志的事件 注册一个写日志事件
            cf.ConvertLog += new ConvertFile.ConvertLogHandler(Syslog);
            //这里的cf这个类提供的方法,仅仅利用里面的转换方法算了           
             cf.Convert(filepath);         
             if (cf.ChangeResult == EnumChangeStatus.SUCCESS)
             {
                 res.attachId = attachId;                               
                 res.filepath = filepath;
                 res.changestaus = cf.ChangeResult;
             }
             else
             {
                 res.changeMessage = cf.convertMessage;
                 res.changestaus = cf.ChangeResult;
             }
            return res;            

        }
        /// <summary>
        /// 供注册的日志事件
        /// </summary>
        /// <param name="messages"></param>
        private static void Syslog(string messages)
        {
            string separateLine = "\r\n============================================================================================\r\n";
            string log = string.Format(@"{0} {1} {2} {0}", separateLine, DateTime.Now.ToString(), messages);
            // 如果上传课程文件夹不存在,则创建
            if (!Directory.Exists(ConverConst.toolpath + "Log\\"))
                Directory.CreateDirectory(ConverConst.toolpath + "Log\\");
            StreamWriter sw = new StreamWriter(ConverConst.toolpath+"Log\\"+DateTime.Now.ToString("yyyy-MM-dd")+".txt", true);
            sw.WriteLine(log);
            sw.Close();
        }
        /// <summary>
        /// 文件转换委托
        /// </summary>
        /// <param name="filepath">文件路径</param>  
        /// <returns> </returns>
        private delegate object DEGAsyncChangingFile(string filepath,string attachId);

    }

    /// <summary>
    /// 结果集对象
    /// </summary>
    public class ResultObj
    {
        /// <summary>
        /// 文件路径
        /// </summary>
        public string filepath;
        /// <summary>
        /// id
        /// </summary>
        public string attachId;
        /// <summary>
        /// 转换结果消息 
        /// </summary>
        public string changeMessage;
        /// <summary>
        /// 转换状态
        /// </summary>
        public EnumChangeStatus changestaus;
    }

    /// <summary>
    ///  文件转换后,业务逻辑所需要调用的委托
    /// </summary>
    /// <param name="filepath">文件路径</param>  
    public delegate void anysChangingHandlerCallBack(string filepath,string attchid);


版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

C# Control的Invoke和BeginInvoke及其实现机制(消息循环)

与C++不同,C#语言禁止在创建某个控件的线程外对控件进行访问,否则会引起访问违规的异常。但有些时候,我们的确需要从其他线程对控件,此时就需要借助于Invoke和BeginInvoke之手了。我们先来...
  • Baple
  • Baple
  • 2014-03-19 13:35
  • 4910

C# invoke和beginInvoke

UI线程以外的线程不能直接更新UI界面,那如何在工作线程中更新UI界面呢? invoke和beginInvoke的出现是为了解决跨线程更新UI显示的问题,invoke是同步的执行,而beginIn...

c# 异步调用 BeginInvoke与EndInvoke方法

为什么要进行异步回调?众所周知,普通方法运行,是单线程的,如果中途有大型操作(如:读取大文件,大批量操作数据库,网络传输等),都会导致方法阻塞,表现在界面上就是,程序卡或者死掉,界面元素不动了,不响应...

C# 多线程 用委托实现异步_调用委托的BeginInvoke和EndInvoke方法

1.C#中的每一个委托都内置了BeginInvoke和EndInvoke方法,如果委托的方法列表里只有一个方法,那么这个方法就可以异步执行(不在当前线程里执行,另开辟一个线程执行)。委托的BeginI...

C#多线程之使用委托进行异步编程

.NET Framework 允许你异步调用任何方法。 为此,应定义与你要调用的方法具有相同签名的委托;公共语言运行时会自动使用适当的签名为该委托定义 BeginInvoke 和 EndInvoke ...

C# 多线程操作界面时,使用Invoke与BeginInvoke的区别

Invoke方法的参数很简单,一个委托,一个参数表(可选),而Invoke方法的主要功能就是帮助你在 UI线程(即创建控件的线程)上调用委托所指定的方法。Invoke方法首先检查发出调用的线程(即当前...

C#线程系列(1):BeginInvoke和EndInvoke方法

一、线程概述 在操作系统中一个进程至少要包含一个线程,然后,在某些时候需要在同一个进程中同时执行多项任务,或是为了提供程序的性能,将要执行的任务分解成多个子任务执行。这就需要在同一个进程中开启多个线...

C#线程系列讲座(1):BeginInvoke和EndInvoke方法

转自:http://www.cnblogs.com/nokiaguy/archive/2008/07/13/1241817.html 开发语言:C#3.0 IDE:Visual Stu...

C#线程系列讲座(1):BeginInvoke和EndInvoke方法

http://www.cnblogs.com/nokiaguy/archive/2008/07/13/1241817.html1.  BeginInvoke和EndInvoke方法2.  Thread...

C#中Invoke与BeginInvoke的区别(另附使用循环创建多个线程)

C#中Invoke与BeginInvoke的区别(另附使用循环创建多个线程) using System; using System.Collections.Generic; using Syst...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)