C# 大文件上传

IHttpModule 分块上传大文件

IHttpModule 分块上传大文件

来源:http://www.cnblogs.com/HeroBeast/archive/2008/03/18/1084874.html

1.一般的在Asp.net里上传文件都是10m左右,要做到大文件上传,必须要改web.config,不过改了web.config有时候也上传不成功,那是每次上传的文件太大,浏览器在这个过程中会超时,采用分块上传的方法就可以避免这种情况。

2.分块上传就是利用post的方法,把数据分块上传,每块上传的数据量少,不会引起超时的问题。不说了,看代码吧。

using System;

using System.Data;

using System.Configuration;

using System.Web;

using System.Web.Security;

using System.Web.UI;

using System.Web.UI.HtmlControls;

using System.Web.UI.WebControls;

using System.Web.UI.WebControls.WebParts;

using System.IO;

using System.Reflection;

using System.Threading;

using System.Text;

using System.Globalization;

/// <summary>

/// 实现IHttpModule接口

/// </summary>

public class HttpUploadModule : IHttpModule

{

public HttpUploadModule()

{

}

public void Init(HttpApplication application)

{

//订阅事件

application.BeginRequest += new EventHandler(this.Application_BeginRequest);

}

public void Dispose()

{

}

private void Application_BeginRequest(Object sender, EventArgs e)

{

HttpApplication app = sender as HttpApplication;

HttpWorkerRequest request = GetWorkerRequest(app.Context);

Encoding encoding = app.Context.Request.ContentEncoding;

int bytesRead = 0;  // 已读数据大小

int read;           // 当前读取的块的大小

int count = 8192;   // 分块大小

byte[] buffer;      // 保存所有上传的数据

if (request != null)

{

// 返回 HTTP 请求正文已被读取的部分。

byte[] tempBuff = request.GetPreloadedEntityBody(); //要上传的文件

// 如果是附件上传

if (tempBuff != null && IsUploadRequest(app.Request))    //判断是不是附件上传

{

// 获取上传大小

//

long length = long.Parse(request.GetKnownRequestHeader(HttpWorkerRequest.HeaderContentLength));

buffer = new byte[length];

count = tempBuff.Length; // 分块大小

// 将已上传数据复制过去

//

Buffer.BlockCopy(tempBuff,  //源数据

0,                      //从开始读

buffer,                 //目标容器

bytesRead,              //指定存储的开始位置

count);                 //要复制的字节数。

// 开始记录已上传大小

bytesRead = tempBuff.Length;

// 循环分块读取,直到所有数据读取结束

while (request.IsClientConnected() && !request.IsEntireEntityBodyIsPreloaded() && bytesRead < length)

{

// 如果最后一块大小小于分块大小,则重新分块

if (bytesRead + count > length)

{

count = (int)(length - bytesRead);

tempBuff = new byte[count];

}

// 分块读取

read = request.ReadEntityBody(tempBuff, count);

// 复制已读数据块

Buffer.BlockCopy(tempBuff, 0, buffer, bytesRead, read);

// 记录已上传大小

bytesRead += read;

}

if (request.IsClientConnected() && !request.IsEntireEntityBodyIsPreloaded())

{

// 传入已上传完的数据

InjectTextParts(request, buffer);

}

}

}

}

HttpWorkerRequest GetWorkerRequest(HttpContext context)

{

IServiceProvider provider = (IServiceProvider)HttpContext.Current;

return (HttpWorkerRequest)provider.GetService(typeof(HttpWorkerRequest));

}

/// <summary>

/// 传入已上传完的数据

/// </summary>

/// <param name="request"></param>

/// <param name="textParts"></param>

void InjectTextParts(HttpWorkerRequest request, byte[] textParts)

{

BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.NonPublic;

Type type = request.GetType();

while ((type != null) && (type.FullName != "System.Web.Hosting.ISAPIWorkerRequest"))

{

type = type.BaseType;

}

if (type != null)

{

type.GetField("_contentAvailLength", bindingFlags).SetValue(request, textParts.Length);

type.GetField("_contentTotalLength", bindingFlags).SetValue(request, textParts.Length);

type.GetField("_preloadedContent", bindingFlags).SetValue(request, textParts);

type.GetField("_preloadedContentRead", bindingFlags).SetValue(request, true);

}

}

private static bool StringStartsWithAnotherIgnoreCase(string s1, string s2)

{

return (string.Compare(s1, 0, s2, 0, s2.Length, true, CultureInfo.InvariantCulture) == 0);

}

/// <summary>

/// 是否为附件上传

/// 判断的根据是ContentType中有无multipart/form-data

/// </summary>

/// <param name="request"></param>

/// <returns></returns>

bool IsUploadRequest(HttpRequest request)

{

return StringStartsWithAnotherIgnoreCase(request.ContentType, "multipart/form-data");

}

}

3.用法

(1)修改web.config

<httpModules>

<add name="up" type="HttpUploadModule"/>

</httpModules>

<httpRuntime maxRequestLength="2000000" executionTimeout="300" />

(2)aspx

<form id="form1" runat="server" enctype="multipart/form-data">

<div>

<asp:FileUpload ID="FileUpload1" runat="server" />

