Volley的框架解读准备一HttpURLConnection

本文详细介绍了HTTP请求的全过程,包括设置连接参数、发送请求、获取响应等步骤,并对比了setRequestProperty与addRequestProperty方法的不同之处。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

请求响应流程

请求响应流程

设置连接参数的方法

  • setAllowUserInteraction
  • setDoInput
  • setDoOutput
  • setIfModifiedSince
  • setUseCaches
  • setDefaultAllowUserInteraction
  • setDefaultUseCaches

设置请求头或响应头

HTTP请求允许一个key带多个用逗号分开的values,但是HttpURLConnection只提供了单个操作的方法:
  • setRequestProperty(key,value)
  • addRequestProperty(key,value)
setRequestProperty和addRequestProperty的区别就是,setRequestProperty会覆盖已经存在的key的所有values,有清零重新赋值的作用。而addRequestProperty则是在原来key的基础上继续添加其他value。

发送URL请求

建立实际连接之后,就是发送请求,把请求参数传到服务器,这就需要使用outputStream把请求参数传给服务器:
  • getOutputStream

获取响应

请求发送成功之后,即可获取响应的状态码,如果成功既可以读取响应中的数据,获取这些数据的方法包括:
  • getContent
  • getHeaderField
  • getInputStream
对于大部分请求来说,getInputStream和getContent是用的最多的。 相应的信息头用以下方法获取:
  • getContentEncoding
  • getContentLength
  • getContentType
  • getDate
  • getExpiration
  • getLastModifed

HttpURLConnection

任何网络连接都需要经过socket才能连接,HttpURLConnection不需要设置socket,所以,HttpURLConnection并不是底层的连接,而是在底层连接上的一个请求。 这就是为什么HttpURLConneciton只是一个抽象类,自身不能被实例化的原因。HttpURLConnection只能通过URL.openConnection()方法创建具体的实例。 虽然底层的网络连接可以被多个HttpURLConnection实例共享,但每一个HttpURLConnection实例只能发送一个请求。请求结束之后,应该调用HttpURLConnection实例的InputStream或OutputStream的close()方法以释放请求的网络资源,不过这种方式对于持久化连接没用。对于持久化连接,得用disconnect()方法关闭底层连接的socket。

创建HttpURLConnection

URL url = new URL("http://localhost:8080/xxx.do");

URLConnection rulConnection = url.openConnection();// 此处的urlConnection对象实际上是根据URL的
// 请求协议(此处是http)生成的URLConnection类
// 的子类HttpURLConnection,故此处最好将其转化
// 为HttpURLConnection类型的对象,以便用到
// HttpURLConnection更多的API.如下:

HttpURLConnection httpUrlConnection = (HttpURLConnection) rulConnection;

设置HttpURLConnection参数


// 设定请求的方法为"POST",默认是
GEThttpUrlConnection.setRequestMethod("POST");
// 设置是否向httpUrlConnection输出,因为这个是post请求,参数要放在
// http正文内,因此需要设为true, 默认情况下是false;
httpUrlConnection.setDoOutput(true);

// 设置是否从httpUrlConnection读入,默认情况下是true;
httpUrlConnection.setDoInput(true);

// Post 请求不能使用缓存
httpUrlConnection.setUseCaches(false);

// 设定传送的内容类型是可序列化的java对象
// (如果不设此项,在传送序列化对象时,当WEB服务默认的不是这种类型时可能抛java.io.EOFException)
httpUrlConnection.setRequestProperty("Content-type", "application/x-java-serialized-object");

// 连接,从上述url.openConnection()至此的配置必须要在connect之前完成,
httpUrlConnection.connect();

URLConnection建立连接


// 此处getOutputStream会隐含的进行connect(即:如同调用上面的connect()方法,
    // 所以在开发中不调用上述的connect()也可以)。
    OutputStream outStrm = httpUrlConnection.getOutputStream();
getInputStream()也是同理。

HttpURLConnection发送请求


// 现在通过输出流对象构建对象输出流对象,以实现输出可序列化的对象。
ObjectOutputStream objOutputStrm = new ObjectOutputStream(outStrm);
// 向对象输出流写出数据,这些数据将存到内存缓冲区中
objOutputStrm.writeObject(new String("我是测试数据"));

// 刷新对象输出流,将任何字节都写入潜在的流中(些处为ObjectOutputStream)
objOutputStm.flush();

