网络请求

一、HTTP和HTTPS的区别

https协议需要到ca申请证书,一般免费证书很少,需要交费。http是超文本传输协议,信息是明文传输,
https 则是具有安全性的ssl加密传输协议,http和https使用的是完全不同的连接方式用的端口也不一样,前者是80,后者是443。
http的连接很简单,是无状态的,HTTPS协议是由SSL+HTTP协议构建的可进行加密传输、身份认证的网络协议 要比http协议安全
 

二、Http头信息

1.  Content-Type(MediaType)

  即是Internet Media Type,互联网媒体类型;也叫做MIME类型,在Http协议消息头中,使用Content-Type来表示具体请求中的媒体类型信息。

[html]  view plain  copy
 
 
  1. 类型格式:type/subtype(;parameter)? type  
  2. 主类型,任意的字符串,如text,如果是*号代表所有;   
  3. subtype 子类型,任意的字符串,如html,如果是*号代表所有;   
  4. parameter 可选,一些参数,如Accept请求头的q参数, Content-Type的 charset参数。   

 例如: Content-Type: text/html;charset:utf-8;

 

 常见的媒体格式类型如下:

  •     text/html : HTML格式
  •     text/plain :纯文本格式      
  •     text/xml :  XML格式
  •     image/gif :gif图片格式    
  •     image/jpeg :jpg图片格式 
  •     image/png:png图片格式

   以application开头的媒体格式类型:

  •    application/xhtml+xml :XHTML格式
  •    application/xml     : XML数据格式
  •    application/atom+xml  :Atom XML聚合格式    
  •    application/json    : JSON数据格式
  •    application/pdf       :pdf格式  
  •    application/msword  : Word文档格式
  •    application/octet-stream : 二进制流数据(如常见的文件下载)
  •    application/x-www-form-urlencoded : <form encType=””>中默认的encType,form表单数据被编码为key/value格式发送到服务器(表单默认的提交数据的格式)

   另外一种常见的媒体格式是上传文件之时使用的:

  •     multipart/form-data : 需要在表单中进行文件上传时,就需要使用该格式

     以上就是我们在日常的开发中,经常会用到的若干content-type的内容格式。

 

 
Content-Type
Content-Type实体报头域用语指明发送给接收者的实体正文的媒体类型。eg:
Content-Type:text/html;charset=ISO-8859-1
Content-Type:text/html;charset=GB2312
Accept
Accept请求报头域用于指定客户端接受哪些类型的信息。eg:Accept:image/gif,表明客户端希望接受GIF图象格式的资源;Accept:text/html,表明客户端希望接受html文本。
 

Requests部分

