导读:CCR可以轻松的承载STA组件或者与它互操作:组件应该创建一个只有一个线程的CCR Dispatcher实例,并且在Dispatcher的构造函数中指定线程套间策略。DispatcherQueue实例就可以在与需要和遗留代码交互 的地方使用这个dispatcher来激活处理函数(activating handlers)。这些处理函数就可以安全的访问COM或者WinForm对象,同时对其他的CCR组件隐藏它们的STA关系,以便其他CCR组件可以 简单的投递元素到常规的CCR ports而不需要关心port上的处理函数使用的是什么dispatcher。

原文链接

线程套间约束

当与某些遗留的Windows组件,特别是COM对象交互时需要特定的线程套件策略。甚至最近的框架,例如.net WinForms,也需要在运行WinForm的线程上应用单线程套间(SingleThreadedApartment)策略。

CCR可以轻松的承载STA组件或者与它互操作:组件应该创建一个只有一个线程的CCR Dispatcher实例,并且在Dispatcher的构造函数中指定线程套间策略。DispatcherQueue实例就可以在与需要和遗留代码交 互的地方使用这个dispatcher来激活处理函数(activating handlers)。这些处理函数就可以安全的访问COM或者WinForm对象,同时对其他的CCR组件隐藏它们的STA关系,以便其他CCR组件可以 简单的投递元素到常规的CCR ports而不需要关心port上的处理函数使用的是什么dispatcher。

CCR WinFrom适配库(Ccr.Adapters.Winforms.dll)是一个在CCR中运行.net WinForm的便利方法。

与应用的主线程协调

CCR软件组件经常在一个CLR应用的上下文中中执行,例如一个独立执行程序。.net运行时使用一个操作系统 线程启动一个程序,当线程退出时终止程序。由于CCR应用是异步、并发的,所以它们在没有消息被发送时处于非激活状态,并且几乎不会阻塞任何线程。CCR Dispatcher将保持线程在一种高效的休眠状态,但是如果其它的线程被作为后台线程创建,应用程序将会退出,那么CCR还在执行。

一个与CLR启动的同步世界接口的常见模式是使用System.Threading.AutoResetEvent来阻塞应用的主线程,直到CCR应用完成。AutoResetEvent事件可以被任何CCR处理函数触发。

例25.


void InteropBlockingExample()

{

    // create OS event used for signalling

    AutoResetEvent signal = new AutoResetEvent(false);

    // schedule a CCR task that will execute in parallel with the rest of

    // this method

    Arbiter.Activate(

        _taskQueue,

        new Task<AutoResetEvent>(signal, SomeTask)

    );

    // block main application thread form exiting

    signal.WaitOne();

}

void ThrottlingExample()

{

    int maximumDepth = ;

    Dispatcher dispatcher = new Dispatcher(, "throttling example");

    DispatcherQueue depthThrottledQueue = new DispatcherQueue("ConstrainQueueDepthDiscard",

          dispatcher,

          TaskExecutionPolicy.ConstrainQueueDepthDiscardTasks,

          maximumDepth);

    Port<int> intPort = new Port<int>();

    Arbiter.Activate(depthThrottledQueue,

       Arbiter.Receive(true, intPort,

       delegate(int i)

       {

           // only some items will be received since throttling will discard most of them

           Console.WriteLine(i);

       })

    );

    // post items as fast as possible so that the depth policy is activated and discards

    // all the oldest items in the dispatcher queue

    for (int i = ; i < maximumDepth * ; i++)

    {

        intPort.Post(i);

    }

}

/// <summary>

/// Handler that executes in parallel with main application thread

/// </summary>

/// <param name="signal"></param>

void SomeTask(AutoResetEvent signal)

{

    try

    {

        for (int i = ; i < ; i++)

        {

            int k = i * i / ;

        }

    }

    finally

    {

        // done, signal main application thread

        signal.Set();

    }

}
在上面的例子中,我们演示了一个使用操作系统事件来阻塞应用程序主线程的小例子。

简化.NET异步编程模式

CCR可以用于任何实现了异步变成模型模式的.net类型。它实际上极大的简化了异步模式,并且当用于C#时,完全不再需要delegate和接续传递(continuation passing)。CCR迭带器调度支持允许你直接返回并继续(yield)挂起的I/O操作并且实现的易读的代码和过去只能在同步代码中实现的模式。

例27.


IEnumerator<ITask> CcrReadEnumerator(string filename)

