Invoke与BeginInvoke

2024-01-04 07:08
文章标签 invoke begininvoke

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

在 Invoke 或者 BeginInvoke 的使用中无一例外地使用了委托 Delegate ,至于委托的本质请参考我的另一随笔: 对 .net事件的看法 。

 

一、为什么 Control类提供了 Invoke和 BeginInvoke机制?

关于这个问题的最主要的原因已经是 dotnet程序员众所周知的,我在此费点笔墨再次记录到自己的日志,以便日后提醒一下自己。

1、 windows程序消息机制

Windows GUI程序是基于消息机制的,有个主线程维护着一个消息泵。这个消息泵让 windows程序生生不息。

                                                   Windows GUI   程序的消息循环  

 

Windows程序有个消息队列,窗体上的所有消息是这个队列里面消息的最主要来源。这里的 while循环使用了 GetMessage()这个方法,这是个阻塞方法,也就是队列为空时方法就会被阻塞,从而这个 while循环停止运动,这避免了一个程序把 cpu无缘无故地耗尽,让其它程序难以得到响应。当然在某些需要 cpu最大限度运动的程序里面就可以使用另外的方法,例如某些 3d游戏或者及时战略游戏中,一般会使用 PeekMessage()这个方法,它不会被 windows阻塞,从而保证整个游戏的流畅和比较高的帧速。

这个主线程维护着整个窗体以及上面的子控件。当它得到一个消息,就会调用 DispatchMessage方法派遣消息,这会引起对窗体上的窗口过程的调用。窗口过程里面当然是程序员提供的窗体数据更新代码和其它代码。

2、 dotnet里面的消息循环

public static void Main(string[] args)

{

   Form f = new Form();

   Application.Run(f);

}

Dotnet窗体程序封装了上述的 while循环,这个循环就是通过 Application.Run方法启动的。

3、线程外操作 GUI控件的问题

如果从另外一个线程操作 windows窗体上的控件,就会和主线程产生竞争,造成不可预料的结果,甚至死锁。因此 windows GUI编程有一个规则,就是只能通过创建控件的线程来操作控件的数据,否则就可能产生不可预料的结果。

因此, dotnet 里面,为了方便地解决这些问题, Control 类实现了 ISynchronizeInvoke 接口,提供了 Invoke 和 BeginInvoke 方法来提供让其它线程更新 GUI 界面控件的机制。

public interface ISynchronizeInvoke

{

        [HostProtection (SecurityAction .LinkDemand, Synchronization=true , ExternalThreading=true )]

        IAsyncResult BeginInvoke(Delegate method, object [] args);

        object EndInvoke(IAsyncResult result);

        object Invoke(Delegate method, object [] args);

        bool InvokeRequired { get ; }

}

}

如果从线程外操作 windows窗体控件,那么就需要使用 Invoke或者 BeginInvoke方法,通过一个委托把调用封送到控件所属的线程上执行。

二、消息机制 ---线程间和进程间通信机制

1、 window消息发送

Windows消息机制是 windows平台上的线程或者进程间通信机制之一。 Windows消息值其实就是定义的一个数据结构,最重要的是消息的类型,它就是一个整数;然后就是消息的参数。消息的参数可以表示很多东西。

Windows 提供了一些 api 用来向一个线程的消息队列发送消息。因此,一个线程可以向另一个线程的消息队列发送消息从而告诉对方做什么,这样就完成了线程间的通信。有些 api 发送消息需要一个窗口句柄,这种函数可以把消息发送到指定窗口的主线程消息队列;而有些则可以直接通过线程句柄,把消息发送到该线程消息队列中。

                                                   

用消息机制通信

 

SendMessage是 windows api,用来把一个消息发送到一个窗口的消息队列。这个方法是个阻塞方法,也就是操作系统会确保消息的确发送到目的消息队列,并且该消息被处理完毕以后,该函数才返回。返回之前,调用者将会被暂时阻塞。

PostMessage也是一个用来发送消息到窗口消息队列的 api函数,但这个方法是非阻塞的。也就是它会马上返回,而不管消息是否真的发送到目的地,也就是调用者不会被阻塞。

2、 Invoke and BeginInvoke

 

                                                        Invoke or BeginInvoke

 

Invoke或者 BeginInvoke方 法都需要一个委托对象作为参数。委托类似于回调函数的地址,因此调用者通过这两个方法就可以把需要调用的函数地址封送给界面线程。这些方法里面如果包含了 更改控件状态的代码,那么由于最终执行这个方法的是界面线程,从而避免了竞争条件,避免了不可预料的问题。如果其它线程直接操作界面线程所属的控件,那么 将会产生竞争条件,造成不可预料的结果。

使用 Invoke完成一个委托方法的封送,就类似于使用 SendMessage方法来给界面线程发送消息,是一个同步方法。也就是说在 Invoke封送的方法被执行完毕前, Invoke方法不会返回,从而调用者线程将被阻塞。

使用 BeginInvoke方法封送一个委托方法,类似于使用 PostMessage进行通信,这是一个异步方法。也就是该方法封送完毕后马上返回,不会等待委托方法的执行结束,调用者线程将不会被阻塞。但是调用者也可以使用 EndInvoke方法或者其它类似 WaitHandle机制等待异步操作的完成。

但是在内部实现上, Invoke和 BeginInvoke都是用了 PostMessage方法,从而避免了 SendMessage带来的问题。而 Invoke方法的同步阻塞是靠 WaitHandle机制来完成的。

3、使用场合问题

如果你的后台线程在更新一个 UI控件的状态后不需要等待,而是要继续往下处理,那么你就应该使用 BeginInvoke来进行异步处理。

