之前的文章已经介绍过如何使用HttpClusterApi进行去中心化的HTTP集群服务访问,这一章主要详细讲述如何使用HttpClusterApi,主要包括如何定义节点,创建服务接口和使用接口描述不同情况下的WebApi服务。最后通过HttpClusterApi简便地调用百度的一些云服务接口。

介绍

通过名称大概也能了解它的功能,通过它可以描述和调用不同的webapi服务应用;它提供了不同Url对应不同服务绑定和创建接口代理调用功能,从而让使用者通过接口的方式即可完成webapi的调用并不像传统WebHttpRequest或HttpClient那样编写大量繁琐的基础代码。

方法

组件提供了以下方法来添加服务地址

HttpClusterApi AddHost(string url, string host, int weight = )//设置url对应节点的host信息,默认权重是10,权重值可以根据自己实际情况来配置
HttpClusterApi AddHost(string url, params string[] host)//设置url对应节点的host信息,默认权重是10
HttpClusterApi SetNode(string url, IApiNode node)//设置url对应的节点信息

Url参数是一个正则表达式,用于匹配请求的url;当url值为*的时候该host是最低级匹配项,当没有任何匹配host的情况下才使用'*'对应的host.

INodeSourceHandler NodeSourceHandler { get; set; }

在之前的文章已经有所介绍,是用于自定义集群信息源,通过这个属性可以绑定集群信息来源,这样使用者就可以通过数据库,缓存或第方服务来加载集群负载信息.

 public T Create<T>()

方法是创建一个接口代理,通过代理操作即可实现具体的HTTP请求,组件对接口的定义具有一定的限制性,虽然不用继承某些基础接口,但要通过一些组件提供的Attribute定义在方法或参数上.

 public ClusterStats Stats()

方法获取统计信息,可以得到每个url对应不同host的调用情况统计.

线程安全问题

HttpClusterApi是一个线程安全类,定义后可以在任意线程中同时使用;由于内部使用连接池的方式进行请求,所以并不用担心数据访问冲突的情况出现.Create<T>返回的代理对象也是线程安全的,所以只需要创建一个接口实例接口即可随意调用.

接口定义

组件通过接口的方式来进行请求处理,所以并不需要编写任何基础的HTTP通讯代码,为了达到实际应用的需要,组件提供一些Attribute来代替基础代码的编写.下面介绍一些这些Attribute的作用.

请求描述

组件暂只支持GET,POST,PUT,DELETE这几种请求描述,这些属性只能标记在接口方法上,用于描述方法指向那种请求方式.Route属性用于描述请求的路径,默认是根路径并使用方法名称作为URL

   [AttributeUsage(AttributeTargets.Method)]
public class GetAttribute : Attribute
{
public string Route { get; set; }
}
[AttributeUsage(AttributeTargets.Method)]
public class PostAttribute : Attribute
{
public string Route { get; set; }
}
[AttributeUsage(AttributeTargets.Method)]
public class DelAttribute : Attribute
{
public string Route { get; set; }
}
[AttributeUsage(AttributeTargets.Method)]
public class PutAttribute : Attribute
{
public string Route { get; set; }
}

数据转换器

数据转换器用于描述请求和响应数据流的处理方式,组件暂只提供两种转换器FormUrlFormaterJsonFormater,分别对应的Content-Type是:application/jsonapplication/x-www-form-urlencoded.如果这两种都不能满足实际应用需要的情况下可以通过继承以下对象

    [AttributeUsage(AttributeTargets.Method | AttributeTargets.Interface | AttributeTargets.Class)]
public abstract class FormaterAttribute : Attribute, IClientBodyFormater
{
public abstract string ContentType { get; } public abstract void Serialization(object data, PipeStream stream); public abstract object Deserialization(BeetleX.Buffers.PipeStream stream, Type type, int length);
}

数据转换器可以标识在接口或方法上,如果方法有标识的情况下则方法标识的转换器优先使用.如果没有标记转换器组件默认使用FormUrlFormater

参数描述

