摘要
  看完本文可以知道,使用java原生编写get/post请求的步骤,进行网络请求时应该注意的地方。
  这里使用java自带的HttpUrlConnection对象进行网络请求,
  请求步骤大致分为五步
  1、创建URL对象
  2、打开URL连接
  3、设置通用属性
  4、建立实际的连接
  5、读取返回的内容
 
正文
 
http网络请求中比较常用的请求方式有get和post请求,其它的如head, put, 以及自定义的OPTIONS等请求方式在这里不做讨论
 
java中使用的请求对象一般是HttpUrlConnection,
其它框架封装的如httpClient,这个apache CommonsHttpClient项目已经不再被维护,随着java版本的更新,这个工具不推荐再使用了,
而其他了,如okhttp,Retrofit2等网络请求框架主要使用于android端的网络请求,
在服务器端进行网络请求使用自己简单封装的http工具类即可。
 
我们请求的地址需要使用URL对象将字符串地址进行封装
即URL connURL = new URL("http://www.xxxx./")
网络请求的步骤
#GET请求
 
    首先要对请求的参数进行中文编码
    java.net.URLEncoder.encode(parameters.get(name),
                                    "UTF-8")
    否则服务器接收到的中文数据会变成乱码,这是人为的错误。
    将编码的参数和url进行拼接形成真正的请求地址
String url = url + "?" + params
    1、创建URL对象
URL connURL = new URL("http://www.xxxx./")
    2、打开url链接获得HttpUrlConnection
HttpURLConnection httpConn = (HttpURLConnection) connURL.openConnection();
     3、设置通用的属性,  ----请求头部信息
       也就是模仿浏览器的请求头部的信息
         Connection、Accept、User-Agent、Cookie 等其它的请求头header信息
httpConn.setRequestProperty("Accept", "*/*");
  Connection:表示是否需要持久链接值为“Keep-Alive   ,持久链接可以减少下载的时间,但服务器必须要返回响应头Content-Length,返回内容的长度。
        http的socket,要经过三次握手通信才能确保连接可用,
 
http连接的两种方式:串行连接,持久连接
持久连接的实现方式有两种:HTTP/1.0+的请求头Connection:keep-alive与HTTP/1.1的持久连接
 
串行连接:1、打开连接
2、请求
3、处理
4、响应
5、关闭连接
重复上面步骤 持久连接:1、打开连接
2、请求
3、处理
4、响应
第二次直接到2操作,
发送请求
持久连接第一次请求后不会关闭连接,下次再发送请求时就不需要再建立连接了

参考资料

Http持久连接与HttpClient连接池

     

其实就是减少了http请求时四次握手(关闭连接1次握手)的时间,
      HTTP 1.1默认进行持久连接
        
    4、 建立连接

httpConn.connect();
    5、 获取响应头部的信息 ,这个步骤可以省略
httpConn.getHeaderField("keyName");
  这里主要是获取到cookie的信息,以便下次访问时不再需要登录,直接通过cookie获取数据
  (当然,随着各大网站的特意防御,只获取到cookie,下次携带cookie可能会被拦截,即不能抓取到想要的数据)
    
如果要对response的响应头部的所有信息进行遍历,则代码如下
// 响应头部获取
Map<String, List<String>> headers = httpConn.getHeaderFields(); // 遍历所有的响应头字段
for (String key : headers.keySet()) { System.out.println(key+": "+httpConn.getHeaderField(key));
}

这是模拟访问www.baidu.com返回的response头部信息

 
6、读取响应的内容,网页html文件,或者特定的返回值数据
// 定义BufferedReader输入流来读取URL的响应,并设置编码方式
    BufferedReader in= null; in = new BufferedReader(new InputStreamReader(httpConn
.getInputStream(), "UTF-8"));//通用编码格式为utf-8
String line;
// 读取返回的内容
while ((line = in.readLine()) != null) {
result += line;
}
httpConn.disconnect();//主动断开httpConn连接
还有需要关闭,字符输入流BufferedReader
这个需要在finally块中进行关闭
finally{
try {
if (in != null) {
in.close();
}
} catch (IOException ex) {
ex.printStackTrace();
}
}

上面是一个get请求的正常流程,当然其中的读取内容的方式有很多种,有些可以直接保存在电脑的文件中,这些方式先不考虑

一个get请求需要注意的几点:

  • 请求参数,中文,其它的转义符号需要进行编码
如下代码,中文和特殊意义字符都会被进行编码,而英语字母则不需要编码
     String url = "http://www.baidu.com";
String ss="你是逗逼吗";
try {
System.out.println(java.net.URLEncoder.encode(url,"UTF-8"));
System.out.println(java.net.URLEncoder.encode(ss,"UTF-8"));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}

输出结果:

  

  • get请求头部最好设置个代理,否则有可能被网站给拦截,拒接访问

,如这里设置为火狐浏览器,模拟浏览器的请求头即可

