表单提交协议规定:
要先将 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.表示一个表单项的对象

  1. /// <summary>
  2. /// 表单数据项
  3. /// </summary>
  4. public class FormItemModel
  5. {
  6. /// <summary>
  7. /// 表单键,request["key"]
  8. /// </summary>
  9. public string Key { set; get; }
  10. /// <summary>
  11. /// 表单值,上传文件时忽略,request["key"].value
  12. /// </summary>
  13. public string Value { set; get; }
  14. /// <summary>
  15. /// 是否是文件
  16. /// </summary>
  17. public bool IsFile
  18. {
  19. get
  20. {
  21. if (FileContent==null || FileContent.Length == 0)
  22. return false;
  23. if (FileContent != null && FileContent.Length > 0 && string.IsNullOrWhiteSpace(FileName))
  24. throw new Exception("上传文件时 FileName 属性值不能为空");
  25. return true;
  26. }
  27. }
  28. /// <summary>
  29. /// 上传的文件名
  30. /// </summary>
  31. public string FileName { set; get; }
  32. /// <summary>
  33. /// 上传的文件内容
  34. /// </summary>
  35. public Stream FileContent { set; get; }
  36. }

2.提交表单主逻辑实现

  1. /// <summary>
  2. /// 使用Post方法获取字符串结果
  3. /// </summary>
  4. /// <param name="url"></param>
  5. /// <param name="formItems">Post表单内容</param>
  6. /// <param name="cookieContainer"></param>
  7. /// <param name="timeOut">默认20秒</param>
  8. /// <param name="encoding">响应内容的编码类型(默认utf-8)</param>
  9. /// <returns></returns>
  10. public static string PostForm(string url, List<FormItemModel> formItems, CookieContainer cookieContainer = null, string refererUrl = null, Encoding encoding = null,int timeOut = 20000)
  11. {
  12. HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
  13. #region 初始化请求对象
  14. request.Method = "POST";
  15. request.Timeout = timeOut;
  16. request.Accept = "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8";
  17. request.KeepAlive = true;
  18. request.UserAgent = "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/31.0.1650.57 Safari/537.36";
  19. if (!string.IsNullOrEmpty(refererUrl))
  20. request.Referer = refererUrl;
  21. if (cookieContainer != null)
  22. request.CookieContainer = cookieContainer;
  23. #endregion
  24. string boundary = "----" + DateTime.Now.Ticks.ToString("x");//分隔符
  25. request.ContentType = string.Format("multipart/form-data; boundary={0}", boundary);
  26. //请求流
  27. var postStream = new MemoryStream();
  28. #region 处理Form表单请求内容
  29. //是否用Form上传文件
  30. var formUploadFile = formItems != null && formItems.Count > 0;
  31. if (formUploadFile)
  32. {
  33. //文件数据模板
  34. string fileFormdataTemplate =
  35. "\r\n--" + boundary +
  36. "\r\nContent-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"" +
  37. "\r\nContent-Type: application/octet-stream" +
  38. "\r\n\r\n";
  39. //文本数据模板
  40. string dataFormdataTemplate =
  41. "\r\n--" + boundary +
  42. "\r\nContent-Disposition: form-data; name=\"{0}\"" +
  43. "\r\n\r\n{1}";
  44. foreach (var item in formItems)
  45. {
  46. string formdata = null;
  47. if (item.IsFile)
  48. {
  49. //上传文件
  50. formdata = string.Format(
  51. fileFormdataTemplate,
  52. item.Key, //表单键
  53. item.FileName);
  54. }
  55. else
  56. {
  57. //上传文本
  58. formdata = string.Format(
  59. dataFormdataTemplate,
  60. item.Key,
  61. item.Value);
  62. }
  63. //统一处理
  64. byte[] formdataBytes = null;
  65. //第一行不需要换行
  66. if (postStream.Length == 0)
  67. formdataBytes = Encoding.UTF8.GetBytes(formdata.Substring(2, formdata.Length - 2));
  68. else
  69. formdataBytes = Encoding.UTF8.GetBytes(formdata);
  70. postStream.Write(formdataBytes, 0, formdataBytes.Length);
  71. //写入文件内容
  72. if (item.FileContent != null && item.FileContent.Length>0)
  73. {
  74. using (var stream = item.FileContent)
  75. {
  76. byte[] buffer = new byte[1024];
  77. int bytesRead = 0;
  78. while ((bytesRead = stream.Read(buffer, 0, buffer.Length)) != 0)
  79. {
  80. postStream.Write(buffer, 0, bytesRead);
  81. }
  82. }
  83. }
  84. }
  85. //结尾
  86. var footer = Encoding.UTF8.GetBytes("\r\n--" + boundary + "--\r\n");
  87. postStream.Write(footer, 0, footer.Length);
  88. }
  89. else
  90. {
  91. request.ContentType = "application/x-www-form-urlencoded";
  92. }
  93. #endregion
  94. request.ContentLength = postStream.Length;
  95. #region 输入二进制流
  96. if (postStream != null)
  97. {
  98. postStream.Position = 0;
  99. //直接写入流
  100. Stream requestStream = request.GetRequestStream();
  101. byte[] buffer = new byte[1024];
  102. int bytesRead = 0;
  103. while ((bytesRead = postStream.Read(buffer, 0, buffer.Length)) != 0)
  104. {
  105. requestStream.Write(buffer, 0, bytesRead);
  106. }
  107. ////debug
  108. //postStream.Seek(0, SeekOrigin.Begin);
  109. //StreamReader sr = new StreamReader(postStream);
  110. //var postStr = sr.ReadToEnd();
  111. postStream.Close();//关闭文件访问
  112. }
  113. #endregion
  114. HttpWebResponse response = (HttpWebResponse)request.GetResponse();
  115. if (cookieContainer != null)
  116. {
  117. response.Cookies = cookieContainer.GetCookies(response.ResponseUri);
  118. }
  119. using (Stream responseStream = response.GetResponseStream())
  120. {
  121. using (StreamReader myStreamReader = new StreamReader(responseStream, encoding ?? Encoding.UTF8))
  122. {
  123. string retString = myStreamReader.ReadToEnd();
  124. return retString;
  125. }
  126. }
  127. }

