1. 代码

/// <summary>
/// 文件上传下载控制器
/// </summary>
public class FileController : ApiController
{
/// <summary>
/// 上传文件
/// </summary>
/// http://192.168.11.67:8077/file
/// <returns></returns>
[HttpPost]
public Task<IList<UploadFile>> Upload()
{
// 获取请求客户端的IP地址
var ipAddr = GetRemoteClientIPAddress(); try
{
//如果文件路径不存在,创建路径
if (!Directory.Exists(UploadFolder)) Directory.CreateDirectory(UploadFolder); if (Request.Content.IsMimeMultipartContent())
{
// 读取文件数据
var task = Request.Content.ReadAsMultipartAsync().ContinueWith(t =>
{
// 如果上传过程失败或被取消,则返回错误信息
if (t.IsFaulted || t.IsCanceled)
{
throw new HttpResponseException(Request.CreateResponse(HttpStatusCode.InternalServerError, "上传过程出错或被取消"));
} try
{
// 读取请求数据
var provider = t.Result; IList<UploadFile> files = new List<UploadFile>();
// 有可能一次上传多个文件,循环保存文件
foreach (var filecontent in provider.Contents)
{
// 读取文件对应的byte数组
var byteArray = filecontent.ReadAsByteArrayAsync().Result;
// 获取文件信息
var fileinfo = GetFileInfo(filecontent, byteArray.Length);
// 保存文件
if (fileinfo.StatusCode == 1)
{
fileinfo.StatusCode = FileHelper.WriteFile(fileinfo.UploadPath, byteArray, fileinfo.UploadName) ? 1 : 0;
}
// 记录文件信息
files.Add(fileinfo); if (fileinfo.StatusCode == 1)
{
Trace.WriteLine("客户端【" + ipAddr + "】请求上传文件" + fileinfo.UploadPath + "/" + fileinfo.UploadName + "成功");
WriteLog.WriteLogger.Info("客户端【" + ipAddr + "】请求上传文件" + fileinfo.UploadPath + "/" + fileinfo.UploadName + "成功");
}
else
{
Trace.WriteLine("客户端【" + ipAddr + "】请求上传文件" + fileinfo.UploadPath + "/" + fileinfo.UploadName + "失败");
WriteLog.WriteLogger.Info("客户端【" + ipAddr + "】请求上传文件" + fileinfo.UploadPath + "/" + fileinfo.UploadName + "成功");
}
}
return files;
}
catch (Exception ex)
{
Trace.WriteLine("客户端【" + ipAddr + "】上传文件失败!");
WriteLog.WriteLogger.Error("客户端【" + ipAddr + "】上传文件失败!", ex);
throw new HttpResponseException(Request.CreateResponse(HttpStatusCode.InternalServerError, ex));
}
});
return task;
}
else
{
// 请求格式错误
throw new HttpResponseException(Request.CreateResponse(HttpStatusCode.UnsupportedMediaType, "请求内容格式不正确!"));
}
}
catch (Exception ex)
{
Trace.WriteLine("客户端【" + ipAddr + "】上传文件失败!");
WriteLog.WriteLogger.Error("客户端【" + ipAddr + "】上传文件失败!", ex);
// 返回异常到请求端
if (ex is HttpResponseException) throw ex; else throw new HttpResponseException(Request.CreateResponse(HttpStatusCode.BadRequest, ex.Message));
}
} /// <summary>
/// 下载文件 注意:加上 attachment 这个header,在浏览器访问 url 链接时会弹出下载框
/// http://192.168.11.67:8077/file?name=filename
/// </summary>
/// <param name="name">文件相对路径</param>
/// <returns></returns>
[HttpGet]
public HttpResponseMessage Download(string name)
{
// 客户端请求IP
var ipAddr = GetRemoteClientIPAddress(); var result = new HttpResponseMessage();
// 读取文件
FileInfo file = new FileInfo(Path.Combine(UploadFolder, name.TrimStart('/', '\\')));
if (file.Exists)
{
var isImg = IsImage(name);
// 如果是图片,返回流,否则弹出下载框下载文件
if (isImg)
{
try
{
// 返回图片流
result.StatusCode = HttpStatusCode.OK;
result.Content = new StreamContent(new MemoryStream(File.ReadAllBytes(file.FullName)));
result.Content.Headers.ContentType = new MediaTypeHeaderValue("image/" + name.Substring(name.LastIndexOf('.') + 1));
Trace.WriteLine("客户端【" + ipAddr + "】请求访问图片【" + name + "】成功");
WriteLog.WriteLogger.Info("客户端【" + ipAddr + "】请求访问图片【" + name + "】成功");
}
catch (Exception ex)
{
result.StatusCode = HttpStatusCode.InternalServerError;
Trace.WriteLine("客户端【" + ipAddr + "】请求访问图片【" + name + "】失败");
WriteLog.WriteLogger.Error("客户端【" + ipAddr + "】请求访问图片【" + name + "】失败!", ex);
}
}
else
{
try
{
// 返回文件流
FileStream fs = new FileStream(file.FullName, FileMode.Open);
result = new HttpResponseMessage(HttpStatusCode.OK);
result.Content = new StreamContent(fs);
result.Content.Headers.ContentType = new MediaTypeHeaderValue("application/octet-stream");
// 加上attachment,浏览器会自动弹出下载框
result.Content.Headers.ContentDisposition = new ContentDispositionHeaderValue("attachment");
result.Content.Headers.ContentDisposition.FileName = file.Name;
//fs.Close();
Trace.WriteLine("客户端【" + ipAddr + "】请求下载文件【" + name + "】成功");
WriteLog.WriteLogger.Info("客户端【" + ipAddr + "】请求下载文件【" + name + "】成功");
}
catch (Exception ex)
{
result = new HttpResponseMessage(HttpStatusCode.InternalServerError);
Trace.WriteLine("客户端【" + ipAddr + "】请求下载文件【" + name + "】失败");
WriteLog.WriteLogger.Error("客户端【" + ipAddr + "】请求下载文件【" + name + "】失败!", ex);
}
}
}
else
{
result.StatusCode = HttpStatusCode.NotFound;
result.Content = new StringContent("找不到文件:" + name);
Trace.WriteLine("服务器上查找不到名为【" + name + "】的文件");
WriteLog.WriteLogger.Info("服务器上查找不到名为【" + name + "】的文件");
}
return result;
} /// <summary>
/// 上传存储文件夹
/// </summary>
private string UploadFolder = System.Configuration.ConfigurationManager.AppSettings.Get("FileSaveFolder"); /// <summary>
/// 图片后缀
/// </summary>
private string[] PicSuffixArray = { "png", "jpg", "jpeg" }; /// <summary>
/// 返回文件上传时的文件信息
/// </summary>
/// <param name="content">请求标头</param>
/// <param name="sizelength">文件字节流长度</param>
/// <returns></returns>
[NonAction()]
public UploadFile GetFileInfo(HttpContent content, int sizelength)
{
try
{
// 文件保存路径(源路径+前缀路径)
string savepath = UploadFolder;
string filenamestar = "";
if (!string.IsNullOrEmpty(content.Headers.ContentDisposition.FileNameStar))
{
filenamestar = content.Headers.ContentDisposition.FileNameStar.TrimStart('/', '\\').Trim(Path.GetInvalidPathChars());
savepath = Path.Combine(savepath, filenamestar);
}
// 文件名
string filename;
if (!string.IsNullOrEmpty(content.Headers.ContentDisposition.FileName))
{
filename = content.Headers.ContentDisposition.FileName.TrimStart('/', '\\').Trim(Path.GetInvalidFileNameChars());
}
else
{
filename = Guid.NewGuid().ToString();
}
while (File.Exists(Path.Combine(savepath, filename)))
{
// 文件名重复在后面增加日期
filename = filename.Insert(filename.LastIndexOf('.'), "_" + DateTime.Now.ToString("yyyyMMdd"));
}
// 计算文件大小
var size = string.Empty;
if (sizelength > 1024 * 1024)
{
size = (sizelength * 100 / 1024 / 1024 / 100.00).ToString() + "MB";
}
else
{
if (sizelength > 1024)
{
size = (sizelength * 100 / 1024 / 100.00).ToString() + "KB";
}
else
{
size = sizelength.ToString() + "B";
}
}
return new UploadFile(savepath, filename, string.Format("?name={0}", Path.Combine(filenamestar, filename)), size);
}
catch (Exception ex)
{
WriteLog.WriteLogger.Error("上传文件时获取文件信息失败!", ex);
return new UploadFile("", "", "上传失败", "", 1);
}
} /// <summary>
/// 判断是否为图片
/// </summary>
/// <param name="fileName"></param>
/// <returns></returns>
[NonAction()]
private bool IsImage(string fileName)
{
bool result = false;
try
{
fileName = fileName.ToLower();
var suffix = fileName.Substring(fileName.LastIndexOf(".") + 1);
if (PicSuffixArray.Where(x => x == suffix.ToLower()).Count() > 0)
{
result = true;
}
}
catch (Exception ex)
{
WriteLog.WriteLogger.Error("判断文件" + fileName + "是否图片", ex);
}
return result;
} /// <summary>
/// 获取请求客户的 ip 地址
/// </summary>
/// <returns></returns>
[NonAction()]
private string GetRemoteClientIPAddress()
{
var result = string.Empty; try
{
if (Request.Properties.ContainsKey("MS_OwinContext"))
result = ((OwinContext)Request.Properties["MS_OwinContext"]).Request.RemoteIpAddress;
}
catch(Exception ex)
{
WriteLog.WriteLogger.Error("获取请求客户的 ip 地址出错!", ex);
} return result;
}
} /// <summary>
/// 上传文件信息
/// </summary>
public class UploadFile
{
/// <summary>
///
/// </summary>
/// <param name="name">上传时的文件原名</param>
/// <param name="paramurl">上传后访问路径</param>
/// <param name="size">文件大小</param>
/// <param name="namestar">存储的文件路径</param>
public UploadFile(string path, string name, string paramurl, string size, int statuscode = 1)
{
UploadPath = path;
UploadName = name;
ParamUrl = paramurl;
Size = size;
StatusCode = statuscode;
} /// <summary>
/// 上传文件结果,1是成功
/// </summary>
public int StatusCode { get; set; } /// <summary>
/// 上传时的文件前缀路径
/// </summary>
public string UploadPath { get; set; } /// <summary>
/// 上传时的文件名
/// </summary>
public string UploadName { get; set; } /// <summary>
/// 文件访问的路径参数
/// </summary>
public string ParamUrl { get; set; } /// <summary>
/// 文件大小
/// </summary>
public string Size { get; set; }
}

