• android 上传文件(表单),表单填写格式分析


    原文地址:http://blog.sina.com.cn/s/blog_68494364010116gk.html

    在Android的客户端编程中(特别是SNS 类型的客户端),经常需要实现注册功能Activity,要用户输入用户名,密码,邮箱,照片后注册。但这时就有一个问题,在HTML中用form表单就 能实现如上的注册表单,需要的信息会自动封装为完整的HTTP协议,但在Android中如何把这些参数和需要上传的文件封装为HTTP协议呢?

    我们可以先做个试验,看一下form表单到底封装了什么样的信息。

    第一步:编写一个Servlet,把接收到的HTTP信息保存在一个文件中,代码如下:

    1.     public void doPost(HttpServletRequest request, HttpServletResponse response)
    2.  
    3.            throws ServletException, IOException {
    4.  
    5.  
    6.  
    7.        //获取输入流,是HTTP协议中的实体内容
    8.  
    9.        ServletInputStream  sis=request.getInputStream();
    10.  
    11.      
    12.  
    13.        //缓冲区
    14.  
    15.        byte buffer[]=new byte[1024];
    16.  
    17.       
    18.  
    19.        FileOutputStream fos=new FileOutputStream("d:\file.log");
    20.  
    21.       
    22.  
    23.        int len=sis.read(buffer, 0, 1024);
    24.  
    25.       
    26.  
    27.        //把流里的信息循环读入到file.log文件中
    28.  
    29.        while( len!=-1 )
    30.  
    31.        {
    32.  
    33.            fos.write(buffer, 0, len);
    34.  
    35.            len=sis.readLine(buffer, 0, 1024);
    36.  
    37.        }
    38.  
    39.       
    40.  
    41.        fos.close();
    42.  
    43.        sis.close();
    44.  
    45.       
    46.  
    47.     }
    48.  
    49.  

    第二步:实现如下一个表单页面, 详细的代码如下:

    1. <form action="servlet/ReceiveFile" method="post" enctype="multipart/form-data">
    2.  
    3.     第一个参数<input type="text" name="name1"/> <br/>
    4.  
    5.     第二个参数<input type="text" name="name2"/> <br/>
    6.  
    7.     第一个上传的文件<input type="file" name="file1"/> <br/>
    8.  
    9.     第二个上传的文件<input type="file" name="file2"/> <br/>
    10.  
    11.     <input type="submit" value="提交">
    12.  
    13. </form>

    注意了,由于要上传附件,所以一定要设置enctype为multipart/form-data,才可以实现附件的上传。

    第三步:填写完信息后按“提交”按钮后,在D盘下查找file.log文件用记事本打开,数据如下:

    —————————–7d92221b604bc

    Content-Disposition: form-data; name=”name1″

    hello

    —————————–7d92221b604bc

    Content-Disposition: form-data; name=”name2″

    world

    —————————–7d92221b604bc

    Content-Disposition: form-data; name=”file1″; filename=”C:2.GIF”

    Content-Type: image/gif

    GIF89a

          €   € €€   €€ € €€€€€览?                                                                                     3  f       3  33 3f 3 3 3 f  f3 ff f f f   ? 檉 櫃 櫶 ?   ? 蘤 虣 烫 ?   3 f   3  3 33 f3 ? ? 33 33333f33?3?33f 3f33ff3f?f?f3 3?3檉3櫃3櫶3?3 3?3蘤3虣3烫3?3 333f3??f  f 3f ff 檉 蘤 f3 f33f3ff3檉3蘤3ff ff3fffff檉f蘤ff f?f檉f櫃f櫶f?f f?f蘤f虣f烫f?f f3fff檉蘤   3 f 櫃 虣 ? ?3?f?櫃3虣3檉 檉3檉f檉櫃f虣f櫃 櫃3櫃f櫃櫃櫶櫃櫶 櫶3櫶f櫶櫃烫櫶? ?3?f?櫃虣   3 f 櫶 烫 ? ?3?f?櫶3烫3蘤 蘤3蘤f蘤櫶f烫f虣 虣3虣f虣櫶櫶虣烫 烫3烫f烫櫶烫烫? ?3?f?櫶烫   3 f ? ? 3 333f3?3?3f f3fff?f?f ?檉櫃櫶??蘤虣烫? 3f??!?   ,   

      e   ??羵Q鸚M!C囑lH馉脝远5荑p釩?3R?R愣?MV39V5?�谈re琷?试   3??qn?薵Q燚c?獖i郸EW艗赥戟j ;

    —————————–7d92221b604bc

    Content-Disposition: form-data; name=”file2″; filename=”C:2.txt”

    Content-Type: text/plain

    hello everyone!!!

    —————————–7d92221b604bc–

           从表单源码可知,表单上传的数据有4个:参数name1和name2,文件file1和file2

    首先从file.log观察两个参数name1和name2的情况。这时候使用UltraEdit打开file.log(因为有些字符在记事本里显示不出来,所以要用16进制编辑器)

           结合16进制数据和记事本显示的数据可知上传参数部分的格式规律:

    1.       第一行是“—————————–7d92221b604bc”作为分隔符,然后是“ ”(即16进制编辑器显示的0D 0A)回车换行符。

    2.       第二行

    (1)       首先是HTTP中的扩展头部分“Content-Disposition: form-data;”,表示上传的是表单数据。

    (2)       “name=”name1″”参数的名称。

    (3)       “ ”(即16进制编辑器显示的0D 0A)回车换行符。

    3.       第三行:“ ”(即16进制编辑器显示的0D 0A)回车换行符。

    4.       第四行:参数的值,最后是“ ”(即16进制编辑器显示的0D 0A)回车换行符。

    由观察可得,表单上传的每个参数都是按照以上1—4的格式构造HTTP协议中的参数部分。

    结合16进制数据和记事本显示的数据可知上传文件部分的格式规律:

    1.       第一行是“—————————–7d92221b604bc”作为分隔符,然后是“ ”(即16进制编辑器显示的0D 0A)回车换行符。

    2.       第二行:

    a)         首先是HTTP中的扩展头部分“Content-Disposition: form-data;”,表示上传的是表单数据。

    b)        “name=”file2″;”参数的名称。

    c)        “filename=”C:2.txt””参数的值。

    d)        “ ”(即16进制编辑器显示的0D 0A)回车换行符。

    3.       第三行:HTTP中的实体头部分“Content-Type: text/plain”:表示所接收到得实体内容的文件格式。计算机的应用中有多种多种通用的文件格式,人们为每种通用格式都定义了一个名称,称为 MIME,MIME的英文全称是”Multipurpose Internet Mail Extensions” (多功能Internet 邮件扩充服务)

    4.       第四行:“ ”(即16进制编辑器显示的0D 0A)回车换行符。

    5.       第五行开始:上传的内容的二进制数。

    6.       最后是结束标志“—————————–7d92221b604bc–”,注意:这个结束标志和分隔符的区别是最后多了“–”部分。

    但现在还有一个问题,就是分隔符“—————————–7d92221b604bc”是怎么确定的呢?是不是一定要“7d92221b604bc”这串数字?

            我们以前的分析只是观察了HTTP请求的实体部分,可以借用工具观察完整的HTTP请求看一看有没有什么线索?

      在IE下用HttpWatch,在Firefox下用Httpfox这个插件,可以实现网页数据的抓包,从图4可看出,原来在Content-Type部分指定了分隔符所用的字符串。

     
    根据以上总结的注册表单中的参数传递和文件上传的规律,我们可以能写出Android中实现一个用户注册功能(包括个人信息填写和上传图片部分)的工具类,

    首先,要有一个javaBean类FormFile封装文件的信息:

    1. public class FormFile {
    2.  
    3.  private byte[] data;
    4.  
    5.  private String filname;
    6.  
    7.  private String formname;
    8.  
    9.  private String contentType = "application/octet-stream"; //需要查阅相关的资料
    10.  
    11.  public FormFile(String filname, byte[] data, String formname, String contentType) {
    12.   this.data = data;
    13.   this.filname = filname;
    14.   this.formname = formname;
    15.   if(contentType!=null) this.contentType = contentType;
    16.  }
    17.  
    18.  public byte[] getData() {
    19.   return data;
    20.  }
    21.  
    22.  public void setData(byte[] data) {
    23.   this.data = data;
    24.  }
    25.  
    26.  public String getFilname() {
    27.   return filname;
    28.  }
    29.  
    30.  public void setFilname(String filname) {
    31.   this.filname = filname;
    32.  }
    33.  
    34.  public String getFormname() {
    35.   return formname;
    36.  }
    37.  
    38.  public void setFormname(String formname) {
    39.   this.formname = formname;
    40.  }
    41.  
    42.  public String getContentType() {
    43.   return contentType;
    44.  }
    45.  
    46.  public void setContentType(String contentType) {
    47.   this.contentType = contentType;
    48.  }
    49.  
    50. }
    51.  

     
    实现文件上传的代码如下:

     
    public static String post(String actionUrl, Map<String, String> params, FormFile[] files) {  
        try {             
            String BOUNDARY = “———7d4a6d158c9″; //数据分隔线  
            String MULTIPART_FORM_DATA = “multipart/form-data”;  
              
            URL url = new URL(actionUrl);  
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();  
            conn.setDoInput(true);//允许输入  
            conn.setDoOutput(true);//允许输出  
            conn.setUseCaches(false);//不使用Cache  
            conn.setRequestMethod(”POST”);            
            conn.setRequestProperty(”Connection”, “Keep-Alive”);  
            conn.setRequestProperty(”Charset”, “UTF-8″);  
            conn.setRequestProperty(”Content-Type”, MULTIPART_FORM_DATA + “; boundary=” + BOUNDARY);  
     
            StringBuilder sb = new StringBuilder();  
              
            //上传的表单参数部分,格式请参考文章  
            for (Map.Entry<String, String> entry : params.entrySet()) {//构建表单字段内容  
                sb.append(”–”);  
                sb.append(BOUNDARY);  
                sb.append(” ”);  
                sb.append(”Content-Disposition: form-data; name=”"+ entry.getKey() + “” ”);  
                sb.append(entry.getValue());  
                sb.append(” ”);  
            }  
            DataOutputStream outStream = new DataOutputStream(conn.getOutputStream());  
            outStream.write(sb.toString().getBytes());//发送表单字段数据  
             
            //上传的文件部分,格式请参考文章  
            for(FormFile file : files){  
                StringBuilder split = new StringBuilder();  
                split.append(”–”);  
                split.append(BOUNDARY);  
                split.append(” ”);  
                split.append(”Content-Disposition: form-data;name=”"+ file.getFormname()+””;filename=”"+ file.getFilname() + “” ”);  
                split.append(”Content-Type: “+ file.getContentType()+” ”);  
                outStream.write(split.toString().getBytes());  
                outStream.write(file.getData(), 0, file.getData().length);  
                outStream.write(” ”.getBytes());  
            }  
            byte[] end_data = (”–” + BOUNDARY + “– ”).getBytes();//数据结束标志           
            outStream.write(end_data);  
            outStream.flush();  
            int cah = conn.getResponseCode();  
            if (cah != 200) throw new RuntimeException(”请求url失败”);  
            InputStream is = conn.getInputStream();  
            int ch;  
            StringBuilder b = new StringBuilder();  
            while( (ch = is.read()) != -1 ){  
                b.append((char)ch);  
            }  
            outStream.close();  
            conn.disconnect();  
            return b.toString();  
        } catch (Exception e) {  
            throw new RuntimeException(e);  
        }  

  • 相关阅读:
    Android存储数据方式(转)
    Android实现双进程守护 (转)
    Android DOM、SAX、Pull解析XML(转)
    TCP/IP和Socket的关系(转)
    Socket通信原理和实践
    [转]Android中Intent传递对象的两种方法(Serializable,Parcelable)
    内存堆和栈的区别
    hdu 1754 线段树
    hdu 1166 线段树
    zoj 3686 线段树
  • 原文地址:https://www.cnblogs.com/zyppac/p/3784393.html
Copyright © 2020-2023  润新知