将Asp.Net Core和corefx移植到.Net 4.0
引言
因为工作内容的原因需要兼容 XP,而 XP 最多支持到.Net Framework 4.0。因此无法享受到 .Net Core 带来的一堆很好用的库,好在无论 corefx 还是 Asp.Net Core 都是开源的,我们可以自行修改编译出支持 .Net 4.0 的版本。
技术可行性
Net 4.0 相比 4.5 和 netstandard 1.0,主要的差别有:
- System.Threading.Tasks.Task 类型。.Net 4.0 的 Task 没有 GetAwaiter 成员,编译器无法生成使用 async await 的代码。好在编译器查找 Task.GetAwaiter 并不是直接查找成员方法,如果是扩展方法也可以,那么我们通过对 Task 编写 GetAwaiter 扩展方法就可以在 .Net 4.0 中使用 async await 了。
- Reflection 类别的 API。.Net 4.5 对反射 API 进行了重构,分离出了 TypeInfo 用于运行时反射。在 .Net 4.0 中,我们可以自行编写一个 TypeInfo 类,但是 .Net 4.5 中 TypeInfo 继承自 Type。据我观察实现中 GetType 返回的对象实际上就是一个 TypeInfo 对象,这一点我们无法通过自己编写的 TypeInfo 做到,不过好在除了 mscorlib,其他库并没有用到这层继承关系,因此为了不产生问题我实现的 TypeInfo 没有继承自 Type。
- WeakReference<T>。.Net 4.5 中的 WeakReference<T> 并没有继承自 WeakReference,它的终结器方法是一个 InternalCall,也就是在 clr 中实现的。而我们无法修改 clr 的实现,因此我实现的 WeakReference<T> 继承自 WeakReference,重用了它的终结器。
- 其它一些类型中 API 的缺失。主要包括 GC、Cryptography。对于没有的类型,我们可以添加实现,但对于已存在的类型是没有办法进行修改的。虽然 clr 中有 TypeForwardTo,配合 assembly redirecting 技术可以替换类型的实现,但 .Net 4.0 编译的程序集默认引用了 mscorlib,而 mscorlib 并不能被 redirect,所以对于 mscorlib 中已存在类型的 API 缺失——这一点暂时没有办法解决。
已经移植的项目
corefx
- System.Runtime:添加了 ExceptionDispatchInfo、IReadOnlyCollection<T>等一些只读集合的接口、生成异步方法所需要的 AsyncStateMachineAttribute、以及使用 MVVM 中很常用的 CallerMemberNameAttribute 等
- System.AppContext:添加了 AppContext 类
- System.Runtime.CompilerServices.Unsafe:添加了 Unsafe 类(ref 和 指针转换、直接读写内存等)
- System.Threading:添加了 Volatile 类
- System.Threading.Tasks:添加支持 async await 相关的类
- System.Security.Cryptography.Algorithms:添加了 IncrementalHash 的实现
Asp.Net Core
- Microsoft.Extensions.DependencyInjection
- Microsoft.Extensions.Options
- Microsoft.Extensions.Configuration
这些写过 Asp,Net Core 的应该很熟悉,他们也可以用在普通的 .Net 桌面程序中
一些开源项目
- Autofac:一个功能很强大的 IoC 实现
- AutoMapper:对象间的映射
- MaterialDesignThemes:WPF 的 MaterialDesign
示例
- 新建一个 .Net 4.0 项目
- 在 Nuget 程序包源里加上 https://www.myget.org/F/dotnet40/api/v3/index.json,并将优先级调到最上面
- install-package System.Threading.Tasks -Version 4.3.0-net40(注意一定要加上 Version)
- 在 app.config 加上 assembly redirecting
<dependentAssembly>
<assemblyIdentity name="System.Runtime" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
<bindingRedirect oldVersion="0.0.0.0-4.1.1.0" newVersion="4.1.1.0" />
</dependentAssembly>
<dependentAssembly>
<assemblyIdentity name="System.Threading" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
<bindingRedirect oldVersion="0.0.0.0-4.0.12.0" newVersion="4.0.12.0" />
</dependentAssembly>
<dependentAssembly>
<assemblyIdentity name="System.Threading.Tasks" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
<bindingRedirect oldVersion="0.0.0.0-4.0.12.0" newVersion="4.0.12.0" />
</dependentAssembly>
- 然后你就可以愉快的 async await 了
下面的示例是使用了
Caliburn.Micro
Microsoft.Extensions.DependencyInjection
Microsoft.Extensions.Configuration
Autofac
Autofac.Extensions.DependencyInjection
AutoMapper
AutoMapper.Extensions.Microsoft.DependencyInjection
public class AppBootstrapper : BootstrapperBase
{
public IConfiguration Configuration { get; }
public IServiceProvider ServiceProvider { get; private set; }
private IContainer _container; public AppBootstrapper()
{
Configuration = LoadConfiguration();
Initialize();
} private IConfiguration LoadConfiguration()
{
var builder = new ConfigurationBuilder()
.SetBasePath(Directory.GetCurrentDirectory())
.AddJsonFile("config.json", false, false);
return builder.Build();
} protected override void Configure()
{
var serviceCollection = new ServiceCollection();
ServiceProvider = ConfigureServices(serviceCollection);
} public IServiceProvider ConfigureServices(IServiceCollection services)
{
services.AddOptions();
services.AddAutoMapper(AssemblySource.Instance.ToArray());
services.AddSingleton<IWindowManager>(new WindowManager());
services.AddSingleton<IEventAggregator>(new EventAggregator()); services.AddSingleton(p => _container); var builder = new ContainerBuilder();
builder.Populate(services);
builder.RegisterAssemblyModules(AssemblySource.Instance.ToArray()); _container = builder.Build();
return new AutofacServiceProvider(_container);
}
}
看起来和在 Asp.Net Core 中没什么差别。
总结
虽然工作环境限制我们只能使用 .Net 4.0,但俗话说没有条件,创造条件也要上。将它们移植到 .Net 4.0 也是跟上 .Net Core 和开源的步伐的一种努力吧。
关于这些包和相关的版本号可以在 https://www.myget.org/feed/Packages/dotnet40 查看
关于移植到 .Net 4.0 的计划我创建了一个 github 组织,里面包含移植的所有项目 https://github.com/dotnet40/
最后,感谢大家花时间阅读!
将Asp.Net Core和corefx移植到.Net 4.0的更多相关文章
- Asp.Net Core 项目实战之权限管理系统(0) 无中生有
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
- Asp.net Core的代码移植技巧,半天将SqlSugarORM转成Core
.net core中有哪些被抛弃的类 1.DataTable DataRow SqlDataAdapter DataRow DataColumn DataColumn 虽然这些类不是我ORM核心功能 ...
- ASP.NET Core 2.1 : 十.升级现有Core2.0 项目到2.1
.NET Core 2.1 终于发布了, 赶紧升级一下. 一. 安装SDK 首先现在并安装 SDK(64-bit) 安装完毕后如果新建项目可以看到已经有2.1的选项了 二. 更新现有2.0项目到2.1 ...
- 细说ASP.NET Core与OWIN的关系
前言 最近这段时间除了工作,所有的时间都是在移植我以前实现的一个Owin框架,相当移植到到Core的话肯定会有很多坑,这个大家都懂,以后几篇文章可能会围绕这个说下,暂时就叫<Dotnet Cor ...
- Asp.Net Core 项目实战之权限管理系统(4) 依赖注入、仓储、服务的多项目分层实现
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
- Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
- Asp.Net Core 项目实战之权限管理系统(2) 功能及实体设计
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
- Asp.Net Core 项目实战之权限管理系统(3) 通过EntityFramework Core使用PostgreSQL
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
- Asp.Net Core 项目实战之权限管理系统(5) 用户登录
0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...
随机推荐
- cas单点登录系统:客户端(client)详细配置
最近一直在研究cas登录中心这一块的应用,分享一下记录的一些笔记和心得.后面会把cas-server端的配置和重构,另外还有这几天再搞nginx+cas的https反向代理配置,以及cas的证书相关的 ...
- java面试基础题(三)
程序员面试之九阴真经 谈谈final, finally, finalize的区别: final:::修饰符(关键字)如果一个类被声明为final,意味着它不能再派生出新的子类,不能作为父类被继承.因此 ...
- github--hello,world(参考官网)
官网:https://guides.github.com/activities/hello-world/ 一共分为5步. 1.为你的项目新建仓库(repository): 2.新建分支(branch) ...
- JavaScript对象之document对象
DOM对象之document对象 DOM对象:当网页被加载时,浏览器会创建页面的文档对象模型(Document Object Model). HTML DOM 模型被构造为对象的树. 打开网页后,首先 ...
- 使用Github+Hexo框架搭建部署自己的博客
前言 Hexo 是一个快速.简洁且高效的博客框架.Hexo 使用 Markdown (或其他渲染引擎 )解析文章, 在几秒内,即可利用靓丽的主题生成静态网页. 安装 安装前提 安装 Hexo 相当简单 ...
- MySQL NULL值
我们已经看到SQL SELECT命令和WHERE子句一起使用,来从MySQL表中提取数据, 但是,当我们试图给出一个条件,比较字段或列值设置为NULL,它确不能正常工作. 为了处理这种情况,MySQL ...
- python3中socket套接字的编码问题解决
一.TCP 1.tcp服务器创建 #创建服务器 from socket import * from time import ctime #导入ctime HOST = '' #任意主机 PORT = ...
- 6.vue如何上传到svn
node_module是不需要上传的,先删掉,然后用tortoiseSvn的TortoiseSVN Repository Browser,ADD folder,选择工程文件,就行,checkout下来 ...
- CSS样式表之background背景
[CSS常用背景属性]:background (缩写形式) background-color:背景色 background-image:背景图 url中放图片地址,背景图和背景色同时存在时,背景图覆 ...
- EF查询百万级数据的性能测试
一.起因 个人还是比较喜欢EF的,毕竟不用写Sql,开发效率高,操作简单,不过总是听人说EF的性能不是很好,也看过别人做的测试,但是看了就以为真的是那样.但是实际上到底是怎么样,说实话我真的不知道. ...