我是如何把 Java 项目移植到 .NET 5.0 的

伴随着 IP 位置库 的上线,笔者的“童年梦想”又成真了一个。为了分发这份来之不易的数据库,笔者找到了 ip2region 项目。该项目提供了一种体积小且查询速度极快的离线IP位置数据库文件格式,同时提供了多种语言支持的查询客户端。但 ip2region 项目的作者并未提供除 Java 以外的数据库文件生成代码,笔者打算为该项目移植 .NET 5.0 的数据库文件生成器,并在本文中记录下移植过程。

移植前准备

ip2region 的 Java 版数据库生成器 代码并不复杂,源代码文件只有 8 个。以笔者粗浅的 Java 经验来看,因为 C# 与 Java 大体相似,移植过程中无需对程序的结构和命名进行变更,也无需对处理逻辑进行调整。移植需要做的就是让程序可以编译通过,基本上就算成功。

开始移植

笔者新建了一个名为 IP2RegionDotNetDbMaker 的 .NET 5.0 控制台应用程序,删掉 Program.cs 文件并将所有的 Java 文件复制到项目中。

下一步操作很暴力,就是直接将源代码的后缀从 .java 改为 .cs 。为此,笔者在 LINQPad 中写了一段小代码,来完成这个操作:

var dir = @"D:\coderbusy.com\demo\IP2RegionDotNetDbMaker\IP2RegionDotNetDbMaker";var javaFiles = Directory.GetFiles(dir, "*.java");foreach (var javaFile in javaFiles){var _ = Path.GetDirectoryName(javaFile);var fileName = Path.GetFileNameWithoutExtension(javaFile);var csFile = Path.Combine(_, fileName + ".cs");File.Move(javaFile, csFile);}

在暴力改名之后的源代码文件里,不出意外的报了很多错误:

需要先把 package 和 import 这两种语句去掉,然后把缺失的命名空间给加上。

var dir = @"D:\coderbusy.com\demo\IP2RegionDotNetDbMaker\IP2RegionDotNetDbMaker";var files = Directory.GetFiles(dir, "*.cs");foreach (var file in files){var lines = File.ReadAllLines(file, Encoding.UTF8);var builder = new StringBuilder();builder.AppendLine($"using System;{Environment.NewLine}namespace IP2RegionDotNetDbMaker{Environment.NewLine}{{");foreach (var line in lines){if (line.StartsWith("package ")){continue;}if (line.StartsWith("import ")){continue;}builder.AppendLine(line);}builder.AppendLine("}");var content = builder.ToString();File.WriteAllText(file, content, Encoding.UTF8);}

异常声明在 C# 中不支持,可以通过正则将其替换掉:

在替换时,确定开启“使用正则表达式”,查找项为:throws ([\w ,]+)Exception 替换项保持为空。之后,替换掉所有的 @Override 和 final 关键字。

在 C# 中 out 是一个关键字不能被当作类型使用,Java 编程中常用的 System.out.println 方法需要被替换成 Console.WriteLine ,直接全局替换搞定。

重构 DbMakerConfigException 类型:

using System;namespace IP2RegionDotNetDbMaker
{/*** configuration exception* * @author chenxin<chenxin619315@gmail.com>*/public class DbMakerConfigException : Exception{public DbMakerConfigException(string info) : base(info){}}}

在 .NET 5.0 中 Mock 实现 Java 所需的 API

新建 Mock.cs 文件,用于存放 Java API 到 C# API 的Mock 代码。使用扩展方法对 String 类型进行扩展,并实现 Java API 所用的方法:

    public static class Extensions{public static Int32 length(this string str){if (String.IsNullOrWhiteSpace(str)){return 0;}return str.Length;}public static string trim(this string str){return str.Trim();}public static char charAt(this string str, Int32 i){return str[i];}public static int indexOf(this string str, string value){return str.IndexOf(value);}public static int indexOf(this string str, char value, Int32 start){return str.IndexOf(value, start);}public static string substring(this string str, Int32 startIndex){return str.Substring(startIndex);}public static string substring(this string str, Int32 startIndex, Int32 endIndex){return str.Substring(startIndex, endIndex - startIndex);}public static bool equals(this string str1, string str2){return String.Equals(str1, str2, StringComparison.InvariantCultureIgnoreCase);}public static string[] split(this string str, string separator){return str.Split(separator);}public static byte[] getBytes(this string str){return Encoding.UTF8.GetBytes(str);}public static byte[] getBytes(this string str, string encoding){return Encoding.GetEncoding(encoding).GetBytes(str);}public static bool endsWith(this string str, string value){return str.EndsWith(value);}}

