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 ...
随机推荐
- php 命令空间总结
本文移到:http://www.phpgay.com/Article/detail/classid/2/id/53.html 1.命名空间的语法: <?php namespace 名称; ?&g ...
- jQuery插件AjaxFileUpload文件上传实现Javascript多文件上传功能
Ajax file upload plugin是一个功能强大的文件上传jQuery插件,可自定义链接.或其它元素庖代传统的file表单上传结果,可实现Ajax动态提示文件上传 过程,同时支撑多文 ...
- 在Windows下搭建Gitlab服务器
一.GitLab简介 GitLab 是一个用于仓库管理系统的开源项目.使用Git作为代码管理工具,并在此基础上搭建起来的web服务. 可通过Web界面进行访问公开的或者私人项目.它拥有与Github类 ...
- RAC 10g administration
10g RAC administration See OCFS Oracle Cluster Filesystem, ASM, TNSnames configuration, Oracle Datab ...
- session效率
(1)-不恰当的request.getSession() 在HttpServlet中,HttpSession对象通常在request.getSession(true)方法调用时才创建. HttpSes ...
- 操作系统 - unix和windows下进程异同
在UNIX系统中,只有一个系统调用可以用来创建新进程:fork.这个系统调用会创建一个与调用进程相同的副本.在调用了fork之后,这两个进程(父进程和子进程)拥有相同的存储映像.同样的环境字符串和同样 ...
- Oracle数据库容灾备份技术探讨
Oracle数据库容灾备份技术探讨 三种Oracle灾备技术 对于Oracle数据库的灾备技术,我们可以从Data Guard,GoldenGate和CDP角度去考虑. Oracle Data Gua ...
- java--加强之 jdk1.5简单新特性,枚举,注解
转载请申明出处:http://blog.csdn.net/xmxkf/article/details/9944041 Jdk1.51新特性(静态导入,可变参数,加强for循环,自动拆装箱) 08.ja ...
- Linux内核中断和异常分析(上)
中断,通常被定义为一个事件.打个比方,你烧热水,水沸腾了,这时候你要去关掉烧热水的电磁炉,然后再去办之前手中停不下来的事情.那么热水沸腾就是打断你正常工作的一个信号机制.当然,还有其它的情况,我们以后 ...
- 十大ios开发者喜爱的开源库
十大ios开发者喜爱的开源库 (转自博客园) 2014-08-17 14:07:58| 分类: objective-c | 标签:ios 开源库 |举报|字号 订阅 下载LOFTER我的照片书 ...