如果你的后台线程需要操作 UI控件,并且需要等到该操作执行完毕才能继续执行,那么你就应该使用 Invoke。否则,在后台线程和主截面线程共享某些状态数据的情况下,如果不同步调用,而是各自继续执行的话,可能会造成执行序列上的问题,虽然不发生死锁,但是会出现不可预料的显示结果或者数据处理错误。

可以看到 ISynchronizeInvoke有一个属性, InvokeRequired。这个属性就是用来在编程的时候确定,一个对象访问 UI控件的时候是否需要使用 Invoke或者 BeginInvoke来进行封送。如果不需要那么就可以直接更新。在调用者对象和 UI对象同属一个线程的时候这个属性返回 false。在后面的代码分析中我们可以看到, Control类对这一属性的实现就是在判断调用者和控件是否属于同一个线程的。

三、 Delegate.BeginInvoke

通过一个委托来进行同步方法的异步调用,也是 .net提供的异步调用机制之一。但是 Delegate.BeginInvoke方法是从 ThreadPool取出一个线程来执行这个方法,以获得异步执行效果的。也就是说,如果采用这种方式提交多个异步委托,那么这些调用的顺序无法得到保证。而且由于是使用线程池里面的线程来完成任务,使用频繁,会对系统的性能造成影响。

Delegate.BeginInvoke也是讲一个委托方法封送到其它线程,从而通过异步机制执行一个方法。调用者线程则可以在完成封送以后去继续它的工作。但是这个方法封送到的最终执行线程是运行库从 ThreadPool里面选取的一个线程。

这里需要纠正一个误区,那就是Control类上的异步调用BeginInvoke并没有开辟新的线程完成委托任务,而是让界面控件的所属线程完成委托任务的。看来异步操作就是开辟新线程的说法不一定准确。 

FROM:http://www.cppblog.com/baby-fly/archive/2010/04/01/111245.html

这篇关于Invoke与BeginInvoke的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python Invoke自动化任务库的使用

《PythonInvoke自动化任务库的使用》Invoke是一个强大的Python库,用于编写自动化脚本,本文就来介绍一下PythonInvoke自动化任务库的使用,具有一定的参考价值,感兴趣的可以... 目录什么是 Invoke?如何安装 Invoke?Invoke 基础1. 运行测试2. 构建文档3.

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

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

探索Invoke:Python自动化任务的瑞士军刀

文章目录 探索Invoke:Python自动化任务的瑞士军刀背景:为何选择Invoke?`invoke`是什么?如何安装`invoke`?简单的`invoke`库函数使用方法场景应用:`invoke`在实际项目中的使用场景一:自动化测试场景二:代码格式化场景三:部署应用 常见问题与解决方案问题一:命令执行失败问题二:权限不足问题三:并发执行问题 总结 探索Invoke:P

Cglib的MethodPro的invoke和invokeSuper的区别;为何invokeSuper可以内部调用方法依然能增强,为何jdk代理不能实现invokeSuper的功能

个人总结 类继承测试代码 package test1;public class B {public static void main(String[] args) {B2 b2=new B2();b2.a1();}}class B1{public void a1() {System.out.println("a1");this.a2();}public void a2() {System.o

JVM的五个 invoke 指令详解

文章目录 概述invoke 指令概览1. invokespecial2. invokevirtual3. invokestatic4. invokeinterface5. invokedynamic 总结 概述 Java 虚拟机 (JVM) 是 Java 语言的核心组件之一,负责执行 Java 字节码。在 JVM 中,invoke 指令用于执行方法调用。本文将详细介绍 JVM 设

Invoke a function in computer

面试中被偶然问起知不知道一个函数调用另一个函数再计算机中是如何进行的,这应该属于汇编语言的范畴,完全忘了,特此简单记录一下相关的general idea。 方式一: 最常见的调用和传递参数形式就是依赖于这个线程的栈。每个线程都有自己的栈,调用函数的时候,相应的参数被push到stack的顶端,然后被调用函数过来提取相应个数的顶部参数即可。 方式二: 当函数参数个数很少或者大小很小的时候,可

汇编proto、proc、invoke伪指令与函数声明、函数定义、函数调用

一、proto伪指令–函数声明 功能和高级语言中的函数声明一样,在代码最前面写函数声明,在后面写函数定义 proto伪指令的格式 函数名 proto [距离] [语言] [参数1]:数据类型,[参数2]:数据类型,……代码示例: Asm_Function_1 proto stdcall arg1:dword,arg2:dword一些注意事项 32位汇编不需要指定距离可以指定语言也可也不指定,

UnityAPI学习之延时调用(Invoke)

延时调用(Invoke) 当我们进行简单函数的延时调用不想使用协程时,我们可以使用Invoke()函数 using System.Collections;using System.Collections.Generic;using UnityEngine;public class NO15_Invoke : MonoBehaviour{//显示在每次生成Gris以后的两秒Gris开始跑

C#工具---使用Signature Tool自动生成P/Invoke调用Windows API的C#函数声明

转载来源:http://blog.csdn.net/Donjuan/article/details/3865026 PS;这个工具不错啊!  在网上看到很多网友在.NET程序中调用Win32 API,或者调用自己的VC DLL里面提供的函数的时候,总是被生成正确的C函数在C#中的正确声明而困扰,而生成C++中结构体在C#中的声明 - 天,没有什么比这个更让人恶心的事情了。因为:

STS部署maven项目到tomcat,出现Cannot invoke Tomcat manager: Error writing to server

此处我是用的是tomcat8 + maven3.5   在tomcat部署文件夹 /conf/tomcat-users.xml 配置用户tomcat 并授权 <role rolename="admin-gui"/> <role rolename="admin-script"/> <role rolename="manager-gui"/> <role rolename="manage