ASP.NET?MVC5實(shí)現(xiàn)文件上傳與地址變化處理(5)
一.上傳文件和重復(fù)文件處理
文件處理的原則是:不在數(shù)據(jù)庫(kù)中保存文件,只在數(shù)據(jù)庫(kù)中保存文件信息(Hash值等)。采取文件的MD5重命名文件在一般情況足夠處理文件的重復(fù)問(wèn)題,強(qiáng)迫癥傾向則可以考慮將MD5和其他摘要算法結(jié)合。
public static string Save(HttpPostedFileBase file, string path)
{
var root = "~/Upload/" + path + "/";
var phicyPath = HostingEnvironment.MapPath(root);
Directory.CreateDirectory(phicyPath);
var fileName = Md5(file.InputStream) + file.FileName.Substring(file.FileName.LastIndexOf('.'));
file.SaveAs(phicyPath + fileName);
return fileName;
}二.單獨(dú)文件上傳
網(wǎng)站Logo、分類(lèi)圖標(biāo)等各種場(chǎng)景需要單獨(dú)文件上傳的處理。通過(guò)使用UIHintAttribute或自定義繼承自UIHintAttribute的特性我們將文件上傳的前端邏輯的重復(fù)代碼消滅,使用統(tǒng)一的視圖文件處理。曾經(jīng)使用過(guò)Uplodify和AjaxFileUploader,前者存在flash依賴和cookie問(wèn)題,后者基本已經(jīng)過(guò)時(shí)。此處我們采用KindEditor中的文件上傳組件作為演示。非Flash的支持IE6+的方案的核心都是通過(guò)iframe方式實(shí)現(xiàn)偽AJax上傳,核心還是通過(guò)html form post到服務(wù)器。
public class UploadModel
{
[Display(Name = "圖標(biāo)")]
[UIHint("Upload")]
public string Image { get; set; }
[Display(Name = "簡(jiǎn)單模式")]
[UIHint("Editor")]
[AdditionalMetadata("useSimple", true)]
public string Text1 { get; set; }
[Display(Name = "標(biāo)準(zhǔn)模式")]
[UIHint("Editor")]
public string Text2 { get; set; }
}在我們的實(shí)際項(xiàng)目中采取繼承UIHintAttribute的方式,其中的path路徑指定存儲(chǔ)的下級(jí)地址,類(lèi)似的還有DropDownAttribute、EditorAtrribute等等。僅供參考。
[AttributeUsage(AttributeTargets.Property)]
public class UploadAttribute : UIHintAttribute, IMetadataAware
{
public string Path { get; private set; }
public UploadAttribute(string path = "")
: base("Upload")
{
this.Path = path;
}
public virtual void OnMetadataCreated(ModelMetadata metadata)
{
metadata.AdditionalValues.Add("Path", this.Path);
}
}Razor:在Shared中添加EditorTemplates文件夾,新建Upload.cshtml文件。
<script>
KindEditor.ready(function (K) {
var editor = K.editor({
allowFileManager: false,
allowImageUpload: true,
formatUploadUrl: false,
uploadJson: '@url',
});
K('#btn_@id').click(function () {
editor.loadPlugin('insertfile', function () {
editor.plugin.fileDialog({
fileUrl: K('#@id').val(),
clickFn: function (url, title) {
K('#@id').val(url);
$('#image_@id').attr('src', url);
editor.hideDialog();
}
});
});
});
});
$('#rest_@id').click(function () {
$('#@id').attr('value', '');
$('#image_@id').attr('src', '@Url.Content("~/Images/default.png")');
});
</script>
三.編輯器中的文件上傳
編輯器中的文件上傳和單獨(dú)文件上傳的主要區(qū)別是上傳后返回值的處理,編輯器需要將url插入到編輯的位置。編輯器采用過(guò)CKeditor和UMeditor,兩者都需要我改源代碼才能處理路徑問(wèn)題。上傳地址和返回值的配置如果不能方便的視圖中調(diào)整的編輯器,我個(gè)人不認(rèn)為是好編輯器,這就好比一個(gè)類(lèi)庫(kù)沒(méi)法擴(kuò)展和自定義配置一樣。仍然采用KindEditor作為演示。Editor.cshtml的
<script type="text/javascript">
var editor;
KindEditor.ready(function (K) {
editor = K.create('textarea[name="@Html.IdForModel()"]', {
resizeType: 1,
allowPreviewEmoticons: false,
allowImageUpload: true,
uploadJson: '@UploadManager.UploadUrl',
formatUploadUrl: false,
allowFileManager: false
@if(useSimple)
{
<text>, items: [
'fontname', 'fontsize', '|', 'forecolor', 'hilitecolor', 'bold', 'italic', 'underline',
'removeformat', '|', 'justifyleft', 'justifycenter', 'justifyright', 'insertorderedlist',
'insertunorderedlist', '|', 'emoticons', 'image', 'link']
</text>
}
});
});
</script>四.處理文章中的圖片路徑
重頭戲來(lái)了,這個(gè)看似問(wèn)題可以回避,其實(shí)真的無(wú)法回避。更換目錄、域名和端口,使用子域名或其他域名作為圖片服務(wù)器等等,這些情況讓我們必須處理好這個(gè)問(wèn)題,否則日后會(huì)浪費(fèi)更多的時(shí)間。這不是小問(wèn)題,打開(kāi)支持插入圖片的各個(gè)網(wǎng)站的編輯器,查看一下圖片的路徑,大多是絕對(duì)url的,又或者只基于根目錄的。如果你以產(chǎn)品的形式提供給客戶,更不可能要求客戶自己挨個(gè)替換文章中的路徑了。
1.在數(shù)據(jù)庫(kù)中不存儲(chǔ)文件路徑,使用URL路徑作為存儲(chǔ)。
2.使用html base元素解決相對(duì)路徑的引用問(wèn)題。
就是base元素,可能有的人認(rèn)為這個(gè)base可有可無(wú),但在處理圖片路徑的問(wèn)題上,沒(méi)有比base更簡(jiǎn)潔更優(yōu)雅的方案了。至少我沒(méi)有也沒(méi)找到過(guò)。其實(shí)可以把全部的靜態(tài)資源都移除到外部存儲(chǔ),如果你需要。在測(cè)試時(shí),我們切換回使用本地存儲(chǔ)。
@{
var baseUrl = UploadManager.UrlPrefix;
}
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link href="~/favicon.ico" rel="shortcut icon" type="image/x-icon" />
<title>@ViewBag.Title</title>
<base href="@baseUrl" />
<script src="~/Scripts/jquery-1.11.2.min.js"></script>
@RenderSection("head",false)
</head>
<body>
@RenderBody()
</body>
</html>五.處理上傳地址的變化
我們需要獨(dú)立的圖片服務(wù)器處理上傳或者使用第三方的圖片存儲(chǔ)服務(wù)時(shí),我們的上傳地址改變了,如果剛剛提到的圖片路徑一樣,因此我們將上傳路徑和圖片路徑都采取配置的方式方便更改,我們就曾經(jīng)切換到又拍云又切換到自有的服務(wù)器。在我的實(shí)際使用時(shí)配置在數(shù)據(jù)中使用時(shí)采用緩存。為了便于演示我們直接使用配置文件。
首先定義配置文件的處理程序
public class UploadConfig : IConfigurationSectionHandler
{
public object Create(object parent, object configContext, System.Xml.XmlNode section)
{
var config = new UploadConfig();
var urloadUrlNode = section.SelectSingleNode("UploadUrl");
if (urloadUrlNode != null && urloadUrlNode.Attributes != null && urloadUrlNode.Attributes["href"] != null)
{
config.UploadUrl = Convert.ToString(urloadUrlNode.Attributes["href"].Value);
}
var urlPrefixNode = section.SelectSingleNode("UrlPrefix");
if (urlPrefixNode != null && urlPrefixNode.Attributes != null && urlPrefixNode.Attributes["href"] != null)
{
config.UrlPrefix = Convert.ToString(urlPrefixNode.Attributes["href"].Value);
}
return config;
}
public string UploadUrl { get; private set; }
public string UrlPrefix { get; private set; }
}
在web.config中配置
<configSections> <section name="UploadConfig" type="SimpleFileManager.UploadConfig, SimpleFileManager" requirePermission="false" /> </configSections> <UploadConfig> <UploadUrl href="~/File/Upload/" /> <UrlPrefix href="~/Upload/" /> </UploadConfig>
使用UploadMange緩存和管理配置
public static class UploadManager
{
private static string uploadUrl;
private static string urlPrefix;
static UploadManager()
{
var config = ConfigurationManager.GetSection("UploadConfig") as UploadConfig;
var url = config != null && !string.IsNullOrEmpty(config.UploadUrl) ? config.UploadUrl : "~/File/Upload";
uploadUrl = url.StartsWith("~") ? UploadHelper.GetUrlFromVisualPath(url) : url;
var prefix = config != null && !string.IsNullOrEmpty(config.UrlPrefix) ? config.UrlPrefix : "~/Upload";
urlPrefix = prefix.StartsWith("~") ? UploadHelper.GetUrlFromVisualPath(prefix) : prefix;
}
public static string UploadUrl
{
get
{
return uploadUrl;
}
}
public static string UrlPrefix
{
get
{
return urlPrefix;
}
}
}
文件Hash的Md5、返回值的Json處理、完整URL的生成和文件的保存這些具體技術(shù)的依賴為了便于演示,統(tǒng)一放置在UploadHelper中,因?yàn)檫@些不是重點(diǎn)。實(shí)際應(yīng)用中可以采取接口隔離并通過(guò)IoC注入的方式解耦。