在HTTP请求中主要有三种类型的参数,分别是Header,QueryString和Body;如果接口方法标记为POST或PUT的情况,方法参数默认是Body参数类型,会写入到HTTP的Body中.组件提供两个Attribute来描述参数应用到Header,和QueryString

CHeader

可以用于接口或方法上,并可以同时标记多个;当用在接口或方法的时候需要指定name和value.如果用在参数name和value都可以缺省,自动拿参数名和参数值.CHeader的内容会自动写入HTTP的header里.

CQuery

可以用于接口或方法上,并可以同时标记多个;当用在接口或方法的时候需要指定name和value.如果用在参数name和value都可以缺省,自动拿参数名和参数值.CQuery的内容会自动附加到URL上,并进行URL编码

SSL访问

组件并不需要针对SSL做特别的配置,会自动根据添加的host是不是https来确定,如果是https情况下组件会自动针对这host启用SSL配置.

百度车牌识别接口实现

考虑到实际应用的便利性,以下简单使用组件描述一下百度云下的车牌识别调用.

引用组件

接口功能定义

调用这个接口主要涉及到两个方法,先是获取access_token,然后再调用识别方法;针对这两个方法定义接口如下:

[BaiduApiFormater]
public interface IBaiduApi
{
[Get(Route = "oauth/2.0/token")]
[CQuery("client_id", "HiSRszPD******mqHfIz7VOrg")]
[CQuery("client_secret", "y7MxdEIItD******YqtalU6b4iF")]
Task<token> GetToken(string grant_type = "client_credentials");
[Post(Route = "rest/2.0/ocr/v1/license_plate")]
Task<PlateData> Plate([CQuery]string access_token, string image, string multi_detect = "false");
}

由于百度的数据提交是application/x-www-form-urlencoded返回是josn;所以需要简单地实现一个对应的数据解释器

public class BaiduApiFormater : BeetleX.FastHttpApi.Clients.FormUrlFormater
{
public override object Deserialization(PipeStream stream, Type type, int length)
{
using (stream.LockFree())
{
if (type == null)
{
using (System.IO.StreamReader streamReader = new System.IO.StreamReader(stream))
using (JsonTextReader reader = new JsonTextReader(streamReader))
{
JsonSerializer jsonSerializer = JsonSerializer.CreateDefault();
object token = jsonSerializer.Deserialize(reader);
return token;
}
}
else
{
using (StreamReader streamReader = new StreamReader(stream))
{
JsonSerializer serializer = new JsonSerializer();
object result = serializer.Deserialize(streamReader, type);
return result;
}
}
}
}
}

相关工作准备好之后就可以使用接口进行功能调用了

BeetleX.FastHttpApi.Clients.HttpClusterApi httpClusterApi = new BeetleX.FastHttpApi.Clients.HttpClusterApi();
httpClusterApi.AddHost("*", "https://aip.baidubce.com");
IBaiduApi api = httpClusterApi.Create<IBaiduApi>();
var token = await api.GetToken();
var result = await api.Plate(token.access_token, GetImageData());

这样一个百度车牌识别的接口调用就完成了,使用上是不是比传统的WebHttpRequest和httpclient要方便很多?