Mock 实现 StringBuilder 类型:

    public class StringBuilder{private readonly System.Text.StringBuilder _builder = new System.Text.StringBuilder();internal StringBuilder append(object value){_builder.Append(value);return this;}internal string toString(){return _builder.ToString();}}

Mock 实现 File 类型:

    public class File{public File(string ipSrcFile){_fileInfo = new FileInfo(ipSrcFile);}private FileInfo _fileInfo;public FileInfo FileInfo => _fileInfo;internal bool exists(){return _fileInfo.Exists;}}

Mock 实现 LinkedList 类型:

    public class LinkedList<T> : List<T>{internal T getFirst(){return this[0];}internal void add(T item){this.Add(item);}internal T getLast(){return this[this.Count - 1];}internal IEnumerable<T> iterator(){return this;}}

Mock 实现 HashMap 类型:

    public class HashMap<K, V> : Dictionary<K, V>{internal void put(K k, V v){this[k] = v;}internal bool containsKey(K key){return this.ContainsKey(key);}internal V get(K k){if (containsKey(k)){return this[k];}return default;}}

Mock 实现 FileReader 类型:

    public class FileReader{private File globalRegionFile;private Queue<String> _lines = new Queue<string>();public FileReader(File file){this.globalRegionFile = file;using (var fs = file.FileInfo.OpenRead()){using (var sr = new StreamReader(fs)){while (!sr.EndOfStream){var line = sr.ReadLine();_lines.Enqueue(line);}}}}internal string readLine(){if (_lines.TryDequeue(out var line)){return line;}return null;}internal void close(){}}

Mock 实现 BufferedReader 类型:

    public class BufferedReader{private FileReader fileReader;public BufferedReader(FileReader fileReader){this.fileReader = fileReader;}internal string readLine(){return fileReader.readLine();}internal void close(){fileReader.close();}}

Mock 实现 RandomAccessFile 类型:

    public class RandomAccessFile{private string dbFile;private Stream stream;internal void seek(long v){stream.Seek(v, SeekOrigin.Begin);}internal void write(byte[] vs){stream.Write(vs);}internal void readFully(byte[] dbBinStr, int v, int length){stream.Read(dbBinStr, v, length);}private string v;public RandomAccessFile(string dbFile, string v){this.dbFile = dbFile;this.v = v;this.stream = new FileStream(dbFile, FileMode.OpenOrCreate, FileAccess.ReadWrite);}public long length(){return this.stream.Length;}internal void close(){if (stream == null){return;}this.stream.Flush();this.stream.Close();this.stream = null;}internal long getFilePointer(){return stream.Position;}internal void write(int v){var bytes = BitConverter.GetBytes(v);stream.Write(bytes);}}

语法与属性修正

经过以上的 Mock 操作,报错部分便仅仅涉及语法和部分属性。

C# 中并不存在“扩展属性”类似的东西,所以 Java 中以“小驼峰”命名的 length 字段调用,据需要改为“大驼峰”方式的 Length 。位运算符 >>> 也需要改为 >> ,同时,还有几个语法错误需要修正。比如:C# 中并不支持 Java 中的 for(Type e:collection) 语法,需要用 foreach 来替代。之后,项目就可以编译通过了。

结果验证

将 data 目录拷贝至 bin 目录,使用以下命令便可启动生成:

dbMaker -src ./data/ip.merge.txt -region ./data/global_region.csv

伴随着大量的控制台输出,笔者似乎找到了黑客帝国的感觉。经过一小会儿的等待,生成已经成功执行。

通过二进制对比,该结果文件仅在行尾的日期存储部分与源文件不同:

