• C# STA和MTA线程设置


    参考资料:
    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_ROOTCLSID{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线程为需要维护内部隐藏的窗口类 和消息队列而增加的。

  • 相关阅读:
    Linux C 面试题总结
    linux下的缓存机制及清理buffer/cache/swap的方法梳理
    接入WebSocket记录 + 一些个人经验
    Linux基础系列—Linux体系结构和Linux内核结构
    typedef和define具体的详细区别
    RANSAC与 最小二乘(LS, Least Squares)拟合直线的效果比较
    深入理解C/C++混合编程优秀博文赏析与学习
    “error LNK2019: 无法解析的外部符号”之分析
    CUDA和OpenGL互操作经典博文赏析和学习
    [原创]C/C++语言中,如何在main.c或main.cpp中调用另一个.c文件
  • 原文地址:https://www.cnblogs.com/xuekai-to-sharp/p/3510148.html
Copyright © 2020-2023  润新知