zoukankan      html  css  js  c++  java
  • C#模拟POST提交上传文件

    要在客户端向服务器上传文件,我们就必须模拟一个POSTmultipart/form-data类型的请求,Content-Type必须是multipart/form-data。
      以multipart/form-data编码的POST请求格式与application/x-www-form-urlencoded完全不同,multipart/form-data需要首先在HTTP请求头设置一个分隔符,例如ABCD:
      我们模拟的提交要设定 content-type不同于非含附件的post时候的content-type,这里需要: ("Content-Type","multipart/form-data; boundary=ABCD");
      然后,将每个字段用“–分隔符”分隔,最后一个“–分隔符–”表示结束。例如,要上传一个title字段"Today"和一个文件C:\1.txt,HTTP正文如下:
    –ABCD
    Content-Disposition: form-data; name="title"
    \r\n
    Today
    –ABCD
    Content-Disposition: form-data; name="1.txt"; filename="C:\1.txt"
    Content-Type: text/plain
    \r\n
    <这里是1.txt文件的内容>
    –ABCD–
    \r\n
      请注意,每一行都必须以\r\n结束,包括最后一行。
    如果用Sniffer程序检测IE发送的POST请求,可以发现IE的分隔符类似于——7d4a6d158c9,这是IE产生的一个随机数,目的是防止上传文件中出现分隔符导致服务器无法正确识别文件起始位置。我们可以写一个固定的分隔符,只要足够复杂即可。
    我们暂且不说如何去模拟数据,通过一个简单的form看看当请求发生时,客户端提交了什么样的数据给服务端。
    下面是一个简单的html form,两个文本输入框,一个文件上传(这里我选择一张图片),注意有文件上传的form的enctype属性。
    复制代码代码如下:
    <form action="sql.aspx"method="post"enctype="multipart/form-data">
    <input id="Text1"name="content"type="text"/><br />
    <input id="Text2"name="uploadImg"type="text"/><br />
    <input id="File1"type="file"name="image0"/><br />
    <input id="Submit1"type="submit"value="submit"/>
    </form>
    为 了查看表单提交时,向服务端post了什么数据,这里我使用Fiddler来查看。Fiddler确实是个不错的工具,注意当url主机地址是 localhost时Fiddler捕获不到,需要再localhost后加一点(.)即可,打开Fiddler,浏览带上面form的page,输入数 据提交,此时在Fiddler中可看到post的数据了。
    分析其中的数据不难得出,一个表单中的数据域(input type="text")对应的格式为
    —————————–7da119c1004a6
    Content-Disposition: form-data; name="content"
    this is a txt value
    一个文件(input type="file")对应的格式为(通常为表单最后一个参数)
    —————————–7da119c1004a6
    Content-Disposition: form-data; name="image0"; filename="E:\CAI\875.jpg"
    Content-Type: image/pjpeg
    [文件内容]
    结尾处是—————————–7da119c1004a6–
    有了上面的数据做参考,按照其格式组织数据,post到服务端,同样可以达到html form提交的效果。要特别注意其格式:如回车换行,差一个都可能得不到正确的响应,还有请求的Content-Length一定计算对。下面是一个参考:
    代码
    复制代码代码如下:
    public string POSTfile(string v1,string v2, string file)
    {
    string boundary ="—————————"+ DateTime.Now.Ticks.ToString("x");
    //请求
    WebRequest req = WebRequest.Create(@"http://localhost.:4944/WebSite1/getfile.aspx");
    req.Method ="POST";
    req.ContentType ="multipart/form-data; boundary="+ boundary;
    //组织表单数据
    StringBuilder sb = new StringBuilder();
    sb.Append("–"+ boundary);
    sb.Append("\r\n");
    sb.Append("Content-Disposition: form-data; name=\"content\"");
    sb.Append("\r\n\r\n");
    sb.Append(v1);
    sb.Append("\r\n");
    sb.Append("–"+ boundary);
    sb.Append("\r\n");
    sb.Append("Content-Disposition: form-data; name=\"uploadImg\"");
    sb.Append("\r\n\r\n");
    sb.Append("v2");
    sb.Append("\r\n");
    sb.Append("–"+ boundary);
    sb.Append("\r\n");
    sb.Append("Content-Disposition: form-data; name=\"image0\"; filename=\"e:\\a.jpg\"");
    sb.Append("\r\n");
    sb.Append("Content-Type: image/pjpeg");
    sb.Append("\r\n\r\n");
    string head = sb.ToString();
    byte[] form_data = Encoding.UTF8.GetBytes(head);
    //结尾
    byte[] foot_data = Encoding.UTF8.GetBytes("\r\n–"+ boundary +"–\r\n");
    //文件
    FileStream fileStream = new FileStream(file, FileMode.Open, FileAccess.Read);
    //post总长度
    long length = form_data.Length + fileStream.Length + foot_data.Length;
    req.ContentLength = length;
    Stream requestStream = req.GetRequestStream();
    //发送表单参数
    requestStream.Write(form_data, 0, form_data.Length);
    //文件内容
    byte[] buffer = new Byte[checked((uint)Math.Min(4096, (int)fileStream.Length))];
    int bytesRead = 0;
    while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0)
    requestStream.Write(buffer, 0, bytesRead);
    //结尾
    requestStream.Write(foot_data, 0, foot_data.Length);
    requestStream.Close();
    //响应
    WebResponse pos = req.GetResponse();
    StreamReader sr = new StreamReader(pos.GetResponseStream(), Encoding.UTF8);
    string html = sr.ReadToEnd().Trim();
    sr.Close();
    if (pos != null)
    {
    pos.Close();
    pos = null;
    }
    if (req != null)
    {
    req = null;
    }
    return html;
    }

  • 相关阅读:
    等保测评(一)
    一个基于RNN的实体抽取工具
    如何画UML类图
    mysql存储过程整理
    记一次mysql事务未提交导致锁未释放的问题
    开启·元宇宙·区块链金融
    Nacos启动报错:[db-load-error]load jdbc.properties error
    使用Bazel编译TypeScript
    Win10上Docker无法正常启动 出现install WSL2 kernel update的情况
    VSCode调试PHP环境配置 phpstudy vscode xdebug调试
  • 原文地址:https://www.cnblogs.com/zcm123/p/3038254.html
Copyright © 2011-2022 走看看