Header解释示例
Accept指定客户端能够接收的内容类型Accept: text/plain, text/html
Accept-Charset浏览器可以接受的字符编码集。Accept-Charset: iso-8859-5
Accept-Encoding指定浏览器可以支持的web服务器返回内容压缩编码类型。Accept-Encoding: compress, gzip
Accept-Language浏览器可接受的语言Accept-Language: en,zh
Accept-Ranges可以请求网页实体的一个或者多个子范围字段Accept-Ranges: bytes
AuthorizationHTTP授权的授权证书Authorization: Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ==
Cache-Control指定请求和响应遵循的缓存机制Cache-Control: no-cache
Connection表示是否需要持久连接。(HTTP 1.1默认进行持久连接)Connection: close
CookieHTTP请求发送时,会把保存在该请求域名下的所有cookie值一起发送给web服务器。Cookie: $Version=1; Skin=new;
Content-Length请求的内容长度Content-Length: 348
Content-Type请求的与实体对应的MIME信息Content-Type: application/x-www-form-urlencoded
Date请求发送的日期和时间Date: Tue, 15 Nov 2010 08:12:31 GMT
Expect请求的特定的服务器行为Expect: 100-continue
From发出请求的用户的EmailFrom: user@email.com
Host指定请求的服务器的域名和端口号Host: www.zcmhi.com
If-Match只有请求内容与实体相匹配才有效If-Match: “737060cd8c284d8af7ad3082f209582d”
If-Modified-Since如果请求的部分在指定时间之后被修改则请求成功,未被修改则返回304代码If-Modified-Since: Sat, 29 Oct 2010 19:43:31 GMT
If-None-Match如果内容未改变返回304代码,参数为服务器先前发送的Etag,与服务器回应的Etag比较判断是否改变If-None-Match: “737060cd8c284d8af7ad3082f209582d”
If-Range如果实体未改变,服务器发送客户端丢失的部分,否则发送整个实体。参数也为EtagIf-Range: “737060cd8c284d8af7ad3082f209582d”
If-Unmodified-Since只在实体在指定时间之后未被修改才请求成功If-Unmodified-Since: Sat, 29 Oct 2010 19:43:31 GMT
Max-Forwards限制信息通过代理和网关传送的时间Max-Forwards: 10
Pragma用来包含实现特定的指令Pragma: no-cache
Proxy-Authorization连接到代理的授权证书Proxy-Authorization: Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ==
Range只请求实体的一部分,指定范围Range: bytes=500-999
Referer先前网页的地址,当前请求网页紧随其后,即来路Referer: http://www.zcmhi.com/archives/71.html
TE客户端愿意接受的传输编码,并通知服务器接受接受尾加头信息TE: trailers,deflate;q=0.5
Upgrade向服务器指定某种传输协议以便服务器进行转换(如果支持)Upgrade: HTTP/2.0, SHTTP/1.3, IRC/6.9, RTA/x11
User-AgentUser-Agent的内容包含发出请求的用户信息User-Agent: Mozilla/5.0 (Linux; X11)
Via通知中间网关或代理服务器地址,通信协议Via: 1.0 fred, 1.1 nowhere.com (Apache/1.1)
Warning关于消息实体的警告信息Warn: 199 Miscellaneous warning

Responses 部分 

Header解释示例
Accept-Ranges表明服务器是否支持指定范围请求及哪种类型的分段请求Accept-Ranges: bytes
Age从原始服务器到代理缓存形成的估算时间(以秒计,非负)Age: 12
Allow对某网络资源的有效的请求行为,不允许则返回405Allow: GET, HEAD
Cache-Control告诉所有的缓存机制是否可以缓存及哪种类型Cache-Control: no-cache
Content-Encodingweb服务器支持的返回内容压缩编码类型。Content-Encoding: gzip
Content-Language响应体的语言Content-Language: en,zh
Content-Length响应体的长度Content-Length: 348
Content-Location请求资源可替代的备用的另一地址Content-Location: /index.htm
Content-MD5返回资源的MD5校验值Content-MD5: Q2hlY2sgSW50ZWdyaXR5IQ==
Content-Range在整个返回体中本部分的字节位置Content-Range: bytes 21010-47021/47022
Content-Type返回内容的MIME类型Content-Type: text/html; charset=utf-8
Date原始服务器消息发出的时间Date: Tue, 15 Nov 2010 08:12:31 GMT
ETag请求变量的实体标签的当前值ETag: “737060cd8c284d8af7ad3082f209582d”
Expires响应过期的日期和时间Expires: Thu, 01 Dec 2010 16:00:00 GMT
Last-Modified请求资源的最后修改时间Last-Modified: Tue, 15 Nov 2010 12:45:26 GMT
Location用来重定向接收方到非请求URL的位置来完成请求或标识新的资源Location: http://www.zcmhi.com/archives/94.html
Pragma包括实现特定的指令,它可应用到响应链上的任何接收方Pragma: no-cache
Proxy-Authenticate它指出认证方案和可应用到代理的该URL上的参数Proxy-Authenticate: Basic
refresh应用于重定向或一个新的资源被创造,在5秒之后重定向(由网景提出,被大部分浏览器支持)
 

 

