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 天前
AI工作岗位的就业分层?
开发语言·人工智能·学习·面试·职场和发展
时寒的笔记1 天前
js逆向_webpack讲解加载器&酷某音乐案例
开发语言·javascript·webpack
t***5441 天前
能否给出更多现代C++架构设计模式?
java·开发语言·c++
励志的小陈1 天前
数据结构--队列(C语言实现)
c语言·开发语言·数据结构
mjhcsp1 天前
C++信息论超详解析
开发语言·c++
無限進步D1 天前
Java 基础算法训练
java·开发语言·算法·入门
map1e_zjc1 天前
Java SpringBoot学习记录(4)
java·开发语言·学习
小毛驴8501 天前
多线程同步打标记的几种实现方案
java·开发语言·python
Mr_Xuhhh1 天前
递归之美:合并两个有序链表的优雅解法
java·开发语言
bluebonnet271 天前
【Python】一些PEP提案(五):注解的延迟求值
开发语言·python