1,原理:

WinRT是一个新的类库,应用程序可以用它访问操作系统的功能.

在内部,WinRT以组件的形式实现.COM Component Object Model…

WinRT使用.net元数据来描述其API

当C#引用COM对象的时候,实际上是获得一个RCW引用,该引用内部引用WINRT组件

类似,将一个CLR对象传递给 WINRT API 实际上时将 CCW引用传递(com callerable Wrapper)

2,投射:====WinRT核心概念

文件名和命名空间: .winmd文件本身的名称必须和包含winrt组件的命名空间匹配. 比如,MyRT.PLC.winmd 文件必须在 MyRT.PLC命名空间或者其子命名空间定义RT组件.

通用基类型.

RT组件不共用基类.CLR投射的时候,感觉像从Object派生

核心数据类型....bool 整数,浮点数,16位字符,字符串和void.

类:支持继承和多态,但是没有使用基本上.

结构: 支持结构. clr进行了某些投射,比如Point,Rect,Size,TimeSpan

可控结构: 将 Foundation.IReference<T>投射成Nullable<T>

枚举: 枚举值作为int传递.有符号时离散型,无符号是标志值类型[flag]

方法:不支持Ref,支持Out(不可以同时输入输出.相当于INOUT)

属性:不支持有参属性和只读属性

委托:只能为参数类型和返回类型指定WinRT组件. WinRT委托没有BeginInvoke和EndInvoke方法

事件:

public delegate void TypedEventHandler<TSender,TResult>(TSender sender,TResult args)
EventHandler<T>

异常:使用HREsult指明状态. CLR投射成异常状态.其HResult属性用来包含HResult值.

字符串:WinRT系统不允许字符串为null.可以传递String.Empty给WinRT API.

日期: Foundation.DateTime结构代表一个UTC时间. CLR将其投射成 DateTimeOffset结构.在实列中,其将UTC时间转换成本地时间,然相反,clr将其作为UTC时间传给WinRT API

URI:WinRT只支持绝对URL

IClosable/IDispose   RT对象映射称为 Dispose方法,注意.Close方法不能执行任何IO操作.所以.需要显示调用存储方法来存储数据.

数组:支持一维0基数组

集合:CLR使用CCW来包装集合. 会跨边界操作.

  •     IITerable<T>   IEnumerable<T>
  •     IVector<T>    IList<T>
  •    IVectorView<T> IReadOnlyList<T>
  • IMap<K,V>           IDictionary<TKey,TValue>
  • IMapView<K,V>     IReadOnlyDictionary<TKey,TValue>
  • IKeyValuePair<K,V>  KeyValuePair<TKey,TValue>

框架投射

       1,从.NET代码中调用异步WinRTAPI  API所在模块地址.

C:\Program Files (x86)\Windows Kits\8.1\References\CommonConfiguration\Neutral\Annotated\Windows.winnmd

C:\Windows\SysWOW64\WinMetadataC:\Windows\System32\WinMetadata

  public static void WinRTAsyncIntro()
{
//Provides access to common locations that contain user content.
//This includes content from a user's local libraries (such as Documents, Pictures, Music, and Videos),
//HomeGroup, removable devices, and media server devices.
IAsyncOperation<StorageFile> asyncOp = KnownFolders.MusicLibrary.GetFileAsync("Song.mp3");//在音乐中查找一首歌.
asyncOp.Completed = OpCompleted;
Console.WriteLine("Operation...Completed!");
} private static void OpCompleted(IAsyncOperation<StorageFile> asyncInfo, AsyncStatus asyncStatus)
{
switch (asyncStatus)
{
case AsyncStatus.Completed:
Console.WriteLine("Completed!");
break;
case AsyncStatus.Canceled:
Console.WriteLine("Canceled");
break;
case AsyncStatus.Error:
Exception exception = asyncInfo.ErrorCode;//将异常转化成异常类.
Console.WriteLine("Error"+exception.HResult.ToString());
break;
default:
break; }
}

1该类是个异步操作函数,用来在公共文件夹的音乐库中查找一首歌

2 注意,在函数结束后,线程还在运行(除非进程结束).

3,当线程完成后,触发事件,调用完成函数

在完成函数里面,进行处理:

注意:没有出现显示的异常. 无论任何情况,都会调用完成函数,在完成函数里面处理3中情况

1,正常完成: status=completed

2,任务取消:status=canceled  //调用所有IAsyncXXX提供的Cancel方法.

3,发生异常:status=error 并且  异常的ErrorCode包装成了一个异常类.在其HResult中存放了异常的HResult.

4,在处理完毕所有情况后,调用IAsyncXXX.Close()方法进行资源的清理.