Refresh: 5; url=
http://www.zcmhi.com/archives/94.html
Retry-After如果实体暂时不可取,通知客户端在指定时间之后再次尝试Retry-After: 120
Serverweb服务器软件名称Server: Apache/1.3.27 (Unix) (Red-Hat/Linux)
Set-Cookie设置Http CookieSet-Cookie: UserID=JohnDoe; Max-Age=3600; Version=1
Trailer指出头域在分块传输编码的尾部存在Trailer: Max-Forwards
Transfer-Encoding文件传输编码Transfer-Encoding:chunked
Vary告诉下游代理是使用缓存响应还是从原始服务器请求Vary: *
Via告知代理客户端响应是通过哪里发送的Via: 1.0 fred, 1.1 nowhere.com (Apache/1.1)
Warning警告实体可能存在的问题Warning: 199 Miscellaneous warning
WWW-Authenticate表明客户端请求实体应该使用的授权方案WWW-Authenticate: Basic
 
HTTP 请求消息头部实例:

 
    Host:rss.sina.com.cn
    User-Agent:Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.8.1.14) Gecko/20080404 Firefox/2.0.0.14
    Accept:text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5
    Accept-Language:zh-cn,zh;q=0.5
    Accept-Encoding:gzip,deflate
    Accept-Charset:gb2312,utf-8;q=0.7,*;q=0.7
    Keep-Alive:300
    Connection:keep-alive
    Cookie:userId=C5bYpXrimdmsiQmsBPnE1Vn8ZQmdWSm3WRlEB3vRwTnRtW   <-- Cookie 
    If-Modified-Since:Sun, 01 Jun 2008 12:05:30 GMT
    Cache-Control:max-age=0
 
 
 
HTTP 响应消息头部实例:

    Status:OK - 200                <-- 响应状态码,表示 web 服务器处理的结果。 
    Date:Sun, 01 Jun 2008 12:35:47 GMT
    Server:Apache/2.0.61 (Unix)
    Last-Modified:Sun, 01 Jun 2008 12:35:30 GMT
    Accept-Ranges:bytes
    Content-Length:18616
    Cache-Control:max-age=120
    Expires:Sun, 01 Jun 2008 12:37:47 GMT
    Content-Type:application/xml
    Age:2
    X-Cache:HIT from 236-41.D07071951.sina.com.cn <--反向代理服务器使用的 HTTP 头部
    Via:1.0 236-41.D07071951.sina.com.cn:80 (squid/2.6.STABLE13)
    Connection:close
 
 

三、文件(图片)的上传与下载

Content-Disposition: Multipart/form-data是上传文件的一种方式
 
1、采用HttpUrlConnection,用post请求以流的方式上传到服务器中
2、第三方框架:
AsyncHttpClient
HttpClient
                volley    
Retrofit
3、利用Socket
4、利用FTP
 

四、Android网络请求

 
1、传输数据基于Gson(GsonRequest)
当从客户端提交数据时(也就是请求参数),提交的也是json字符串,只不过在我们的代码中的会将请求参数以
对象bean、Map集合、JSONObject等进行封装,将其作为参数发起网络请求
请求参数为普通bean对象:
request.setBody(JSON.toJSONString(requestData).getBytes);
请求参数为Map结合:
Map<String, Object> loginInfo = new HashMap<>();
request.setBody(JSON.toJSONString(loginInfo).getBytes());
请求参数为JSONObject:
JSONObject postData=new JSONObject();
request.setBody(postData.toJSONString().getBytes());
       返回的response的数据类型为一个你想要的对象
 
2、传输数据基于String(StringRequeset)
 
StringRequest stringRequest = new StringRequest("http://www.baidu.com",
                  new Response.Listener<String>() {
                   @Override
                   public void onResponse(String response) {
                    Log.d("TAG", response);
                   }
                  }, new Response.ErrorListener() {
                   @Override
                   public void onErrorResponse(VolleyError error) {
                    Log.e("TAG", error.getMessage(), error);
                   }
                  });
 
 
       返回的response的数据类型为字符串
以上两种只是返回的对象不同而已。
 
    总结:
        服务端和移动端的交互基础都是 数据流,而在两者中传递的数据包括 字符串,图片,文件等,作为最基本的数据字符串(移动端大量数据都由其组成),为了便于逻辑上的处理,我们会选择一种特定规范并已成熟的数据格式(使用其他的格式也可以,只要你能读懂,这让我想到,服务端直接传过来一个对象也不是不可以的,只要做好解析就可以了,对,他们的交互单单 只是流),而开发人员为了便于开发,制定了一种 可视化的按某种规范的数据格式,方便阅读和解析(而市场上以存在大量的解析框架)。
 
3、上传图片
(1)利用HttpUrlConnection
服务器端代码:
 
服务器servlet代码
publicvoid doPost(HttpServletRequest request, HttpServletResponse response) 
           throws ServletException, IOException {  
 
           String temp=request.getSession().getServletContext().getRealPath("/")+"temp"; //临时目录
           System.out.println("temp="+temp);
           String loadpath=request.getSession().getServletContext().getRealPath("/")+"Image"; //上传文件存放目录
           System.out.println("loadpath="+loadpath);
           DiskFileUpload fu =new DiskFileUpload();
           fu.setSizeMax(1*1024*1024); // 设置允许用户上传文件大小,单位:字节
           fu.setSizeThreshold(4096);   // 设置最多只允许在内存中存储的数据,单位:字节
           fu.setRepositoryPath(temp); // 设置一旦文件大小超过getSizeThreshold()的值时数据存放在硬盘的目录
 
           //开始读取上传信息
int index=0;
           List fileItems =null;
 
 
                                try {
                                        fileItems = fu.parseRequest(request);
                                         System.out.println("fileItems="+fileItems);
                                } catch (Exception e) {
                                        e.printStackTrace();
                                }
 
 
           Iterator iter = fileItems.iterator(); // 依次处理每个上传的文件
           while (iter.hasNext())
           {
               FileItem item = (FileItem)iter.next();// 忽略其他不是文件域的所有表单信息
               if (!item.isFormField())
               {
                   String name = item.getName();//获取上传文件名,包括路径
                   name=name.substring(name.lastIndexOf("\\")+1);//从全路径中提取文件名
                   long size = item.getSize();
                   if((name==null||name.equals("")) && size==0)
                         continue;
                   int point = name.indexOf(".");
                   name=(new Date()).getTime()+name.substring(point,name.length())+index;
                   index++;
                   File fNew=new File(loadpath, name);
                   try {
                                        item.write(fNew);
                                } catch (Exception e) {
                                        // TODO Auto-generated catch block
                                        e.printStackTrace();
                                }
 
 
               }
               else//取出不是文件域的所有表单信息
               {
                   String fieldvalue = item.getString();
            //如果包含中文应写为:(转为UTF-8编码)
                   //String fieldvalue = new String(item.getString().getBytes(),"UTF-8");
               }
           }
           String text1="11";
           response.sendRedirect("result.jsp?text1="+ text1);
    }
 
 
android端代码:
 
