上篇(.Net基础体系和跨框架开发普及)介绍了.Net当前生态下的大概情况,也分享了简单实现的过程,这篇文章就是讲解我的OSS.Common项目扩展.Net Standard 支持的过程,主要集中在:方案的选择,移植检测,移植过程,常见问题的解决思路,以及nuget的打包部署这几个方面。

  在开始之前简单介绍下OSS.Common项目的涉及的内容,方便了解后边所遇到的问题。这个类库是相对简单的一个基础类库,提供主要有以下的内容:1. 基础用户系统设备信息实体定义,2. 主流加解密方案实现(md5,aes,sha1,hmacsha1), 3.  常规实体DTO转化,静态扩展方法(时间,字符串等等处理), 4. 基础日志,缓存,异步辅助静态类及默认方案实现(采用了简单Provider模式,使用者可以注册自己的实现方案), 5. 全局结果,分页实体定义。  其主要作用就是完成对常见碎化方法的收集规整,方便在业务逻辑中减少不必要的消耗,了解之后我们进入具体的扩展过程。

  一. 方案的选择

  这个其实在上篇文章中已经做了介绍,当前.net core ,.net Framework,mono for Xarmain等都有自己的运行时,虽然使用的都是C#语法,但是类库在不使用可移植或者标准库的前提下不能直接互相调用。随着.net standard 2.0的即将推出,.net core(asp.net core) 的应用场景会越来越普遍,对旧有项目的兼容需求会越来越强烈,oss.common也是遇到这个问题,所以我将对它进行.net standard支持的扩展。

  由于当前项目现在在好几个.net framework的项目中还在使用,为了旧项目中对net45的版本的支持不能丢失,所以我会保留两套解决方案,一个为.net Standard 提供支持,一个为.net framework使用,两个类库项目,共享同一套代码文件,针对Framework特有功能通过条件编译来控制。github上目录结构已更新,欢迎查看

  二. 移植检测

  在移植之前我们需要对移植有个大概评估,了解需要代码改动的覆盖面积,确定代码的可移植性,这里推荐使用微软官方提供的移植检测工具(ApiPort),或者使用它的VS扩展。这里我使用的是vs插件,安装完插件之后,打开解决方案,查看右键菜单会有如下两个选项:

  首先,点击第二个选项,配置要检测的移植对比版本,如下图:

  完成对应的检测版本之后点击确定,点击第一个选项,执行分析过程,会生成html和xsl两种报表,html报表界面如下所示:

报表中会给出对应版本的接口覆盖情况,以及相关的建议,可以说是比较详细了。

  三.移植过程

  经过上边的检测,可以看出oss.common项目 在.net standard1.4下,大概超过20%的代码不能直接提供支持,我看了一下,主要集中在涉及配置,缓存,反射等特有属性相关代码中,这个还算在预期之中,不过看到一堆的红叉叉还是一阵头疼,没办法,自己的类库,哭着也要码完,下边介绍下移植的步骤。

  1. 添加项目文件

  为了项目直观和方便管理,我将原来的OSS.Common类库修改名称为OSS.Common.NET45,新建一个OSS.Common的标准库项目,两个项目文件放在同一目录下,说明一下,vs2015如果要建标准库项目需要先建可移植类库,在类库属性页修改,如果不清楚请看上一篇文章介绍。

  这个时候如果你直接生成OSS.Common.NET45的项目,是会出现报错的,哪怕你没有做任何实际的代码的操作,主要是因为添加可移植类库需要project.json的文件进行依赖管理,当他们在同一目录下时,nuget会把project.json中的依赖默认执行还原操作,虽然你当前是在生成OSS.Common.NET45项目,没办法,就是这么傻,如果你遇到了这个错误,在当前目录中再建一个对应当前项目文件的project.json文件就好了,这里我添加了OSS.Common.Net45.project.json文件,文件中添加如下代码:

{
"frameworks": {
"net45": {}
},
"runtimes": {
"win": {}
}
}

  2. 代码集成

  新建好对应的解决方案之后,把代码文件附件到新建的标准库下,这个时候直接生成会有很多错误,这个时候我们就需要祭出条件编译这个大招了,因为以后主要是维护标准库,所以我在旧NET45的旧项目上新建了NETFW的条件编译符号 ,剩下的就是一个个错误完善了。

  在处理兼容的过程中,主要会面临这几个问题,1. 标准库完全不支持   2.  标准库和Framework的调用方法不一样, 3. 可以间接完成标准库的实现

  这里我把我遇到的情况各举一个例子供大家参考:

  1.  标准库完全不支持,这个最典型的就是缓存模块,在.net standard下,System.Runtime.Caching类库完全被移除了,没办法,只能使用#if NETFW 完全把Module模块下的默认Cache实现给屏蔽了,只能在Framework下才能使用默认实现(本来打算自己实现一个缓存类的,不过发现可能会带来不可预知bug,作废)。

  2. 标准库和Framework的调用方法不一样,举个例子就是Type类型下的IsEnum属性,在net standard下需要.gettypeinfo().IsEnum才可以,举例代码:

#if NETFW
if (!enType.IsEnum)
#else
if (!enType.GetTypeInfo().IsEnum)
#endif

  3. 可以间接完成标准库的实现,这常见的如 list的ConvertAll方法,在Framework下有默认实现的,标准库下是没有的,这里我在ConvertExtention类自己定义了个一个:

#if !NET40
public static List<TResult> ConvertAll<TPara, TResult>(this List<TPara> list, Func<TPara, TResult> func)
{
if (list == null)
return null;
var resultList = new List<TResult>(list.Count);
list.ForEach(e => resultList.Add(func(e)));
return resultList;
}
#endif

