实现 B/S 服务器
先看一个简单例子:
packagedemo.net.bs_demo;importjava.io.IOException;importjava.io.InputStream;importjava.net.ServerSocket;importjava.net.Socket;publicclassBaseDemo{publicstaticvoidmain(String[]args)throwsIOException{ServerSocketserver=newServerSocket(8000);Socketsocket=server.accept();InputStreamin=socket.getInputStream();byte[]bytes=newbyte[1024];intlen=in.read(bytes);System.out.println(newString(bytes,0,len));socket.close();server.close();}}
在浏览器访问:http://localhost:8000/后,服务器端输出如下:
GET/HTTP/1.1Host:localhost:8000Connection:keep-aliveUpgrade-Insecure-Requests:1User-Agent:Mozilla/5.0(WindowsNT10.0;WOW64)AppleWebKit/537.36(KHTML,likeGecko)Chrome/76.0.3809.132Safari/537.36Sec-Fetch-Mode:navigateSec-Fetch-User:?1Accept:text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3Sec-Fetch-Site:noneAccept-Encoding:gzip,deflate,brAccept-Language:zh-CN,zh;q=0.9Cookie:_ga=GA1.1.1778445080.1563089064;csrftoken=ZnWKg5ZuuSYd3dwts5iqnfjQvSnp21n9OiuFZAKz11WdGZuXENLhAxYckiLHFTsI
接下来是一个处理浏览器的访问请求,并返回请求页面的服务器端代码
页面放在和 src 同级的目录 webDemo 下
packagedemo.net.bs_demo;importjava.io.IOException;importjava.net.*;importjava.io.*;publicclassBrowserServer{publicstaticvoidmain(String[]args)throwsIOException{//创建一个服务器ServerSocket,和系统要指定的端口号ServerSocketserver=newServerSocket(8080);//使用accept方法获取到请求的客户端对象(浏览器)Socketsocket=server.accept();//使用Socket对象中的方法getInputStream,获取到网络字节输入流InputStream对象InputStreamis=socket.getInputStream();//使用网络字节输入流InputStream对象中的方法read读取客户端的请求信息/*byte[] bytes = new byte[1024];int len = 0;while((len = is.read(bytes))!=-1){System.out.println(new String(bytes,0,len));}*///把is网络字节输入流对象,转换为字符缓冲输入流,使用缓冲流更快BufferedReaderbr=newBufferedReader(newInputStreamReader(is));//把客户端请求信息的第一行读取出来 "GET /webDemo/index.html HTTP/1.1"Stringline=br.readLine();//System.out.println(line);//把读取的信息进行切割,只要中间部分 "/webDemo/index.html"String[]arr=line.split(" ");//把路径前边的/去掉,进行截取 "webDemo/index.html"Stringhtmlpath=arr[1].substring(1);//System.out.println(htmlpath);//创建一个本地字节输入流,构造方法中绑定要读取的html路径FileInputStreamfis=newFileInputStream(htmlpath);//使用Socket中的方法getOutputStream获取网络字节输出流OutputStream对象OutputStreamos=socket.getOutputStream();// 写入HTTP协议响应头,固定写法os.write("HTTP/1.1 200 OK\r\n".getBytes());os.write("Content-Type:text/html\r\n".getBytes());// 必须要写入空行,否则浏览器不解析os.write("\r\n".getBytes());//一读一写复制文件,把服务读取的html文件回写到客户端intlen=0;byte[]bytes=newbyte[1024];while((len=fis.read(bytes))!=-1){os.write(bytes,0,len);}//释放资源fis.close();socket.close();server.close();}}
在浏览器访问:http://localhost:8080/webDemo/index.html,会出现相应的页面,但页面中的图片显示不正常
发现浏览器中出现很多的叉子,说明浏览器没有读取到图片信息导致。
浏览器工作原理是遇到图片会开启一个线程进行单独的访问,因此在服务器端加入线程技术。
优化后的服务器端代码如下:
packagedemo.net.bs_demo;importjava.io.*;importjava.net.*;publicclassBrowserServerThread{publicstaticvoidmain(String[]args)throwsIOException{//创建一个服务器ServerSocket,和系统要指定的端口号ServerSocketserver=newServerSocket(8080);System.out.println("服务器启动 ...");/*浏览器解析服务器回写的html页面,页面中如果有图片,那么浏览器就会单独的开启一个线程,读取服务器的图片我们就的让服务器一直处于监听状态,客户端请求一次,服务器就回写一次*/while(true){//使用accept方法获取到请求的客户端对象(浏览器)Socketsocket=server.accept();newThread(newRunnable(){@Overridepublicvoidrun(){try{//使用Socket对象中的方法getInputStream,获取到网络字节输入流InputStream对象InputStreamis=socket.getInputStream();//使用网络字节输入流InputStream对象中的方法read读取客户端的请求信息/*byte[] bytes = new byte[1024];int len = 0;while((len = is.read(bytes))!=-1){System.out.println(new String(bytes,0,len));}*///把is网络字节输入流对象,转换为字符缓冲输入流,使用缓冲流更快BufferedReaderbr=newBufferedReader(newInputStreamReader(is));//把客户端请求信息的第一行读取出来 "GET /webDemo/index.html HTTP/1.1"Stringline=br.readLine();//System.out.println(line);//把读取的信息进行切割,只要中间部分 "/webDemo/index.html"String[]arr=line.split(" ");//把路径前边的/去掉,进行截取 "webDemo/index.html"Stringhtmlpath=arr[1].substring(1);//System.out.println(htmlpath);//创建一个本地字节输入流,构造方法中绑定要读取的html路径FileInputStreamfis=newFileInputStream(htmlpath);//使用Socket中的方法getOutputStream获取网络字节输出流OutputStream对象OutputStreamos=socket.getOutputStream();// 写入HTTP协议响应头,固定写法os.write("HTTP/1.1 200 OK\r\n".getBytes());os.write("Content-Type:text/html\r\n".getBytes());// 必须要写入空行,否则浏览器不解析os.write("\r\n".getBytes());//一读一写复制文件,把服务读取的html文件回写到客户端intlen=0;byte[]bytes=newbyte[1024];while((len=fis.read(bytes))!=-1){os.write(bytes,0,len);}//释放资源fis.close();socket.close();}catch(IOExceptione){e.printStackTrace();}}}).start();}//server.close();}}
再次在浏览器访问:http://localhost:8080/webDemo/index.html,会出现相应的页面,并且页面中的图片显示正常
如果觉得文章报错的话,麻烦动动小手点赞哟
中秋假期结束,明天继续上班、学习、睡觉、地铁、吃饭的无限循环
2019.09.15 23:55
假期终于把之前学习的知识全部整理完了
以下是在公司内网本部分的学习记录,无奈网络隔离,只能上图片了
声明:本文部分素材转载自互联网,如有侵权立即删除 。
© 版权声明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!邮箱:cxysz1@tom.com
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是赞助,收取费用仅维持本站的日常运营所需!
7. 如遇到加密压缩包,请使用WINRAR解压,如遇到无法解压的请联系管理员!
8. 精力有限,不少源码未能详细测试(解密),不能分辨部分源码是病毒还是误报,所以没有进行任何修改,大家使用前请进行甄别
丞旭猿论坛
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是赞助,收取费用仅维持本站的日常运营所需!
7. 如遇到加密压缩包,请使用WINRAR解压,如遇到无法解压的请联系管理员!
8. 精力有限,不少源码未能详细测试(解密),不能分辨部分源码是病毒还是误报,所以没有进行任何修改,大家使用前请进行甄别
丞旭猿论坛
THE END
暂无评论内容