public class PhotoUpload extends Activity {
    private String newName ="image.jpg";
    private String uploadFile ="/sdcard/image.JPG";
    private String actionUrl ="http://192.168.0.71:8086/HelloWord/myForm";
    private TextView mText1;
    private TextView mText2;
    private Button mButton;
    @Override
      publicvoid onCreate(Bundle savedInstanceState)
      {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.photo_upload);
        mText1 = (TextView) findViewById(R.id.myText2);
        //"文件路径:\n"+
        mText1.setText(uploadFile);
        mText2 = (TextView) findViewById(R.id.myText3);
        //"上传网址:\n"+
        mText2.setText(actionUrl);
        /* 设置mButton的onClick事件处理 */    
        mButton = (Button) findViewById(R.id.myButton);
        mButton.setOnClickListener(new View.OnClickListener()
        {
          publicvoid onClick(View v)
          {
            uploadFile();
          }
        });
      }
      /* 上传文件至Server的方法 */
      privatevoid uploadFile()
      {
        String end ="\r\n";
        String twoHyphens ="--";
        String boundary ="*****";
        try
        {
          URL url =new URL(actionUrl);
          HttpURLConnection con=(HttpURLConnection)url.openConnection();
          /* 允许Input、Output,不使用Cache */
          con.setDoInput(true);
          con.setDoOutput(true);
          con.setUseCaches(false);
          /* 设置传送的method=POST */
          con.setRequestMethod("POST");
          /* setRequestProperty */
          con.setRequestProperty("Connection", "Keep-Alive");
          con.setRequestProperty("Charset", "UTF-8");
          con.setRequestProperty("Content-Type",
                             "multipart/form-data;boundary="+boundary);
          /* 设置DataOutputStream */
          DataOutputStream ds =
            new DataOutputStream(con.getOutputStream());
          ds.writeBytes(twoHyphens + boundary + end);
          ds.writeBytes("Content-Disposition: form-data; "+
                        "name=\"file1\";filename=\""+
                        newName +"\""+ end);
          ds.writeBytes(end);  
          /* 取得文件的FileInputStream */
          FileInputStream fStream =new FileInputStream(uploadFile);
          /* 设置每次写入1024bytes */
          int bufferSize =1024;
          byte[] buffer =newbyte[bufferSize];
          int length =-1;
          /* 从文件读取数据至缓冲区 */
          while((length = fStream.read(buffer)) !=-1)
          {
            /* 将资料写入DataOutputStream中 */
            ds.write(buffer, 0, length);
          }
          ds.writeBytes(end);
          ds.writeBytes(twoHyphens + boundary + twoHyphens + end);
          /* close streams */
          fStream.close();
          ds.flush();
          /* 取得Response内容 */
          InputStream is = con.getInputStream();
          int ch;
          StringBuffer b =new StringBuffer();
          while( ( ch = is.read() ) !=-1 )
          {
            b.append( (char)ch );
          }
          /* 将Response显示于Dialog */
          showDialog("上传成功"+b.toString().trim());
          /* 关闭DataOutputStream */
          ds.close();
        }
        catch(Exception e)
        {
          showDialog("上传失败"+e);
        }
      }
      /* 显示Dialog的method */
      privatevoid showDialog(String mess)
      {
        new AlertDialog.Builder(PhotoUpload.this).setTitle("Message")
         .setMessage(mess)
         .setNegativeButton("确定",new DialogInterface.OnClickListener()
         {
           publicvoid onClick(DialogInterface dialog, int which)
           {          
           }
         })
         .show();
      }
    }
 
 

 

1.HTTP multipart/form-data 上传报文格式