通过阅读代码,文件末尾部分的数据是生成的时间戳和一小段声明信息。文件尾的不一致并不会对使用造成影响。这表示,这次移植是成功的。

开源地址

目前,该代码已经上传至 Gitee ,地址是:

https://gitee.com/coderbusy/demo/tree/master/IP2RegionDotNetDbMaker

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/306542.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

qt制作一个画板_如何直接用Sketch制作动画|Sketch插件|

本期的sketch插件来为大家讲解如何制作动画&#xff0c;看来看看马克笔设计留学的MUzi老师的教程吧&#xff01;安装1.下载并解压 Anima Tookit.ziphttps://timeline.animaapp.com/2.双击 Anima Tookit.sketchplugin 完成安装使用1.执行 Plugins > Anima Tookit > Show/H…

算法设计与分析——动态规划——最大字段和问题

动态规划解决问题是自底向上。原问题的规模是n个元 素。这n个元素不好考虑&#xff0c;我们先考虑n-1个元素&#xff0c;这样还不好考 虑&#xff0c;我们考虑n-2个元素&#xff0c;这样依次递减&#xff0c;最后问题规模变成一个 元素。但是我们发现&#xff0c;在递减的过程中…

如何在 Asp.Net Core 中 管理敏感数据

译文链接&#xff1a;https://www.infoworld.com/article/3576292/how-to-work-with-user-secrets-in-asp-net-core.html在应用程序开发时&#xff0c;你肯定会有一些特别需要保护的数据&#xff0c;这些数据通常是非常机密的&#xff0c;敏感的&#xff0c;禁止和别人共享&…

C#开源项目:SiMay远程控制管理系统

C#开源项目&#xff1a;SiMay远程控制管理系统Gitee仓库截图下方基于原项目仓库readme系统介绍SiMay远程控制管理系统是一个Windows远程控制系统&#xff0c;底层基于IOCP的异步通信模型&#xff0c;能对海量客户端实时监控&#xff0c;目前功能已实现&#xff1a;逐行扫描远程…

算法设计与分析——回溯法——01背包问题

//0-1背包问题 回溯法求解 #include<bits/stdc.h> #include <iostream>using namespace std; template<class Typew,class Typep> class Knap {public:Typep Bound(int i);void Backtrack(int i);Typew c; //背包容量int n; //物品数Typew *w; //物品重量数…

排列组合思维导图_排列组合——排列数专题

在上篇关于排列组合主要考点的介绍中&#xff0c;正男老师提到&#xff1a;排列组合考点通常可以拆分为排列数考点和组合数考点。排列数考点相关试题可以细分为2类&#xff0c;分别为&#xff1a;穷举问题和限制条件问题。本期正男老师就从近六年内的5道涉及排列数考点的真题入…

高级的说服,从不讲道理

大家好&#xff0c;我是Z哥。你会发现有一些人&#xff0c;他们好像说话从来都没有说服过别人&#xff0c;天天被别人牵着鼻子走。但有些人一说的话你就觉得特别对&#xff0c;就应该按照他说的办。很明显&#xff0c;我们都希望自己是后者。能不能说服人&#xff0c;不管在生活…

算法设计与分析——分支限界法——n皇后问题

一、问题描述 问题描述&#xff1a;在nn格的棋盘上放置彼此不受攻击的n个皇后。按照国际象棋的规则&#xff0c;皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。n皇后问题等价于在n*n的棋盘上放置n个皇后&#xff0c;任何2个皇后不放在同一行或同一列或同一斜线上。 …

IdentityServer4系列 | 授权码模式

一、前言在上一篇关于简化模式中&#xff0c;通过客户端以浏览器的形式请求「IdentityServer」服务获取访问令牌&#xff0c;从而请求获取受保护的资源&#xff0c;但由于token携带在url中&#xff0c;安全性方面不能保证。因此&#xff0c;我们可以考虑通过其他方式来解决这个…

算法设计与分析——算法思想总结

算法设计与分析 1、分治法 分治法的基本思想是将一个规模为n的问题分解为k个规模较小的子问题&#xff0c;这些子问题相互独立且与原问题相同。递归的解这些子问题&#xff0c;然后将各子问题的解合并得到原问题的解。 分治法所能解决的问题一般具有以下几个特征&#xff1a…

pearson相关系数_Pearson(皮尔逊)相关系数

由于使用的统计相关系数比较频繁&#xff0c;所以这里就利用几篇文章简单介绍一下这些系数。相关系数&#xff1a;考察两个事物(在数据里我们称之为变量)之间的相关程度。如果有两个变量&#xff1a;X、Y&#xff0c;最终计算出的相关系数的含义可以有如下理解&#xff1a;(1)、…

聊一聊ABP vNext的模块化系统

官网&#xff1a;https://abp.io/开源&#xff1a;https://github.com/abpframework/abp EasyAbp&#xff1a;https://easyabp.io/Abp 模块&#xff1a;https://abp.io/packages模块化系统ABP vNext 的世界观在 Abp vNext 框架里面&#xff0c;模块系统是整个框架的基石&#x…

双离合档把上按钮作用_英特尔展示双屏幕概念笔记本:带有双铰链

本文转自&#xff1a;IT之家作者&#xff1a;嗜橙近日&#xff0c;英特尔在位于圣克拉拉总部深处的一个半秘密实验室里&#xff0c;公布了配备两个屏幕的概念笔记本电脑。在近日的台北电脑展上&#xff0c;华硕发布了双屏笔记本&#xff1b;不久之后&#xff0c;英特尔也公布了…

浏览器眼中的0

0作为一个特殊的符号&#xff0c;经常会跟浏览器打交道&#xff0c;在不同的场景下&#xff0c;0代表的意思不尽相同&#xff0c;因此浏览器眼中的0不一定就是符合人们感官上的认识&#xff0c;那究竟浏览器会怎么对待它呢&#xff0c;今天我们就来探究一下各种场景中0的含义及…

剑指offer——01二维数组中的查找.

class Solution { public:bool Find(int target, vector<vector<int> > array) {int m array.size();//得到该二维数组的行数if(m0) return false;//如果行数为0则直接退出int n array[0].size();//得到该二维数组的列数if(n0) return false;//如果列数为0则直接退…

GraphQL:验证与授权

GraphQL 既是一种用于 API 的查询语言也是一个满足你数据查询的运行时。GraphQL 对你的 API 中的数据提供了一套易于理解的完整描述&#xff0c;使得客户端能够准确地获得它需要的数据&#xff0c;而且没有任何冗余&#xff0c;也让 API 更容易地随着时间推移而演进&#xff0c…

python 创建目录_第二天:Python中目录及文件操作

Python创建目录使用os模块mkdir创建一级目录&#xff0c;但不会创建父级目录#创建test目录&#xff0c;成功 import os os.mkdir(E:test)##若无父级目录python,则创建test目录失败 import os os.mkdir(E:pythontest)makedirs可创建父级目录import os os.makedirs(E:pythontest)…

计算机科普小知识——U盘格式化

在格式化U盘的时候我们需要选择文件系统类型&#xff0c;分别有FAT32&#xff08;默认&#xff09;&#xff0c;NTFS和exFAT这三种格式 常见格式一&#xff1a;FAT32 FAT32是windows传统的文件格式&#xff0c;对每个分区只有4GB的容量&#xff0c;是任何一种usb存储设备都会预…

.NET 5干货来袭 嘉宾李杨桂素伟

Azure Show大家好&#xff0c;欢迎来到Azure Show第八期&#xff0c;好久不见&#xff01;因为工作的原因有两个月没和大家见面&#xff0c;但12月Azure Show回归&#xff0c;会有更多大家感兴趣的话题&#xff0c;也有更多嘉宾与大家见面。除了这一期&#xff0c;在12月我们还…

setnx和expire合成一条指令_Python 为什么只需一条语句“a,b=b,a”,就能直接交换两个变量?...

从接触 Python 时起&#xff0c;我就觉得 Python 的元组解包&#xff08;unpacking&#xff09;挺有意思&#xff0c;非常简洁好用。最显而易见的例子就是多重赋值&#xff0c;即在一条语句中同时给多个变量赋值&#xff1a;>>> x, y 1, 2 >>> print(x, y) …