以上就是ASP.NET MVC5如何實(shí)現(xiàn)文件上傳與地址變化處理的全部過(guò)程,希望對(duì)大家的學(xué)習(xí)有所幫助。
- ASP.NET MVC HttpPostedFileBase文件上傳的實(shí)例代碼
- ASP.NET MVC4 利用uploadify.js多文件上傳
- asp.net core mvc實(shí)現(xiàn)文件上傳實(shí)例
- asp.net mvc 實(shí)現(xiàn)文件上傳帶進(jìn)度條的思路與方法
- 解決ASP.NET Core Mvc文件上傳限制問(wèn)題實(shí)例
- ASP.NET MVC文件上傳教程(二)
- ASP.NET MVC 文件上傳教程(一)
- ASP.NET MVC處理文件上傳的小例子
- 用Fine Uploader+ASP.NET MVC實(shí)現(xiàn)ajax文件上傳[代碼示例]
- ASP.NET MVC實(shí)現(xiàn)批量文件上傳
相關(guān)文章
.net mvc頁(yè)面UI之Jquery博客日歷控件實(shí)現(xiàn)代碼
最近在做一個(gè)博客系統(tǒng),其他需要用到博客日歷控件,網(wǎng)上搜索了很多資料,其中大部分都是javascript的,經(jīng)過(guò)總結(jié)使用jquery實(shí)現(xiàn)了博客日歷效果。代碼如下2013-09-09
EasyUI Tree+Asp.net實(shí)現(xiàn)權(quán)限樹(shù)或目錄樹(shù)導(dǎo)航的簡(jiǎn)單實(shí)例
本篇文章主要是對(duì)EasyUI Tree+Asp.net實(shí)現(xiàn)權(quán)限樹(shù)或目錄樹(shù)導(dǎo)航的簡(jiǎn)單實(shí)例進(jìn)行了介紹,需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助2014-02-02
asp.net 站點(diǎn)URLRewrite使用小記
asp.net的底層運(yùn)作已經(jīng)也亂談過(guò)一番, 今天記一下URLRewrite的方法。2009-11-11
Sqlite 常用函數(shù)封裝提高Codeeer的效率
以下是頻繁用到的Sqlite函數(shù),內(nèi)容格式相對(duì)固定,封裝一下有助于提高開(kāi)發(fā)效率^_^至少提高Codeeer的效率了2012-12-12
詳細(xì)介紹.NET中的動(dòng)態(tài)編譯技術(shù)
這篇文章詳細(xì)介紹了.NET中的動(dòng)態(tài)編譯技術(shù),有需要的朋友可以參考一下2013-11-11
靈活掌握asp.net中g(shù)ridview控件的多種使用方法(下)
這篇文章繼續(xù)向大家推薦如何靈活掌握asp.net中g(shù)ridview控件的多種使用方法,感興趣的小伙伴們可以參考一下2015-11-11
ASP.NET MVC用存儲(chǔ)過(guò)程批量添加修改數(shù)據(jù)操作
這篇文章主要介紹了ASP.NET MVC用存儲(chǔ)過(guò)程批量添加修改數(shù)據(jù)的方法,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-02-02

