C# 一步步解决DotNetZip因超长路径(MAX_PATH)报错的问题
当前位置:点晴教程→知识管理交流
→『 技术文档交流 』
一、前言超长路径(MAX_PATH)的问题,在很多地方都可能遇到,常见的解决办法无非三种:添加前缀\\?\、app.config添加配置、修改注册表等。 而对于其它第三方的DLL,我们如何去从外部解决呢?答案是:反射。 本篇文章,我们就以DotNetZip为例,一步步来解决超长路径的问题。 **相信看完的你,一定会有所收获! 二、一步步定位问题首先,我们看下使用DotNetZip的ZipEntry.AddDictionary()方法添加一个目录后,相关属性的变化。 我们可以看到,在添加前,ZipFile.Entries是空的。 在添加后,变化如下: 会遇到超长路径(MAX_PATH)报错,肯定会有一个完整的路径,当然存在临时拼接的情况,但是我们先找一找。 最后,我们在“非公开成员”中找到一个属性:LocalFileName,其值正是完整的文件路径,如下图所示: 为了方便观察,我们将LocalFileName属性置顶显示: 之后我们就可以看到,默认显示的都是LocalFileName了: 现在我们找到这个关键的LocalFileName,虽然还不能百分百确定和这有关,但是可以以此为准进行后面的排查和验证了。 三、反编译梳理逻辑像DotNetZip,是开源的,我们可以直接找到源码来查看,对于非开源的,我们就只能反编译了。 这里常用的工具是dnSpy,我们使用dnSpy打开DotNetZip,然后定位到ZipEntry,再找到LocalFileName属性: 可以看到只有get,没有set,这样的话,如果直接反射操作此属性,则需要额外的添加set访问器,非常麻烦,此时不作考虑。 我们再看,LocalFileName其实是返回的变量:_LocalFileName,我们再找下此变量: 这样的话就简单了,我们反射直接操作此变量: 我们在处理时,先进行判断是否超长,再反射处理,因为反射是很耗资源的。 至于扩展方法,定义如下: 说一下这里面为什么要以248作为判断条件,是因为260是文件路径限制,而对于目录则是以248为限制,所以我们以最小的为主进行判断。 四、验证这里就不贴图了,验证发现已经可以正常处理超长路径了,不会再报错了。
五、开源因为DotNetZip已经不再积极维护,且Nuget上的版本也标记了弃用和风险,所以也就不提PR了, 但在这么多压缩解压缩库中,DotNetZip在易用性上是无与伦比的,所以在很多时候用用还是挺不错的。也基于此想法,正好也将自己之前封装的方法给开源了吧,基于DotNetZip,就两个方法:Zip、UnZip。简单易用、解决了超长路径(MAX_PATH)问题、支持密码、分卷。 开源地址:https://gitee.com/lesliexin/lesliexin.simplezip NuGet:https://www.nuget.org/packages/LeslieXin.SimpleZip 六、结语这次也是遇到了这个超长路径(MAX_PATH)的问题,才想起来修复下,顺便也开源下自己的库。 本篇文章最重要的是给读者一个遇到超长路径(MAX_PATH)问题时一个实操指导,毕竟此问题不止会发生在DotNetZip中,可能发生在任何与文件路径相关的地方,当再次遇到此问题时,该如何一步步去排查定位和处理。 转自https://www.cnblogs.com/lesliexin/p/18930533 该文章在 2025/9/9 9:53:28 编辑过 |
关键字查询
相关文章
正在查询... |