在网络编程过程中需要向服务器上传文件。

Multipart/form-data是上传文件的一种方式。

        /// <summary>
/// 上传工程文件
/// </summary>
/// <returns></returns>
public async Task<HttpResponseMessage> UploadProjectFile()
{
ProjectFile postData = new ProjectFile();
IEnumerable<string> stringValues;
//获取请求头文件相关参数
Request.Headers.TryGetValues("token", out stringValues);
postData.token = stringValues.FirstOrDefault(); Request.Headers.TryGetValues("uid", out stringValues);
postData.uid = int.Parse(stringValues.FirstOrDefault()); Request.Headers.TryGetValues("project_id", out stringValues);
postData.project_id = int.Parse(stringValues.FirstOrDefault()); Request.Headers.TryGetValues("md5", out stringValues);
postData.md5 = stringValues.FirstOrDefault(); Request.Headers.TryGetValues("file_name", out stringValues);
postData.file_name = stringValues.FirstOrDefault();
//验证Token
var result = CheckToken(postData.token);
if (!result)
{
return CreateResponseError(, "请求拒绝");
}
//获取文件名,这里的文件名必须带扩展名
string fileName = postData.file_name; int projectId = postData.project_id;
//获取应用程序的当前工作目录
string rootPath = HostingEnvironment.MapPath("~/");
//通过Path类的Combine方法可以合并路径。获取当前工程文件夹地址
string directoryPath = Path.Combine(rootPath, "BgerProject", projectId.ToString());
//创建目录时如果目录已存在,则不会重新创建目录,且不会报错。创建目录时会自动创建路径中各级不存在的目录。
Directory.CreateDirectory(directoryPath);
//获取当前工程文件地址
string filePath = Path.Combine(directoryPath, fileName);
if (Request.Content.IsMimeMultipartContent())
{
Dictionary<string, string> dic = new Dictionary<string, string>();
var provider = new MultipartFormDataMemoryStreamProvider(); await Request.Content.ReadAsMultipartAsync(provider);
foreach (var item in provider.FileContents)
{
//Trace.WriteLine(item.Headers.ContentDisposition.FileName);//获取上传文件实际的文件名
//Trace.WriteLine("Server file path: " + item.LocalFileName);//获取上传文件在服务上默认的文件名
var stream = await item.ReadAsStreamAsync();
if (Path.GetExtension(item.Headers.ContentDisposition.FileName.Replace("\"", "")) == ".mp4")
{
//通过正则表达式判断是视频还是音频
Regex regex = new Regex("voice", RegexOptions.Compiled);
Match m = regex.Match(item.Headers.ContentDisposition.Name.ToLower());
if (m.Success)
{
//音频直接保存处理 //判断是否已存在该音频
if (!File.Exists(filePath))
{
using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
}
}
else
{
//视频进行压缩处理
//保存原片(如果路径不存在,创建路径) using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
//压缩并保存上传的视频
//压缩后的存放路径(如果路径不存在,创建路径)
////判断是否已存在该视频
//if (!File.Exists(filePath))
//{
// //压缩保存视频
// if (!FFMPEGHelper.CreateNewVideo(originalFilePath, filePath))
// {
// continue;
// };
//}
}
}
else
{
//保存原片(如果路径不存在,创建路径) using (StreamWriter sw = new StreamWriter(filePath))
{
stream.CopyTo(sw.BaseStream);
sw.Flush();
}
//压缩并保存上传的图片
//压缩后的存放路径(如果路径不存在,创建路径) ////判断是否已存在该图片
//if (!File.Exists(filePath))
//{
// var pxLimit = 481;
// //判断是否需要压缩
// System.Drawing.Image iSource = System.Drawing.Image.FromFile(originalFilePath);
// if (iSource.Width <= pxLimit || iSource.Height <= pxLimit)
// {
// //无需压缩
// System.IO.File.Copy(originalFilePath, filePath);
// }
// else
// {
// var min = Math.Min((int)iSource.Width, (int)iSource.Height);
// if (min > pxLimit)
// {
// //缩放倍率
// var rate = (double)pxLimit / min;
// var width = (int)Math.Ceiling(rate * (double)iSource.Width);
// var height = (int)Math.Ceiling(rate * (double)iSource.Height); // //压缩保存图片
// if (!ImageHelper.GetPicThumbnail(originalFilePath, filePath, height, width, 100))
// {
// continue;
// };
// }
// }
//}
}
}
return CreateResponse("请求成功"); }
return CreateResponseError(,"缺少MIME内容"); }
 using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.IO;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Threading;
