The HTTP protocol
In network communication, data sent and received between two computers is binary stream, which can be understood as a long string containing only zeros and ones. Pure binary streams are meaningless unless they are specified artificially. For example, if A loving passer-by defines A set of rules (rule A) in which 00 represents me, 11 represents you, and 01 represents love, then when computer A sends A binary stream 000111 to computer B, if both A and B follow rule A. Then B knows that A told her I love you. If another disgruntled passer-by, B, defines A set of rules (Rule B) in which 00 is me, 11 is you, and 01 is hate, and if A and B communicate according to Rule B, then B receives the binary stream above and knows that A told her I hate you. The same binary stream, with different rules, can be interpreted with completely different meanings. So for computers on a network to communicate with each other, you need a standard set of rules, or protocols. HTTP is one of the most basic protocols in network communication. Browser through a HTTP request to the server (transmit binary stream to the server, and tell this binary stream USES the protocol is HTTP server), the server receives the request, using the HTTP protocol for binary stream parsing, understand what is the specific need to access the web browser page, find the corresponding page resources, Convert it to a binary stream based on HTTP and send it back to the browser. This completes a request and response for the web page.
HTTP protocol distinguishes between requesters and responders when communicating. Or the client, or the server. The client sends an HTTP request to the server to inform it of the required resources. The server sends an HTTP response to the client to return the required resources to the client. Binary streams that comply with the HTTP protocol are called HTTP packets. HTTP request packets sent from the client to the server. HTTP response packets are sent from the server to the client. Their formats are defined as follows.
HTTP request packet
Let’s look at an actual HTTP request. Open FireFox, press F12 to open developer tools, type www.baidu.com in the address bar, and press Enter. The following HTTP request is displayed.
GET / HTTP / 1.1
Host: www.baidu.com
User-Agent: Mozilla / 5.0 (Windows NT 10.0; Win64; x64; The rv: 84.0) Gecko / 20100101 Firefox 84.0Accept: text/html,application/xhtml+xml,application/xml; Q = 0.9, image/webp, * / *; Q = 0.8Accept-Language: zh-CN,zh; Q = 0.8, useful - TW; Q = 0.7, useful - HK; Q = 0.5, en - US; Q = 0.3, en. Q = 0.2Accept-Encoding: gzip, deflate, br
Connection: keep-alive
Cookie: BAIDUID=BD904AF3599D963DAF4E36903E9D0561;
Copy the code
The first line of a packet is called the request line and consists of the request method, the URL of the request (about the URL), and the HTTP version. The request line above indicates that the request method is GET, the request URL is /, and the protocol is HTTP version 1.1. Each subsequent line appears as a key-value pair, with keys on the left and values on the right, separated by a colon:. These key-value pairs are called request headers, which mainly describe some metadata about the request. For example, Host: www.baidu.com indicates that the requested domain name is www.baidu.com, and user-agent: Mozilla/5.0…. Accept: text/ HTML…. Represents the format of the response body expected to be received by this request, indicating that plain text or HTML data is expected to be received.
The above request message does not contain the request body. Now submit the form data in the JSP to make a POST request and view the request message.
POST /login HTTP / 1.1
Host: localhost:8025
User-Agent: Mozilla / 5.0 (Windows NT 10.0; Win64; x64; The rv: 84.0) Gecko / 20100101 Firefox 84.0Accept-Language: zh-CN,zh; Q = 0.8, useful - TW; Q = 0.7, useful - HK; Q = 0.5, en - US; Q = 0.3, en. Q = 0.2Accept-Encoding: gzip, deflate
Content-Type: application/x-www-form-urlencoded; charset=UTF-8
Content-Length: 26
Connection: keep-alive
user=yogurtzz&password=123
Copy the code
You can see that after the multi-line header, separated by an empty line, there is extra content, which is the request body
Therefore, an HTTP request packet consists of three parts:
- Request line: Contains the request method, the URL of the request, and the HTTP protocol version (separated by Spaces).
- Request header: Contains the metadata of the request (about metadata)
- Request body: Contains the body data of the request (the header is separated from the body by an empty line)
HTTP response packet
Again, look at an actual HTTP response packet
HTTP / 1.1 200 OK
Content-Type: text/html;
Transfer-Encoding: chunked
Date: Wed, 30 Dec 2020 03:20:06 GMT
Keep-Alive: timeout=60
Connection: keep-alive
<html>
<head>
</head>
<body>
success
</body>
</html>
Copy the code
Similar to an HTTP request packet, a response packet consists of three parts
- Response line: contains the HTTP protocol version, HTTP status code, and status code description
- Response header: Contains metadata for the response
- Response body: The body data of the response
After receiving the response packet, the browser automatically renders an HTML page, as follows
HTTP response status code
The status code describes the HTTP response. It consists of three digits. The first digit indicates the type of the response
category | instructions | |
---|---|---|
1xx | Informational (Informational status code) | The received request is being processed |
2xx | Success (Success Status Code) | The request is successfully processed |
3xx | Redirection (Redirection status code) | Additional action is required to complete the request |
4xx | Client Error | The server cannot process the request |
5xx | Server Error | The server failed to process the request |
For the definition of the status code, see RFC2616
Generally speaking, you only need to be familiar with the following common status codes, of which the bold ones are very common.
Status code | Status code Description | instructions |
---|---|---|
200 | OK | The request is successful |
301 | Move Permanently | Permanent redirection |
302 | Found | Temporary redirection |
400 | Bad Request | A syntax error occurred in the request packet. Procedure |
401 | Unauthorized | Requires authentication |
403 | Forbidden | The request was rejected by the server because the IP address was not authorized |
404 | Not Found | The requested resource could not be found on the server |
500 | Internal Server Error | Server internal error |
HTTP request methods
HTTP request methods are used to inform the server of the client’s intent
Request method | intentions | The HTTP version |
---|---|---|
GET | Access to resources | 1.0/1.1 |
POST | Transport entities | 1.0/1.1 |
PUT | Transfer files | 1.0/1.1 |
HEAD | Get the response header | 1.0/1.1 |
DELETE | Delete the file | 1.0/1.1 |
OPTIONS | Ask for supported request methods | 1.1 |
TRACE | Tracking path | 1.1 |
CONNECT | Call for a tunnel | 1.1 |
In Web development, only GET and POST are usually used. In layman’s terms, the request parameters of GET are placed in the URL and usually have no request body (there is no rule that A GET request cannot carry a request body, but it is usually not done). Suppose you have the following form
<form action="login" method="get">User name:<input type="text" name="user"/>Password:<input type="password" name="password">
<input type="submit" value="Login"/>
</form>
Copy the code
Press F12 to open the developer tool. The following message is displayed
GET /login? user=yogurtzz&password=123 HTTP / 1.1
Host: localhost:8080
User-Agent: Mozilla / 5.0 (Windows NT 10.0; Win64; x64; The rv: 84.0) Gecko / 20100101 Firefox 84.0Accept: text/html,application/xhtml+xml,application/xml; Q = 0.9, image/webp, * / *; Q = 0.8Accept-Language: zh-CN,zh; Q = 0.8, useful - TW; Q = 0.7, useful - HK; Q = 0.5, en - US; Q = 0.3, en. Q = 0.2Accept-Encoding: gzip, deflate
Connection: keep-alive
Referer: http://localhost:8080/login.jsp
Cookie: JSESSIONID=FD82B6FF51C8B2B22407AC0892BCBA86
Upgrade-Insecure-Requests: 1
Copy the code
As you can see, the form data submitted by GET is passed by appending the queryString to the URL, so the requested parameters are directly reflected in the browser’s address bar, as shown below. So we would say that a GET request is less secure because it displays the requested data directly.
Let’s change the request method of the form to POST
<form action="login" method="post">User name:<input type="text" name="user"/>Password:<input type="password" name="password">
<input type="submit" value="Login"/>
</form>
Copy the code
Fill in the user name and password again and click login. The obtained request packet is as follows
POST /login HTTP / 1.1
Host: localhost:8080
User-Agent: Mozilla / 5.0 (Windows NT 10.0; Win64; x64; The rv: 84.0) Gecko / 20100101 Firefox 84.0Accept: text/html,application/xhtml+xml,application/xml; Q = 0.9, image/webp, * / *; Q = 0.8Accept-Language: zh-CN,zh; Q = 0.8, useful - TW; Q = 0.7, useful - HK; Q = 0.5, en - US; Q = 0.3, en. Q = 0.2Accept-Encoding: gzip, deflate
Content-Type: application/x-www-form-urlencoded
Content-Length: 26
Origin: http://localhost:8080
Connection: keep-alive
Referer: http://localhost:8080/login.jsp
Cookie: JSESSIONID=A061396E89A92B916AB1FA8298354BAE
Upgrade-Insecure-Requests: 1
user=yogurtzz&password=123
Copy the code
Request parameters are placed in the body of the request and are not displayed in the browser address bar, so we say POST is relatively secure.
HTTP summary
-
Definition: HTTP, Hyper Text Transfer Protocol
-
Role: The specification of client and server data interaction format
-
Message format:
- Request line (response line)
- Request header (response header)
- Request body (response body)
-
The characteristics of
-
simple
The HTTP protocol is simple and not complex
-
flexible
The HTTP protocol can send any type of data (MIME type) as long as the client and server can handle it. This can be specified through the Content-Type header, which can be set in both request and response headers.
-
There is no connection
In earlier VERSIONS of HTTP, short links were used, meaning that a separate HTTP connection was established for each request and response, and the connection was disconnected after completing a request response. The reason for this is that in the early networks, the traffic was not very large and the requests were intermittent and sudden. Maintaining persistent connections would make the connection channel idle for a long time, occupying network resources. Nowadays, network requests are frequent and persistent. Several HTTP requests may be sent consecutively in a single session. If an HTTP connection is established for each request and response, the frequent establishment and closure of connections will cost performance unnecessarily. Prior to HTTP 1.1, it was possible to establish an HTTP persistent Connection by setting the request header Connection: keep-alive. HTTP 1.1 has long connections by default. Clients can send multiple requests over a single HTTP long connection. To disconnect, set Connection: close.
-
stateless
The HTTP protocol has no memory capability. That is, each HTTP request is independent and complete. The latter request has no information about the former request. This design makes the HTTP protocol simple and fast enough, because there is no need to maintain state information and transition between states. Shortcoming is also very obvious, it is a single request all the information must be fully included in a request, make a single request the amount of data that could be very big, and was previously transmitted in the request of the same data, may need to repeat transmission in subsequent requests (if need to use the front in the subsequent requests request information).
In some cases, the HTTP protocol needs to remember some state. For example, in the case of a user login, the user should only log in once and be able to access all the resources of a site, instead of having to log in again each time they request resources. This scenario of maintaining a Session can be solved by using cookies and Session techniques.
例 句 : Ateful and stateless was a life ateful that was built over time.
-
A stateful
A: What did you have for lunch?
B: I had spareribs with potatoes.
A: How does it taste?
B: A sweet one.
-
stateless
A: What did you have for lunch?
B: I had spareribs with potatoes.
A: How does it taste?
B: How does it taste?
Each stateless request is independent and has no context. There is a context when there is a state.
-
-
The appendix
metadata
Metadata is data of data, or a description of data. For example, in the HTTP request packet sent by the client, the request body is name=yogurt. In the request header, content-Length: 11 indicates that the Length of the request body (bytes) is 11. The data transmitted during the request is name=yogurt, and the request header Content-Length: 11 describes the Length of data, that is, metadata. Another example is the Content-Type header, which specifies the format of the requested body data. In front and back end docking development, when the front end requests the back end and transmits parameters, the two most commonly used Content-Types are Application/X-ww-form-urlencoded and Application/JSON, the former is form data, The backend can use Spring’s @RequestParam to receive request parameters, which are JSON data, which the backend can receive using @RequestBody.
URL
Uniform Resource Locator: Uniform Resource Locator. Enter the URL, such as http://www.baidu.com, when accessing a Web page using a browser
Related to URLS, there is also a concept called Uniform Resource Identifier (URI). A URI uses a string to uniquely identify a resource on a network, while a URL in addition describes the location of the resource, that is, where it can be found.
For example, an ID card can be used to uniquely identify a specific person, but you need that person’s address to find them.
An absolute URI has the following format
http://user:[email protected]:8808/dir/index.html?date=2021#chapter1
scheme | User Info (Optional) | host | Port (optional) | path | Query (optional) | Fragment (optional) |
---|---|---|---|---|---|---|
The name of the protocol | Login Information (User + Password) | Server address | The port number | Resource path | Query string | Fragment identifier |
http |
user:pass |
www.xxx.com |
8808 |
/dir/index.html |
date=2021 |
chapter1 |
The server address can be a DNS resolvable domain name such as www.xxx.com, an IP address such as 192.168.0.133 (ipv4), or an ipv6 address enclosed in square brackets such as 0:0:0:0:0:0:0:0:1.
For fragment identifiers, assume the index.html content is as follows
<! DOCTYPEhtml>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<p>This is head</p>
<! -- 10,000 words omitted here -->
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/><br/>
<p id="chapter1"> This is Chapter 1</p>
</body>
</html>
Copy the code
Go to http://localhost:8080/index.html, the result is as follows. To see This is Chapter1, drag the scroll bar down
If add a fragment identifier, visit http://localhost:8080/index.html#chapter1, and web pages will automatically locate the anchor chapter1 position (automatic positioning to an id attribute for the position of the element, chapter1 for < a > element, Setting its name attribute to chapter1 will do the same.)
Great use of anchors when writing Markdown documents: You can use anchors to freely link to other parts of the document. For example, in this Markdown document, I added an anchor request to the HTTP request message
At this point, ALL I need to do is create a link to the request anchor, and when I view the document, I can click on the link and navigate to the location of the anchor. So if I use markdown syntax,
[Click here](# Request) to create a link that can be clicked directly to the location of the anchor. Very convenient. You can try it => click here
HTTP packet capture experience
To better learn HTTP, you can use a packet capture tool to view HTTP packets. Fiddler, a free HTTP analysis and debugging tool, is recommended. Supports interception, editing, and storage of HTTP packets. It is also very simple to operate. Here is a brief introduction to using Fiddler to capture packets
-
Download & Install
Visit https://www.telerik.com/download/fiddler-everywhere to download Fiddler, and install
-
Fire up Fiddler and click the button to start capturing packets
-
Open your browser and visit a random page, such as www.baidu.com
At first, you may wonder why you can’t see HTTP packets in Fiddler. This may be because you use HTTPS to access external websites instead of configuring HTTPS on Fiddler, so you can’t see them. Click the Settings button in the upper right corner to set HTTPS. Remember to reboot Fiddler when you’re done
-
The HTTP packets visiting www.baidu.com are displayed after packet capture is enabled again. Select and click the HTTP request to view. The specific request and response packets are shown in the right Inspectors.
All parts of the packet, such as the packet header and packet body, are automatically split. To view the Raw packet information, click Raw
During development, local services are often debugged and accessed using http://localhost or http://127.0.0.1. Therefore, Fiddler is required to capture HTTP packets requesting local services. However, without any configuration, HTTP requests to the local service cannot be fetched in Fiddler.
The following two solutions are recommended:
-
Replace localhost with ipv4.fiddler
Suppose you want to visit a local service address for http://localhost:8080/demo, replace the localhost ipv4. Fiddler, URL to http://ipv4.fiddler:8080/demo
Fiddler intercepts the domain name ipv4. Fiddler and automatically converts to localhost, so when Fiddler is enabled, accessing ipv4. Fiddler is equivalent to accessing localhost
-
Replace localhost with the computer name
To view the computer name: Open control Panel -> System, you can see the computer name
Or directly open CMD, command prompt, type command hostname, get the computer name
Suppose you want to visit a local service address for http://localhost:8080/demo, replace the localhost DESKTOP computer name – XXXX, is the URL to http://DESKTOP-XXXX:8080/demo
Reference links:
www.racecoder.com/archives/90…
Docs.telerik.com/fiddler/Obs…
The packet capture test for local service access is as follows
-
HTTP headers
There are some very common headers in HTTP packets, and the most important one for interface calls is the Content-Type
Content-Type
This header is used to specify the format of the data in the packet body (it can be used in both request and response packets). There are three types of headers commonly used in front-end interaction.
application/x-www-form-urlencoded
application/json
multipart/form-data
The following uses the backend interface as an example to illustrate the three values of the header
The form data
For the first, when submitting form data, the browser automatically puts Content-Type: Application/X-www-form-urlencoded in the request header, In this case, the data format in the request body is similar to user= Yogurt&Password =123&age=17. If Spring is used on the backend, @requestParam can be used to obtain the data. Without the @requestParam annotation), as shown in the following code example
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class SimpleController {
@PostMapping("/test")
public String test(String user, String password, int age) {
System.out.println("user = " + user + ", password = " + password + ", age = " + age);
return "success"; }}Copy the code
We use POSTMAN to make a request for the above interface
View captured packets
View the application console output
The JSON data
For the second, when submitting JSON data using the POST method, we need to add the content-Type in the request header: In this case, the content in the RequestBody is a json-formatted string. If Spring is used, the @requestbody can automatically wrap the json string into a JAVA object, as shown in the following code example
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class SimpleController {
@PostMapping("/json")
public String json(@RequestBody Person person) {
System.out.println(person.toString());
return "success"; }}Copy the code
The Person class is defined as follows
public class Person {
private String name;
private Integer age;
private String school;
public void setName(String name) {
this.name = name;
}
public void setAge(Integer age) {
this.age = age;
}
public void setSchool(String school) {
this.school = school;
}
@Override
public String toString(a) {
return "Person{" +
"name='" + name + '\' ' +
", age=" + age +
", school='" + school + '\' ' +
'} '; }}Copy the code
We use POSTMAN to make a request to the/JSON interface above
View the captured packets as follows
View the application console output below
The file data
For the third scenario, which is usually used for file uploading, the MultipartFile class can be used to receive uploaded files on the back end of Spring, as shown in the following example code
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;
import java.io.IOException;
@RestController
public class SimpleController {
@PostMapping("/file")
public String file(MultipartFile file) throws IOException {
String fileName = file.getName();
byte[] bytes = file.getBytes();
String fileContent = new String(bytes);
return "fileName: " + fileName + "\nfileContent: "+ fileContent; }}Copy the code
We use POSTMAN to submit a request and upload a file that looks like this
The POSTMAN request is as follows
The packet capture situation is as follows (Because the packet content is long, the packet content is directly copied without taking screenshots)
Request message:
POST http://127.0.0.1:8025/file HTTP / 1.1 the user-agent: PostmanRuntime / 7.26.8 Accept: * / * cache-control: No-cache postman-token: 0c6504d5-b71C-4b28-8e1f-d770596a796f Host: 127.0.0.1:8025 Accept-encoding: gzip, deflate, br Connection: keep-alive Content-Type: multipart/form-data; boundary=--------------------------604727261679510540033606 Content-Length: 261 ----------------------------604727261679510540033606 Content-Disposition: form-data; name="file"; Filename ="test.txt" content-type: text/plain Where am I from? Where am I going? -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 604727261679510540033606Copy the code
Response message:
HTTP / 1.1 200 the content-type: text/plain. charset=UTF-8 Content-Length: 89 Date: Sat, 16 Jan 2021 14:05:13 GMT Keep-Alive: timeout=60 Connection: Keep-alive fileName: test.txt fileContent: Who am I? Where am I from? Where am I going?Copy the code
Having said that, let’s talk about the file download scenario, which involves another HTTP header: Content-Disposition
Content-Disposition
Instructions:
Use Response.getOutputStream () to get an output stream and write an array of bytes to the output stream.
Next, the response header Content-Disposition is set in two ways
Content-Disposition: attachment; filename=xxx.txt
Content-Disposition: inline; filename=xxx.txt
If attachment is set, the browser saves the file as an attachment. The default file name is XXX.txt
Set to inline, the file is opened directly in the browser when the browser supports the file type
In the process of file download, we set attachment, so that the browser will activate the file download (some browsers will pop up the file download dialog, some browsers will directly download), the following example is used
Suppose I have the next image aw.jpeg on my server
I want to provide an interface so that others can download this image
The back-end code is as follows
import org.springframework.http.HttpHeaders;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
import java.nio.file.Path;
import java.nio.file.Paths;
@Controller
public class SimpleController {
@GetMapping("/img")
public void file(HttpServletResponse response) throws IOException {
String imgPath = "E:\\AW.jpeg";
Path path = Paths.get(imgPath);
FileInputStream in = new FileInputStream(path.toFile());
ServletOutputStream outputStream = response.getOutputStream();
byte[] buffer = new byte[1024];
int len;
while((len = in.read(buffer)) ! = -1) {
outputStream.write(buffer, 0, len);
}
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=AW2.jpeg"); }}Copy the code
Using the browser request/IMG interface, the image is displayed directly in the browser instead of the download window as expected
I started to wonder, so I pressed F12 to open the debugger, requested again, and found a strange phenomenon: there was no Content-disposition in the response header
I set the response header in the backend code. In other words, the response header I set is invalid. So I started my “debug” journey
I first added a simple TXT file download interface, as follows
@GetMapping("/txt")
public void txt(HttpServletResponse response) throws IOException {
response.setCharacterEncoding("utf-8"); // Sets the character encoding in the response stream
response.getWriter().write("Who AM I, where have I come from, where am I going?");
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=question.txt");
}
Copy the code
On the browser, the file download dialog box is opened as expected
In the developer toolbar below, check out the request and see that Content-Disposition was successfully added
After a series of searches, the ServletResponse has a method isCommitted(), which you can find in the source code
According to the comment, a “submitted” response has its response status code and response header written.
So I printed isCommitted() of response before adding the response headers to the /img and/TXT interfaces
@GetMapping("/img")
public void file(HttpServletResponse response) throws IOException {
String imgPath = "E:\\AW.jpeg";
Path path = Paths.get(imgPath);
FileInputStream in = new FileInputStream(path.toFile());
ServletOutputStream outputStream = response.getOutputStream();
byte[] buffer = new byte[1024];
int len;
while((len = in.read(buffer)) ! = -1) {
outputStream.write(buffer, 0, len);
}
System.out.println("/ IMG interface requested, isCommitted =" + response.isCommitted());
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=AW2.jpeg");
}
@GetMapping("/txt")
public void txt(HttpServletResponse response) throws IOException {
response.setCharacterEncoding("utf-8"); // Sets the character encoding in the response stream
response.getWriter().write("Who AM I, where have I come from, where am I going?");
System.out.println("/ TXT interface requested, isCommitted =" + response.isCommitted());
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=question.txt");
}
Copy the code
Then request the two interfaces separately with the browser
In /img interface, response has been committed before adding the response header, while in/TXT interface, response has not been committed. Therefore, the /img interface finally adds the response header invalidation, but the/TXT interface is not invalidation. The question then becomes, when will the commit of response be triggered?
After querying the data, it was found that in my above case, it was most likely that the buffer in response was full, triggering the flush of the buffer data, causing the response to change to the submitted state, which invalidated the subsequent setting of the response header (the submitted response, the response status code and the response header could not be changed).
Let’s modify the code to verify
@GetMapping("/img")
public void file(HttpServletResponse response) throws IOException {
String imgPath = "E:\\AW.jpeg";
Path path = Paths.get(imgPath);
FileInputStream in = new FileInputStream(path.toFile());
ServletOutputStream outputStream = response.getOutputStream();
byte[] buffer = new byte[1024];
int len;
int sumLen = 0;
int bufferSize = response.getBufferSize();
System.out.println("bufferSize = " + bufferSize);
while((len = in.read(buffer)) ! = -1) {
outputStream.write(buffer, 0, len);
sumLen += len;
System.out.println("sumLen = " + sumLen);
System.out.println("isCommitted = " + response.isCommitted());
System.out.println("= = = = = = = = = = = = = = = = = = = = = = = = = =");
}
System.out.println("/ IMG interface requested, isCommitted =" + response.isCommitted());
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=AW2.jpeg");
}
Copy the code
There is a lot of information to print. Key information is displayed as follows
As you can see, the response submission is triggered because the buffer for response is full. For more details, see the debug process in the appendix
So, we just need to set up the response header in the first place.
@GetMapping("/img")
public void file(HttpServletResponse response) throws IOException {
response.addHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=AW2.jpeg");
String imgPath = "E:\\AW.jpeg";
Path path = Paths.get(imgPath);
FileInputStream in = new FileInputStream(path.toFile());
ServletOutputStream outputStream = response.getOutputStream();
byte[] buffer = new byte[1024];
int len;
while((len = in.read(buffer)) ! = -1) {
outputStream.write(buffer, 0, len); }}Copy the code
After the modification, a file download dialog box is displayed in the browser as expected
The Debug process
The debug procedure for the response header cannot be changed because the response buffer is full causing the response to be committed
Then we call Response.iscommitted ()