BeetleX之HttpClusterApi应用详解的更多相关文章

  1. BeetleX之XRPC使用详解

    XRPC是基于BeetleX扩展一个远程接口调用组件,它提供基于接口的方式来实现远程服务调用,在应用上非常简便.组件提供.NETCore2.1和.NETStandard2.0的client版本,因此即 ...

  2. Linq之旅:Linq入门详解(Linq to Objects)

    示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集 ...

  3. 架构设计:远程调用服务架构设计及zookeeper技术详解(下篇)

    一.下篇开头的废话 终于开写下篇了,这也是我写远程调用框架的第三篇文章,前两篇都被博客园作为[编辑推荐]的文章,很兴奋哦,嘿嘿~~~~,本人是个很臭美的人,一定得要截图为证: 今天是2014年的第一天 ...

  4. EntityFramework Core 1.1 Add、Attach、Update、Remove方法如何高效使用详解

    前言 我比较喜欢安静,大概和我喜欢研究和琢磨技术原因相关吧,刚好到了元旦节,这几天可以好好学习下EF Core,同时在项目当中用到EF Core,借此机会给予比较深入的理解,这里我们只讲解和EF 6. ...

  5. Java 字符串格式化详解

    Java 字符串格式化详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 文中如有纰漏,欢迎大家留言指出. 在 Java 的 String 类中,可以使用 format() 方法 ...

  6. Android Notification 详解(一)——基本操作

    Android Notification 详解(一)--基本操作 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Notification 文中如有纰 ...

  7. Android Notification 详解——基本操作

    Android Notification 详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 前几天项目中有用到 Android 通知相关的内容,索性把 Android Notificatio ...

  8. Git初探--笔记整理和Git命令详解

    几个重要的概念 首先先明确几个概念: WorkPlace : 工作区 Index: 暂存区 Repository: 本地仓库/版本库 Remote: 远程仓库 当在Remote(如Github)上面c ...

  9. Drawable实战解析:Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)

    Android XML shape 标签使用详解   一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...

随机推荐

  1. mock打桩之EasyMock

    TDD是测试驱动开发(Test-Driven Development)的英文简称,是敏捷开发中的一项核心实践和技术,也是一种设计方法论.TDD的原理是在开发功能代码之前,先编写单元测试用例代码,测试代 ...

  2. JDBC连接MySQL数据库基础

    主要内容 MySQL的基本操作和基本SQL语法 使用JDBC进行MySQL数据库的开发 使用DriverManager.Connection.PreparedStatement.ResultSet对数 ...

  3. google cache源码详解

    一.引子 缓存有很多种解决方案,常见的是: 1.存储在内存中 : 内存缓存顾名思义直接存储在JVM内存中,JVM宕机那么内存丢失,读写速度快,但受内存大小的限制,且有丢失数据风险. 2.存储在磁盘中: ...

  4. BZOJ_3879_SvT_后缀数组+单调栈

    BZOJ_3879_SvT_后缀数组+单调栈 Description (我并不想告诉你题目名字是什么鬼) 有一个长度为n的仅包含小写字母的字符串S,下标范围为[1,n]. 现在有若干组询问,对于每一个 ...

  5. sql server 任务调度与CPU

    一. 概述 我们知道在操作系统看来, sql server产品与其它应用程序一样,没有特别对待.但内存,硬盘,cpu又是数据库系统最重要的核心资源,所以在sql server 2005及以后出现了SQ ...

  6. 对抗明文口令泄露 —— Web 前端慢 Hash

    (更新:https://www.cnblogs.com/index-html/p/frontend_kdf.html ) 0x00 前言 天下武功,唯快不破.但在密码学中则不同.算法越快,越容易破. ...

  7. SSRS报表服务随笔(rdl报表服务)-报表参数

    上一篇我们说了创建一个简单的显示报表,但在实际工作中,我们有很多要带条件的报表 现在先认识一下报表数据,首次打开SSDT,报表数据在窗口的左侧,要是找不到了,没关系,在工具栏-视图-最下面的报表数据 ...

  8. jdk源码阅读笔记-ArrayList

    一.ArrayList概述 首先我们来说一下ArrayList是什么?它解决了什么问题?ArrayList其实是一个数组,但是有区别于一般的数组,它是一个可以动态改变大小的动态数组.ArrayList ...

  9. 开发人员必备工具 —— JMeter 压测

    在接口开发完以后,开发人员应该学会对自己的接口先进行压测一下,虽然压测的结果并不一定准确,也不能完全反映真实情况,但是如果有问题的话多少是可以看出的,而且也可以及早做优化,做到心里有底.否则,等测试进 ...

  10. vue全家桶安装以及修改webpack配置新增vue项目启动方式

    一.安装node环境(自带npm) 下载地址 二.替换下载源 // 淘宝 NPM 镜像 npm install -g cnpm --registry=https://registry.npm.taob ...