using System.Threading.Tasks; namespace BgerServer.Models
{
/// <summary>
/// 与MultipartFormDataStreamProvider对应,但不将文件直接存入指定位置,而是需要自己指定数据流如何保存
/// </summary>
public class MultipartFormDataMemoryStreamProvider : MultipartStreamProvider
{
private NameValueCollection _formData = new NameValueCollection();
private Collection<bool> _isFormData = new Collection<bool>();
/// <summary>
/// 获取文件对应的HttpContent集合,文件如何读取由实际使用方确定,可以ReadAsByteArrayAsync,也可以ReadAsStreamAsync
/// </summary>
public Collection<HttpContent> FileContents
{
get
{
if (this._isFormData.Count != this.Contents.Count)//两者总数不一致,认为未执行过必须的Request.Content.ReadAsMultipartAsync(provider)方法
{
throw new InvalidOperationException("System.Net.Http.HttpContentMultipartExtensions.ReadAsMultipartAsync must be called first!");
}
return new Collection<HttpContent>(this.Contents.Where((ct, idx) => !this._isFormData[idx]).ToList());
}
}
/// <summary>Gets a <see cref="T:System.Collections.Specialized.NameValueCollection" /> of form data passed as part of the multipart form data.</summary>
/// <returns>The <see cref="T:System.Collections.Specialized.NameValueCollection" /> of form data.</returns>
public NameValueCollection FormData
{
get
{
return this._formData;
}
}
public override async Task ExecutePostProcessingAsync()
{
for (var i = ; i < this.Contents.Count; i++)
{
if (!this._isFormData[i])//非文件
{
continue;
}
var formContent = this.Contents[i];
ContentDispositionHeaderValue contentDisposition = formContent.Headers.ContentDisposition;
string formFieldName = UnquoteToken(contentDisposition.Name) ?? string.Empty;
string formFieldValue = await formContent.ReadAsStringAsync();
this.FormData.Add(formFieldName, formFieldValue);
}
}
public override Stream GetStream(HttpContent parent, HttpContentHeaders headers)
{
if (parent == null)
{
throw new ArgumentNullException("parent");
}
if (headers == null)
{
throw new ArgumentNullException("headers");
}
ContentDispositionHeaderValue contentDisposition = headers.ContentDisposition;
if (contentDisposition == null)
{
throw new InvalidOperationException("Content-Disposition is null");
}
this._isFormData.Add(string.IsNullOrEmpty(contentDisposition.FileName));
return new MemoryStream();
}
/// <summary>
/// 复制自 System.Net.Http.FormattingUtilities 下同名方法,因为该类为internal,不能在其它命名空间下被调用
/// </summary>
/// <param name="token"></param>
/// <returns></returns>
private static string UnquoteToken(string token)
{
if (string.IsNullOrWhiteSpace(token))
{
return token;
}
if (token.StartsWith("\"", StringComparison.Ordinal) && token.EndsWith("\"", StringComparison.Ordinal) && token.Length > )
{
return token.Substring(, token.Length - );
}
return token;
}
}
}