3.调用模拟post表单

  1. var url = "http://127.0.0.1/testformdata.aspx?aa=1&bb=2&ccc=3";
  2. var log1=@"D:\temp\log1.txt";
  3. var log2 = @"D:\temp\log2.txt";
  4. var formDatas = new List<Grass.Net.FormItemModel>();
  5. //添加文件
  6. formDatas.Add(new Grass.Net.FormItemModel()
  7. {
  8. Key="log1",
  9. Value="",
  10. FileName = "log1.txt",
  11. FileContent=File.OpenRead(log1)
  12. });
  13. formDatas.Add(new Grass.Net.FormItemModel()
  14. {
  15. Key = "log2",
  16. Value = "",
  17. FileName = "log2.txt",
  18. FileContent = File.OpenRead(log2)
  19. });
  20. //添加文本
  21. formDatas.Add(new Grass.Net.FormItemModel()
  22. {
  23. Key = "id",
  24. Value = "id-test-id-test-id-test-id-test-id-test-"
  25. });
  26. formDatas.Add(new Grass.Net.FormItemModel()
  27. {
  28. Key = "name",
  29. Value = "name-test-name-test-name-test-name-test-name-test-"
  30. });
  31. //提交表单
  32. var result = PostForm(url, formDatas);
 
转载自 http://blog.csdn.net/xxj_jing/article/details/50221113