2,注意接口关系

基本接口:

 [GuidAttribute(54, 0, 0, 192, 0, 0, 0, 0, 0, 0, 70)]
[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
[Version(100794368)]
public interface IAsyncInfo
{
[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
void Cancel();
[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
void Close(); [SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
Exception ErrorCode { get; }
[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
uint Id { get; }
[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
AsyncStatus Status { get; }
}

AsyncStatus

[SupportedOn(100794368, Platform.Windows)]
[SupportedOn(100794368, Platform.WindowsPhone)]
[Version(100794368)]
public enum AsyncStatus
{
Started = 0,
Completed = 1,
Canceled = 2,
Error = 3
}

几个异步函数比较

IAsyncAction----Task
IAsyncActionWithProgress<TProgress>---带有过程回调
IAsynOperation<Tresut>---带返回参数
IA十一年Operation With Progress<Tresult,TProgress>---带过程回调,带返回参数的异步操作

利用C#提供的async和await简化代码:

 public static async void WinRTAsyncIntroEasy()
{
try
{
StorageFile storageFile = await KnownFolders.MusicLibrary.GetFileAsync("Song.mp3");//正常结果
}//异常完成
catch (OperationCanceledException) { Console.WriteLine("operation is canceled"); }
catch(Exception e) { Console.WriteLine(e.HResult.ToString()); } }

1,要添加引用system.Runtime.WindowsRuntime.dll.

C:\Windows\WinSxS\msil_system.runtime.windowsruntime_* \

2,注意该函数是一个异步函数. 但是在函数内部是同步的.使用await关键字的作用:

1,如果没有await关键字,那么异步操作将导致 GetFileAsync...异步执行,将执行下面的语句.加入await后,函数在这里就返回了,当异步函数没有执行完毕前,其不会执行下面的语句.

2,await进行了数据的转换, 比如    int t=Task<int> XXXAsync() 如果函数正确完成,那么就会将结果返回给t.

3,await在返回后,才继续执行下面的语句.所以,上面的方法,是一个很不错的范式.

 public static async Task TryConnectPlcAsync(CancellationToken token)
{ Plc plc = new Plc(CpuType.Logo0BA8, "192.167.0.0", 0, 1); Task t1 = plc.OpenAsync();//一个PLC异步函数,注意,异步函数的异常只能通过await来接收,它并不会直接导致出错.只是会终止线程. Task t2 = Task.Run(() =>//一种将没有取消功能的异步函数增加取消功能.
{ do
{ log("token is"+token.IsCancellationRequested.ToString());
log(t1.Status.ToString()); if (token.IsCancellationRequested)
{
//token.ThrowIfCancellationRequested();
throw new Exception();
}
if (t1.IsFaulted) throw t1.Exception; } while (true); },token
);
try
{ await t2;
}
catch(Exception e)
{
log(e.Message+"in 1");
//throw; 有的时候,将throw 抛给上层,并且运行完finally之后,就返回.//没的时候,不抛给上层. } finally
{
plc.Close();
Console.WriteLine("plc is closed");
}
//正常完成的情况. Console.WriteLine("ok..."); }
public static async Task TryConnectPLCTest()
{
var tokenSource = new CancellationTokenSource();
var token = tokenSource.Token;
Task t = TryConnectPlcAsync( token);
Thread.Sleep(100);
//tokenSource.Cancel();用于 给 toekn设定一个取消标志.然后交给异步程序进行处理.(向异步程序传递取消标志).
try
{
await t;//必须在这个地方捕捉错误,否则,异步任务的错误捕捉不到.
Console.WriteLine(t.Status);
}
catch(Exception e)
{
log(e.GetType().ToString());
}
finally
{
tokenSource.Dispose();
}
}

注意1:  取消的用法:建立一个cancellationTokenSource对象,并且将其token传递给异步函数,当设定cancellationTokenSource.Cancel()方法的时候,就会向异步函数的token.IsCancelRequest传递true;

注意2:在异步中发生的错误,需要 await Task 中进行捕捉,否则会捕捉不到

注意3:使用一个包装来强制在取消的时候,抛出,任务取消异常.查看Task t2.

与WinRT组件进行操作的更多相关文章

  1. 使用 C++ WinRT 组件

    创建 C++ WinRT 组件 通过 Cpp/WinRT 项目模板创建一个 WinRT 组件工程 CppWinrtComponent.vcxproj,主要接口定义如下: namespace CppWi ...

  2. [UWP] 为WinRT组件创建Nuget包

    Nuget 是 dotnet 开发中必不可少的包管理工具,但不仅仅局限于 dotnet 项目,在 VS 中使用 C++ 开发的时候,也可以使用 Nuget 来引用第三方组件.同样也可以用 Nuget ...

  3. 使用 iview Table 表格组件修改操作的显示隐藏

    使用 iview Table 表格组件修改操作的显示隐藏,如下图 1.如何设置 table 操作按后台传入的状态值去渲染 不同的按钮? 解决方法 我们在vue2中,动态渲染html 使用的是 retu ...

  4. elementUI的动态tabs页的使用,vue的动态组件的操作

    elementUI的动态tabs页的使用,vue的动态组件的操作 有时候我们需要用到动态的tab页,结合不同的页面内容来显示.这里是使用了elementUI的动态tabs页来实现的 <div c ...

  5. WP8.1中C++的winodws运行时组件位移操作的差异

    最近学习WP8.1应用开发,想把C语言的SM3国密算法移植到手机app中.由于把C语言的代码转换成C#代码工作量较大,因此只能用winodws运行时组件来实现. SM3国密算法是一种HASH算法,具体 ...

  6. vue的组件小操作

    项目技术: webpack + vue + element + axois (vue-resource) + less-loader+ ... vue的操作的方法案例: 1.数组数据还未获取到,做出预 ...

  7. C#使用第三方组件Epplus操作Excel表

    Epplus操作Excel基础详解 1.什么是Epplus Epplus是一个使用Open Office XML文件格式,能读写Excel2007/2010文件的开源组件,在导出Excel的时候不需要 ...

  8. C# 开源组件--Word操作组件DocX

    使用模版生成简历 读写表格数据 合并单元格 工具源代码下载 学习使用 使用模版生成简历 下面将以一个简历实例来讲解DocX对表格的操作,先看看生成的效果 private static void Cre ...

  9. 【Ngui 学习系列之一:简单组件的操作】

    一.Buttonunity edit: Sprite作为父对象和背景 -- Collider -- Button script Label 作为子对象和显示文字代码: private UIButton ...

随机推荐

  1. 剑指Offer对答如流系列 - 重建二叉树

    面试题6:重建二叉树 题目描述: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历的结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8} ...

  2. Hyper-V 搭建独臂路由器(单网卡也可以)

    2020年原本难得清闲的春节,由于疫情的原因只能在家里看视频打发时间.打开某奇艺,全是某某公寓的推荐真的是受不了.一群人在那里叽叽喳喳,超前点播更是吃像难看,实在是没意思,所以决定搞一个独臂路由器玩一 ...

  3. Java 添加、读取、删除Excel形状

    本文介绍通过java程序在excel中操作形状(图形)的方法,包括: 1. 添加形状(如设置形状类型/位置/大小.形状颜色填充(单色/渐变色/纹理/图片填充).形状显示或隐藏.形状倾斜角度.添加文本到 ...

  4. djgango装饰器

    from django.http import HttpResponse from django.views import View class MyView(View): def get(self, ...

  5. c和c++中读取数据的方式总结

    目录 c 输出 printf() 输入 scanf getchar(), putchar() gets(), puts() c++ 输入 cin() getline() get() 输出 cout 最 ...

  6. 14、 NAT

    私有IP地址段:10.0.0.0-10.255.255.255/8172.16.0.0-172.31.255.255/12192.168.0.0-192.168.255.255/16 NAT的必要性: ...

  7. 安卓开发实战-记账本APP(五)

    今天将昨天剩余的bug修复,并且完成图标的美化,设置长按删除,模仿支付宝实现金额的动态增加. ①将昨天的布局进行了修改:之前是fragment,改成FrameLayout布局,不再设置name,进而在 ...

  8. OpenCV3入门(五)图像的阈值

    1.图像阈值与二值化 阈值是一种简单的图像分割方法,一幅图像包括目标物体(前景).背景还有噪声,要想从数字图像中直接提取出目标物体,可以设定一个像素值即阈值,然后用图像的每一个像素点和阈值做比较,给出 ...

  9. 【动手学pytorch】softmax回归

    一.什么是softmax? 有一个数组S,其元素为Si ,那么vi 的softmax值,就是该元素的指数与所有元素指数和的比值.具体公式表示为: softmax回归本质上也是一种对数据的估计 二.交叉 ...

  10. PTA 7-10 树的遍历(二叉树基础、层序遍历、STL初体验之queue)

    7-10 树的遍历(25 分) 给定一棵二叉树的后序遍历和中序遍历,请你输出其层序遍历的序列.这里假设键值都是互不相等的正整数. 输入格式: 输入第一行给出一个正整数N(≤30),是二叉树中结点的个数 ...