ASP.NET 服务端接收Multipart/form-data文件的更多相关文章

  1. Asp.Net_ 服务端向客户端写JavaScript脚本

    在Asp.net 服务端处理脚本,一般都用 ClientScriptManager ,即web窗体服务端的this.ClientScript.该对象比较常用的方法: 1.RegisterArrayDe ...

  2. C# TCP socket发送大数据包时,接收端和发送端数据不一致 服务端接收Receive不完全

    简单的c# TCP通讯(TcpListener) C# 的TCP Socket (同步方式) C# 的TCP Socket (异步方式) C# 的tcp Socket设置自定义超时时间 C# TCP ...

  3. 自己实现FormsAuthentication.SetAuthCookie方法,怎样在ASP.NET服务端代码中删除客户端Cookie

    如何手动设置AuthCookie ASP.NET中实现可以自己实现FormsAuthentication.SetAuthCookie方法,控制更为灵活 /// <summary> /// ...

  4. android如何与asp.net服务端共享session

    近期需要实现一个功能,就是需要通过发送短信进行注册,现在想把短信验证码放到服务器的session值中,当客户端收到短信并提交短信码时由asp.net服务端进行判断,那么如何共享这个session那么需 ...

  5. MVC文件上传04-使用客户端jQuery-File-Upload插件和服务端Backload组件实现多文件异步上传

    本篇使用客户端jQuery-File-Upload插件和服务端Badkload组件实现多文件异步上传.MVC文件上传相关兄弟篇: MVC文件上传01-使用jquery异步上传并客户端验证类型和大小  ...

  6. 微信小程序开发之多图片上传+服务端接收

    前言: 业务需求,这次需要做一个小程序同时选中三张图片一起上传到服务端,后端使用的.NET WEBAPI接收数据保存. 使用技术: 在这章中将会使用到微信小程序wx.uploadFile(Object ...

  7. java httpclient发送json 请求 ,go服务端接收

    /***java客户端发送http请求*/package com.xx.httptest; /** * Created by yq on 16/6/27. */ import java.io.IOEx ...

  8. [转]微信小程序开发(二)图片上传+服务端接收

    本文转自:http://blog.csdn.net/sk719887916/article/details/54312573 文/YXJ 地址:http://blog.csdn.net/sk71988 ...

  9. tcp syn-synack-ack 服务端接收ack

    TCP 服务端 接收到ack tcp_v4_rcv() -> tcp_v4_do_rcv() -> tcp_v4_hnd_req() + tcp_child_process()tcp_v4 ...

随机推荐

  1. 栅格数据的批量镶嵌(附Python脚本)

    栅格数据的批量镶嵌(附Python脚本) 博客小序:在数据处理的过程中,会遇到需要大量镶嵌的情况,当数据较多时手动镶嵌较为麻烦,自己最近对分省的DEM数据进行镶嵌,由于利用python进行镶嵌较为方便 ...

  2. 【linux】【qt5】【信号槽示例】

    什么叫信号槽: 信号槽是 Qt 框架引以为豪的机制之一.所谓信号槽,实际就是观察者模式.当某个事件发生之后,比如,按钮检测到自己被点击了一下,它就会发出一个信号(signal).这种发出是没有目的的, ...

  3. 分布式唯一ID生成算法-雪花算法

    在我们的工作中,数据库某些表的字段会用到唯一的,趋势递增的订单编号,我们将介绍两种方法,一种是传统的采用随机数生成的方式,另外一种是采用当前比较流行的“分布式唯一ID生成算法-雪花算法”来实现. 一. ...

  4. 分享一个赚钱方法:用趣分类app在家轻松赚钱

    什么是趣分类 近期,垃圾分类是社会各界和广大市民关心的一个热门话题,随着垃圾分类工作的推进,各地都掀起学习垃圾分类的热潮.为了我们的美好生活,打响"垃圾分类"这场硬仗刻不容缓.据了 ...

  5. 牛客练习赛39 D 动态连通块+并查集 X bitset 优化

    https://ac.nowcoder.com/acm/contest/368/D 题意 小T有n个点,每个点可能是黑色的,可能是白色的.小T对这张图的定义了白连通块和黑连通块:白连通块:图中一个点集 ...

  6. HDU3652:B-number(数位DP)

    Problem Description A wqb-number, or B-number for short, is a non-negative integer whose decimal for ...

  7. CF 988C Equal Sums 思维 第九题 map

    Equal Sums time limit per test 2 seconds memory limit per test 256 megabytes input standard input ou ...

  8. Python 单元测试框架系列:聊聊 Python 的单元测试框架(一):unittest

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  9. [DP]最长公共子串

    题目 给定两个字符串str1和str2, 长度分别稳M和N,返回两个字符串的最长公共子串 解法一 这是一道经典的动态规划题,可以用M*N的二维dp数组求解.dp[i][j]代表以str1[i]和str ...

  10. 【Redis】基础学习概览【汇总】

    一.概述 1.1 简介 1.2 Redis单线程好处 1.3 单线程弊端 1.4 Redis应用场景 二.安装.开启以及关闭 三.Redis基本数据类型 四.SpringBoot整合Redis 五.R ...