c# 模拟表单提交,post form 上传文件、大数据内容
表单提交协议规定:
要先将 HTTP 要求的 Content-Type 设为 multipart/form-data,而且要设定一个 boundary 参数,
这个参数是由应用程序自行产生,它会用来识别每一份资料的边界 (boundary),
用以产生多重信息部份 (message part)。
而 HTTP 服务器可以抓取 HTTP POST 的信息,
基本内容:
1. 每个信息部份都要用 --[BOUNDARY_NAME] 来包装,以分隔出信息的每个部份,而最后要再加上一个 --[BOUNDARY_NAME] 来表示结束。
2. 每个信息部份都要有一个 Content-Disposition: form-data; name="",而 name 设定的就是 HTTP POST 的键值 (key)。
3. 声明区和值区中间要隔两个新行符号(\r\n)。
4. 中间可以夹入二进制资料,但二进制资料必须要格式化为二进制字符串。
5. 若要设定不同信息段的资料型别 (Content-Type),则要在信息段内的声明区设定。
两个form内容模板
boundary = "----" + DateTime.Now.Ticks.ToString("x");//程序生成
1.文本内容
"\r\n--" + boundary +
"\r\nContent-Disposition: form-data; name=\"键\"; filename=\"文件名\"" +
"\r\nContent-Type: application/octet-stream" +
"\r\n\r\n";
2.文件内容
"\r\n--" + boundary +
"\r\nContent-Disposition: form-data; name=\"键\"" +
"\r\n\r\n内容";
代码
1.表示一个表单项的对象
- /// <summary>
- /// 表单数据项
- /// </summary>
- public class FormItemModel
- {
- /// <summary>
- /// 表单键,request["key"]
- /// </summary>
- public string Key { set; get; }
- /// <summary>
- /// 表单值,上传文件时忽略,request["key"].value
- /// </summary>
- public string Value { set; get; }
- /// <summary>
- /// 是否是文件
- /// </summary>
- public bool IsFile
- {
- get
- {
- if (FileContent==null || FileContent.Length == 0)
- return false;
- if (FileContent != null && FileContent.Length > 0 && string.IsNullOrWhiteSpace(FileName))
- throw new Exception("上传文件时 FileName 属性值不能为空");
- return true;
- }
- }
- /// <summary>
- /// 上传的文件名
- /// </summary>
- public string FileName { set; get; }
- /// <summary>
- /// 上传的文件内容
- /// </summary>
- public Stream FileContent { set; get; }
- }
2.提交表单主逻辑实现
- /// <summary>
- /// 使用Post方法获取字符串结果
- /// </summary>
- /// <param name="url"></param>
- /// <param name="formItems">Post表单内容</param>
- /// <param name="cookieContainer"></param>
- /// <param name="timeOut">默认20秒</param>
- /// <param name="encoding">响应内容的编码类型(默认utf-8)</param>
- /// <returns></returns>
- public static string PostForm(string url, List<FormItemModel> formItems, CookieContainer cookieContainer = null, string refererUrl = null, Encoding encoding = null,int timeOut = 20000)
- {
- HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
- #region 初始化请求对象
- request.Method = "POST";
- request.Timeout = timeOut;
- request.Accept = "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8";
- request.KeepAlive = true;
- request.UserAgent = "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/31.0.1650.57 Safari/537.36";
- if (!string.IsNullOrEmpty(refererUrl))
- request.Referer = refererUrl;
- if (cookieContainer != null)
- request.CookieContainer = cookieContainer;
- #endregion
- string boundary = "----" + DateTime.Now.Ticks.ToString("x");//分隔符
- request.ContentType = string.Format("multipart/form-data; boundary={0}", boundary);
- //请求流
- var postStream = new MemoryStream();
- #region 处理Form表单请求内容
- //是否用Form上传文件
- var formUploadFile = formItems != null && formItems.Count > 0;
- if (formUploadFile)
- {
- //文件数据模板
- string fileFormdataTemplate =
- "\r\n--" + boundary +
- "\r\nContent-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"" +
- "\r\nContent-Type: application/octet-stream" +
- "\r\n\r\n";
- //文本数据模板
- string dataFormdataTemplate =
- "\r\n--" + boundary +
- "\r\nContent-Disposition: form-data; name=\"{0}\"" +
- "\r\n\r\n{1}";
- foreach (var item in formItems)
- {
- string formdata = null;
- if (item.IsFile)
- {
- //上传文件
- formdata = string.Format(
- fileFormdataTemplate,
- item.Key, //表单键
- item.FileName);
- }
- else
- {
- //上传文本
- formdata = string.Format(
- dataFormdataTemplate,
- item.Key,
- item.Value);
- }
- //统一处理
- byte[] formdataBytes = null;
- //第一行不需要换行
- if (postStream.Length == 0)
- formdataBytes = Encoding.UTF8.GetBytes(formdata.Substring(2, formdata.Length - 2));
- else
- formdataBytes = Encoding.UTF8.GetBytes(formdata);
- postStream.Write(formdataBytes, 0, formdataBytes.Length);
- //写入文件内容
- if (item.FileContent != null && item.FileContent.Length>0)
- {
- using (var stream = item.FileContent)
- {
- byte[] buffer = new byte[1024];
- int bytesRead = 0;
- while ((bytesRead = stream.Read(buffer, 0, buffer.Length)) != 0)
- {
- postStream.Write(buffer, 0, bytesRead);
- }
- }
- }
- }
- //结尾
- var footer = Encoding.UTF8.GetBytes("\r\n--" + boundary + "--\r\n");
- postStream.Write(footer, 0, footer.Length);
- }
- else
- {
- request.ContentType = "application/x-www-form-urlencoded";
- }
- #endregion
- request.ContentLength = postStream.Length;
- #region 输入二进制流
- if (postStream != null)
- {
- postStream.Position = 0;
- //直接写入流
- Stream requestStream = request.GetRequestStream();
- byte[] buffer = new byte[1024];
- int bytesRead = 0;
- while ((bytesRead = postStream.Read(buffer, 0, buffer.Length)) != 0)
- {
- requestStream.Write(buffer, 0, bytesRead);
- }
- ////debug
- //postStream.Seek(0, SeekOrigin.Begin);
- //StreamReader sr = new StreamReader(postStream);
- //var postStr = sr.ReadToEnd();
- postStream.Close();//关闭文件访问
- }
- #endregion
- HttpWebResponse response = (HttpWebResponse)request.GetResponse();
- if (cookieContainer != null)
- {
- response.Cookies = cookieContainer.GetCookies(response.ResponseUri);
- }
- using (Stream responseStream = response.GetResponseStream())
- {
- using (StreamReader myStreamReader = new StreamReader(responseStream, encoding ?? Encoding.UTF8))
- {
- string retString = myStreamReader.ReadToEnd();
- return retString;
- }
- }
- }
3.调用模拟post表单
- var url = "http://127.0.0.1/testformdata.aspx?aa=1&bb=2&ccc=3";
- var log1=@"D:\temp\log1.txt";
- var log2 = @"D:\temp\log2.txt";
- var formDatas = new List<Grass.Net.FormItemModel>();
- //添加文件
- formDatas.Add(new Grass.Net.FormItemModel()
- {
- Key="log1",
- Value="",
- FileName = "log1.txt",
- FileContent=File.OpenRead(log1)
- });
- formDatas.Add(new Grass.Net.FormItemModel()
- {
- Key = "log2",
- Value = "",
- FileName = "log2.txt",
- FileContent = File.OpenRead(log2)
- });
- //添加文本
- formDatas.Add(new Grass.Net.FormItemModel()
- {
- Key = "id",
- Value = "id-test-id-test-id-test-id-test-id-test-"
- });
- formDatas.Add(new Grass.Net.FormItemModel()
- {
- Key = "name",
- Value = "name-test-name-test-name-test-name-test-name-test-"
- });
- //提交表单
- var result = PostForm(url, formDatas);
c# 模拟表单提交,post form 上传文件、大数据内容的更多相关文章
- c# 模拟表单提交,post form 上传文件、数据内容
转自:https://www.cnblogs.com/DoNetCShap/p/10696277.html 表单提交协议规定:要先将 HTTP 要求的 Content-Type 设为 multipar ...
- [Nginx 2] form表单提交,图片上传
导读:昨晚恶补了一些Nginx服务器的东西,从整体上对Nginx有一个初步的了解.上午去找师哥问了问目前项目中的使用情况,然后就开始上传图片了.这里就简单总结整理一下今天的成果,以后接着提升.简单粗暴 ...
- 上传文件表单file,限制上传文件类型的方法--参数accept
我们使用<input type="file" />来上传文件,但是当你只想要上传某种格式的文件,比如说(jpg)文件时.可以通过accept来限制. <form& ...
- 表单提交---前端页面模拟表单提交(form)
有些时候我们的前端页面总没有<form></form>表单,但是具体的业务时,我们又必须用表单提交才能达到我们想要的结果,LZ最近做了一些关于导出的一些功能,需要调用浏览器默认 ...
- 项目总结15:JavaScript模拟表单提交(实现window.location.href-POST提交数据效果)
JavaScript模拟表单提交(实现window.location.href-POST提交数据效果) 前沿 1-在具体项目开发中,用window.location.href方法下载文件,因windo ...
- 由于想要实现下载的文件可以进行选择,而不是通过<a>标签写死下载文件的参数,所以一直想要使用JFinal结合ajax实现文件下载,但是ajax实现的文件下载并不能触发浏览器的下载文件弹出框,这里通过模拟表单提交实现同样的效果。
由于想要实现下载的文件可以进行选择,而不是通过<a>标签写死下载文件的参数,所以一直想要使用JFinal结合ajax实现文件下载(这样的话ajax可以传递不同的参数),但是ajax实现的文 ...
- C# Winform利用POST传值方式模拟表单提交数据(Winform与网页交互)
其原理是,利用winfrom模拟表单提交数据.将要提交的參数提交给网页,网页运行代码.得到数据.然后Winform程序将网页的全部源码读取下来.这样就达到windows应用程序和web应用程序之间传參 ...
- ajax form表单提交 input file中的文件
ajax form表单提交 input file中的文件 现今的主流浏览器由于ajax提交form表单无法把文件类型数据提交到后台,供后台处理,可是开发中由于某些原因又不得不用ajax提交文件, 为了 ...
- HTTP通信模拟表单提交数据
前面记录过一篇关于http通信,发送数据的文章:http://www.cnblogs.com/hyyq/p/7089040.html,今天要记录的是如何通过http模拟表单提交数据. 一.通过GET请 ...
随机推荐
- jboss使用(eap 6.0以后版本)
以6.3版本为例,下载地址 https://www.jboss.org/products/eap/download/,这个地址现在有问题了,下不了好像要redhat账号还要花钱?不知咋回事 下载完成后 ...
- 24小时学通Linux内核之有关Linux文件系统实现的问题
有时间睡懒觉了,却还是五点多醒了,不过一直躺倒九点多才算起来,昨晚一直在弄飞凌的嵌入式开发板,有些问题没解决,自己电脑系统的问题,虽然Win10发布了,,但我还是好喜欢XP呀,好想回家用用家里的XP来 ...
- DAG的动态规划 (UVA 1025 A Spy in the Metro)
第一遍,刘汝佳提示+题解:回头再看!!! POINT: dp[time][sta]; 在time时刻在车站sta还需要最少等待多长时间: 终点的状态很确定必然是的 dp[T][N] = 0 ---即在 ...
- 2012蓝桥杯C组本科决赛答案
题目: 脱氧核糖核酸即常说的DNA,是一类带有遗传信息的生物大分子.它由4种主要的脱氧核苷酸(dAMP.dGMP.dCMT和dTMP)通过磷酸二酯键连接而成.这4种核苷酸可以分别记为:A.G.C.T. ...
- 小技巧之a标签自动解析URL
我们可能都知道javascript中的window.location对象用来获取当前页面的地址URL,并把浏览器重定向到新的页面.它有protocol.hostname.host.port.searc ...
- js中定义对象的方式有哪些?
1.对象直接量 2.构造函数 3.原型方法 4.动态原型方法
- 前端开发中的一些chrome插件推荐
这篇博客推荐的都是谷歌chrome浏览器插件,理论上,与之相同内核的浏览器都能使用.由于是谷歌插件,所以在天朝的网络,你懂的! 红杏 专为 学者 .程序员.外贸工作者 打造的上网加速器.我们相信,上网 ...
- Linux 命令 - history: 显示或操作历史列表
命令格式 history [-c] [-d offset] [n] history -anrw [filename] history -ps arg [arg...] 命令参数 -c 清除历史列表. ...
- Nginx - Additional Modules, Content and Encoding
The following set of modules provides functionalities having an effect on the contents served to the ...
- vs2012如何创建报表
引自百度文库,介绍的比较详细 http://wenku.baidu.com/view/c405d5f48762caaedc33d405.html