httpConn.setRequestProperty("User-Agent",
"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1)");
  • 连接断开时候记得一定要把输入输出流给关掉,降低异常出现

  如上面的HttpUrlConnection连接,还有bufferReader对象

一个完整的get请求代码

    public static String sendGet(String url, LinkedHashMap<String, String> parameters) {
String result = "";// 返回的结果
BufferedReader in = null;// 读取响应输入流
StringBuffer sb = new StringBuffer();// 存储参数
String params = "";// 编码之后的参数
try {
// 编码请求参数
if (parameters.size() == 1) {
for (String name : parameters.keySet()) {
sb.append(name).append("=").append(
java.net.URLEncoder.encode(parameters.get(name),
"UTF-8"));
}
params = sb.toString();
} else {
for (String name : parameters.keySet()) {
sb.append(name).append("=").append(
java.net.URLEncoder.encode(parameters.get(name),
"UTF-8")).append("&");
}
String temp_params = sb.toString();
params = temp_params.substring(0, temp_params.length() - 1);
}
String full_url = url + "?" + params;
// 创建URL对象
URL connURL = new URL(full_url);
// 打开URL连接
HttpURLConnection httpConn = (HttpURLConnection) connURL
.openConnection();
// 设置通用属性
httpConn.setRequestProperty("Accept", "*/*");
httpConn.setRequestProperty("Connection", "Keep-Alive");
httpConn.setRequestProperty("User-Agent",
"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1)");
// 建立实际的连接
httpConn.connect();
// 响应头部获取
Map<String, List<String>> headers = httpConn.getHeaderFields(); // 遍历所有的响应头字段
for (String key : headers.keySet()) {
//System.out.println(key+": "+httpConn.getHeaderField(key));
}
// 定义BufferedReader输入流来读取URL的响应,并设置编码方式
in = new BufferedReader(new InputStreamReader(httpConn
.getInputStream(), "GBK"));
String line;
// 读取返回的内容
while ((line = in.readLine()) != null) {
result += line;
}
} catch (Exception e) {
e.printStackTrace();
System.out.println("Http请求方法内部问题");
} finally {
try {
if (in != null) {
in.close();
}
} catch (IOException ex) {
ex.printStackTrace();
}
}
return result;
}
#POST网络请求方式
 post请求跟get请求的方式大体流程一样,区别在于传递请求数据的方式不一样,post可以提交的数据大小没有限制,
  而get请求理论上没有大小限制,由于浏览器限制其长度,所以传递数据大小有限制
post请求不需要设置请求头的user-agent
如果要避免登陆,则需要将登陆时返回的请求头中cookie信息在httpUrlConnect中设置请求投信息即可
connection.setRequestProperty("Cookie", "这里放cookie数据");
connection.setRequestProperty("Content-Type",
"application/x-www-form-urlencoded;charset=utf-8");
content-type主要进行设置服务器解析数据的方式,服务器会通过请求头的值将body里面的数据以键值对key  value方式进行解析,
我们在后台只需要用request.getParameter("keyName")就可以将数据读取出来
其它提交数据的格式:
  • multipart/form-data类型主要是上传文件时用到;
  • application/x-www-form-urlencoded类型主要是提交k-v时用到,当然这种方法也可以将json设置在v中提交json数据;
  • application/json类型主要是传递json数据用到,层次比较深的数据;
public static String sendPost(String curl, String param) {
String result = "";// 返回的结果
BufferedReader in = null;// 读取响应输入流
try {
//创建连接
URL url = new URL(curl);
HttpURLConnection connection = (HttpURLConnection) url
.openConnection();
connection.setDoOutput(true); 是否打开outputStream 相对于程序,即我们向远程服务器写入数据,默认为false,不打开
connection.setDoInput(true); 输入流,获取到返回的响应内容, 默认为true,所以get请求时可以不设置这个连接信息
connection.setRequestMethod("POST"); //发送请求的方式
connection.setUseCaches(false); //不使用缓存
connection.setInstanceFollowRedirects(true); //重定向,一般浏览器才需要
connection.setRequestProperty("Content-Type",
"application/x-www-form-urlencoded;charset=utf-8"); //设置服务器解析数据的方式 connection.connect(); //POST请求
BufferedWriter out = new BufferedWriter(new OutputStreamWriter(connection.getOutputStream(),"UTF-8"));
out.write(param);
out.flush();
out.close(); //读取响应
// 定义BufferedReader输入流来读取URL的响应,并设置编码方式
in = new BufferedReader(new InputStreamReader(connection.getInputStream(), "UTF-8"));
String line;
// 读取返回的内容
while ((line = in.readLine()) != null) {
result += line;
}
} catch (Exception e) {
e.printStackTrace();
System.out.println("Http请求方法内部问题");
} finally {
try {
if (in != null) {
in.close();
}
} catch (IOException ex) {
ex.printStackTrace();
}
}
return result;
}

post请求不同提交数据方式有对应的解析方法,json解析和文件上传下次再写个专题

