C#检测文本编码格式

最近有个csv文件带"±",文件是GB2312编码格式,所以使用Encoding.Default/UTF-8读取该文件±会变成乱码,又不好写死Encoding.GetEncoding("GB2312"),所以添加一步检测。尝试了Ude.NET库检测,不好使,检测不出来。所以用AI生成了一段,亲测可用。

复制代码
   public static Encoding DetectFileEncoding(string filePath)
   {
       byte[] bytes = File.ReadAllBytes(filePath);
       if (bytes.Length == 0) return Encoding.UTF8;

       // 1. 检查 BOM (最准确)
       if (bytes.Length >= 3 && bytes[0] == 0xEF && bytes[1] == 0xBB && bytes[2] == 0xBF)
           return Encoding.UTF8;
       if (bytes.Length >= 2 && bytes[0] == 0xFF && bytes[1] == 0xFE)
           return Encoding.Unicode; // UTF-16 LE
       if (bytes.Length >= 2 && bytes[0] == 0xFE && bytes[1] == 0xFF)
           return Encoding.BigEndianUnicode; // UTF-16 BE

       // 2. 注册所有编码支持 (确保 .NET Core/.NET 5+ 能识别 GBK)
       Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);

       // 3. 启发式检测:对比 UTF-8, GB18030, ISO-8859-1

       // 尝试 UTF-8
       // 如果 UTF-8 解码后出现很多替换字符 (U+FFFD),说明大概率不是 UTF-8
       string utf8String = Encoding.UTF8.GetString(bytes);
       int utf8ReplacementCount = utf8String.Count(c => c == '\uFFFD');

       // 如果替换字符很少,可能是 UTF-8 (无 BOM)
       if (utf8ReplacementCount == 0 && IsLikelyText(utf8String))
       {
           return Encoding.UTF8;
       }

       // 尝试 GB18030 (完全兼容 GB2312 和 GBK)
       Encoding gbEncoding = Encoding.GetEncoding("GB18030");
       string gbString = gbEncoding.GetString(bytes);

       // 尝试 ISO-8859-1 (单字节,永远不会失败,但中文会变成乱码)
       Encoding isoEncoding = Encoding.GetEncoding("ISO-8859-1");
       string isoString = isoEncoding.GetString(bytes);

       // 4. 关键判断逻辑:
       // 如果 GBK 解码出的字符串中包含大量汉字,而 ISO 解码出的只是普通拉丁字符或乱码符号
       // 则认为它是 GBK/GB2312

       int chineseCharCount = gbString.Count(c => c >= 0x4E00 && c <= 0x9FFF); // 基本汉字范围
       int isoWeirdCount = isoString.Count(c => c > 0x7F && (c < 0xA0 || char.IsControl(c))); // ISO 中高位字节的奇怪字符

       // 阈值:如果检测到超过 5 个汉字,基本可以确定是 GB 编码
       if (chineseCharCount > 5)
       {
           return gbEncoding;
       }

       // 如果汉字很少,但 UTF-8 也不对,默认回退到系统默认编码 (中文 Windows 下通常是 GBK)
       // 或者根据业务需求,如果是英文为主,可能真的是 ISO-8859-1
       // 这里做一个简单的置信度判断:如果 GB 解码没有乱码替换,优先 GB
       int gbReplacementCount = gbString.Count(c => c == '\uFFFD');

       if (gbReplacementCount < utf8ReplacementCount)
       {
           return gbEncoding;
       }

       // 最后手段:返回系统默认
       return Encoding.Default;
   }

   // 辅助:判断字符串是否像正常文本(非二进制垃圾)
   private static bool IsLikelyText(string text)
   {
       // 简单检查:如果包含大量控制字符,则不是文本
       int controlChars = text.Count(c => char.IsControl(c) && c != '\r' && c != '\n' && c != '\t');
       return controlChars < text.Length * 0.05;
   }

另外,Net8不支持GB2312,需要先在.csproj添加:

<PackageReference Include="System.Text.Encoding.CodePages" Version="7.0.0" />,

再在代码添加:

Encoding.RegisterProvider(CodePagesEncodingProvider.Instance);

才可使用Encoding.GetEncoding("GB2312");

相关推荐
骇客野人1 小时前
JDK8和JDK8以后对jdk的优化,以及为什么如此优化
java·开发语言·windows
乐观勇敢坚强的老彭1 小时前
c++图论
开发语言·c++·图论
27669582921 小时前
悟空租车帮app最新登录算法
开发语言·前端·python·悟空app·租车帮·租车帮app·租车帮登录逆向
ZTLJQ1 小时前
数据采集的工业级武器:Python爬虫框架完全解析
开发语言·爬虫·python
小年糕是糕手2 小时前
【35天从0开始备战蓝桥杯 -- Day6】
开发语言·前端·网络·数据库·c++·蓝桥杯
星轨初途2 小时前
【C/C++底层修炼】拆解动态内存管理:四大动态内存函数、六大错误与柔性数组
c语言·开发语言·c++·经验分享·笔记·柔性数组
froginwe112 小时前
PHP 过滤器
开发语言
0xDevNull2 小时前
Java 视频在线播放功能案例
java·开发语言·音视频