Spring Cloud feign是僞RPC方式解決微服務間的調用。翻看FeignCloudFeign源碼,能夠看到Feign默認使用HttpUrlConnection; 代碼在DefaultFeignLoadBalancedConfiguration 的Client.Default。html
這裏特地說明下,其餘替代方式有OKhttp和HTTPClient,這兩種方式有鏈接池,能夠減小建立鏈接的性能損耗,可是多家實測效果代表,HttpUrlConnection的效率上是最高的,這也是feign默認的緣由,若是在輕型應用不考慮速度考慮CPU的狀況下,能夠考慮替換成OKhttp和HttpClient。git
在序列化方面,springboot中HttpMessageConverters 默認使用jackson2方式進行序列化和反序列化。 jackson的效率在於GSON和fastjson之上。github
正常狀況下使用jackson2支持先後端開發基本沒有什麼問題,可是若是是微服務間頻頻通訊,使用jackson2序列化和反序列化會佔用很多系統資源,而且效率較差。 這裏有個git地址來對比各類序列化和反序列化框架的性能 https://github.com/eishay/jvm-serializers/wiki,部份內容以下:web
Ser Time+Deser Time (ns)
Size, Compressed size [light] in bytesspring
可見jackson在各類測試中都不佔優點,網上如今不少的教程使用protobuf來替換jackson提高feign的性能,可是因爲nafos中使用的是更加方便的protostuff,並且由圖中能夠看出protostuff的性能比protobuf更勝一籌。因此我決定用protostuff來替換。shell
在feign中,protobuff有默認提供的編解碼器,所以參考其餘教程使用便可,可是protostuf卻沒有默認提供,因此咱們必須本身動手擼了。
數據庫
在手擼以前,咱們確定是須要先了解protostuff和feign提供的protobuf編解碼器的,畢竟protostuff的編解碼器,在網上貌似還沒得教程。json
Protostuff在這裏就不作贅述,能夠參考我博客的另外一篇文章:http://www.javashuo.com/article/p-xnhsiprm-mr.html後端
經過查看feign提供的protobuff編解碼器的源碼,能夠看到他是繼承AbstractHttpMessageConverter<Object>,主要實現了supports,readInternal,writeInternal這三個方法。分別是返回類的校驗,解碼器,和編碼器。tomcat
代碼以下:
/**
* @Author 黃新宇
* @Date 2018/10/30 下午1:24
* @Description feign調用中protostuff編解碼器
**/
public class ProtostuffHttpMessageConverter extends AbstractHttpMessageConverter<Object> {
private static final Logger logger = LoggerFactory.getLogger(ProtostuffHttpMessageConverter.class); public static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8"); public static final MediaType PROTOBUF; static { PROTOBUF = new MediaType("application", "x-protobuf", DEFAULT_CHARSET); } public ProtostuffHttpMessageConverter() { super(new MediaType[]{PROTOBUF, MediaType.TEXT_PLAIN, MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML}); } @Override protected boolean supports(Class<?> aClass) { return Object.class.isAssignableFrom(aClass); } @Override protected Object readInternal(Class<?> aClass, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException { MediaType contentType = inputMessage.getHeaders().getContentType(); if (contentType == null) { contentType = PROTOBUF; } if (!PROTOBUF.isCompatibleWith(contentType)) { logger.error("不支持的解碼格式,請用x-protobuf做爲contentType"); } try { return ProtoUtil.deserializeFromByte(inputMessage.getBody(),aClass); } catch (Exception var7) { throw new HttpMessageNotReadableException("Could not read Protobuf message: " + var7.getMessage(), var7); } } @Override protected void writeInternal(Object object, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException { MediaType contentType = outputMessage.getHeaders().getContentType(); if (contentType == null) { contentType = PROTOBUF; } Charset charset = contentType.getCharset(); if (charset == null) { charset = DEFAULT_CHARSET; } if (!PROTOBUF.isCompatibleWith(contentType)) { logger.error("不支持的編碼格式,請用x-protobuf做爲contentType"); } FileCopyUtils.copy(ProtoUtil.serializeToByte(object), outputMessage.getBody()); } }
添加以下配置,ProtostuffHttpMessageConverter在nafos.remote.feign.ProtostuffHttpMessageConverter下
而後在須要用protostuff的feignClient上註明configuration = ProtoFeignConfiguration.class)便可
@FeignClient(value = "service",path ="/nafosRemoteCall/test" , configuration = ProtoFeignConfiguration.class)
詳細代碼請參考nafos-network: https://gitee.com/huangxinyu/BC-NETTYGO
一個基於netty的輕量級高性能服務端框架。
nafos是一個基於netty的高性能服務器框架,其目的在於易上手,易擴展,讓開發人員更致力於業務開發。 在先後端分離的web架構上,或者APP,手遊,nafos都是一個很不錯的選擇。
除此以外,sample中也給出了超簡單的擴展方案,使得nafos在分佈式擴展上能更勝一籌。