java学习-get和post请求的更多相关文章

  1. java学习-http中get请求的非ascii参数如何编码解码探讨

    # 背景: 看着别人项目代码看到一个PathUtils工具类, 里面只有一个方法,String  rebuild(String Path),将路径进行URLDecoder.decode解码,避免路径中 ...

  2. Java学习心得之 HttpClient的GET和POST请求

    作者:枫雪庭 出处:http://www.cnblogs.com/FengXueTing-px/ 欢迎转载 Java学习心得之 HttpClient的GET和POST请求 1. 前言2. GET请求3 ...

  3. 【Java Web开发学习】跨域请求

    [Java Web开发学习]跨域请求 ================================================= 1.使用jsonp ===================== ...

  4. Java学习笔记--通过java.net.URLConnection发送HTTP请求

    http://www.cnblogs.com/nick-huang/p/3859353.html 使用Java API发送 get请求或post请求的步骤: 1. 通过统一资源定位器(java.net ...

  5. 0032 Java学习笔记-类加载机制-初步

    JVM虚拟机 Java虚拟机有自己完善的硬件架构(处理器.堆栈.寄存器等)和指令系统 Java虚拟机是一种能运行Java bytecode的虚拟机 JVM并非专属于Java语言,只要生成的编译文件能匹 ...

  6. java 学习写架构必会几大技术点

    java 学习写架构必会几大技术点 关于学习架构,必须会的几点技术 1. java反射技术 2. xml文件处理 3. properties属性文件处理 4. 线程安全机制 5. annocation ...

  7. Java学习-046-日志抓取合并后排序问题解决方案之 --- log4j 二次定制,实现日志输出添加延时10ms

    自3月25至今,已经好久没有写学习日志了,今天在写日志抓取合并的小方法,发现抓取后的日志并米有依据系统执行的日志顺序排序.日志抓取排列逻辑如下: 通过日志标识,从各个日志文件(例如 use.log,e ...

  8. Java学习-036-JavaWeb_005 -- JSP 动作标识 - forward

    JSP 动作主要作用是根据指定的动作进行相应的处理. 一.param 动作 用来给 HTML 文件和 JSP 文件传递参数的,经常和 forward.include.plugin 动作结合使用,语法格 ...

  9. Java学习-034-JavaWeb_003 -- JSP page 指令

    前文对 JSP 的基础知识进行了初步的讲解,此文主要讲述 JSP page 指令. page 指令用于定义页面的多种属性,例如:脚本语言.编码方式.导入的 Java 包等,page 执行的语法如下: ...

随机推荐

  1. (网络流 模板 Edmonds-Karp)Drainage Ditches --POJ --1273

    链接: http://poj.org/problem?id=1273 Drainage Ditches Time Limit: 1000MS   Memory Limit: 10000K Total ...

  2. 挂载镜像SD卡的FAT32文件系统分区到Linux中

    首先使用fdisk –l命令查看sd卡的分区息 可以看到,sd卡有三个分区,第一个分区是FAT32系统,主要存放内核img文件和dtb文件,第二个分区是Linux系统分区,第三个分区格式未知,实际是U ...

  3. Asp.net Core2.0, 基于 claims 实现权限验证

    https://www.cnblogs.com/KimmyLee/p/6430474.html

  4. DBCC--LOG

    DBCC LOGTo retrieve the transaction log for a given database.对应日志文件较大的数据库,慎用该命令Uasge:DBCC LOG(<db ...

  5. rtmp官方标准规范详细解析

    标准规范学习: rtmp消息结构,包括几个部分: 时戳:4  byte,单位毫秒.超过最大值后会翻转. 长度:消息负载的长度. 类型ID:Type Id 一部分ID范围用于rtmp的控制信令.还有一部 ...

  6. NET 集合交集、并集、差集操作

    , , , , , , , }; , , , , , , , , }; // List1:1 3 5 7 9 11 13 15 Console.WriteLine("List1:" ...

  7. [ASP.NET]ASP.NET中常用的26个优化性能方法

    1. 数据库访问性能优化 数据库的连接和关闭 访问数据库资源需要创建连接.打开连接和关闭连接几个操作.这些过程需要多次与数据库交换信息以通过身份验证,比较耗费服务器资源.ASP.NET中提供了连接池( ...

  8. NET Core2.1 WEB老项目迁移

    .NET Core2.1 版本新增功能不在赘述. NET Core2.1更新链接 如果开发需要安装Net Core2.1SDK,及Runtime. .NET Core2.1安装地址. 接下来是WEB ...

  9. Ocelot中文文档入门

    入门 Ocelot仅适用于.NET Core,目前是根据netstandard2.0构建的,如果Ocelot适合您,这个文档可能会有用. .NET Core 2.1 安装NuGet包 使用nuget安 ...

  10. Delphi - 如何执行Windows、OSX、Linux的外部程序?

    毫无疑问,几乎对所有Delphi程序员来说,不用说如何在Windows下如何执行外部程序了!目前Delphi,真的已经很棒了,Delphi有一套和VCL并驾齐驱的图形界面库,叫做"FireM ...