在做一个客户推广系统的时候,里面有一个模板管理模块,需要管理员添加模板,包括模板的名称、说明和缩略图等,在这里上传图片的功能,我采用了比较传统的方法,进行上传,测试没有问题。但当我发布之后,对存储图片的文件夹创建了虚拟目录,并赋予该目录写入的权限,但是,当我上传图片的时候,总是失败。以前没遇到过这种情况,觉得很是怪异,所以想尽办法去解决。
首先,检查上传目录的权限,我添加了NetWork Service用户的写入、修改权限,结果还是失败,然后我将权限设置为EveryOne还是失败,看来不是权限的问题。
然后,将虚拟目录删除,并在应用程序目录下重新建立了存储图片的文件夹(发布前已经删除),并赋予其写入权限,结果上传成功,也能够正常显示。
接着,我又将上传的图片copy到原来建立的虚拟目录下面,并重新建立了虚拟目录,结果显示成功。
经过这些检查和设想工作,我总结了一下,文件真的能够上传,也能够正常显示,看来是上传时候的目录和虚拟目录间的转换出现问题,然后我继续审查我写的代码。
当前的简略代码,其中Upload即为图片的存储目录,也是我创建虚拟目录的地方,而Spread是其上层目录。 我突然发现,原来Server.Mappath并没有指向图片存储的根目录,感觉有些怪怪的,然后就进行了修改。



















修改后的代码如下,结果能够正常上传和显示,看来问题的确处在这里。



















由于自己的疏忽出现了这个怪问题,但是为什么会出现这样的事情呢。我又进行了深入研究。
Server.MapPath方法
返回Web服务器上指定虚拟路径相对应的物理文件路径。
如果通过这个方法能够得到目录的物理路径,那么
Server.MapPath("~/Spread")+"/Upload"和Server.MapPath("~/spread/Upload"),也没有区别啊。为什么会产生不同的效果呢?而实际的结果是,这两者的路径并不相同。
前者返回的是Upload的原路径,而后者返回的是Upload指向的虚拟路径。如果真是这样,那么疑问就已经揭开了。
于是我做了一个小小的测试程序。
测试程序
建立一个测试用例,在根目录下创建BBS目录,然后在BBS目录下创建Upload目录,在用例发布后,首先测试返回的结果,然后将Upload建立虚拟目录,指向其它的物理路径,查看返回的值。
主程序:





测试结果:
结果证实了推出的结论,的确,Server.MapPath返回的是虚拟路径的物理地址。哎,其实其定义本来就是这样了,不过有的时候就是不碰南墙不死心。
但是还有一个问题:
在上传成功的图片,在显示的时候,路径绑定的是相对路径,用一个Image控件显示,其图片地址如:“~/Spread/Upload/080826094153.jpg”,但是当你将该图片放到虚拟目录中去,将以前的Upload目录给删除,也能够正常显示,而这里并没有用到Server.MapPath。这个问题又把我搞糊涂了,难道IIS自动能够识别其虚拟目录的地址?但是如果是这样,为什么上传的时候,则没有识别出来?
谢谢大家的意见,本问题已经得以及解决。
教训:
<1>注意基本方法的正确理解
<2>在上传文件时,如果需要对其存储目录建立虚拟目录,那么Server.MapPath参数应该是此目录。
<3>一定要注意页面缓存带来的烦扰
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 周边上新:园子的第一款马克杯温暖上架
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?
· 使用C#创建一个MCP客户端