package cn.cctv.net;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
publicclass ImageRequest {
/**
* @param args
*/
publicstaticvoid main(String[] args) throws Exception {
//new一个URL对象
URL url = new URL("http://img.hexun.com/2011-06-21/130726386.jpg");
//打开链接
HttpURLConnection conn = (HttpURLConnection)url.openConnection();
//设置请求方式为"GET"
conn.setRequestMethod("GET");
//超时响应时间为5秒
conn.setConnectTimeout(5 * 1000);
//通过输入流获取图片数据
InputStream inStream = conn.getInputStream();
//得到图片的二进制数据,以二进制封装得到数据,具有通用性
byte[] data = readInputStream(inStream);
//new一个文件对象用来保存图片,默认保存当前工程根目录
File p_w_picpathFile = new File("BeautyGirl.jpg");
//创建输出流
FileOutputStream outStream = new FileOutputStream(p_w_picpathFile);
//写入数据
outStream.write(data);
//关闭输出流
outStream.close();
}
publicstaticbyte[] readInputStream(InputStream inStream) throws Exception{
ByteArrayOutputStream outStream = new ByteArrayOutputStream();
//创建一个Buffer字符串
byte[] buffer = newbyte[1024];
//每次读取的字符串长度,如果为-1,代表全部读取完毕
int len = 0;
//使用一个输入流从buffer里把数据读取出来
while( (len=inStream.read(buffer)) != -1 ){
//用输出流往buffer里写入数据,中间参数代表从哪个位置开始读,len代表读取的长度
outStream.write(buffer, 0, len);
}
//关闭输入流
inStream.close();
//把outStream里的数据写入内存
return outStream.toByteArray();
}
}
*************************************************************************************
在Java中可以使用HttpURLConnection发起这两种请求,了解此类,对于了解soap,和编写servlet的自动测试代码都有很大的帮助。
下面的代码简单描述了如何使用HttpURLConnection发起这两种请求,以及传递参数的方法:
publicstaticfinal String GET_URL ="http://localhost:8080/welcome1";
publicstaticfinal String POST_URL ="http://localhost:8080/welcome1";
publicstaticvoid readContentFromGet() throws IOException ...{
// 拼凑get请求的URL字串,使用URLEncoder.encode对特殊和不可见字符进行编码
String getURL = GET_URL +"?username="
+ URLEncoder.encode("fat man", "utf-8");
URL getUrl =new URL(getURL);
// 根据拼凑的URL,打开连接,URL.openConnection函数会根据URL的类型,
// 返回不同的URLConnection子类的对象,这里URL是一个http,因此实际返回的是HttpURLConnection
HttpURLConnection connection = (HttpURLConnection) getUrl
.openConnection();
// 进行连接,但是实际上get request要在下一句的connection.getInputStream()函数中才会真正发到
// 服务器
connection.connect();
// 取得输入流,并使用Reader读取
BufferedReader reader =new BufferedReader(new InputStreamReader(
connection.getInputStream()));
System.out.println("=============================");
System.out.println("Contents of get request");
System.out.println("=============================");
String lines;
while ((lines = reader.readLine()) !=null) ...{
System.out.println(lines);
}
reader.close();
// 断开连接
connection.disconnect();
System.out.println("=============================");
System.out.println("Contents of get request ends");
System.out.println("=============================");
}
publicstaticvoid readContentFromPost() throws IOException ...{
// Post请求的url,与get不同的是不需要带参数
URL postUrl =new URL(POST_URL);
// 打开连接
HttpURLConnection connection = (HttpURLConnection) postUrl
.openConnection();
// Output to the connection. Default is
// false, set to true because post
// method must write something to the
// connection
// 设置是否向connection输出,因为这个是post请求,参数要放在
// http正文内,因此需要设为true
connection.setDoOutput(true);
// Read from the connection. Default is true.
connection.setDoInput(true);
// Set the post method. Default is GET
connection.setRequestMethod("POST");
// Post cannot use caches
// Post 请求不能使用缓存
connection.setUseCaches(false);
// This method takes effects to
// every instances of this class.
// URLConnection.setFollowRedirects是static函数,作用于所有的URLConnection对象。
// connection.setFollowRedirects(true);
// This methods only
// takes effacts to this
// instance.
// URLConnection.setInstanceFollowRedirects是成员函数,仅作用于当前函数
connection.setInstanceFollowRedirects(true);
// Set the content type to urlencoded,
// because we will write
// some URL-encoded content to the
// connection. Settings above must be set before connect!
// 配置本次连接的Content-type,配置为application/x-www-form-urlencoded的
// 意思是正文是urlencoded编码过的form参数,下面我们可以看到我们对正文内容使用URLEncoder.encode
// 进行编码
connection.setRequestProperty("Content-Type",
"application/x-www-form-urlencoded");
// 连接,从postUrl.openConnection()至此的配置必须要在connect之前完成,
// 要注意的是connection.getOutputStream会隐含的进行connect。
connection.connect();
DataOutputStream out =new DataOutputStream(connection
.getOutputStream());
// The URL-encoded contend
// 正文,正文内容其实跟get的URL中'?'后的参数字符串一致
String content ="firstname="+ URLEncoder.encode("一个大肥人", "utf-8");
// DataOutputStream.writeBytes将字符串中的16位的unicode字符以8位的字符形式写道流里面
out.writeBytes(content);
out.flush();
out.close(); // flush and close
BufferedReader reader =new BufferedReader(new InputStreamReader(
connection.getInputStream()));
String line;
System.out.println("=============================");
System.out.println("Contents of post request");
System.out.println("=============================");
while ((line = reader.readLine()) !=null) ...{
System.out.println(line);
}
System.out.println("=============================");
System.out.println("Contents of post request ends");
System.out.println("=============================");
reader.close();
connection.disconnect();
}
/** *//**
* @param args
*/
publicstaticvoid main(String[] args) ...{
// TODO Auto-generated method stub
try...{
readContentFromGet();
readContentFromPost();
}catch (IOException e) ...{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
上面的readContentFromGet()函数产生了一个get请求,传给servlet一个username参数,值为"fat man"。
readContentFromPost()函数产生了一个post请求,传给servlet一个firstname参数,值为"一个大肥人"。
HttpURLConnection.connect函数,实际上只是建立了一个与服务器的tcp连接,并没有实际发送http请求。无论是post还是get,http请求实际上直到HttpURLConnection.getInputStream()这个函数里面才正式发送出去。
在readContentFromPost()中,顺序是重中之重,对connection对象的一切配置(那一堆set函数)都必须要在connect()函数执行之前完成。而对outputStream的写操作,又必须要在inputStream的读操作之前。这些顺序实际上是由http请求的格式决定的。
http请求实际上由两部分组成,一个是http头,所有关于此次http请求的配置都在http头里面定义,一个是正文content,在connect()函数里面,会根据HttpURLConnection对象的配置值生成http头,因此在调用connect函数之前,就必须把所有的配置准备好。
紧接着http头的是http请求的正文,正文的内容通过outputStream写入,实际上outputStream不是一个网络流,充其量是个字符串流,往里面写入的东西不会立即发送到网络,而是在流关闭后,根据输入的内容生成http正文。
至此,http请求的东西已经准备就绪。在getInputStream()函数调用的时候,就会把准备好的http请求正式发送到服务器了,然后返回一个输入流,用于读取服务器对于此次http请求的返回信息。由于http请求在getInputStream的时候已经发送出去了(包括http头和正文),因此在getInputStream()函数之后对connection对象进行设置(对http头的信息进行修改)或者写入outputStream(对正文进行修改)都是没有意义的了,执行这些操作会导致异常的发生。
************************************************************************************
本节深入学习post请求。
上节说道,post请求的OutputStream实际上不是网络流,而是写入内存,在getInputStream中才真正把写道流里面的内容作为正文与根据之前的配置生成的http request头合并成真正的http request,并在此时才真正向服务器发送。
HttpURLConnection.setChunkedStreamingMode函数可以改变这个模式,设置了ChunkedStreamingMode后,不再等待OutputStream关闭后生成完整的http request一次过发送,而是先发送http request头,正文内容则是网路流的方式实时传送到服务器。实际上是不告诉服务器http正文的长度,这种模式适用于向服务器传送较大的或者是不容易获取长度的数据,如文件。下面以一段代码讲解一下,请与Http学习之使用HttpURLConnection发送post和get请求中的readContentFromPost()函数作对比:
URL postUrl =new URL(POST_URL);
HttpURLConnection connection = (HttpURLConnection) postUrl
.openConnection();
connection.setDoOutput(true);
connection.setDoInput(true);
connection.setRequestMethod("POST");
connection.setUseCaches(false);
connection.setInstanceFollowRedirects(true);
connection.setRequestProperty("Content-Type",
"application/x-www-form-urlencoded");
/**//*
* 与readContentFromPost()最大的不同,设置了块大小为5字节
*/
connection.setChunkedStreamingMode(5);
connection.connect();
/**//*
* 注意,下面的getOutputStream函数工作方式于在readContentFromPost()里面的不同
* 在readContentFromPost()里面该函数仍在准备http request,没有向服务器发送任何数据
* 而在这里由于设置了ChunkedStreamingMode,getOutputStream函数会根据connect之前的配置
* 生成http request头,先发送到服务器。
*/
DataOutputStream out =new DataOutputStream(connection
.getOutputStream());
String content ="firstname="+ URLEncoder.encode("一个大肥人 "+
""+
"asdfasfdasfasdfaasdfasdfasdfdasfs", "utf-8");
out.writeBytes(content);
out.flush();
out.close(); // 到此时服务器已经收到了完整的http request了,而在readContentFromPost()函数里,要等到下一句服务器才能收到http请求。
BufferedReader reader =new BufferedReader(new InputStreamReader(
connection.getInputStream()));
out.flush();
out.close(); // flush and close
String line;
System.out.println("=============================");
System.out.println("Contents of post request");
System.out.println("=============================");
while ((line = reader.readLine()) !=null) ...{
System.out.println(line);
}
System.out.println("=============================");
System.out.println("Contents of post request ends");
System.out.println("=============================");
reader.close();
connection.disconnect();
}
***********************************************************************************
根据GET请求的路径信息和设定的工作目录拼成页面文件的物理路径,去读取页面并返回,如果该路径指向的文件不存在,或者不为html文件,则返回404页面,如果是非法的请求(如POST)则直接干掉连接嘻嘻。
具体代码如下:
*
*/
package org.hello.server;
import java.io.BufferedReader;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.nio.charset.Charset;
import java.util.Date;
/** *//**
* @author pandazxx
*
*/
publicclass SimpleHttpGetServer ...{
/** *//**
* html 文件存放的根路径
*/
privatestaticfinal String ROOT_PATH ="/home/pandazxx/http/";
/** *//**
* 默认的字符编码
*/
privatestaticfinal String DEFAULT_CHARSET = Charset.defaultCharset().name();
/** *//**
* 404页面的存放路径,这个路径一定要存在,因为程序偷懒没再判断了
*/
privatestaticfinal String NOT_FOUND_PAGE_PATH ="/home/pandazxx/http/not_found.html";
/** *//**
* 默认监听端口
*/
privatestaticfinalint DEFAULT_PORT =8080;
private ServerSocket server =null;
publicvoid start(int port) throws IOException ...{
server =new ServerSocket(port);
Socket connection =null;
while (true) ...{
connection = server.accept();
try...{
String filepath = ROOT_PATH
+ readGetRequest(connection.getInputStream());
if (checkFile(filepath)) ...{
OutputStream os = connection.getOutputStream();
writeHTTPResponseHeader(os, getFileSize(filepath));
writeFileContent(os, filepath);
os.close();
}else...{
OutputStream os = connection.getOutputStream();
writeHttpNotFound(os, NOT_FOUND_PAGE_PATH);
}
}catch (HttpServerException e) ...{
// TODO 增加向客户端返回异常信息的代码。
}finally...{
connection.close();
}
}
}
privatevoid writeHttpNotFoundContent(OutputStream os, String filepath) throws IOException ...{
FileInputStream fis =new FileInputStream(filepath);
byte[] buffer =newbyte[2048];
int readCnt =-1;
while ((readCnt = fis.read(buffer)) !=-1) ...{
os.write(buffer, 0, readCnt);
}
}
privatevoid writeHttpNotFound(OutputStream os, String filepath) throws IOException ...{
DataOutputStream dos =new DataOutputStream(os);
String notFoundHeader ="HTTP/1.1 404 Not Found "
+"Date: "+new Date() +""
+"Server: SimpleHttpGetServer "
+"Content-Type: text/html "
+"Content-Length: "+ getFileSize(filepath) +""
+"";
dos.writeBytes(notFoundHeader);
writeHttpNotFoundContent(os, filepath);
}
privatelong getFileSize(String filepath) ...{
File file =new File(filepath);
return file.length();
}
privatevoid writeFileContent(OutputStream os, String filepath) throws IOException ...{
FileInputStream fis =new FileInputStream(filepath);
byte[] buffer =newbyte[2048];
int readCnt =-1;
while ((readCnt = fis.read(buffer)) !=-1) ...{
os.write(buffer, 0, readCnt);
}
}
privatevoid writeHTTPResponseHeader(OutputStream os, long contentSize) throws IOException ...{
DataOutputStream dos =new DataOutputStream(os);
String repHeader ="HTTP/1.1 200 OK "
+"Date: "+new Date() +""
+"Server: SimpleHttpGetServer "
+"Content-Type: text/html;charset="+ DEFAULT_CHARSET +""
+"Content-Length: "+ contentSize +""
+"";
dos.writeBytes(repHeader);
}
privateboolean fileIsHTML(String filepath) ...{
String lowerName = filepath.toLowerCase();
if (lowerName.endsWith(".html") || lowerName.endsWith(".htm")) ...{
returntrue;
}else...{
returnfalse;
}
}
privateboolean checkFile(String filepath) ...{
if (!fileIsHTML(filepath)) ...{
returnfalse;
}
File file =new File(filepath);
if (!file.canRead()) ...{
returnfalse;
}
returntrue;
}
/** *//**
* 偷懒之作,此函数仅仅是判断HTTP Request头是否GET的,获取Get的路径,其余信息一概不管。
* @param is Input stream for the http connection.
* @return The file path of the Get request
* @throws IOException
* @throws HttpServerException
*/
private String readGetRequest(InputStream is) throws IOException,
HttpServerException ...{
String filename =null;
BufferedReader reader =new BufferedReader(new InputStreamReader(is));
String line = reader.readLine();
if (line ==null) ...{
thrownew HttpServerException("Bad get request");
}
filename = parseGetRequest(line);
//is.close();
return filename;
}
private String parseGetRequest(String getRequest)
throws HttpServerException ...{
String[] toks = getRequest.split("/s");
if (toks.length !=3) ...{
thrownew HttpServerException("Bad get request");
}
if (!toks[0].toUpperCase().equals("GET")) ...{
thrownew HttpServerException("Bad get request");
}
if (!toks[2].equals("HTTP/1.1")) ...{
thrownew HttpServerException("Not a HTTP/1.1 request");
}
return toks[1];
}
publicstaticvoid main(String[] args) ...{
SimpleHttpGetServer server =new SimpleHttpGetServer();
try...{
server.start(DEFAULT_PORT);
}catch (IOException e) ...{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}