1.HTTP协议:

  • Hyper Text Transfer Protocol:超文本传输协议
  • 基于TCP协议之上的请求/响应协议
  • 目前使用最广泛的高级协议
    * 使用浏览器浏览网页和服务器交互使用的就是HTTP协议
    * 手机应用上绝大多数程序与服务器之间交互数据使用的也是HTTP协议。
  • HTTP协议是一个纯文本协议
  • HTTP是一个请求/响应协议。浏览器发送一个请求,服务器收到以后,然后发送响应。

HTTP 1.0:

  • 每次请求都会创建一个新的HTTP连接,浏览器在请求一个网页之后,往往还是多次请求图片、CSS、JS等其他资源,而创建TCP连接需要一定的时间,所以HTTP1.0传输效率比较低

HTTP 1.1:
在HTTP1.0之上做了改进,多个HTTP请求可以通过一个TCP连接来完成。这样浏览器只需要和服务器建立一次TCP连接,就可以反复进行HTTP的请求/响应。它的效率比HTTP1.0要高

HTTP 2.0:
又对HTTP多了改进。多个HTTP请求也是通过一次TCP连接来完成的,但是浏览器在发送了一次HTTP请求以后,不需要等待响应,就可以立即发送第2个、第3个,而服务器在接收到请求的同时,又把前面已经接收的请求对应的响应发送出去,也就是说HTTP 2.0 请求/响应是可以异步发送的

2.HTTP请求格式

GET POST
请求 GET / HTTP/1.1 GET方法 路径 HTTP协议版本

Host: www.baidu.com 请求地址

