一、前言
超长路径(MAX_PATH)的问题,在很多地方都可能遇到,常见的解决办法无非三种:添加前缀\\?\、app.config添加配置、修改注册表等。
而对于其它第三方的DLL,我们如何去从外部解决呢?答案是:反射。
本篇文章,我们就以DotNetZip为例,一步步来解决超长路径的问题。
**相信看完的你,一定会有所收获!
**
本文地址:https://www.cnblogs.com/lesliexin/p/18930533
二、一步步定位问题
首先,我们看下使用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中,可能发生在任何与文件路径相关的地方,当再次遇到此问题时,该如何一步步去排查定位和处理。
感谢大家的观看,本人水平有限,文章不足之处欢迎大家评论指正。
-[END]-