• C#异步委托之委托使用的时机浅析


    C#异步委托之委托使用的时机是什么呢?那么让我们开始讲述吧:

    在我的印象里,委托经常出现在三个场合(也许更多,我水平有限)。

    C#异步委托之委托使用的时机1、事件调用(最常用)。

    C#异步委托之委托使用的时机2、线程调用。

    C#异步委托之委托使用的时机3、同步异步调用。

    事件调用(最常用)是委托用的最多的地方。我们为了使窗口之间的数据可以进行传递,经常会使用窗体定义事件。最常见的情况是这样的,用户为了查询一个客户,在窗体B中单击“查询”按钮后,弹出一个客户查询窗体(这里定义为A),在A中查询出指定的结果后,单击“选择”按钮,A关闭,窗体B上就获得了查询的客户信息。基于这种情况,我们一般会在A中定义一个事件,然后会在B中写一个当这个事件发生后调用的方法(函数)。那么怎么把A的事件和B的方法结合起来呢,这个时候就需要用到委托了。实际上,委托就是事件和方法之间的桥梁。如果大家不明白,可以看看class.designer.cs这个文件,当我们在按钮上选择click按钮事件时,就会自动创建一个与之相关的委托,以及事件函数,例如。

    this.btnStart.Click +=   new System.EventHandler(this.btnStart_Click); 

    线程调用时也需要使用委托。为什么呢?我们这么来考虑一下,线程主要表达模块中的程序代码的“执行事实”(深入浅出MFC)。那么线程要调用执行的函数代码,怎么调用函数呢?这么一说,我们就知道了,委托(即函数指针)就是搭起函数和其他东西(事件、线程)之间的桥梁。所以,在线程调用时也要使用委托,通过委托连接要执行的函数。例如:

    C#异步委托之委托使用的实例应用:

    class Test  
    {  
        
    static void Main()   
        {  
            Thread newThread 
    =   new Thread(new ThreadStart(Work.DoWork));  
            newThread.Start();  
        }  
    }  
    class Work   
    {  
        Work() {}  
        
    public static void DoWork() {}  

    以上代码中“new Thread(new ThreadStart(Work.DoWork));”部分就是委托的部分。ThreadStart委托,它表示此线程开始执行时要调用的方法。所以,在线程调用时需要委托。

    C#异步委托之委托使用的相关内容就向你介绍到这里,希望对你了解和学习C#异步委托之委托使用有所帮助。

    如果提示Error: 线程间操作无效: 从不是创建控件“”的线程访问它。
    以下是一个偷懒的解决办法

    Control.CheckForIllegalCrossThreadCalls = false;

    使用前请注意:该方法只是简单的将错误提示禁用了,仍然存在跨线程调用控件的问题。为此可能造成两个线程同时或者循环改变该控件的状态导致线程死锁。如果要根本性的解除问题,得用Invoke方法,因为Invoke是同步的方法,所以执行过程是有先后顺序的。
    举个例子:

    //声明委托
    private delegate void DELLoadData(int rowIndex);

    /// <summary>
    /// 加载任务列表、短信数量、任务数量
    /// </summary>
    /// <param name="rowIndex">CurrentRow</param>
    private void LoadDatas(int rowIndex)
    {
        DoSearchWaitHandle();
        GetAmountByNotRead(LoginUser.loginUserID);
        GetWHAmountByNotRead(LoginUser.loginUserID);
        
    if (dgWaitHandle.Rows.Count > 0)
            dgWaitHandle.Rows[rowIndex].Cells[
    0].Selected = true;
    }

    //执行Invoke委托
    this.Invoke(new DELLoadData(LoadDatas), new object[] { rowIndex });


     以下例子涉及到的内容和本文关连不大,请慎重观看:

    //声明计时器
    private System.Timers.Timer timer = new System.Timers.Timer();

    //初始设置计时器
    private void InitializeMyTimer()
    {
          timer.Interval 
    = 60 * 1000;
          timer.Elapsed 
    += new ElapsedEventHandler(OnTimedEvent);
          timer.Start();
    }

    private void OnTimedEvent(object sender, ElapsedEventArgs e)
    {
        
    try
        {
            
    this.Interval += 1;
            
    // 每隔10分钟刷新一次界面
             if (this.Interval == 10)
            {
                 
    this.Invoke(new DELLoadData(LoadDatas), new object[] { 0 });
                 
    this.Interval = 0;
            }
         }
         
    catch (Exception ex)
         {
             
    //log.Error(Definition.MESSAGE_QUERY_ERROR + ":" + ex.Message);
             
    //Tool.ShowMessage(Definition.MESSAGE_QUERY_ERROR + ":" + ex.Message, MessageBoxIcon.Error);
         }
    }
  • 相关阅读:
    OPENGL学习笔记整理(一)
    OPENGL学习笔记整理(二):纹理知多少?
    OPENGL学习笔记整理(三):缓冲区对象
    细节决定成败(基础知识收集)
    OPENGL学习笔记整理(四):几何图元渲染
    目标规划,利用业余时间。欢迎大家指导。
    微型通用程序框架
    COM在注册表中的相关键值及其意义
    C#编写基于.Net IOCP的高性能服务器(转)
    Java获取XML节点总结之读取XML文档节点
  • 原文地址:https://www.cnblogs.com/xvqm00/p/1564604.html
Copyright © 2020-2023  润新知