User-Agent: Mozilla/5.0 (Linux; Android 9; ALP-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/76.0.3809.89 Mobile Safari/537.36

User-Agent表示浏览器本身,如火狐

Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3

Accept表示能接收什么样的数据

Accept-Language: zh-CN,zh;q=0.9 表示用户的语言
POST: /dataPipeline/uploadData?channel=statistics&version=v1 HTTP/1.1 请求方法 路径 HTTP协议类型

Host: cgicol.amap.com

User-Agent: Dalvik/2.1.0 (Linux; U; Android 9; ALP-AL00 Build/HUAWEIALP-AL00)

Accept: */*

Accept-Language: zh-CN,zh,en

Content-Type: application/x-www-form-urlencoded 数据格式html表单

Content-Length: 264 数据长度264

channel=statistics&version=v1... Body具体的数据
响应 GET响应:HTTP/1.1 302 Found HTTP协议 响应

Content-Type:text/html;charset=utf-8 响应的内容是网页,编码是utf-8

Content-Length:0 响应的长度

响应体和header以空行分隔
HTTP/1.1 200

Content-Type: text/html;charset=ISO-8859-1 指定具体类型

Content-Length: 32086 大小

响应体

HTTP请求分为Header和Body2个部分,Header是必须的。Get请求只有Header没有Body,而Post请求既有Header,又有Body。Header和Body之间用一个空行分隔。
如果浏览器发送的是POST请求,那么浏览器还需要告诉服务器发送的数据格式Content-Type和数据的大小Content-Length,如application/x-www-form-urlencoded表示一个html表单,大小为264个字节。

HTTP响应也分为Header和Body2个部分,Header是必须的,第1行指出HTTP响应码。2XX正确返回,3XX重定向,4XX客户端错误,5XX服务器错误。
Header还可以包含各种属性,例如Content-Type指出Body的类型,Content-Length指出Body大小。
响应体通过一个空行与Header分隔,响应的数据可以是任意的文本或者二进制格式作为body。

HTTP Server

  • 处理HTTP请求,发送HTTP响应
  • 服务器是通过Java EE Servlet API来定义的

HTTP Client

  • 发送HTTP请求,接收HTTP响应
  • java.net.HttpURLConnection处理http的客户端

GET请求示例:

    URL url = new URL("http://www.exampe.com"); //获取URL对象
HttpURLConnection conn = (HttpURLConnection) url.openConnection(); //通过url对象的openConnection()获取HttpURLConnection对象
int code = conn.getResponseCode(); //获取响应码
try(InputStream input = conn.getInputSteam()){ //读取响应数据
//读取响应数据
}
conn.disconnect(); //断开连接

POST请求示例:

    URL url = new URL("http://www.example.com/login");
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("POST"); //默认方法是get
conn.setDoOutput(true); //需要发送请求的数据
//指定数据的Content-Type和Content-Length
byte[] postData = "loginName=test&password=123456".getBytes("UTF-8");
conn.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
conn.setRequestProperty("Content-Length",String.valueOf(postData.length));
try(OutputStream output = conn.getOutputStream()){ //写入请求的数据
output.write(postData);
}
int code = conn.getResponseCode();
try(inputStream input = conn.getInputStream()){
//读取响应数据
}
conn.disconnect();

示例

package com.csj2018;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; class Response{
final int code;
final byte[] data;
public Response(int code,byte[] data){
this.code = code;
this.data = data;
}
@Override
public String toString(){
StringBuilder sb = new StringBuilder(1024);
sb.append("HTTP状态码").append(code).append("\n");
String s = new String(data,StandardCharsets.UTF_8);
if(s.length() > 100){
sb.append(s.substring(0,100)).append("\n...");
}else{
sb.append(s);
}
return sb.toString(); //返回状态码和前100个字节的内容,多于100,以...代表剩余内容
}
}
public class HttpClient {
public static void main(String[] args) throws Exception{
Response resp = get("http://www.douban.com");
System.out.println(resp);
Map<String,String> postMap = new HashMap<>();
postMap.put("form_email","test");
postMap.put("form_password","password");
Response postResp = post("https://www.douban.com/accounts/login","application/x-www-form-urlencoded",toFormData(postMap));
System.out.println(postResp);
}
static Response get(String theUrl){
System.err.println("GET:"+theUrl);
HttpURLConnection conn = null;
try{
URL url = new URL(theUrl); //获取URL对象
conn = (HttpURLConnection) url.openConnection(); //建立连接
ByteArrayOutputStream responseBuffer = new ByteArrayOutputStream(); //获取一个ByteArrayOutputStream对象用于存放读取的内容
try(InputStream input = conn.getInputStream()){//获取连接获取的内容
byte[] buffer = new byte[1024];
for(;;){
int n = input.read(buffer);
System.err.println("每次读取的大小:"+n);
if(n== (-1)){
break;
}
responseBuffer.write(buffer,0,n);//将内容写入到responseBuffer
System.out.println("每次写入后responseBuffer的大小"+responseBuffer.size());
}
}
return new Response(conn.getResponseCode(),responseBuffer.toByteArray());//返回Response对象
}catch (IOException e){
throw new RuntimeException(e);
}finally {
if(conn != null){
conn.disconnect();
}
}
}
static Response post(String theUrl,String contentType,String contentData){
System.err.println("POST:"+theUrl);
HttpURLConnection conn = null;
try{
URL url = new URL(theUrl);
conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("POST"); //请求方法默认get,因此设为post
conn.setDoOutput(true);
//添加请求属性,Content-Type, Content-Length
conn.setRequestProperty("Content-Type",contentType);
conn.setRequestProperty("Content_Length",String.valueOf(contentData.length()));
//设置body
byte[] postData = contentData.getBytes(StandardCharsets.UTF_8); //
try(OutputStream output = conn.getOutputStream()){
output.write(postData);
}
ByteArrayOutputStream responseBuffer = new ByteArrayOutputStream();
try(InputStream input = conn.getInputStream()){
byte[] buffer = new byte[1024];
for(;;){
int n = input.read(buffer);
if(n==(-1)){
break;
}
responseBuffer.write(buffer,0,n);
}
}
return new Response(conn.getResponseCode(),responseBuffer.toByteArray());
}catch (IOException e){
throw new RuntimeException(e);
}finally {
if(conn != null){
conn.disconnect();
}
}
}
static String toFormData(Map<String,String> map) throws IOException{
List<String> list = new ArrayList<>(map.size());
for(String key:map.keySet()){
list.add(key + "=" + URLEncoder.encode(map.get(key),"UTF-8"));
}
return String.join("&",list); //key不编码,value编码,多个参数以&连接
}
}

3.总结:

  • HTTP协议是一个基于TCP的请求/响应协议
  • 广泛用于浏览器、手机app与服务器的数据交互
  • Java提供了HttpURLConnection实现HTTP客户端

廖雪峰Java13网络编程-3其他-1HTTP编程的更多相关文章

  1. 廖雪峰Java13网络编程-1Socket编程-2TCP编程

    1. Socket 在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据 ...

  2. 廖雪峰Java13网络编程-3其他-2RMI远程调用

    1.RMI远程调用: Remote Method Invocation 目的:把一个接口方法暴露给远程 示例: 定义一个接口Clock,它有一个方法能够获取当前的时间,并编写一个实现类,来实现这个接口 ...

  3. 廖雪峰Java13网络编程-1Socket编程-5UDP编程

    1. UDP编程: 不需要建立连接 可以直接发送和接收数据 1.1 客户端 DatagramSocket sock = new DatagramSocket(){} sock.connect(addr ...

  4. 廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程

    TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通 ...

  5. 廖雪峰Java13网络编程-1Socket编程-1网络编程概念

    1.计算机网络 1.1 什么是计算机网络? 两台或更多计算机组成的网络 同一网络内的任意2台计算机都可以直接通信 所有计算机必须遵循同一种网络协议 1.2 什么是互联网 互联网是网络的网络 互联网采用 ...

  6. 廖雪峰Java13网络编程-2Email编程-2接收Email

    1接收Email协议类型 接收Email:收件人通过MUA软件把邮件从MDA抓取到本地计算机的过程. 1.1 POP3 从MUA到MDA使用最广泛的是协议是POP3 Post Office Proto ...

  7. 廖雪峰Java13网络编程-2Email编程-1发送email

    1.邮件发送 1.1传统邮件发送: 传统的邮件是通过邮局投递,从一个邮局到另一个邮局,最终到达用户的邮箱. 1.2电子邮件发送: 与传统邮件类似,它是从用户电脑的邮件软件(如outlook)发送到邮件 ...

  8. 廖雪峰Java1-3流程控制-9break、continue

    break跳出循环 int sum=0; for(int i=1; ;i++) { sum =sum + i; if(i == 100) { break; } } System.out.println ...

  9. 廖雪峰Java1-3流程控制-6 do-while循环

    do-while循环 do-while先执行循环,再判断条件. 条件满足时继续循环:条件不满足时退出:至少循环1次 int sum =0; int n = 1; do{ sum = sum + n; ...

随机推荐

  1. flutter 修饰盒子

    decoration: BoxDecoration( borderRadius: BorderRadius.circular(), //圆角 gradient: RadialGradient( col ...

  2. EasyMock添加行为

    EasyMock使用expect()方法或expectLassCall()方法添加一个功能,一个模拟对象.请看下面的代码片段. 1 //add the behavior of calc service ...

  3. spring boot资源文件配置读取

    一般业务配置,尽量新建自己的配置文件,来读取,而不是配置在application.properties或application-*.properties或yml/yaml配置中. applicatio ...

  4. EFCore学习记录笔记

    1:连接slqlocaldb数据库 (1)在CMD下可以输入sqllocaldb info 查看本机安装的所有的localdb实例 (2)数据库连接字符串为:“Server=(localdb)\\MS ...

  5. springcloud(十六):服务网关 zuul 快速入门

    服务网关是微服务架构中一个不可或缺的部分.通过服务网关统一向外系统提供REST API的过程中,除了具备服务路由.均衡负载功能之外,它还具备了权限控制等功能.Spring Cloud Netflix中 ...

  6. 科普帖:深度学习中GPU和显存分析

    知乎的一篇文章: https://zhuanlan.zhihu.com/p/31558973 关于如何使用nvidia-smi查看显存与GPU使用情况,参考如下链接: https://blog.csd ...

  7. swt java 内嵌ActiveX控件

    这里用的是SWT/JFace开发application中SWT自带的org.eclipse.swt.ole.win32 包可以支持内嵌OLE和ActiveX. 具体用法如下: //创建一个OleFra ...

  8. Apache2.2+tomcat7 负载均衡配置

    思路及步骤:第一步配置tomcat,第二步配置apache 服务器,第三步添加项目到tomcat中并测试 第一步配置tomcat 1,打开 第一个tomcat,conf文件夹下的server.xml ...

  9. 【学术篇】SDOI2011 计算器

    好一道三合一...(然而被我做成了四合一) 其实1 2 3是独立的OvO 然后就可以逐个分析了... 1 快速幂..就不说了..(我省选的时候有这么水的20pts部分分么←_← 2 两种做法(写在标题 ...

  10. LightOJ 1151 Snakes and Ladders 期望dp+高斯消元

    题目传送门 题目大意:10*10的地图,不过可以直接看成1*100的,从1出发,要到达100,每次走的步数用一个大小为6的骰子决定.地图上有很多个通道 A可以直接到B,不过A和B大小不确定   而且 ...