导读: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. objective-C 中两种实现动画的方法(转)

     转发自:http://wayne173.iteye.com/blog/1250232 第一种方法: [UIView beginAnimations:@"Curl"context: ...

  2. 【学习总结】IOS程序运行过程 、UIWindow 、controller 、 UIView 创建过程的总结

    程序启动开始到view显示: 程序启动首先会执行main函数 - > UIApplicationMain函数: 程序启动 (加载框架,info文件,资源等) 执行Main函数 初始化UIAppl ...

  3. sql之left join、right join、inner join的区别(转)

    感谢:http://www.cnblogs.com/pcjim/articles/799302.html ----------------------------------------------- ...

  4. HIBERNATE一对一双向外键联合主键关联

    HIBERNATE一对一双向外键联合主键关联: 一. 创建主键类:这个主键必须实现serializedable接口和重写其中的hashCode方法和equals方法:为主键类添加一个叫做@Embedd ...

  5. 纯CSS制作二级导航

    一.问题描述 做一个类似校园网首页,主要是导航栏的设置,ul默认纵向排列,如何横向排列,同时去掉圆点. 二.问题解决 2.1 先写导航条 用两个ul嵌套,一个ul是横向导航条,另一个是每个小项目下连一 ...

  6. Android 使用httpClient POST 模拟发送 multipart表单内容

    使用的环境:apache-mime4j-0.6.jar,httpcore-4.3.2.jar,httpmime-4.3.3.jar try { HttpPost httpPost = new Http ...

  7. 孟岩的c++ 的学习方法,这何尝有不是做人做事的方法呢?

    “(孟岩)我主张,在具备基础之后,学习任何新东西,都要抓住主线,突出重点.对 于关键理论的学习,要集中精力,速战速决.而旁枝末节和非本质性的知识内容,完全可 以留给实践去零敲碎打. “原因是这样的,任 ...

  8. 关于PHP写APP接口的安全问题探讨(一)

    在探讨这个问题之前,先要确认一点的是,作为一名互联网Coder,无论你是前端或者后端你都要对http请求要有一定的了解,知道http特性,要清楚的了解http里面的Request与Response是什 ...

  9. Android:控件ProgressBar进度条

    各种进度条属于 ProgressBar的子类 设置style: 环形进度条   style="?android:attr/progressBarStyleLarge" 横向进度条, ...

  10. Android:控件布局(线性布局)LinearLayout

    LinearLayout是线性布局控件:要么横向排布,要么竖向排布 决定性属性:必须有的! android:orientation:vertical (垂直方向) .horizontal(水平方向) ...