引言

因为工作内容的原因需要兼容 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
  1. <dependentAssembly>
  2. <assemblyIdentity name="System.Runtime" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
  3. <bindingRedirect oldVersion="0.0.0.0-4.1.1.0" newVersion="4.1.1.0" />
  4. </dependentAssembly>
  5. <dependentAssembly>
  6. <assemblyIdentity name="System.Threading" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
  7. <bindingRedirect oldVersion="0.0.0.0-4.0.12.0" newVersion="4.0.12.0" />
  8. </dependentAssembly>
  9. <dependentAssembly>
  10. <assemblyIdentity name="System.Threading.Tasks" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" />
  11. <bindingRedirect oldVersion="0.0.0.0-4.0.12.0" newVersion="4.0.12.0" />
  12. </dependentAssembly>
  • 然后你就可以愉快的 async await 了

下面的示例是使用了

Caliburn.Micro

Microsoft.Extensions.DependencyInjection

Microsoft.Extensions.Configuration

Autofac

Autofac.Extensions.DependencyInjection

AutoMapper

AutoMapper.Extensions.Microsoft.DependencyInjection

  1. public class AppBootstrapper : BootstrapperBase
  2. {
  3. public IConfiguration Configuration { get; }
  4. public IServiceProvider ServiceProvider { get; private set; }
  5. private IContainer _container;
  6.  
  7. public AppBootstrapper()
  8. {
  9. Configuration = LoadConfiguration();
  10. Initialize();
  11. }
  12.  
  13. private IConfiguration LoadConfiguration()
  14. {
  15. var builder = new ConfigurationBuilder()
  16. .SetBasePath(Directory.GetCurrentDirectory())
  17. .AddJsonFile("config.json", false, false);
  18. return builder.Build();
  19. }
  20.  
  21. protected override void Configure()
  22. {
  23. var serviceCollection = new ServiceCollection();
  24. ServiceProvider = ConfigureServices(serviceCollection);
  25. }
  26.  
  27. public IServiceProvider ConfigureServices(IServiceCollection services)
  28. {
  29. services.AddOptions();
  30. services.AddAutoMapper(AssemblySource.Instance.ToArray());
  31. services.AddSingleton<IWindowManager>(new WindowManager());
  32. services.AddSingleton<IEventAggregator>(new EventAggregator());
  33.  
  34. services.AddSingleton(p => _container);
  35.  
  36. var builder = new ContainerBuilder();
  37. builder.Populate(services);
  38. builder.RegisterAssemblyModules(AssemblySource.Instance.ToArray());
  39.  
  40. _container = builder.Build();
  41. return new AutofacServiceProvider(_container);
  42. }
  43. }

看起来和在 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的更多相关文章

  1. Asp.Net Core 项目实战之权限管理系统(0) 无中生有

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

  2. Asp.net Core的代码移植技巧,半天将SqlSugarORM转成Core

    .net  core中有哪些被抛弃的类 1.DataTable DataRow SqlDataAdapter DataRow DataColumn DataColumn 虽然这些类不是我ORM核心功能 ...

  3. ASP.NET Core 2.1 : 十.升级现有Core2.0 项目到2.1

    .NET Core 2.1 终于发布了, 赶紧升级一下. 一. 安装SDK 首先现在并安装 SDK(64-bit) 安装完毕后如果新建项目可以看到已经有2.1的选项了 二. 更新现有2.0项目到2.1 ...

  4. 细说ASP.NET Core与OWIN的关系

    前言 最近这段时间除了工作,所有的时间都是在移植我以前实现的一个Owin框架,相当移植到到Core的话肯定会有很多坑,这个大家都懂,以后几篇文章可能会围绕这个说下,暂时就叫<Dotnet Cor ...

  5. Asp.Net Core 项目实战之权限管理系统(4) 依赖注入、仓储、服务的多项目分层实现

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

  6. Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

  7. Asp.Net Core 项目实战之权限管理系统(2) 功能及实体设计

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

  8. Asp.Net Core 项目实战之权限管理系统(3) 通过EntityFramework Core使用PostgreSQL

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

  9. Asp.Net Core 项目实战之权限管理系统(5) 用户登录

    0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用AdminLTE搭建前端 2 Asp.Net Core 项目实战之 ...

随机推荐

  1. python读写文件相关内容

    1.python读取文件 f = open('E:/info.txt', 'r')#用read的方式打开a = 0for line in f.readlines():读取没一行,就是读取所有文件的意思 ...

  2. Sublime常用插件

    注:此插件为我自己在用的,仅代表个人,如果发现好用的插件,会不断更新此博文. 1,package control 我们用sublime几乎都会首先安装这个插件,这个插件是管理插件的功能,先安装它,再安 ...

  3. Spring学习(12)--- @Autowired与@Resource 对比

    Spring不但支持自己定义的@Autowired注解,还支持由JSR-250规范定义的几个注解,如:@Resource. @PostConstruct及@PreDestroy. 1. @Autowi ...

  4. Ajax 跨域提交表单

    跨域提交表单,前端ajax不用做任何修改, 只需要在后端调用的方法里面添加一行代码即可. .NET 版 HttpContext.Response.AddHeader("Access-Cont ...

  5. 【知识整理】这可能是最好的RxJava 2.x 入门教程(四)

    这可能是最好的RxJava 2.x入门教程系列专栏 文章链接: 这可能是最好的RxJava 2.x 入门教程(一) 这可能是最好的RxJava 2.x 入门教程(二) 这可能是最好的RxJava 2. ...

  6. 数组 list互转

    数组 list互转 String str[] = list.toArray(new String[]{}); List list= java.util.Arrays.asList(String str ...

  7. hibernate sql查询转换成VO返回list

    hibernate sql查询转换成VO @Override public List<FenxiVo> getTuanDuiFenxiList(FenxiVo FenxiVo,Intege ...

  8. Linux下memcached安装与连接

    前几天技术总监要我在项目中加一个memcached,以前也从来没有配置过,所以就去网上找教程,最终折腾成功.比较坑的就是sasl协议那里. 由于memcached依赖libevents,所以要下载两个 ...

  9. Redis的安装与使用(单节点)

    IP:192.168.4.111 环境:CentOS 6.6 Redis版本:redis-3.0 (考虑到Redis3.0在集群和性能提升方面的特性,rc版为正式版的候选版,而且很快就出正式版) 安装 ...

  10. Linux 下开启ssh服务

    在虚拟机里安装完后,想往虚拟机里拷贝软件 当使用Xmanager ftp连接时候 出现 连接不上的问题,其解决方案如下: 1.修改sshd_config文件,命令为:vim /etc/ssh/sshd ...