c# 模拟表单提交,post form 上传文件、大数据内容的更多相关文章

  1. c# 模拟表单提交,post form 上传文件、数据内容

    转自:https://www.cnblogs.com/DoNetCShap/p/10696277.html 表单提交协议规定:要先将 HTTP 要求的 Content-Type 设为 multipar ...

  2. [Nginx 2] form表单提交,图片上传

    导读:昨晚恶补了一些Nginx服务器的东西,从整体上对Nginx有一个初步的了解.上午去找师哥问了问目前项目中的使用情况,然后就开始上传图片了.这里就简单总结整理一下今天的成果,以后接着提升.简单粗暴 ...

  3. 上传文件表单file,限制上传文件类型的方法--参数accept

    我们使用<input type="file" />来上传文件,但是当你只想要上传某种格式的文件,比如说(jpg)文件时.可以通过accept来限制. <form& ...

  4. 表单提交---前端页面模拟表单提交(form)

    有些时候我们的前端页面总没有<form></form>表单,但是具体的业务时,我们又必须用表单提交才能达到我们想要的结果,LZ最近做了一些关于导出的一些功能,需要调用浏览器默认 ...

  5. 项目总结15:JavaScript模拟表单提交(实现window.location.href-POST提交数据效果)

    JavaScript模拟表单提交(实现window.location.href-POST提交数据效果) 前沿 1-在具体项目开发中,用window.location.href方法下载文件,因windo ...

  6. 由于想要实现下载的文件可以进行选择,而不是通过<a>标签写死下载文件的参数,所以一直想要使用JFinal结合ajax实现文件下载,但是ajax实现的文件下载并不能触发浏览器的下载文件弹出框,这里通过模拟表单提交实现同样的效果。

    由于想要实现下载的文件可以进行选择,而不是通过<a>标签写死下载文件的参数,所以一直想要使用JFinal结合ajax实现文件下载(这样的话ajax可以传递不同的参数),但是ajax实现的文 ...

  7. C# Winform利用POST传值方式模拟表单提交数据(Winform与网页交互)

    其原理是,利用winfrom模拟表单提交数据.将要提交的參数提交给网页,网页运行代码.得到数据.然后Winform程序将网页的全部源码读取下来.这样就达到windows应用程序和web应用程序之间传參 ...

  8. ajax form表单提交 input file中的文件

    ajax form表单提交 input file中的文件 现今的主流浏览器由于ajax提交form表单无法把文件类型数据提交到后台,供后台处理,可是开发中由于某些原因又不得不用ajax提交文件, 为了 ...

  9. HTTP通信模拟表单提交数据

    前面记录过一篇关于http通信,发送数据的文章:http://www.cnblogs.com/hyyq/p/7089040.html,今天要记录的是如何通过http模拟表单提交数据. 一.通过GET请 ...

随机推荐

  1. Android 自学之滚动视图ScrollView

    滚动视图ScrollView由FarmeLayout派生而出,他就是一个用于为普通组件添加垂直滚动条的组件:ScrollView里面最多包含一个组件,而ScrollView的作用就是为该组件添加一个垂 ...

  2. 怎么在html页面和js里判断是否是IE浏览器

    HTML里: HTML代码中,在编写网页代码时,各种浏览器的兼容性是个必须考虑的问题,有些时候无法找到适合所有浏览器的写法,就只能写根据浏览器种类区别的代码,这时就要用到判断代码了.在HTML代码中, ...

  3. HDU 3533 Escape (BFS + 预处理)

    Escape Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  4. 【模拟ACM排名】ZOJ-2593 Ranking (Andrew Stankevich’s Contest #5)

    真心是道水题,但找bug找的我想剁手了/(ㄒoㄒ)/~~ 注意几个坑点, 1.输入,getline(cin); / gets(); 一行输入,注意前面要加getchar();   输入运行记录的时候可 ...

  5. Mina的线程模型

    在Mina的NIO模式中有三种I/O工作线程(这三种线程模型只在NIOSocket中有效,在NIO数据包和虚拟管道中没有,也不需要配置): IoAcceptor/IoConnector线程 IoPro ...

  6. sort排序中的坑

    问题的产生原因: 在一篇阿里面试题的跟帖中,很多人应用sort()方法对数组进行排序.看似合情合理的代码,运行结果却频频出错.为什么呢?因为很多人都忽略掉了一点,那就是sort()排序默认情况下是按A ...

  7. int/double/string使用

    在计算机中存储数据和儿童在抽屉中存放物品很类似. 例如: 要在计算机中存一个数字50,需要两句话. int a;  //将要放的物品告诉家长 a=50;  //将物品放到某个抽屉中 计算机存储变量的过 ...

  8. C# 微信扫码支付 回调页面

    .NET版 微信扫码支付,官方推荐使用[模式二] 一.微信扫码支付模式一: 1.回调页面:官方demo中example文件下的NativeNotifyPage.aspx 2.微信回调地址:http:/ ...

  9. 返璞归真vc++之感言

    本人自述,大专学历,感觉自己也属于好学型学生,历任班上学习委员3年有余,参与学校项目几多个,不知道不觉从11年毕业已有3个年头,3年来,不敢苟同自己的生活方式,奈何人生无奈..从刚开始的电子商务公司转 ...

  10. Ubuntu gedit 折叠插件

    Ubuntu Kylin 14.04 gedit  - Version 3.10.4 (as same as all version of gedit 3.x ) Attention: this pl ...