一种优雅的条件引用第三方.net库的方法
1.遇到的问题
今年我一直在开发一个WebApiClient库,旨在.net下能像java的retrofit一样,方便地请求服务端的http接口。在这restful api盛行的年代,json的身影无处不在,.net framework自然也有json库,System.Web.Script.Serialization.JavaScriptSerializer就是其中一个,但综合性较好的,还是第三方的json.net。在开发WebApiClient过程中,我一直没有依赖json.net,原因是:如果我的WebApiClient依赖json.net的版本是6.0,某个第三方库ThirdLib依赖的json.net版本是7.0,在项目中会无法都使用WebApiClient和ThirdLib,原因是他们依赖的json.net版本不一样而决生冲突,给使用者解决起来很非常麻烦。
2.期望的效果
我期望的效果是,如果引用WebApiClient库的项目有使用到json.net,那么WebApiClient在json序列化时,就使用json.net,否则使用.net framework的System.Web.Script.Serialization.JavaScriptSerializer。这就要求我在开发WebApiClient的时候,不能直接引用json.net来开发,只能在运行时检测应用程序域是否已加载了json.net,从而反射调用json.net来进行序列化。
想要的效果是这样的
/// <summary>
/// 反序列化对象
/// </summary>
/// <param name="json">json</param>
/// <param name="objType">对象类型</param>
/// <returns></returns>
public object Deserialize(string json, Type objType)
{
if (string.IsNullOrEmpty(json))
{
return null;
} if (JsonNet.IsSupported == true)
{
return JsonNet.DeserializeObject(json, objType);
} var serializer = new System.Web.Script.Serialization.JavaScriptSerializer();
return serializer.Deserialize(json, objType);
}
3.解决的方案
在运行时检测应用程序域是否已加载了json.net,从而反射调用json.net来进行序列化。
实现中的难点:
1、如果检测是否加载了Json.net
2、在检测之后,才加载Json.net怎么办
3、怎么样减少反射的性能损耗
难点的解决:
1、使用AppDomain.CurrentDomain.GetAssemblies()方法获取当前程序域目前已加载的程序集,判断是否有Json.net的程序集;
2、监听AppDomain.CurrentDomain.AssemblyLoad事件,订阅新加载的程序集是否为Json.net的程序集;
3、将Newtonsoft.Json.JsonConvert的SerializeObject和DeserializeObject方法生成强类型委托,缓存起来等待调用;
完整代码实现:
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks; namespace WebApiClient
{
/// <summary>
/// 提供Json.net无引用调用
/// </summary>
static class JsonNet
{
/// <summary>
/// Json.Net程序集名称
/// </summary>
private static readonly string jsonNetAssemblyName = "Newtonsoft.Json"; /// <summary>
/// JsonConvert类名
/// </summary>
private static readonly string jsonNetJsonConvertTypeName = "Newtonsoft.Json.JsonConvert"; /// <summary>
/// 序列化方法的委托
/// </summary>
private static Func<object, string> serializeFunc = null; /// <summary>
/// 反序列化方法的委托
/// </summary>
private static Func<string, Type, object> deserializeFunc = null; /// <summary>
/// 获取是否得到支持
/// </summary>
public static bool IsSupported = false; /// <summary>
/// Json.net
/// </summary>
static JsonNet()
{
AppDomain.CurrentDomain.AssemblyLoad += (s, e) => InitJsonNet(e.LoadedAssembly);
InitJsonNet(AppDomain.CurrentDomain.GetAssemblies());
} /// <summary>
/// 序列化对象
/// </summary>
/// <param name="obj">对象</param>
/// <returns></returns>
public static string SerializeObject(object obj)
{
return JsonNet.serializeFunc.Invoke(obj);
} /// <summary>
/// 反序列化为对象
/// </summary>
/// <param name="json">json文本</param>
/// <param name="type">对象类型</param>
/// <returns></returns>
public static object DeserializeObject(string json, Type type)
{
return JsonNet.deserializeFunc.Invoke(json, type);
} /// <summary>
/// 初始化json.net
/// </summary>
/// <param name="assemblies">查找的程序集</param>
private static void InitJsonNet(params Assembly[] assemblies)
{
if (JsonNet.IsSupported == true)
{
return;
} var jsonNetAssembly = assemblies
.FirstOrDefault(item => item.GetName().Name.Equals(jsonNetAssemblyName, StringComparison.OrdinalIgnoreCase)); if (jsonNetAssembly == null)
{
return;
} var jsonConvertType = jsonNetAssembly.GetType(jsonNetJsonConvertTypeName, false);
if (jsonConvertType == null)
{
return;
} serializeFunc = CreateSerializeObjectFunc(jsonConvertType);
deserializeFunc = CreateDeserializeObjectFunc(jsonConvertType);
JsonNet.IsSupported = serializeFunc != null && deserializeFunc != null;
} /// <summary>
/// 创建SerializeObject方法的委托
/// </summary>
/// <param name="classType">JsonConvert类型</param>
/// <returns></returns>
private static Func<object, string> CreateSerializeObjectFunc(Type classType)
{
var method = classType.GetMethod("SerializeObject", new[] { typeof(object) });
if (method == null)
{
return null;
}
return (Func<object, string>)method.CreateDelegate(typeof(Func<object, string>));
} /// <summary>
/// 创建DeserializeObject方法的委托
/// </summary>
/// <param name="classType">JsonConvert类型</param>
/// <returns></returns>
private static Func<string, Type, object> CreateDeserializeObjectFunc(Type classType)
{
var method = classType.GetMethod("DeserializeObject", new[] { typeof(string), typeof(Type) });
if (method == null)
{
return null;
}
return (Func<string, Type, object>)method.CreateDelegate(typeof(Func<string, Type, object>));
}
}
}
关于WebApiClient
这是一个让你描述做什么请求,而不是叫你如何请求的httpClient客户端项目,关注一下,你会眼前一亮,get到新思想。
github:https://github.com/xljiulang/WebApiClient
一种优雅的条件引用第三方.net库的方法的更多相关文章
- 【Angular】关于angular引用第三方组件库无法改变其组件样式 :host ::ng-deep
[Angular]关于angular引用第三方组件库无法改变其组件样式 :host ::ng-deep css修改:无效 .ant-input-affix-wrapper .ant-input:not ...
- 如何在 FineUIMvc 中引用第三方 JavaScript 库
声明:FineUIMvc(基础版)是免费软件,本系列文章适用于基础版. 引入第三方颜色选择器 在 FineUIMvc 中使用第三方 JavaScript 遵循一定的约定,也非常简单. 下面以官网示例为 ...
- ionic3.0--angular4.0 引入第三方插件库的方法
ionic3.0 引入第三方插件 (swiper),方法很多,现详细说明下官方推荐(typings)做法. 1.全局安装Typings 1. npm install -g typings 2.搜索你 ...
- Use Swift Dynamic Framework (如何科学地引用第三方 Swift 库)
转自:http://andelf.github.io/blog/2014/07/07/use-swift-dynamic-library/ CocoaPods 由于完全使用静态链接解决方法,过度依赖 ...
- css引用第三方字体库
对应的CSS文件中如下方式进行字体库的引用: @font-face { font-family: '造字工房情书'; src: url('../fonts/MFQingShu_Noncommercia ...
- Xamarin引用第三方包错误解决方法
http://www.cnblogs.com/ThenDog/p/7623720.html
- 解决vue webApp使用lib-flexible和px2rem引用第三方ui库后,样式变小问题
首先,需要卸载项目中的postcss-px2rem. npm uninstall postcss-px2rem --save-dev 其次,安装postcss-px2rem-exclude npm i ...
- NDK 链接第三方静态库的方法
将NDK编译的第三方静态拷贝到JNI目录下,在Android.mk中添加如下代码 以openssl静态库(libcrypto-static.a)为例 第一种链接方法:LOCAL_LDFLAGS := ...
- vs增加第三方依赖库的方法总结
转自http://blog.csdn.net/raodotcong/article/details/8998379 先说说普通的两种方法: 方法1: 通过设置工程配置来添加lib库. 转自网上: A ...
随机推荐
- phpstorm快捷键记录
快捷键记录 Ctrl + N 按类名查找Ctrl + Shift + N 按文件名查找,快速查找文件Ctrl + Shift+Alt+N 根据函数名查找Ctrl + F 当前文件查找Ctrl + Sh ...
- [CSS 混合模式]——mix-blend-mode/background-blend-mode简介
mix-blend-mode/background-blend-mode CSS3真是有很多的神奇的地方,这个两个元素你知道吗? 这是张大大拿过来的图,关于混合模式,借图一用. mix-blend-m ...
- 利用KindEditor实现公司通讯录的维护
引言: 本人所属施工单位,在建项目较多,通讯录是以项目为单位挂接在公司内部网站通讯录板块,以静态页面展示.一直以来都是项目部办公室通过电话.邮件等方式通知总部信息部门变更通讯录,日常维护的工作量较大. ...
- springBoot actuator监控配置及使用
准备环境: 一个springBoot工程 第一步:添加以下依赖 <dependency> <groupId>org.springframework.boot</group ...
- 如何使用 Secret?- 每天5分钟玩转 Docker 容器技术(108)
我们经常要向容器传递敏感信息,最常见的莫过于密码了.比如: docker run -e MYSQL_ROOT_PASSWORD=my-secret-pw -d mysql 在启动 MySQL 容器时我 ...
- Python笔记·第六章——字典 (dict) 的增删改查及其他方法
字典是python中唯一的映射类型,采用键值对(key-value)的形式存储数据.python对key进行哈希函数运算,根据计算的结果决定value的存储地址,所以字典是无序存储的,且key必须是可 ...
- C#将dataGridView中显示的数据导出到Excel(大数据量超有用版)
开发中非常多情况下须要将dataGridView控件中显示的数据结果以Excel或者Word的形式导出来,本例就来实现这个功能. 因为从数据库中查找出某些数据列可能不是必需显示出来,在dataGrid ...
- codevs1050
题目地址:http://codevs.cn/problem/1050/ 分析: 最開始想直接用状压做,发现怎么都想不出来.就和当年的多行多米诺骨牌(这道题至少最后还是把普通状压做法看懂了). 直到听到 ...
- Caffe-5.2-(GPU完整流程)训练(依据googlenet微调)
上一篇使用caffenet的模型微调.但由于caffenet有220M太大,測试速度太慢.因此换为googlenet. 1. 训练 迭代了2800次时死机,大概20分钟. 使用的是2000次的模型. ...
- 谈谈单元測试之(二):測试工具 JUnit 3
前言 上一篇文章<为什么要进行烦人的单元測试?>讨论了一下现阶段软件开发中,程序猿们測试情况的现状.这篇文章中,我打算介绍一下单元測试的工具(插件).而且推荐大家以后在开发中,真正的用上单 ...