native aot 下 file.readalltext 会因 encoding.getencoding 动态查表失败而抛出 notsupportedexception,应显式传入 encoding.utf8;directory.getfiles 通配符受限,需手动过滤;filestream 参数须为字面量;资源路径须用 appcontext.basedirectory 而非 assembly.location。

Native AOT 下 File.ReadAllText 会直接失败
Native AOT 编译时,.NET 的反射和动态代码生成被大幅限制,而 File.ReadAllText 底层依赖 Encoding.GetEncoding 动态查表(比如识别 "utf-8" 字符串),这在 AOT 中默认不可用。你不会看到编译错误,但运行时抛出 NotSupportedException: No data is available for encoding 65001。
实操建议:
- 显式传入
Encoding.UTF8实例,避免字符串编码名解析:File.ReadAllText(path, Encoding.UTF8) - 若需其他编码(如
GBK),必须在RuntimeFeature.IsDynamicCodeSupported为 true 时才可用——AOT 下它恒为 false,所以 GBK、Shift-JIS 等非 UTF 编码基本不可用 - 提前在
rd.xml中保留编码类型(不推荐):添加<type name="System.Text.Encoding" dynamic="Required All"></type>,但这会增大二进制体积且不保证所有编码都生效
AOT 中 Directory.GetFiles 的通配符行为受限
Directory.GetFiles(path, "*.txt") 在 AOT 下可能返回空数组或抛出 ArgumentException,因为 Windows API 层的通配符匹配逻辑依赖部分 JIT 生成的辅助代码,AOT 模式下这部分被裁剪。
实操建议:
- 改用无通配符的
Directory.GetFiles(path)+ 手动Path.GetExtension过滤:Directory.GetFiles(path).Where(f => Path.GetExtension(f).Equals(".txt", StringComparison.OrdinalIgnoreCase)) - 避免嵌套通配符(如
"**/*.cs"),Directory.EnumerateFiles(path, "*", SearchOption.AllDirectories)在 AOT 中支持有限,推荐用Directory.GetDirectories递归遍历 - 注意:AOT 构建时若未启用
System.IO.FileSystem元数据保留,File.GetAttributes可能返回默认值,影响IsDirectory判断
FileStream 构造函数参数必须静态可知
AOT 不允许运行时拼接 flag 值,比如 new FileStream(path, (FileMode)modeValue, FileAccess.Read) 中的 modeValue 若来自变量或配置,会导致链接期失败或运行时异常。
实操建议:
- 所有
FileMode、FileAccess、FileShare必须写死为字面量:new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read) - 不要用位运算组合
FileOptions(如FileOptions.Asynchronous | FileOptions.SequentialScan),AOT 链接器无法跟踪这种动态组合;改用单个预定义值,或拆成多个明确分支 - 异步 I/O(
ReadAsync)在 AOT 下可用,但底层仍依赖线程池——确保发布时未禁用ThreadPool相关功能(即不要设DOTNET_SYSTEM_THREADING_DISABLETHREADPOOL=1)
资源文件路径在 AOT 中不能靠 Assembly.Location 推导
Assembly.GetExecutingAssembly().Location 在 AOT 下返回空字符串或无效路径,因为原生二进制没有传统 .dll 路径概念;试图拼接 Path.Combine(Path.GetDirectoryName(asm.Location), "data.json") 必然失败。
实操建议:
- 把必需的文件打到应用目录(如
publish输出根),用AppContext.BaseDirectory获取启动路径:Path.Combine(AppContext.BaseDirectory, "config.json") - 嵌入资源(
EmbeddedResource)仍可用,但读取方式必须用Assembly.GetManifestResourceStream,且资源名需完全匹配(区分大小写),不能靠反射枚举 - 若需跨平台路径兼容,避免硬编码
"\data\",统一用Path.Join或Path.Combine,它们在 AOT 下已完全静态化
最麻烦的不是语法报错,而是那些“看起来能跑、但换环境就静默失败”的路径推导和编码解析——AOT 不给你 runtime 补救的机会,所有不确定路径、动态编码名、反射式 IO 都得在写代码时就砍掉。