// 关闭流对象。此时,不能再向对象输出流写入任何数据,先前写入的数据存在于内存缓冲区中,
// 在调用下边的getInputStream()函数时才把准备好的http请求正式发送到服务器
objOutputStm.close();

HttpURLConneciton获取响应

 // 调用HttpURLConnection连接对象的getInputStream()函数,  

InputStream inStrm = httpConn.getInputStream(); 

设置POST参数

OutputStream os = httpConn.getOutputStream();
String param = new String();
param = “CorpID=” + CorpID +
“&LoginName=” + LoginName+
“&send_no=” + phoneNumber +
“&msg=” + java.net.URLEncoder.encode(msg,”GBK”); ;
os.write(param.getBytes());

超时设置,防止 网络异常的情况下,可能会导致程序僵死而不继续往下执行

System.setProperty("sun.net.client.defaultConnectTimeout", "30000");  

System.setProperty("sun.net.client.defaultReadTimeout", "30000");  



其中: sun.net.client.defaultConnectTimeout:连接主机的超时时间(单位:毫秒)  

sun.net.client.defaultReadTimeout:从主机读取数据的超时时间(单位:毫秒)  



JDK 1.5以前的版本,只能通过设置这两个系统属性来控制网络超时。在1.5中,还可以使用HttpURLConnection的父类URLConnection的以下两个方法:  

setConnectTimeout:设置连接主机超时(单位:毫秒)  

setReadTimeout:设置从主机读取数据超时(单位:毫秒)  



例如:  

HttpURLConnection urlCon = (HttpURLConnection)url.openConnection();  

urlCon.setConnectTimeout(30000);  

urlCon.setReadTimeout(30000);  

追加内容之:android封装HttpURLConnection用Post实现用户登录
public void loginByPost(String userName, String userPass) {  

        try {  

            // 请求的地址  
            String spec = "http://172.16.237.200:8080/video/login.do";  
            // 根据地址创建URL对象  
            URL url = new URL(spec);  
            // 根据URL对象打开链接  
            HttpURLConnection urlConnection = (HttpURLConnection) url  
                    .openConnection();  
            // 设置请求的方式  
            urlConnection.setRequestMethod("POST");  
            // 设置请求的超时时间  
            urlConnection.setReadTimeout(5000);  
            urlConnection.setConnectTimeout(5000);  
            // 传递的数据  
            String data = "username=" + URLEncoder.encode(userName, "UTF-8")  
                    + "&userpass=" + URLEncoder.encode(userPass, "UTF-8");  
            // 设置请求的头  
            urlConnection.setRequestProperty("Connection", "keep-alive");  
            // 设置请求的头  
            urlConnection.setRequestProperty("Content-Type",  
                    "application/x-www-form-urlencoded");  
            // 设置请求的头  
            urlConnection.setRequestProperty("Content-Length",  
                    String.valueOf(data.getBytes().length));  
            // 设置请求的头  
            urlConnection  
                    .setRequestProperty("User-Agent",  
                            "Mozilla/5.0 (Windows NT 6.3; WOW64; rv:27.0) Gecko/20100101 Firefox/27.0");  

            urlConnection.setDoOutput(true); // 发送POST请求必须设置允许输出  
            urlConnection.setDoInput(true); // 发送POST请求必须设置允许输入  
                                            //setDoInput的默认值就是true  
            //获取输出流  
            OutputStream os = urlConnection.getOutputStream();  
            os.write(data.getBytes());  
            os.flush();  
            if (urlConnection.getResponseCode() == 200) {  
                // 获取响应的输入流对象  
                InputStream is = urlConnection.getInputStream();  
                // 创建字节输出流对象  
                ByteArrayOutputStream baos = new ByteArrayOutputStream();  
                // 定义读取的长度  
                int len = 0;  
                // 定义缓冲区  
                byte buffer[] = new byte[1024];  
                // 按照缓冲区的大小,循环读取  
                while ((len = is.read(buffer)) != -1) {  
                    // 根据读取的长度写入到os对象中  
                    baos.write(buffer, 0, len);  
                }  
                // 释放资源  
                is.close();  
                baos.close();  
                // 返回字符串  
                final String result = new String(baos.toByteArray());  

                // 通过runOnUiThread方法进行修改主线程的控件内容  
                LoginActivity.this.runOnUiThread(new Runnable() {  
                    @Override  
                    public void run() {  
                        // 在这里把返回的数据写在控件上 会出现什么情况尼  
                        tv_result.setText(result);  
                    }  
                });  

            } else {  
                System.out.println("链接失败.........");  
            }  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值