假设接受文件的网页程序位于 http://192.168.24.56/logsys/home/uploadIspeedLog!doDefault.html.假设我们要发送一个图片文件,文件名为“kn.jpg”,

  首先客户端链接 192.168.24.56 后, 应该发送如下http 请求:

  POST/logsys/home/uploadIspeedLog!doDefault.html HTTP/1.1 

  Accept: text/plain, */* 
  Accept-Language: zh-cn 
  Host: 192.168.24.56
  Content-Type:multipart/form-data;boundary=-----------------------------7db372eb000e2
  User-Agent: WinHttpClient 
  Content-Length: 3693
  Connection: Keep-Alive

  -------------------------------7db372eb000e2

  Content-Disposition: form-data; name="file"; filename="kn.jpg"

  Content-Type: image/jpeg

  (此处省略jpeg文件二进制数据...)

  -------------------------------7db372eb000e2--

  此内容必须一字不差,包括最后的回车,红色字体部分就是协议的头。给服务器上传数据时,并非协议头每个字段都得说明,其中,content-type是必须的,它包括一个类似标志性质的名为boundary的标志,它可以是随便输入的字符串。对后面的具体内容也是必须的。它用来分辨一段内容的开始。Content-Length: 3693 ,这里的3693是要上传文件的总长度。绿色字体部分就是需要上传的数据,可以是文本,也可以是图片等。数据内容前面需要有Content-Disposition, Content-Type以及Content-Transfer-Encoding等说明字段。最后的紫色部分就是协议的结尾了。

  注意这一行:

  Content-Type: multipart/form-data; boundary=---------------------------7db372eb000e2  

  根据 rfc1867, multipart/form-data是必须的. 
  ---------------------------7db372eb000e2 是分隔符,分隔多个文件、表单项。其中b372eb000e2 是即时生成的一个数字,用以确保整个分隔符不会在文件或表单项的内容中出现。Form每个部分用分隔符分割,分隔符之前必须加上"--"着两个字符(即--{boundary})才能被http协议认为是Form的分隔符,表示结束的话用在正确的分隔符后面添加"--"表示结束。

  前面的 ---------------------------7d 是 IE 特有的标志,Mozila 为---------------------------71. 


 

五、网络请求和提交数据

1、服务端和移动端的交互基础都是数据流,而在两者中传递的数据包括字符串,图片,文件等,作为最基本的数据字符串(移动端大量数据都由其组成),为了便于逻辑上的处理,我们会选择一种特定规范并已成熟的数据格式(使用其他的格式也可以,只要你能读懂,这让我想到,服务端直接传过来一个对象也不是不可以的,只要做好解析就可以了,对,他们的交互单单只是流),而开发人员为了便于开发,制定了一种可视化的按某种规范的数据格式,方便阅读和解析(而市场上以存在大量的解析框架)。

2、请求参数(以Volley为例)

(1)如果是无参单纯请求网络获取数据,只需输入URL即可,请求方法一般为GET;

            如果有参数也可直接在其后拼接:

             String path = "http://wthrcdn.etouch.cn/weather_mini?city="+URLEncoder.encode(cityname,"utf-8");

(2)如果是需要请求参数的话,这就要看以何种形式提交请求参数了,请求方法一般为POST

第一种:以键值对的形式,方便拼接在URL中(Volley中默认提交请求参数)

byte[] getBody()


 
encodedParams.append(URLEncoder.encode(entry.getKey(), paramsEncoding));
                encodedParams.append('=');
                encodedParams.append(URLEncoder.encode(entry.getValue(), paramsEncoding));
                encodedParams.append('&');
 
 
也就是常见的以键值对的形式并拼接'='和'&',
例如:a=b&c=d&f=e
 
第二种:以Json串的形式,用post的传递
 
//重写getBody()方法
@Override
    public byte[] getBody() {
        return body;
    }
//设置请求体参数
    public void setBody(byte[] body) {
        this.body = body;
        Log.e("请求体是:",new String(body));
    }
//传递请求参数
request.setBody(JSON.toJSONString(loginInfo).getBytes());
 
 
例如: {"password":"666666","username":"zhengrong"}
 
3、请求头
以键值对的形式添加 Cookie(除了添加请求参数,也可以添加请求头,还包括 token),也就是 getHeaders()
 
//重写getHeaders方法
@Override
    @Override
    public Map<String, String> getHeaders() throws AuthFailureError {
        return headers;
    }
//设置请求头
public void addHeader(String key, String value) {
        headers.put(key, value);
    }
//传递请求头
request.addHeader("Cookie", "token=" + Preferences.getAccessToken());
 
 
 
4、上传文件,用 POST
也是重写 byte[] getBody()方法,也就是重组请求参数
 
private void uploadFile()
  {
    String end = "/r/n";
    String Hyphens = "--";
    String boundary = "*****";
    try
    {
      URL url = new URL(actionUrl);
      HttpURLConnection con = (HttpURLConnection) url.openConnection();
      /* 允许Input、Output,不使用Cache */
      con.setDoInput(true);
      con.setDoOutput(true);
      con.setUseCaches(false);
      /* 设定传送的method=POST */
      con.setRequestMethod("POST");
      /* setRequestProperty */
      con.setRequestProperty("Connection", "Keep-Alive");
      con.setRequestProperty("Charset", "UTF-8");
      con.setRequestProperty("Content-Type",
          "multipart/form-data;boundary=" + boundary);
      /* 设定DataOutputStream */
      DataOutputStream ds = new DataOutputStream(con.getOutputStream());
      ds.writeBytes(Hyphens + boundary + end);
      ds.writeBytes("Content-Disposition: form-data; "
          + "name=/"file1/";filename=/"" + newName + "/"" + end);
      ds.writeBytes(end);
      /* 取得文件的FileInputStream */
      FileInputStream fStream = new FileInputStream(uploadFile);
      /* 设定每次写入1024bytes */
      int bufferSize = 1024;
      byte[] buffer = new byte[bufferSize];
      int length = -1;
      /* 从文件读取数据到缓冲区 */
      while ((length = fStream.read(buffer)) != -1)
      {
        /* 将数据写入DataOutputStream中 */
        ds.write(buffer, 0, length);
      }
      ds.writeBytes(end);
      ds.writeBytes(Hyphens + boundary + Hyphens + end);
      fStream.close();
      ds.flush();
      /* 取得Response内容 */
      InputStream is = con.getInputStream();
      int ch;
      StringBuffer b = new StringBuffer();
      while ((ch = is.read()) != -1)
      {
        b.append((char) ch);
      }
      System.out.println("上传成功");
      Toast.makeText(MainActivity.this, "上传成功", Toast.LENGTH_LONG)
          .show();
      ds.close();
    } catch (Exception e)
    {
      System.out.println("上传失败" + e.getMessage());
      Toast.makeText(MainActivity.this, "上传失败" + e.getMessage(),
          Toast.LENGTH_LONG).show();
    }
  }
}
 
 
 