当然还会有其他的一些问题,不过还好,基本都已经解决,如果有不清楚的可以去下载oss.common代码自行查看

  四. nuget打包部署

  这个相对简单,在两个解决方案中分别生成对应的dll,在lib文件夹中分别添加net45 和 netstandard1.4 文件夹添加对应的dll就行。

需要注意的一点就是,最好添加个各自的依赖,举个例子,标准库的Hmacsha1加密算法在“System.Security.Cryptography.Algorithms” dll程序集下,如果在调用项目中没有引用这个dll,生成是不会报错的,但是当代码执行调用的时候就会弹出程序集未找到的错误,当然如果发现这个问题也可以通过nuget线上安装命令(install-package)安装。

  给大家看下我的nuget文件配置:

如有其它疑问,欢迎关注公众号(osscoder):

OSS.Common扩展.Net Standard支持实例分享的更多相关文章

  1. .Net Standard扩展支持实例分享

    上篇(.Net基础体系和跨框架开发普及)介绍了.Net当前生态下的大概情况,也分享了简单实现的过程,这篇文章就是讲解我的OSS.Common项目扩展.Net Standard 支持的过程,主要集中在: ...

  2. OSS.Common获取枚举字典列表标准库支持

    上篇(.Net Standard扩展支持实例分享)介绍了OSS.Common的标准库支持扩展,也列举了可能遇到问题的解决方案.由于时间有限,同时.net standard暂时还没有提供对Descrip ...

  3. PHPnow开启PHP扩展里openssl支持的方法

    PHPnow 是 Win32 下绿色的 Apache + PHP + MySQL 环境套件包.简易安装.快速搭建支持虚拟主机的 PHP 环境.更多介绍<PHP服务套件 PHPnow1.5.6&g ...

  4. 效率神器 Workflow 实例分享

    WorkflowShare Workflow实例分享,Github链接:WorkflowShare logo.jpg 苹果公司收购 Workflow 并将其完全免费,作为一款效率类 APP,Workf ...

  5. Python模拟登陆淘宝并统计淘宝消费情况的代码实例分享

    Python模拟登陆淘宝并统计淘宝消费情况的代码实例分享 支付宝十年账单上的数字有点吓人,但它统计的项目太多,只是想看看到底单纯在淘宝上支出了多少,于是写了段脚本,统计任意时间段淘宝订单的消费情况,看 ...

  6. Java实现MD5加密及解密的代码实例分享

    链接:http://www.jb51.net/article/86027.htm Java实现MD5加密及解密的代码实例分享 作者:厦门大学陈黎栋 字体:[增加 减小] 类型:转载 时间:2016-0 ...

  7. Android经典项目开发之天气APP实例分享

    原文:Android经典项目开发之天气APP实例分享 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/mzc186/article/details/5 ...

  8. C# 9.0新特性详解系列之二:扩展方法GetEnumerator支持foreach循环

    1.介绍 我们知道,我们要使一个类型支持foreach循环,就需要这个类型满足下面条件之一: 该类型实例如果实现了下列接口中的其中之一: System.Collections.IEnumerable ...

  9. YOLO3升级优化版!Poly-YOLO:支持实例分割!

    YOLO3升级优化版!Poly-YOLO:支持实例分割! POLY-YOLO: HIGHER SPEED, MORE PRECISE DETECTION AND INSTANCE SEGMENTATI ...

随机推荐

  1. OC语言的特性(二)-Block

    本篇文章的主要内容 了解何谓block. 了解block的使用方法. Block 是iOS在4.0版本之后新增的程序语法. 在iOS SDK 4.0之后,Block几乎出现在所有新版的API之中,换句 ...

  2. Sping--IOC概念

    1. 新建项目, 引入spring包(sping, common-annotation, common-logging包), 还有junit包. user.java: package com.bjsx ...

  3. Keil MDK下如何设置非零初始化变量(转)

    源:Keil MDK下如何设置非零初始化变量 一些工控产品,当系统复位后(非上电复位),可能要求保持住复位前RAM中的数据,用来快速恢复现场,或者不至于因瞬间复位而重启现场设备.而keil mdk在默 ...

  4. sublime text 调出结果输出框

    sublime是一个非常好用的代码编辑器,同时可以build program 但是在执行代码的过程中,如果进行了查找等操作,下面原来显示输出框的地方被查找界面替代,而程序结果输出框就会"消失 ...

  5. (一)Javascript基础知识

    一,五种基本数据类型和一种复合数据类型. 五种基本数据类型 1,undefined 2,null 3,string 4,number 5,boolean 6,复合数据类型(Object,Array,D ...

  6. UVa 10176 - Ocean Deep ! - Make it shallow !!

    题目大意:判断一个很大的二进制能否被131071整除.在二进制转十进制的过程中不断取模,最后判断结果是否是0就可以了. #include <cstdio> #include <cst ...

  7. 使用命令创建github代码仓库,push本地仓库到github远程代码仓库

    1.利用命令创建github远程代码仓库 在将本地代码push到github远程代码仓库之前,总是需要新建github代码仓库,在将本地仓库关联到github远程仓库.其中最为繁琐的操作是建立gith ...

  8. linux在线预览pdf文件开发思路

    准备:swftools,flexpaper 基本思路: 1,将pdf文件转化成swf文件 2,使用flexpaper预览swf文件 主要代码: 1,在linux中安装swftools.官网下载swft ...

  9. PHP 性能追踪及分析工具(XHPROF)

    原文:https://gold.xitu.io/post/5860d23f128fe10069e1cfbf XHPROF:Facebook 开源的轻量级PHP性能分析工具. 它报告函数级别的请求次数和 ...

  10. R语言中的if-else语句写法

    结构  1 :  if()  xx  else    yy    一行: 结构  2:   if()  {xx} else  {yy} 或者   if(){ xx }else    #此处不能两行写 ...