热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

从一个实际的例子来了解线程套间模型(.NETandCOMinterop)

问题实例:前些天有一个开发者在微软MSDN.NET开发者论坛上问了一个这样的问题:“IcreatemultipleinstancesofMSScript

问题实例:

前些天有一个开发者在微软MSDN .NET开发者论坛上问了一个这样的问题:

I create multiple instances of MSScriptControlClass on on different threads.

Calling an ExecuteStatement on one thread blocks other instances of MSScriptControlClass from executing on their own thread. I have an old Delphi Binary that seems to allow this behavior. I assume they use the same COM object.

Any ideas?

Thanks in advance.

大概意思是说他创建了一个.NET的程序,并且在多个线程上都创建了MSScriptControlClass的实例,但问题是任意一个线程上的脚本执行都会阻止其他线程的脚本执行。并且他曾经用Delphi使用这个COM组件的时候并没有这个问题。

问题重现:

为了重现这个问题,我用下面的步骤做了一个简单控制台程序来进行测试:

  1. 在Visual Studio里创建了一个C#的控制台程序。
  2. 添加了一个名为”Microsoft  Script Control 1.0”的COM引用。
  3. 添加了下列代码:

static void Main(string[] args){ThreadStart ts = new ThreadStart(LongRunningTask);Thread t = new Thread(ts);t.Start();ThreadStart ts1 = new ThreadStart(LongRunningTask1);Thread t1 = new Thread(ts1);t1.Start();}static void LongRunningTask(){MSScriptControl.ScriptControlClass scc = new MSScriptControl.ScriptControlClass();scc.Language = "Vbscript";scc.AllowUI = true;scc.ExecuteStatement(Properties.Resources.command1);}static void LongRunningTask1(){MSScriptControl.ScriptControlClass scc = new MSScriptControl.ScriptControlClass();scc.Language = "Vbscript";scc.AllowUI = true;scc.ExecuteStatement(Properties.Resources.command2);}

 

以下是Properties.Resources.command1和command2的内容:

commnd1:

For i=1 To 100 Step 1

if i mod 10 = 0 Then MsgBox(i)

Next

command2:

For i=100 To 200 Step 1

if i mod 10 = 0 Then MsgBox(i)

Next

 

期望的结果是有两个消息框弹出,并且一个显示10而另一个显示100. 那样就能证明脚本的执行可以并行进行。

但是得到的结果却是,要么弹出的是10,要么弹出的是100. 也就是说,线程间确实受到了影响。

问题分析:

这到底是为什么呢?如何来解决这个问题呢?

第一个印入我脑海的是:不同的threads之间对COM 对象的操作被进了了同步保护。所以让我想到了COM 线程套间模型。

先简单介绍一下COM线程套间模型(用来保证COM的线程安全):

单线程套间模型(STA):我们可以通过调用CoInitialize 或者CoInitializeEx(NULL, COINIT_APARTMENTTHREADED)来创建STA。 在.NET中的话,CLR 运行时会根据我们的设置去调用相应的方法,并不需要我们显示调用。每个调用了CoInitialize的线程都会有一个独立的套间。

多线程套间模型(MTA):我们可以通过调用CoInitializeEx(NULL, COINIT_MULTITHREADED) 来创建MTA。在.NET中的话,CLR 运行时会根据我们的设置去调用相应的方法,并不需要我们显示调用。所有调用了CoInitializeEx(NULL, COINIT_MULTITHREADED)的线程共享同一个套间。

根据上述描述,多线程套间模型是最有可能出现线程间同步控制的。那么我们来验证一下我们在.NET测试程序中创建的线程是否是MTA。这里我有必要强调一下.Net中托管线程的线程模型。在.NET中,线程模型不是.NET本身的线程安全/线程同步机制,纯粹是为了和之前的线程套间模型兼容。.NET本身是用完成端口、锁、互斥量等等来完成线程同步/线程安全的。

我们只需要在每个线程中调用下面的代码就可以看到当前的设置:

Console.WriteLine(Thread.CurrentThread.GetApartmentState().ToString());

得到的结果是:

主线程:MTA

线程t:  Unknown

线程 t1: Unknown

 

出现Unknown是因为我并没有显示指定用那种线程模型,所以我们得到了默认的Unknown。不过CLR运行时在Unknown的情况下也是调用了CoInitializeEx(NULL, COINIT_MULTITHREADED),这就意味着线程t和t1都是MTA。现在我们就解释了为什么两个线程会出现互相阻塞的情况。

那么STA模式将是我们解决问题的关键。但是不要急,因为这个COM组件并不一定支持STA,我们需要进一步验证。

Windows SDK提供了一个工具叫做OLE/COM Object Viewer, 那么我们就来看看我们需要调用的接口是否支持。下图就是我找到的关键接口。大家可以清楚的看到高亮的部分显示的是Both,那就是说STA和MTA都支持。

                       

问题解决:

那么我们接下来试一下是不是STA可以解决问题,我在之前创建线程的代码的基础上做出了以下修改:

ThreadStart ts = new ThreadStart(LongRunningTask);Thread t = new Thread(ts);t.SetApartmentState(ApartmentState.STA);t.Start();ThreadStart ts1 = new ThreadStart(LongRunningTask1);Thread t1 = new Thread(ts1);t1.SetApartmentState(ApartmentState.STA);t1.Start();

  

尝试再次运行这个控制台程序,我们会发现这次同时出现了显着着10和100的两个消息框。

结论及注意事项:

通过对上述问题的分析与解决,希望大家可以认识:

  1. 在多线程.NET程序中调用COM组件的时候,一定要注意线程套间模型的设置。
  2. 对于Scriptcontrol来说,它的开发者实现了多线程调用的线程同步和线程安全策略。但是并不意味着所有的COM组件都不能在MTA下进行多线程并行操作,这也是取决于COM服务是如何实现的。
  3. 我们可以通过OLE/COM Object Viewer来查看某个COM服务支持那种线程套间模型,并且对调用线程进行相应设置。

 

相关文档参考:

Understanding and Using COM Threading Models

http://msdn.microsoft.com/en-us/library/ms809971.aspx

INFO: Descriptions and Workings of OLE Threading Models

http://support.microsoft.com/kb/150777

Processes, Threads, and Apartments

http://msdn.microsoft.com/en-us/library/windows/desktop/ms693344(v=vs.85).aspx

Managed and Unmanaged Threading

http://msdn.microsoft.com/en-us/library/5s8ee185.aspx

COM Clients and Servers

http://msdn.microsoft.com/en-us/library/windows/desktop/ms683835(v=vs.85).aspx

An Overview of Managed/Unmanaged Code Interoperability

http://msdn.microsoft.com/en-us/library/ms973872.aspx


转载于:https://www.cnblogs.com/alanyao/p/3370168.html


推荐阅读
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • IhaveconfiguredanactionforaremotenotificationwhenitarrivestomyiOsapp.Iwanttwodiff ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • 本文介绍了一个Java猜拳小游戏的代码,通过使用Scanner类获取用户输入的拳的数字,并随机生成计算机的拳,然后判断胜负。该游戏可以选择剪刀、石头、布三种拳,通过比较两者的拳来决定胜负。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • iOS Swift中如何实现自动登录?
    本文介绍了在iOS Swift中如何实现自动登录的方法,包括使用故事板、SWRevealViewController等技术,以及解决用户注销后重新登录自动跳转到主页的问题。 ... [详细]
author-avatar
手机用户2502920725
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有