import java.net. * ;
import java.io. * ;
import java.util.Properties;
import java.util.Enumeration;
/**
Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的
是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜。本文根据HTTP
协议规范,用Java.net.Socket类实现一个HTTP协议客户端程序.
<pre>
1.Socket类:
了解TCP/IP协议集通信的读者知道,协议间的通信是通过Socket完成的。在
Java.net包中,Socket类就是对Socket的具体实现。它通过连接到主机后,返回一个
I/O流,实现协议间的信息交换。
2 . HTTP协议
HTTP协议同其它TCP/IP协议集中的协议一样,是遵循客户/服务器模型工作的。客
户端发往服务端的信息格式如下:
------------------------------
请求方法 URL HTTP协议的版本号
提交的元信息
**空行**
实体
------------------------------
请求方法是对这次连接工作的说明,目前HTTP协议已经发展到1.1版,它包括GET、
HEAD、POST、DELETE、OPTIONS、TRACE、PUT七种。元信息是关于当前请求的信息。通
过分析元信息,可以检查实体数据是否完整,接收过程是否出错,类型是否匹配等。元
信息的引入使HTTP协议通信更加稳妥可靠。实体是请求的具体内容。
将上述报文发往Web服务器,如果成功,应答格式如下:
--------------------------------
HTTP协议的版本号 应答状态码 应答状态码说明
接收的元信息
**空行**
实体
--------------------------------
以上报文发向客户端,并且接收成功,彼此间关闭连接,完成一次握手。
下面用最常用的GET方法,来说明具体的报文应用
----------------------------------
GET http://www.youhost.com HTTP/1.0
accept: www/source; text/html; image/gif; image/jpeg; */*
User_Agent: myAgent
**空行**
-----------------------------------
这个报文是向www.youhost.com主机请求一个缺省HTML文档。客户端HTTP协议版本
号是1.0版,元信息包括可接收的文件格式,用户代理,每一段之间用回车换行符分
隔,最后以一个空行结束。发向服务器后,如果执行过程正常,服务器返回以下代码:
------------------------------------
HTTP/1.1 200 OK
Date: Tue, 14 Sep 1999 02:19:57 GMT
Server: Apache/1.2.6
Connection: close
Content-Type: text/html
**空行**
<html><head></head><body></body></html>
------------------------------------
HTTP/1.1表示这个HTTP服务器是1.1版,200是服务器对客户请求的应答状态码,OK
是对应答状态码的解释,之后是这个文档的元信息和文档正文。(相关应答状态码和元
信息的解释请参阅Inetrnet标准草案:RFC2616)。
注: 程序中只实现GET、HEAD、POST三种方法。其他几种因不常使用,暂且忽略。
</pre>
*/
public class Http{
protected Socket client;
protected BufferedOutputStream sender;
protected BufferedInputStream receiver;
protected ByteArrayInputStream byteStream;
protected URL target;
private int responseCode = - 1 ;
private String responseMessage = "" ;
private String serverVersion = "" ;
private Properties header = new Properties();
public Http(){}
public Http(String url){
GET(url);
}
/* GET方法根据URL,会请求文件、数据库查询结果、程序运行结果等多种内容 */
public void GET(String url){
try {
checkHTTP(url);
openServer(target.getHost(),target.getPort());
String cmd = " GET " + getURLFormat(target) + " HTTP/1.0\r\n " +
getBaseHeads() + " \r\n " ;
sendMessage(cmd);
receiveMessage();
}
catch (ProtocolException p){
p.printStackTrace();
return ;
}
catch (UnknownHostException e){
e.printStackTrace();
return ;
}
catch (IOException i){
i.printStackTrace();
return ;
}
}
/*
* HEAD方法只请求URL的元信息,不包括URL本身。若怀疑本机和服务器上的
* 文件相同,用这个方法检查最快捷有效。
*/
public void HEAD(String url){
try {
checkHTTP(url);
openServer(target.getHost(),target.getPort());
String cmd = " HEAD " + getURLFormat(target) + " HTTP/1.0\r\n " +
getBaseHeads() + " \r\n " ;
sendMessage(cmd);
receiveMessage();
}
catch (ProtocolException p){
p.printStackTrace();
return ;
}
catch (UnknownHostException e){
e.printStackTrace();
return ;
}
catch (IOException i){
i.printStackTrace();
return ;
}
}
/*
* POST方法是向服务器传送数据,以便服务器做出相应的处理。例如网页上常用的
* 提交表格。
*/
public void POST(String url,String content){
try {
checkHTTP(url);
openServer(target.getHost(),target.getPort());
String cmd = " POST " + getURLFormat(target) + " HTTP/1.0\r\n " +
getBaseHeads();
cmd += " Content-type: application/x-www-form-urlencoded\r\n " ;
cmd += " Content-length: " + content.length() + " \r\n\r\n " ;
cmd += content + " \r\n " ;
sendMessage(cmd);
receiveMessage();
}
catch (ProtocolException p){
p.printStackTrace();
return ;
}
catch (UnknownHostException e){
e.printStackTrace();
return ;
}
catch (IOException i){
i.printStackTrace();
return ;
}
}
protected void checkHTTP(String url) throws ProtocolException{
try {
URL target = new URL(url);
if (target == null ||
! target.getProtocol().toUpperCase().equals( " HTTP " )){
throw new ProtocolException( " 这不是HTTP协议 " );
}
this .target = target;
}
catch (MalformedURLException m){
throw new ProtocolException( " 协议格式错误 " );
}
}
/*
* 与Web服务器连接。若找不到Web服务器,InetAddress会引发UnknownHostException
* 异常。若Socket连接失败,会引发IOException异常。
*/
protected void openServer(String host, int port) throws UnknownHostException,
IOException{
header.clear();
responseMessage = "" ;
responseCode = - 1 ;
if (client != null ){
closeServer();
}
if (byteStream != null ){
byteStream.close();
byteStream = null ;
}
InetAddress address = InetAddress.getByName(host);
client = new Socket(address,port == - 1 ? 80 : port);
client.setSoTimeout( 5000 );
sender = new BufferedOutputStream(client.getOutputStream());
receiver = new BufferedInputStream(client.getInputStream());
}
/* 关闭与Web服务器的连接 */
protected void closeServer() throws IOException{
if (client == null ){
return ;
}
try {
client.close();
sender.close();
receiver.close();
}
catch (IOException i){
throw i;
}
client = null ;
sender = null ;
receiver = null ;
}
protected String getURLFormat(URL target){
String spec = " http:// " + target.getHost();
if (target.getPort() != - 1 ){
spec += " : " + target.getPort();
}
return spec += target.getFile();
}
/* 向Web服务器传送数据 */
protected void sendMessage(String data) throws IOException{
sender.write(data.getBytes(), 0 ,data.length());
sender.flush();
}
/* 接收来自Web服务器的数据 */
protected void receiveMessage() throws IOException{
byte data[] = new byte [ 1024 ];
int count = 0 ;
int word = - 1 ;
// 解析第一行
while ( (word = receiver.read()) != - 1 ){
if (word == ' \r ' || word == ' \n ' ){
word = receiver.read();
if (word == ' \n ' ) {
word = receiver.read();
}
break ;
}
if (count == data.length) {
data = addCapacity(data);
}
data[count ++ ] = ( byte ) word;
}
String message = new String(data, 0 ,count);
int mark = message.indexOf( 32 );
serverVersion = message.substring( 0 ,mark);
while (mark < message.length() && message.charAt(mark + 1 ) == 32 ) {
mark ++ ;
}
responseCode = Integer.parseInt(message.substring(mark + 1 ,mark += 4 ));
responseMessage = message.substring(mark,message.length()).trim();
// 应答状态码和处理请读者添加
switch (responseCode){
case 400 :
throw new IOException( " 错误请求 " );
case 404 :
throw new FileNotFoundException(getURLFormat(target));
case 503 :
throw new IOException( " 服务器不可用 " );
}
if (word == - 1 ){
throw new ProtocolException( " 信息接收异常终止 " );
}
int symbol = - 1 ;
count = 0 ;
// 解析元信息
while (word != ' \r ' && word != ' \n ' && word > - 1 ){
if (word == ' \t ' ) {
word = 32 ;
}
if (count == data.length) {
data = addCapacity(data);
}
data[count ++ ] = ( byte ) word;
parseLine:{
while ( (symbol = receiver.read()) > - 1 ){
switch (symbol){
case ' \t ' :
symbol = 32 ;
break ;
case ' \r ' :
case ' \n ' :
word = receiver.read();
if (symbol == ' \r ' && word == ' \n ' ){
word = receiver.read();
if (word == ' \r ' ) {
word = receiver.read();
}
}
if (word == ' \r ' || word == ' \n ' || word > 32 ){
break parseLine;
}
symbol = 32 ;
break ;
}
if (count == data.length) {
data = addCapacity(data);
}
data[count ++ ] = ( byte ) symbol;
}
word = - 1 ;
}
message = new String(data, 0 ,count);
mark = message.indexOf( ' : ' );
String key = null ;
if (mark > 0 ) {
key = message.substring( 0 ,mark);
}
mark ++ ;
while (mark < message.length() && message.charAt(mark) <= 32 ) {
mark ++ ;
}
String value = message.substring(mark,message.length());
header.put(key,value);
count = 0 ;
}
// 获得正文数据
while ( (word = receiver.read()) != - 1 ){
if (count == data.length) {
data = addCapacity(data);
}
data[count ++ ] = ( byte ) word;
}
if (count > 0 ) {
byteStream = new ByteArrayInputStream(data, 0 ,count);
}
data = null ;
closeServer();
}
public String getResponseMessage(){
return responseMessage;
}
public int getResponseCode(){
return responseCode;
}
public String getServerVersion(){
return serverVersion;
}
public InputStream getInputStream(){
return byteStream;
}
public synchronized String getHeaderKey( int i){
if (i >= header.size()){
return null ;
}
Enumeration enumss = header.propertyNames();
String key = null ;
for ( int j = 0 ; j <= i; j ++ ){
key = (String) enumss.nextElement();
}
return key;
}
public synchronized String getHeaderValue( int i){
if (i >= header.size()){
return null ;
}
return header.getProperty(getHeaderKey(i));
}
public synchronized String getHeaderValue(String key){
return header.getProperty(key);
}
protected String getBaseHeads(){
String inf = " User-Agent: ZealHttp/1.0\r\nAccept: www/source; text/html; image/gif; */*\r\n " ;
return inf;
}
private byte [] addCapacity( byte rece[]){
byte temp[] = new byte [rece.length + 1024 ];
System.arraycopy(rece, 0 ,temp, 0 ,rece.length);
return temp;
}
}
java 客户端 稳定 java客户端技术
转载本文章为转载内容,我们尊重原作者对文章享有的著作权。如有内容错误或侵权问题,欢迎原作者联系我们进行内容更正或删除文章。
提问和评论都可以,用心的回复会被更多人看到
评论
发布评论
相关文章
-
java 软件客户端 java开发客户端
DemoApp20 移动商务应用项目技术点· 基于5.0 SDK以上版本的客户端推送 实现推送关键的类包括: PushApplication - 实现推送客户端的接口PushApplicationDescriptor -根据应用属性创建和管理推送客户端,端口,URL,App ID等。
java 软件客户端 blackberry java 报表 javascript