{

    var resultPort = new Port<IAsyncResult>();

    // stage 1: open file and start the asynchronous operation

    using (FileStream fs = new FileStream(filename, 

        FileMode.Open, FileAccess.Read, FileShare.Read, , FileOptions.Asynchronous))

    {

        Byte[] data = new Byte[fs.Length];

        fs.BeginRead(data, , data.Length, resultPort.Post, null);    

        // stage 2: suspend execution until operation is complete

        yield return Arbiter.Receive(false, resultPort, delegate { });

        // stage 3: retrieve result of operation just by assigned variable to CCR port

        var ar = (IAsyncResult)resultPort;

        try

        {   Int32 bytesRead = fs.EndRead(ar); }

        catch

        {   

            // handle I/O exception 

        }

        ProcessData(data);

    }

}

例子27演示了CCR如何使用文件流的异步编程模型BeginRead/EndRead方法而又不需要传递delegate。取而代之的是,我们提供一个CCR Port的Post方法,所以异步结果被直接投递到一个CCR Port。代码返回并继续在一个port上进行接受操作。yield return语句允许我们编写逻辑上顺序的代码而不需要使用一个操作系统线程。代码保留了异步的可伸缩性和overlapped操作,但是读起来就像是同步、顺序的代码

例28

/// <summary>

/// Read from one stream into a Http request stream, asynchronously

/// </summary>

public virtual IEnumerator<ITask> UploadHttpStream(string contentUrl,

    Stream contentStream, PortSet<HttpStatusCode, Exception> resultPort)

{

    // Create HTTP request

    HttpWebRequest webRequest = (HttpWebRequest)WebRequest.Create(contentUrl);

    webRequest.Method = "POST";

    HttpStatusCode status = HttpStatusCode.OK;

    Exception ex = null;

    using (Stream requestStream = webRequest.GetRequestStream())

    {

        byte[] readBuffer = new byte[];

        int bytesRead = -;

        // With CCR and iterators you can do loops and asynchronous I/O

        do

        {

            // use CCR stream adapter (or a custom APM adapter) to schedule operation

            var ioResultPort = StreamAdapter.Read(contentStream, readBuffer, , );

            // yield to result (success or failure)

            yield return (Choice)ioResultPort;

            // check for error

            ex = ioResultPort;

            if (ex != null)

            {

                resultPort.Post(ex);

                // exit on error

                yield break;

            }

            bytesRead = ioResultPort;

            var writeResultPort = StreamAdapter.Write(requestStream, readBuffer, , bytesRead);

            // yield to write operation

            yield return (Choice)writeResultPort;

            // check for write error

            ex = writeResultPort;

            if (ex != null)

            {

                resultPort.Post(ex);

                yield break;

            }

        } while (bytesRead > );

        requestStream.Flush();

    }

    // Use built-in CCR adapter for reading HTTP response

    var webResultPort = WebRequestAdapter.ReadResponse(webRequest, _taskQueue);

    // yield to web response operation

    yield return (Choice)webResultPort;

    // check for any exceptions

    GetErrorDetails((Exception)webResultPort, out status);

    resultPort.Post(status);

}
例子28展示了一个更复杂的异步变成模型交互:一个CCR迭带器被调度来从一个流异步的读取数据,然后再异步的把数据写到一个HTTP请求流中。值得注意的是,CCR可以表达内部包含异步操作的while和for循环。

参考

MSDN Magazine: Concurrent Affairs -- Concurrency and Coordination Runtime