六、网络框架与Socket

关于UrlConnection连接和Socket连接的区别,只知道其中的原理如下:
抽象一点的说,Socket只是一个供上层调用的抽象接口,隐躲了传输层协议的细节。
urlconnection 基于Http协议,Http协议是应用层协议,对传输层Tcp协议进行了封装,是无状态协议,不需要你往考虑线程、同步、状态治理等,内部是通过socket进行连接和收发数据的,不过一般在数据传输完成之后需要封闭socket连接。
直接使用Socket进行网络通讯得考虑线程治理、客户状态监控等,但是不用发送头信息等,更省流量。


 

七、JSON

    字符串,对象(基本元素:键值对),数组(基本元素:字符串、对象、数组)
{K:V}              值是一个字符串(boolean值、数字、null), 键值对的形式,以逗号分开
 
{K:{k,obj}}     值是一个对象
 
{K:["str"]}    值是一个数组,数组中是字符串(以index为key,str为value)
 
{K:[{k:v}]}      值是一个数组,数组中是对象( 以index为key,对象为value
 
        在HiJson中,前面有图案表示普通的键值对;的表示一个普通的对象;有的表示value为数组。
表示数组中的对象
每个元素都是以逗号分隔。
        可以简单记忆为:有三种类型,基本类型、对象、数组;
key一般都是字符串(str),value可以是三种类型的任意一种,当值为数组时,数组中的值可以是三种类型中的任意一种(字符串、对象居多)
 
 

转载于:https://www.cnblogs.com/mesaz/p/11141337.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值