<asp:Button ID="Button1" runat="server" onclick="Button1_Click" Text="Button" />

</div>

</form>

(3)aspx.cs

protected void Button1_Click(object sender, EventArgs e)

{

string strDesPath = "D:\";

string strFileName = this.FileUpload1.PostedFile.FileName;

strFileName = strDesPath + strFileName;//.Substring(strFileName.LastIndexOf("\"));

//

this.FileUpload1.PostedFile.SaveAs(strFileName);

Response.Write("文件保存到了:" + strFileName);

}

4.大文件上传的限制

虽然可以上传大文件,但是这个大小也是有限制的,不能超过2G的大小。

分类: VS2005

绿色通道: 好文要顶 关注我 收藏该文与我联系

时间: 2024-10-15 05:32:47

C# 大文件上传的相关文章

基于Nodejs的大文件上传之断点续传

接着<扒一扒Nodejs formidable的onPart>和<也说文件上传之兼容IE789的进度条---丢掉flash>:前面已完成兼容IE789的大文件上传:无flash的低版本进度条,高版本的分段上传,并已为断点续传做好铺垫: 说什么做好铺垫,原本以为Nodejs端已没问题,只剩前端依靠HTML5接着监听abort事件,保存中断时上传到第几块了(断点续传只支持文件比较大,然后意外上传中断了,暂时定50M开启断点续传吧),通过文件内容hash和该文件唯一上传token来记录断

【原创】用JAVA实现大文件上传及显示进度信息

用JAVA实现大文件上传及显示进度信息 ---解析HTTP MultiPart协议 一. 大文件上传基础描述: 各种WEB框架中,对于浏览器上传文件的请求,都有自己的处理对象负责对Http MultiPart协议内容进行解析,并供开发人员调用请求的表单内容. 比如: Spring 框架中使用类似CommonsMultipartFile对象处理表二进制文件信息. 而.NET 中使用HtmlInputFile/ HttpPostedFile对象处理二进制文件信息. 优点:使用框架内置对象可以很方便的

Socket大文件上传

1 public sealed class SocketData 2 { 3 private SocketData() 4 { 5 } 6 7 public static SendFileMode SendFile(Socket socket, string fileName, int maxBufferLength) 8 { 9 SendFileMode flag = SendFileMode.Success; 10 try 11 { 12 using (Stream fs = new Fil

IIS中的大文件上传问题解决方法

IIS出于安全考虑限制了大文件的上传,而网上百度到的大部分解决方法都是用一个管理员权限的记事本打开一个文件修改参数,但是我发现里面根本没有网上所说的那些参数,最后自己找到了修改发布文件的webconfig的方法解决的IIS对大文件上传的限制. 首先在system.web中加入以下代码 [csharp] view plain copy <httpRuntime maxRequestLength="2097151"//最大上传长度 useFullyQualifiedRedirectU

Nodejs+HTML5兼容IE789的大文件上传完整版

业余将大文件上传重新梳理了一遍,后端基于Nodejs:有几个要点感觉很好玩: 兼容性:IE789为代表: 跨域上传:document.domain||middlePage: 多文件上传:input['type=file'] multiple: 拖拽上传:drag drop: 大文件分段:files.slice(s,e): 断点续传:localStorage: 接收分段的文件:formidable.onPart: 陆续写入分段文件:fs.write(fd,bf,offset,length,posi

gitlab使用过程中遇到大文件上传或下载失败的问题,总结一下

环境如下:gitlab服务器redhat,客户端环境mac os,如果是其他环境遇到问题仅供参考 如果gitlab上传代码提示: error: RPC failed; result=22, HTTP code = 411 该问题是由于客户端设置的http_post_buffer大小不足导致的,解决方法如下: 进入到工程所在的终端目录下执行: git config http.postBuffer 524288000 如果gitlab上传代码提示: error: RPC failed; result

因用了NeatUpload大文件上传控件而导致Nonfile portion &gt; 4194304 bytes错误的解决方法

今天遇到一个问题,就是"NeatUpload大文件上传控件而导致Nonfile portion > 4194304 bytes错误",百度后发现了一个解决方法,跟大家分享下: NeatUpload是一个开源的大文件上传控件,非常的强大,支持文件类型过滤.上传进度条显示.多文件上传等强大的功能. 但部署至项目后,有些地方用普通的FileUpload上传时却发生了一个错误(Nonfile portion > 4194304 bytes,文件大于默认值4M),因如果用NeatUp

将小文件合并大文件上传

自定义方法将本地多个小文件合并成一个大文件上传到HDFS上. package test; import java.net.URI; import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.fs.FSDataInputStream; import org.apache.hadoop.fs.FSDataOutputStream; import org.apache.hadoop.fs.FileStatus; impo

BootStrap Progressbar 实现大文件上传的进度条

1.首先实现大文件上传,如果是几兆或者几十兆的文件就用基本的上传方式就可以了,但是如果是大文件上传的话最好是用分片上传的方式.我这里主要是使用在客户端进行分片读取到服务器段,然后保存,到了服务器段读取完了之后将分片数据进行组合. 2.前端代码如下: <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="UploadTest2.aspx.cs" Inherits="Htm