• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

C#STA和MTA线程设置

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

参考资料:
http://www.yesky.com/20010207/158097.shtml
http://www.ftponline.com/china/XmlFile.aspx?ID=242
http://research.microsoft.com/~chadv/java_com2.htm
http://blogs.msdn.com/jfoscoding/archive/2005/04/07/406341.aspx


.NET支持两种线程模型:STA和MTA。
STA(single threaded apartments)。apartment只是一个逻辑上的概念,它可以包含一个或多个线程。一个AppDomain可以包括一个或多个apartment。STA是指该apartment中只能包含一个thread。
MTA(multi threaded apartments)。指该apartment中可以包含多个thread。
STA and MTA 之间最大的区别就是MTA 可以在同一个apartment 中使用所有的共享资源并发执行多个线程。 而多个STA虽然可以共享数据,但是不能并发执行线程,存在性能问题。

线程的创建:
当创建一个新的STA线程时,CLR会在该AppDomain中创建一个apartment和thread(从属于该apartment)。如果是创建 MTA线程,则会CLR会检查该AppDomain是否存在一个存放MTA的apartment,如果存在仅创建该线程到该MTA中,否则就创建一个 MTA和thread(从属于该apartment)。
我们可以设置线程的属性。例如 t.ApartmentState = ApartmentState.STA;

线程的使用区别:
我们应该仅仅在访问STA-based 的COM组件时才使用STA线程模式。可以在注册表的HKEY_CLASSES_ROOT\CLSID\{Class ID of the COM component} \InProcServer32 下查看到该COM的线程模式。如果该值是Apartment,则说明该COM只能以STA模式运行。其他的值有Free(MTA),Both(STA+MTA),Single(只能在一个单一的线程中执行)。
其他情况下,我们应该使用MTA的线程,虽然需要我们费心线程间资源的同步问题。


示例:
在一个windows form的程序中实现从某个word文档复制图片并保存的方案。
具体是:打开word文档,将图片信息复制到粘贴板中,然后从粘贴板中取得图片信息,再保存到本地目录中。

using System;  
using System.Collections.Generic;  
using System.Linq;  
using System.Text;  
using System.Threading;  
using Microsoft.Office.Interop.Word;  
using System.Windows.Forms;  
using System.Drawing;  
namespace ConsoleApplication1  
{  
    class STAANDMTA  
    {  
        static void Main()  
        {  
            Thread t = new Thread(new ThreadStart(CopyImages));  
            t.ApartmentState = ApartmentState.STA;  
            t.Start();  
        }  
  
        private static void CopyImages()  
        {  
            Microsoft.Office.Interop.Word.Application app = null;  
            Microsoft.Office.Interop.Word.Document doc = null;  
  
            object missing = System.Reflection.Missing.Value;   
  
  
  
  
            app = new Microsoft.Office.Interop.Word.Application();  
              
            try  
            {  
                object fileName = @"E:\A.doc";  
                doc = app.Documents.Open(ref fileName, ref missing, ref missing, ref missing, ref missing, ref missing, ref missing, ref missing, ref missing, ref missing,  
                    ref missing, ref missing, ref missing, ref missing, ref missing, ref missing);  
  
                int count = doc.InlineShapes.Count;  
                for (int i = 1; i <= count; i++)  
                {  
                    doc.InlineShapes[i].Range.Copy();//复制到粘贴板  
  
                    if (System.Windows.Forms.Clipboard.GetDataObject() != null)  
                    {  
                        IDataObject data = Clipboard.GetDataObject();  
  
                        if (data.GetDataPresent(DataFormats.Bitmap))  
                        {  
                            Image image = (Image)data.GetData(DataFormats.Bitmap, true);  
                            image.Save("E:\\" + i.ToString() + ".jpg", System.Drawing.Imaging.ImageFormat.Jpeg);  
                        }  
                        else  
                        {  
                            //lst_Items.Items.Add(doc.Name + ";无正确图片数据");  
                        }  
                    }  
                    else  
                    {  
//lst_Items.Items.Add(doc.Name + ";粘贴板为空");  
                    }  
                }  
  
            }  
            catch (Exception ex)  
            {  
                //lst_Items.Items.Add(doc.Name + "发生错误;" + ex.Message);  
            }  
            finally  
            {  
                if (doc != null)  
                    doc.Close(ref missing, ref missing, ref missing);  
                if (app != null)  
                    app.Quit(ref missing, ref missing, ref missing);  
            }  
        }  
    }  
}  


说明:
如果在某个按钮的事件中,直接调用该方法,那么界面将变得没有响应。所以我们需要考虑使用多线程来解决这个问题。Thread t = new Thread(new TheardStart(CopyImages); t.Start();
如果是这样,则程序会发生错误.。要么显示出现异常,要么没异常但是Clipboard为空,取不到任何数据!为什么呢?
因为Word.Application 是Automation并且STA-Based,不能在没有指定ThreadApartment的线程中被调用。所以导致了各种错误,所以需要在 t.Start();前面加上t.Apartment = ApartmentState.STA;这样就完全正常了。
对于MTA的多线程我们就见的比较多了,不再举例了。

另外一点,经过监视任务管理器发现,在执行Thread t = new Thread(new TheardStart(CopyImages);t.Apartment = ApartmentState.STA; t.Start();之后该程序的进程中线程数从3个增加到6个,如果创建的是MTA的线程则只增加1。我的理解是STA线程为需要维护内部隐藏的窗口类 和消息队列而增加的。


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
C#写入文件发布时间:2022-07-14
下一篇:
C++高性能无锁日志系统发布时间:2022-07-14
热门推荐
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap