WebApiClient库支持AOT
1 库简介
WebApiClient是开源在github上的一个http客户端库,内部基于HttpClient开发,只需要定义c#接口(interface),并打上相关特性,即可异步调用http-api。该库支持framework4.5+、netstandard1.3、netcoreapp2.1,包含以下特性:
- 原生的支持面向切面编程;
- 内置丰富的特性,支持自定义特性;
- 灵活和Filter、GlobalFilter和IParameterable;
- 功能强大的序列化工具;
- 与外部HttpMessageHandler无缝衔接;
- 独一无二的请求异常条件重试功能和异常处理链式语法功能
2 WebApiClient.JIT
JIT,即Just-in-time,动态(即时)编译,边运行边编译,WebApiClient.JIT在运行时,在需要的时候使用Emit创建Http请求接口的代理类。
2.1 动态代理
[HttpHost("http://www.webapiclient.com")]
public interface IMyWebApi : IHttpApi
{
// GET webapi/user?account=laojiu
// Return 原始string内容
[HttpGet("/webapi/user")]
ITask<string> GetUserByAccountAsync(string account);
// POST webapi/user
// Body Account=laojiu&password=123456
// Return json或xml内容
[HttpPost("/webapi/user")]
ITask<UserInfo> UpdateUserWithFormAsync([FormContent] UserInfo user);
}
例如,开发者定义了如上请求接口,在调用到 HttpApiClient.Create<IMyWebApi>()时,才会自动创建一个动态模块,动态模块里定义一个实现了IMyWebApi接口的类型,使用IL实现接口的方法,最后实例化并返回此类型的实例,这些过程,都离不开Emit和JIT。
2.2 优势和劣势
优势:不依赖于编译器,在运行时用到接口的时候,按需默默地为开发者实现了请求接口的代理类,这些接口的定义可以在外部程序集,同时编写接口的语言只要是能编译为中间公共语言的就得到支持。
劣势:由于严重依赖于JIT,对于编译阶段就将IL翻译到平台本机指令的地方,将无法得到支持,比如unity3d、ios或uwp,因为需要在编译阶段将IL再编译为平台本机指令,而动态代理IL在这个时期还不存在,同时编译器也会禁止代码里使用Emit相关Api,因为运行时不会再有JIT的存在。
3 WebApiClient.AOT
AOT,Ahead Of Time,指运行前编译,WebApiClient.AOT在编译阶段,就自动插入Http请求接口的代理类IL指令。
3.1 编译时插入代理类IL
这类需求实际上是自动帮开发者补充接口实现类的代码,然后合并到项目一起编译。考虑到开发语言的众多,比如c#、vb.net、c++/cli和f#等等,而且语言特性也在不断变化,所以WebApiClient没有从代码语法分析和自动补充接口实现类代码这个角度来解决这个问题,而是在编译过程中,等待输出程序集之后,再往程序集文件里写入代理类的IL代码,这样就可以规避上层编程语言的语法分析难题。
.net的编译过程,实际上为msbuild的一系列任务组合完成,我们的项目文件.csproj,实际上msbuild脚本,WebApiClient需要做的是,给msbuild增加一个自定义的任务,用于在完成CoreCompile任务之后,使用mono.cecli修改编译得到的程序集,插入代理类的IL。
<?xml version="1.0" encoding="utf-16"?>
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<Runtime Condition="'$(MSBuildRuntimeType)' == 'Core'">netcoreapp1.0</Runtime>
<Runtime Condition="'$(MSBuildRuntimeType)' != 'Core'">net45</Runtime>
<TaskAssembly>$(MSBuildThisFileDirectory)..\BuildTask\$(Runtime)\WebApiClient.AOT.Task.dll</TaskAssembly>
<TargetAssembly>$(MSBuildProjectDirectory)\$(IntermediateOutputPath)$(TargetFileName)</TargetAssembly>
</PropertyGroup>
<UsingTask TaskName="WebApiClient.AOT.Task.ProxyTask" AssemblyFile="$(TaskAssembly)" />
<Target Name="BuildProxy" AfterTargets="CoreCompile" DependsOnTargets="ResolveProjectReferences;ResolveAssemblyReferences">
<Message Text="BuildProxy: $(TargetAssembly)" Importance="high" />
<WebApiClient.AOT.Task.ProxyTask TargetAssembly="$(TargetAssembly)" References="@(_ResolveAssemblyReferenceResolvedFiles)" />
</Target>
</Project>
3.2 优势和劣势
优势:支持的平台更广泛,在编译时就可以静态检查接口声明是否分的符合框架约定。
劣势:依赖于编译器和编译过程,必须在项目引用nuget包才支持,直接引用WebApiClient.AOT库开发的话,编译时不会插入IL代理指令。
4 如何选择
如果你不是用于unity3d、ios或uwp这些平台应用开发,选择WebApiClient.JIT将比较方便;如果你不确定未来项目是否也会用于上面这些平台,就选择WebApiClient.AOT。不过不管怎么样,两个提供Api是完全一致,区别只在于你看不到的编译过程或运行时过程。
WebApiClient库支持AOT的更多相关文章
- Python-数据库支持
10.Python-数据库支持 使用数据库的好处: a.支持数据的并发访问,多个用户同时对基于磁盘的数据进行读写而不造成任何文件的损坏: b.支持根据多个数据字段或属性进行复杂的搜索: 1.如何操作数 ...
- OSS.Common获取枚举字典列表标准库支持
上篇(.Net Standard扩展支持实例分享)介绍了OSS.Common的标准库支持扩展,也列举了可能遇到问题的解决方案.由于时间有限,同时.net standard暂时还没有提供对Descrip ...
- c++ 11开始语言本身和标准库支持并发编程
c++ 11开始语言本身和标准库支持并发编程,意味着真正要到编译器从语言和标准库层面开始稳定,估计得到17标准出来.14稳定之后的事情了,根据历史经验,新特性的引入到稳定被广泛采用至少要一个大版本的跨 ...
- 自动检测GD库支持的图像类型
以下代码通过自动检测GD库支持的图像类型 来写出移直性更好的PHP代码 <?php if(function_exists("imagegif")){ header(" ...
- 微软企业库支持 MySql
微软企业库支持 MySql 三步让企业库支持 mysql 数据库 1.创建 MySqlDatabaseData 类 using Microsoft.Practices.EnterpriseLibr ...
- android WebP解析开源库-支持高清无损
在我们的项目中需要支持WebP高清无损图片,推荐一个我们已经使用的解析开源库给大家:https://github.com/keshuangjie/WebpExample/tree/master/lib ...
- 修改GDAL库支持IRSP6数据
使用GDAL库发现不能打开IRSP6的数据,不过看GDAL提供的文件格式里面却是支持IRSP6的数据的,具体可以参考网页http://www.gdal.org/frmt_fast.html.下面图1是 ...
- 修改GDAL库支持RPC像方改正模型
最近在做基于RPC的像方改正模型,方便对数据进行测试,修改了GDAL库中的RPC纠正模型,使之可以支持RPC像方改正参数. 下面是RPC模型的公式,rn,cn为归一化之后的图像行列号坐标,PLH为归一 ...
- 使GDAL库支持中文路径或中文文件名的处理方法
之前生成的gdal 2.1.1动态库,在通过命令行执行时,遇到有中文路径或中文图像名时,GDALOpen函数不能正确的被调用,如下图: 解决方法: 1. 在所有使用GDALAllRegist ...
随机推荐
- OC语言(六)
四十六.block类型 用来封装代码,可以在任何时候执行. 与函数的区别:可以在运行时动态产生. block的标志:^ 有形参.有返回值 int (^sumblock)(int,int) = ^(in ...
- SpriteBuilder中关节的Breaking force属性
在SpriteBuilder中三种物理关节都包含Breaking force区域在属性框中. 该属性被设置成关节可以承受的压力临界值.如果关节的压力超出了Breaking force中设置的值,则关节 ...
- eclipse 设置maven来自动下载源码与doc
通常我们通过maven来使用各种库文件,想要真正了解别人的类实现方法,需要查看别人的源码,maven给我们提供了这个便利,它不仅可以下载各种库文件,还会下载对应的源码和doc文档. 一.在工具栏找到W ...
- Java 条形码生成(一维条形码)
utl:http://mianhuaman.iteye.com/blog/1013945 在这里给大家介绍一个java 生成条形码 jbarcode.jar 生成条形码 支持EAN13, EAN8, ...
- ERP-非财务人员的财务培训教(五)------资本结构筹划
一.融资渠道 二.融筹资管理 第五部分 资本结构筹划 一.融资渠道 l 银行借款 优点:不影响企业的营运资本,不给企 ...
- Hbase 备份的方式
HBase 备份的方式有三种: 1.下线备份 (1)停止集群. (2)Distcp (3)restore 2.在线备份 -replication 3.在线北大 -CopyTable 4.在线备份-Ex ...
- 在linux下搭建STM32工程
转载自康神博客:http://blog.csdn.net/u013298300/article/details/50243935 在LINUX下开始一个STM32工程 在LINUX下开始一个STM32 ...
- LeetCode(60)-ZigZag Conversion
题目: The string "PAYPALISHIRING" is written in a zigzag pattern on a given number of rows l ...
- ruby用来发送互联网邮件
其实只要你任性的可以,用telnet也是可以发邮件的哦.不过本猫没那么任性,还是用KISS原则来发邮件吧.本篇博文只介绍了如何发邮件,但没涉及收邮件的事,以后如有机会会单独开一篇博文介绍. ruby通 ...
- $("li")是对象类型不是数组类型