- 浏览: 219453 次
- 性别:
- 来自: 上海
文章分类
最新评论
-
wly20110420:
很详细,谢谢分享!
java中String\十六进制String\byte[]之间相互转换函数 -
星星283320052:
...
随心畅言 -
贾懂凯:
...
随心畅言
用java socket实现了一个简单的http服务器, 可以处理GET, POST,以及带一个附件的multipart类型的POST。虽然中途遇到了很多问题, 不过通过在论坛和几个高手交流了一下,问题都解决了。如果你觉得程序有些地方看不明白,可以参看这个帖子:http://topic.csdn.net/u/20090625/22/59a5bfc8-a6b6-445d-9829-ea6d462a4fe6.html .
虽然解析http头不是很规范,本来应该用原始的字节流, 我采用了一个折衷的方案,用DataInputStream.
本代码的实用性==0,但是可以帮助很好地了解http协议,然后其他的应用层协议大都如此。
如果你从来都没有了解过http协议,建议先搜索阅读一下,或者你还可以用下面的代码来简单的看一看到底浏览器和服务器之间都相互发送了什么数据。
MyHttpClient.java: 模拟浏览器的行为, 向服务器发送get/post请求,然后打印出服务器返回的消息。这样就可以查看当一个请求到来之后, 服务器到底都给浏览器发送了哪些消息。
package socket; import java.io.*; import java.net.*; public class MyHttpClient { public static void main(String[] args) throws Exception{ InetAddress inet = InetAddress.getByName("www.baidu.com"); System.out.println(inet.getHostAddress()); Socket socket = new Socket(inet.getHostAddress(),80); InputStream in = socket.getInputStream(); OutputStream out = socket.getOutputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader(in)); PrintWriter writer = new PrintWriter(out); writer.println("GET /home.html HTTP/1.1");//home.html是关于百度的页面 writer.println("Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/msword, application/vnd.ms-excel, application/vnd.ms-powerpoint, */*"); writer.println("Accept-Language: en-us,zh-cn;q=0.5"); writer.println("Accept-Encoding: gzip, deflate"); writer.println("Host: www.baidu.com"); writer.println("User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)"); writer.println("Connection: Keep-Alive"); writer.println(); writer.flush(); String line = reader.readLine(); while(line!=null){ System.out.println(line); line = reader.readLine(); } reader.close(); writer.close(); } }
MyServer.java: 模拟server端接收浏览器的请求,然后把整个请求的报文打印出来。程序运行之后直接用浏览器测试。
package socket; import java.io.*; import java.net.*; public class MyServer { public static void main(String[] args) throws IOException{ ServerSocket svrSocket = new ServerSocket(8080); while(true){ Socket socket = svrSocket.accept(); //足够大的一个缓冲区 byte[] buf = new byte[1024*1024]; InputStream in = socket.getInputStream(); int byteRead = in.read(buf, 0, 1024*1024); String dataString = new String(buf, 0, byteRead); System.out.println(dataString); in.close(); socket.close(); } } }
主程序MyHttpServer.
package socket; import java.io.*; import java.net.*; /** * MyHttpServer 实现一个简单的HTTP服务器端,可以获取用户提交的内容 * 并给用户一个response * 因为时间的关系,对http头的处理显得不规范 * 对于上传附件,暂时只能解析只上传一个附件而且附件位置在第一个的情况 * 转载请注明来自http://blog.csdn.net/sunxing007 * **/ public class MyHttpServer { //服务器根目录,post.html, upload.html都放在该位置 public static String WEB_ROOT = "c:/root"; //端口 private int port; //用户请求的文件的url private String requestPath; //mltipart/form-data方式提交post的分隔符, private String boundary = null; //post提交请求的正文的长度 private int contentLength = 0; public MyHttpServer(String root, int port) { WEB_ROOT = root; this.port = port; requestPath = null; } //处理GET请求 private void doGet(DataInputStream reader, OutputStream out) throws Exception { if (new File(WEB_ROOT + this.requestPath).exists()) { //从服务器根目录下找到用户请求的文件并发送回浏览器 InputStream fileIn = new FileInputStream(WEB_ROOT + this.requestPath); byte[] buf = new byte[fileIn.available()]; fileIn.read(buf); out.write(buf); out.close(); fileIn.close(); reader.close(); System.out.println("request complete."); } } //处理post请求 private void doPost(DataInputStream reader, OutputStream out) throws Exception { String line = reader.readLine(); while (line != null) { System.out.println(line); line = reader.readLine(); if ("".equals(line)) { break; } else if (line.indexOf("Content-Length") != -1) { this.contentLength = Integer.parseInt(line.substring(line.indexOf("Content-Length") + 16)); } //表明要上传附件, 跳转到doMultiPart方法。 else if(line.indexOf("multipart/form-data")!= -1){ //得multiltipart的分隔符 this.boundary = line.substring(line.indexOf("boundary") + 9); this.doMultiPart(reader, out); return; } } //继续读取普通post(没有附件)提交的数据 System.out.println("begin reading posted data......"); String dataLine = null; //用户发送的post数据正文 byte[] buf = {}; int size = 0; if (this.contentLength != 0) { buf = new byte[this.contentLength]; while(size<this.contentLength){ int c = reader.read(); buf[size++] = (byte)c; } System.out.println("The data user posted: " + new String(buf, 0, size)); } //发送回浏览器的内容 String response = ""; response += "HTTP/1.1 200 OK/n"; response += "Server: Sunpache 1.0/n"; response += "Content-Type: text/html/n"; response += "Last-Modified: Mon, 11 Jan 1998 13:23:42 GMT/n"; response += "Accept-ranges: bytes"; response += "/n"; String body = "<html><head><title>test server</title></head><body><p>post ok:</p>" + new String(buf, 0, size) + "</body></html>"; System.out.println(body); out.write(response.getBytes()); out.write(body.getBytes()); out.flush(); reader.close(); out.close(); System.out.println("request complete."); } //处理附件 private void doMultiPart(DataInputStream reader, OutputStream out) throws Exception { System.out.println("doMultiPart ......"); String line = reader.readLine(); while (line != null) { System.out.println(line); line = reader.readLine(); if ("".equals(line)) { break; } else if (line.indexOf("Content-Length") != -1) { this.contentLength = Integer.parseInt(line.substring(line.indexOf("Content-Length") + 16)); System.out.println("contentLength: " + this.contentLength); } else if (line.indexOf("boundary") != -1) { //获取multipart分隔符 this.boundary = line.substring(line.indexOf("boundary") + 9); } } System.out.println("begin get data......"); /*下面的注释是一个浏览器发送带附件的请求的全文,所有中文都是说明性的文字***** <HTTP头部内容略> ............ Cache-Control: no-cache <这里有一个空行,表明接下来的内容都是要提交的正文> -----------------------------7d925134501f6<这是multipart分隔符> Content-Disposition: form-data; name="myfile"; filename="mywork.doc" Content-Type: text/plain <附件正文>........................................ ................................................. -----------------------------7d925134501f6<这是multipart分隔符> Content-Disposition: form-data; name="myname"<其他字段或附件> <这里有一个空行> <其他字段或附件的内容> -----------------------------7d925134501f6--<这是multipart分隔符,最后一个分隔符多两个-> ****************************************************************/ /** * 上面的注释是一个带附件的multipart类型的POST的全文模型, * 要把附件去出来,就是要找到附件正文的起始位置和结束位置 * **/ if (this.contentLength != 0) { //把所有的提交的正文,包括附件和其他字段都先读到buf. byte[] buf = new byte[this.contentLength]; int totalRead = 0; int size = 0; while (totalRead < this.contentLength) { size = reader.read(buf, totalRead, this.contentLength - totalRead); totalRead += size; } //用buf构造一个字符串,可以用字符串方便的计算出附件所在的位置 String dataString = new String(buf, 0, totalRead); System.out.println("the data user posted:/n" + dataString); int pos = dataString.indexOf(boundary); //以下略过4行就是第一个附件的位置 pos = dataString.indexOf("/n", pos) + 1; pos = dataString.indexOf("/n", pos) + 1; pos = dataString.indexOf("/n", pos) + 1; pos = dataString.indexOf("/n", pos) + 1; //附件开始位置 int start = dataString.substring(0, pos).getBytes().length; pos = dataString.indexOf(boundary, pos) - 4; //附件结束位置 int end = dataString.substring(0, pos).getBytes().length; //以下找出filename int fileNameBegin = dataString.indexOf("filename") + 10; int fileNameEnd = dataString.indexOf("/n", fileNameBegin); String fileName = dataString.substring(fileNameBegin, fileNameEnd); /** * 有时候上传的文件显示完整的文件名路径,比如c:/my file/somedir/project.doc * 但有时候只显示文件的名字,比如myphoto.jpg. * 所以需要做一个判断。 */ if(fileName.lastIndexOf("//")!=-1){ fileName = fileName.substring(fileName.lastIndexOf("//") + 1); } fileName = fileName.substring(0, fileName.length()-2); OutputStream fileOut = new FileOutputStream("c://" + fileName); fileOut.write(buf, start, end-start); fileOut.close(); fileOut.close(); } String response = ""; response += "HTTP/1.1 200 OK/n"; response += "Server: Sunpache 1.0/n"; response += "Content-Type: text/html/n"; response += "Last-Modified: Mon, 11 Jan 1998 13:23:42 GMT/n"; response += "Accept-ranges: bytes"; response += "/n"; out.write("<html><head><title>test server</title></head><body><p>Post is ok</p></body></html>".getBytes()); out.flush(); reader.close(); System.out.println("request complete."); } public void service() throws Exception { ServerSocket serverSocket = new ServerSocket(this.port); System.out.println("server is ok."); //开启serverSocket等待用户请求到来,然后根据请求的类别作处理 //在这里我只针对GET和POST作了处理 //其中POST具有解析单个附件的能力 while (true) { Socket socket = serverSocket.accept(); System.out.println("new request coming."); DataInputStream reader = new DataInputStream((socket.getInputStream())); String line = reader.readLine(); String method = line.substring(0, 4).trim(); OutputStream out = socket.getOutputStream(); this.requestPath = line.split(" ")[1]; System.out.println(method); if ("GET".equalsIgnoreCase(method)) { System.out.println("do get......"); this.doGet(reader, out); } else if ("POST".equalsIgnoreCase(method)) { System.out.println("do post......"); this.doPost(reader, out); } socket.close(); System.out.println("socket closed."); } } public static void main(String args[]) throws Exception { MyHttpServer server = new MyHttpServer("c:/root", 8080); server.service(); } }
测试文件post.html, upload.html都放在上面程序定义的WEB_ROOT下面。
post.html:处理普通的post请求
<html> <head> <title>test my server</title> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> </head> <body> <p>upload</p> 来自http://blog.csdn.net/sunxing007<br> <form name="UploadForm" method="post" action="http://localhost:8080/"> <input type="text" name="myname" /><br> <select name="myage"> <option value="18">18</option> <option value="20">20</option> <option value="22">22</option> </select><br> <input type="submit"value="Sutmit"> </form> </body> </html>
upload.html:测试带附件的post请求
<head> <title>my page</title> <style> table{ border-collapse: collapse; } </style> </head> <body> 来自http://blog.csdn.net/sunxing007<br> <form action='http://localhost:8080/' method='post' enctype='multipart/form-data'> file: <input type='file' name='myfile' /><br> <input type='submit' /> </form> </body> </html>
一切准备妥当,并且MyHttpServer运行之后, 在浏览器输入http://localhost:8080/post.html和http://localhost:8080/upload.html即可进行测试.
发表评论
-
@RequestMapping 用法详解之地址映射
2015-08-31 11:32 0前段时间项目中用到了RESTful模式来开发程序,但是当用P ... -
focus获取不到光标
2014-03-07 18:10 523focus第一能获取,重新调用不能获取光标,解决方案: i ... -
根据XML Schema生成java类
2014-02-26 13:41 0根据XML Schema生成java类 JAXB提供了 ... -
CLOB/BLOB与String互转。
2013-12-21 17:28 7517String s1="1231dsdgasd ... -
json转换
2013-11-27 17:03 667j1、DTO:Data Transfer Objec ... -
org.hibernate.LazyInitializationException: failed to lazily initialize a collect
2013-04-26 15:37 828这个异常:org.hibernate.LazyInitial ... -
String Stringbuffer和StringBuilder的区别?
2013-04-10 10:27 610java.lang.StringBuffer线程安全的可变字 ... -
获取当前时间的前一天和后一天
2013-01-17 19:36 1308package jp.co.hitachi_solution ... -
js 上传和下载本地文件
2012-11-29 15:24 1251function writeFile(path,data){ ... -
java 文件上传
2012-11-29 15:11 676//同步销单信息 function synchrony ... -
java 文件下载
2012-11-29 15:04 644//下载任务 public String synchronyT ... -
Specified VM install not found: type Standard VM, name jre
2012-11-29 14:10 1007编译Ant脚本时出错: "Specified VM ... -
警告:不能读取 AppletViewer 的属性文件:
2012-10-30 15:04 633警告:不能读取 AppletViewer 的属性文件: C:\ ... -
SAXBuilder基本用法
2012-08-29 13:38 976xml为我们在网络交换 ... -
jdom学习:读取xml文件
2012-08-27 16:50 595用JDOM读取XML文件需先用org.jdom.input.S ... -
Java获取当前日期的前一个月,前一天的时间
2012-08-23 16:08 19713Calendar calendar = Calendar.ge ... -
Hql语句根据时间查询,如何查询昨天、今天、上月、本月的数据
2012-08-10 17:26 13326这个只要你能根据今天的时间算出你需要的时间就完全可以了 如下 ... -
cron 表达式的格式
2012-07-12 10:35 802cron 表达式的格式 Quartz cron 表达式的格式 ... -
java中String\十六进制String\byte[]之间相互转换函数
2012-03-16 16:14 41449java二进制,字节数组,字符,十六进制,BCD编码转换200 ... -
BCD 转 String, String 转 BCD
2012-03-16 16:08 6911/// <summary> / ...
相关推荐
用java实现的http服务器端例程
用java实现的http服务器端例程java
用java实现的http服务器端例程.rar 用java实现的http服务器端例程.rar
HTTP1.1 服务器端及servlet引擎 java实现
实验的目的: 1) 掌握Java Socket编程应用; 2) 阅读HTTP协议相关规范; 3) 基于Java Socket构建简单的HTTP的客户端和服务器; 4) 构建支持并发的HTTP服务器。
java实现的简单http web服务器,能响应浏览器端GET请求,并返回响应资源,服务器端界面能显示相应信息。包含实验报告。
先运行服务器端,再运行客户端。就可以实现通信,客户端想服务器端发送数据。服务器端作出响应,适合初学者 代码简单
java 实现绘制指针时钟 和多线程服务器java 实现绘制指针时钟 和多线程服务器java 实现绘制指针时钟 和多线程服务器java 实现绘制指针时钟 和多线程服务器java 实现绘制指针时钟 和多线程服务器java 实现绘制指针...
Java Socket编程 PC端程序 用来传输文件 包括服务器、客户端
基于Java、Socket实现的一个HTTP服务器,能够支持GET请求,支持日志写入,支持数据库管理、能够满足静态页面的发布。... 对于研究并开发RFC协议、实现Socket服务器端程序、实现协议交互与JAVA语言编程有重要参考意义。
java与网络一共7个案例 简单多线程服务器 用java实现http服务器端 用javaemail发邮件 java版msn java实现hppt验证等
修改 APP_KEY,APP_CLIENT_ID,APP_CLIENT_SECRET即可...封装环信全部API... ...参考环信开发者文档: http://docs.easemob.com/im/100serverintegration/40fileoperation 官网 http://www.easemob.com/
SOCKET TCP 文件传输 客户端 服务器端 client svever 实现简易文件传输功能
通过服务器端和客户端,实现文件的传输,从客户端向服务器端发送文件,不同于tftp协议
Java实现HTTP连接与浏览,Java源码下载,输入html文件地址或网址,显示页面和HTML源文件,一步步的实现过程请下载本实例的Java源码,代码中包括丰富的注释,对学习有帮助。 Java实现的FTP连接与数据浏览程序 1个...
Java实现HTTP连接与浏览,Java源码下载,输入html文件地址或网址,显示页面和HTML源文件,一步步的实现过程请下载本实例的Java源码,代码中包括丰富的注释,对学习有帮助。 Java实现的FTP连接与数据浏览程序 1个...
用java进行客户端的applet (小程序)开发的技术已广为使用,而用java进行服务器端的servlet(服务器小程序)开发则尚需揭开其神秘的面纱,本书正是基于这样的目的编写的。全书从java服务器的体系结构、开发工具和...
实现HTTP协议 II. 实现HTTPS协议 III. 实现FASTCGI协议(Client端) 运行HTTP/HTTPS服务器 运行开发包下jar文件 java -jar http-server-version-{version}.jar 源码编译 http: 编译源码并...
app包:Android服务器端代码, coaplib: java服务器代码以及java客户端测试代码 , 以下为补充说明内容 COAP协议: CoAP服务器则提供了人们能轻松看懂的URI,如 /thermometers/5 。在可发现性的使用惯例里,所有资源都...
java.rmi.server 提供支持服务器端 RMI 的类和接口。 java.security 为安全框架提供类和接口。 java.security.acl 此包中的类和接口已经被 java.security 包中的类取代。 java.security.cert 提供用于解析和管理...