2. Http 系列

2.1 发起请求

使用 HttpWebRequest 发起 Http 请求:https://www.cnblogs.com/MichaelLoveSna/p/14501036.html

使用 WebClient 发起 Http 请求 :https://www.cnblogs.com/MichaelLoveSna/p/14501582.html

使用 HttpClient 发起 Http 请求:https://www.cnblogs.com/MichaelLoveSna/p/14501592.html

使用 HttpClient 发起上传文件、下载文件请求:https://www.cnblogs.com/MichaelLoveSna/p/14501603.html

2.2 接受请求

使用 HttpListener 接受 Http 请求:https://www.cnblogs.com/MichaelLoveSna/p/14501628.html

使用 WepApp 接受 Http 请求:https://www.cnblogs.com/MichaelLoveSna/p/14501612.html

使用 WepApp 处理文件上传、下载请求:https://www.cnblogs.com/MichaelLoveSna/p/14501616.html

C# 应用 - 使用 WepApp 处理文件上传、下载请求的更多相关文章

  1. Struts的文件上传下载

    Struts的文件上传下载 1.文件上传 Struts2的文件上传也是使用fileUpload的组件,这个组默认是集合在框架里面的.且是使用拦截器:<interceptor name=" ...

  2. Android okHttp网络请求之文件上传下载

    前言: 前面介绍了基于okHttp的get.post基本使用(http://www.cnblogs.com/whoislcj/p/5526431.html),今天来实现一下基于okHttp的文件上传. ...

  3. Selenium2学习-039-WebUI自动化实战实例-文件上传下载

    通常在 WebUI 自动化测试过程中必然会涉及到文件上传的自动化测试需求,而开发在进行相应的技术实现是不同的,粗略可划分为两类:input标签类(类型为file)和非input标签类(例如:div.a ...

  4. 艺萌文件上传下载及自动更新系统(基于networkComms开源TCP通信框架)

    1.艺萌文件上传下载及自动更新系统,基于Winform技术,采用CS架构,开发工具为vs2010,.net2.0版本(可以很容易升级为3.5和4.0版本)开发语言c#. 本系统主要帮助客户学习基于TC ...

  5. 艺萌TCP文件上传下载及自动更新系统介绍(TCP文件传输)(一)

    艺萌TCP文件上传下载及自动更新系统介绍(TCP文件传输) 该系统基于开源的networkComms通讯框架,此通讯框架以前是收费的,目前已经免费并开元,作者是英国的,开发时间5年多,框架很稳定. 项 ...

  6. ssh框架文件上传下载

    <!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. SpringMVC——返回JSON数据&&文件上传下载

    --------------------------------------------返回JSON数据------------------------------------------------ ...

  8. 【FTP】FTP文件上传下载-支持断点续传

    Jar包:apache的commons-net包: 支持断点续传 支持进度监控(有时出不来,搞不清原因) 相关知识点 编码格式: UTF-8等; 文件类型: 包括[BINARY_FILE_TYPE(常 ...

  9. NetworkComms 文件上传下载和客户端自动升级(非开源)

    演示程序下载地址:http://pan.baidu.com/s/1geVfmcr 淘宝地址:https://shop183793329.taobao.com 联系QQ号:3201175853 许可:购 ...

随机推荐

  1. 【转】K8S中部署Helm

    K8S中的包管理工具 1. 客户端Helm(即Helm)  通过脚本安装:curl https://raw.githubusercontent.com/helm/helm/master/scripts ...

  2. HDU - 5115 Dire Wolf (非原创)

    Dire wolves, also known as Dark wolves, are extraordinarily large and powerful wolves. Many, if not ...

  3. Kattis amazingadventures Amazing Adventures(费用流路径)题解

    题意: 在一个\(100*100\)的方格中,要求从\(b\)走到\(g\),途中经过\(c\)但不经过\(u\),并且不能走已经做过的路.如果可以,就求出路径. 思路: 拆点建费用流,看能不能从\( ...

  4. xss之htmlspecialchars

    源代码: 可以看到是用htmlspecialchars 对get参数message进行处理,但是他默认不对单引号进行处理的. 只对预定义的字符进行处理: & (和号)成为 & &quo ...

  5. bash variables plus operator All In One

    bash variables plus operator All In One Errors missing pass params #!/usr/bin/env bash # echo emoji ...

  6. TypeScript tuple 元组

    TypeScript tuple 元组 元组类型允许您用固定数量的元素表示数组,这些元素的类型是已知的,但不必相同. "use strict"; /** * * @author x ...

  7. 微信公众号 & 付费阅读

    微信公众号 & 付费阅读 付费功能 付费阅读 付费功能使用说明 1.付费功能介绍 开通了付费功能的公众号,运营者可以在编辑时对原创文章的部分或全部内容设置收费.对于付费图文,用户未付费前可免费 ...

  8. 谁能成为数据储存领域领头羊?永久数据存储--NGK的终极使命!

    区块链的目的是永远存储交易网络的历史.NGK技术团队能够永久存储其去中心化账本的副本.这是其日后能进行审计关键.一些著名的团队,如Solana和SKALE,现在正在为此与NGK进行最后的集成,我们预计 ...

  9. 离场定高转弯DF与CF的对比

    也许是刚学会CAD的缘故,配合风螺旋插件,画图的感觉真是蛮爽的,忍不住画了一张又一张. 接着昨天的离场保护区,我们来聊一下PBN指定高度转弯保护区的画法.指定高度转弯的计算本身没有太多复杂的地方,真正 ...

  10. ES6 声明变量的六种方法

    ES5 只有两种声明变量的方法: var 命令和 function 命令. ES6 除了添加 let 和 const 命令, 后面章节还会提到, 另外两种声明变量的方法: import 命令和 cla ...