与非CCR代码互操作的更多相关文章

  1. Java类加载信息的顺序:包括静态代码快、静态类变量、非静态代码快、构造方法、普通方法

    JVM运行之前会执行一个叫做类加载器的子系统,叫做ClassLoader,那么类里面那么多“元素”,究竟是个什么顺序呢,写几行代码测试一下,通过给每个方法和代码快和静态变量打上断点来测试: class ...

  2. ARC代码和非ARC代码 混用

    选中工程->TARGETS->相应的target然后选中右侧的“Build Phases”,向下就找到“Compile Sources”了.然后在相应的文件后面添加:-fobjc-arc参 ...

  3. java 子类、父类中静态代码块、字段,非静态代码块、字段以及构造函数的初始化顺序和次数

    一个类中的数据初始化顺序是面试官非常喜欢出的面试题之一,本文用一个实例来介绍java中子类.父类中静态代码块.字段,非静态代码块.字段以及构造函数的执行顺序和次数. 一.包结构

  4. 使用非java代码编程

    使用非JAVA代码     JAVA语言及其标准API(应用程序编程接口)应付应用程序的编写已绰绰有余.但在某些情况下,还是必须使用非JAVA编码.例如,我们有时要访问操作系统的专用特性,与特殊的硬件 ...

  5. Java静态代码块与非静态代码块

    静态代码块,格式是 static{ },随着类的加载而加载,且只执行一次. 在程序中,执行的优先级最高. 非静态代码块,格式是{ },在创建对象的时候运行(即new一个对象的时候),每创建一次对象就执 ...

  6. Java非静态代码块和静态代码块

    类中存在两种特殊的代码块,即非静态代码块和静态代码块,前者是直接由 { } 括起来的代码,而后者是由 static{ } 括起来的代码. 非静态代码块在类初始化创建实例时,将会被提取到类的构造器中执行 ...

  7. Java中静态变量、静态代码块、非静态代码块以及静态方法的加载顺序

    在研究单例设计模式的时候,用到了静态变量和静态方法的内容,出于兴趣,这里简单了解一下这四个模块在类初始化的时候的加载顺序. 经过研究发现,它们的加载顺序为: 1.非静态代码块 2.静态变量或者静态代码 ...

  8. 关于一致/非一致代码段与TSS 关系的个人看法

    [0]概念定义 0.1)一致代码段: 简单理解,就是操作系统拿出来被共享的代码段,可以被低特权级的用户直接调用访问的代码, 但是特权级高的程序不允许访问特权级低的数据. 通常这些共享代码,是" ...

  9. node总结--回调函数阻塞和非阻塞代码实例

    阻塞代码实例: var fs = require("fs"); var data = fs.readFileSync('input.txt'); console.log(data. ...

随机推荐

  1. javascript面向对象思想2

    上篇说到面向对象可以帮我们梳理页面的逻辑的文章(http://www.cnblogs.com/hetaojs/p/6024013.html),很多朋友看了说我这种写法是初级的面向对象小儿科,确实是初级 ...

  2. iOS本地动态验证码生成-b

    用于ios本地动态生成验证码,效果如下: demo.gif 导入CoreGraphics.framework用于绘制图形 封装UIView,便捷使用,代码如下: AuthcodeView.h #imp ...

  3. iPhone手机屏幕的尺寸

    以下是 iPhone的型号和对应的屏幕宽高 英寸  宽 高  厚度 3.5   320 480 4s      ipad   系列   4   320 568 5   5s   4.7  375 66 ...

  4. tcpprep 对IPV6的支持

    在采用tcpreplay对包实施回放前,需要对包执行预处理,tcpprep就是完成这个任务的.tcpprep要做的处理就是生成一个cache文件,根据tcpprep wiki的介绍http://tcp ...

  5. 1874: [BeiJing2009 WinterCamp]取石子游戏 - BZOJ

    Description小H和小Z正在玩一个取石子游戏. 取石子游戏的规则是这样的,每个人每次可以从一堆石子中取出若干个石子,每次取石子的个数有限制,谁不能取石子时就会输掉游戏. 小H先进行操作,他想问 ...

  6. 如何安装Favicon

    如何安装Favicon favicon.ico图像放在根目录下(也可以是其他目录)在页面源文件的<head></head>标签之间插入 <link rel="s ...

  7. 关于ax+by=c的解x,y的min(|x|+|y|)值问题

    首先我们移动一下项,并强行让a>b. 然后我们可以画出这样一个图像 我们发现,在线段l与x轴交点处的下方,x,y的绝度值是递增的,所以我们不考虑那个最小点在下端. 之后我们发现在点的上端,因为斜 ...

  8. Fiddler 日志

    Fiddler 日志(Logging) 在开发扩展插件及编写FiddlerScript时对调试程序非常有用. 1.输出日志 在FiddlerScript脚本中,你可以这样输出输出日志: Fiddler ...

  9. 【leetcode】Word Break II (hard)★

    Given a string s and a dictionary of words dict, add spaces in s to construct a sentence where each ...

  10. POJ3122Pie(二分)

    http://poj.org/problem?id=3122 题意 :这个题最主要的就是审题要仔细,翻译不要漏句子.题目讲的是我要过生日,要给好友分馅饼(还有自己也想要一块),怕引起不公,所以每个人大 ...