C/S 与 B/S
☆ C/S ( Client/Server ) 客户端和服务端的特点
1、客户端和服务端的软件都需要程序员进行编写。
2、客户端维护起来较为麻烦。(缺陷)
3、客户端的存在可以将一部分运算分离到客户端来运行,减轻了服务器端的压力。(优势)
☆ B/S ( Browse/Server ) 浏览器和服务端的特点
1、客户端不用程序员编写,直接使用系统中具备的浏览器软件作为客户端即可。程序员只需要编写服务器端就OK了。
2、维护起来也很容易,因为只要维护服务器即可。(优势)
3、所有的运算都在服务器端,相对压力较大。(缺陷)
编程练习1
1、自定义一个服务器,接收浏览器发来的信息。显示浏览器发送了什么信息,并向浏览器发送简单的网页信息。
只实现了单线程~~~多线程我写了~只是由于不知道的原因~会出现Socket异常关闭的情况~!所以并没有实现~
单线程:成功~
package cn.hncu.bs; import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.net.ServerSocket; import java.net.Socket; /** * @author 陈浩翔 * * @version 1.0 2016-5-11 */ public class MyServer { public static void main(String[] args) { try { while (true) { ServerSocket server = new ServerSocket(80); Socket s = server.accept(); System.out.println(s.getInetAddress().getHostAddress() + "...来访"); InputStream in = s.getInputStream(); byte buf[] = new byte[1024]; int len = in.read(buf); String text = new String(buf, 0, len); System.out.println(text); BufferedReader bin = new BufferedReader(new FileReader( "chx.html")); // 要加true刷新PrintWriter流 PrintWriter pw = new PrintWriter(s.getOutputStream(), true); String line = null; while ((line = bin.readLine()) != null) { pw.println(line); } in.close(); pw.close(); s.close(); server.close(); } } catch (IOException e) { e.printStackTrace(); } } }
多线程:失败了的例子:
求解:
package cn.hncu.bs; import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.net.ServerSocket; import java.net.Socket; /** * @author 陈浩翔 * * 2016-5-10 */ public class MyServer { public static void main(String[] args) { try { while(true){ ServerSocket server = new ServerSocket(80); //多客户同时访问 Socket s = server.accept(); new Thread(new ServerThread(s)).start(); } } catch (IOException e) { e.printStackTrace(); } } } class ServerThread implements Runnable{ Socket s = null; public ServerThread(Socket s) { this.s = s; } @Override public void run() { try { String ip = s.getInetAddress().getHostAddress(); System.out.println(ip+"...来访问了~"); InputStream in = s.getInputStream(); byte buf[] = new byte[1024]; int len = in.read(buf); String text = new String(buf,0,len); System.out.println(text); BufferedReader bin = new BufferedReader(new FileReader("chx.html")); //要加true刷新PrintWriter流 PrintWriter pw = new PrintWriter(s.getOutputStream(),true); String line=null; while((line=bin.readLine())!=null){ System.out.println(line); pw.println(line); } in.close(); pw.close(); s.close(); } catch (IOException e) { e.printStackTrace(); } } }
我老师给的解释是:
在练习MyServer时采用多线程都出现了多次请求与应该的情况,这很可能是TCP的三次握手造成的,即客户浏览器跟服务器进行几次请求与应答通讯,而服务器在第二次(同一次请求中的第2次握手)收到信息时却用另一个线程去接,因此握手就实现不了,而服务器却多次输出收到消息
编程练习2
2、模拟一个浏览器客户端向服务器发请求,接收并显示响应消息。
package cn.hncu.bs; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.PrintWriter; import java.net.Socket; import java.net.UnknownHostException; /** * @author 陈浩翔 * * @version 1.0 2016-5-11 */ public class MyBrowse { public static void main(String[] args) { String ipAdd = "www.hncu.net"; try { Socket s = new Socket(ipAdd, 80); //向服务器发送http协议请求头,以让服务器认识我们,给我们正确地响应 OutputStream out = s.getOutputStream(); System.out.println(s.getPort());//端口 PrintWriter pw = new PrintWriter(out,true); pw.println("GET / HTTP/1.1");//请求行包含: 请求方式(GET POST) 空格 请求的资源路径 空格 http的协议版本 pw.println("Accept: */*;"); pw.println("Host: "+ipAdd);//下面这些key-value是请求头 //out.println("User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; rv:11.0) like Gecko"); //out.println("Accept-Encoding: gzip, deflate"); pw.println("Connection: close");//如果没有这一句,程序不会自己结束的 pw.println();//空行---http协议请求头结束之后,必须要一个空行 System.out.println("请求完毕!"); //接收服务器的响应 InputStream in = s.getInputStream(); int len=0; byte b[] = new byte[1024]; while((len=in.read(b))!=-1){ String str = new String(b,0,len,"UTF-8"); System.out.println(str); } s.close(); } catch (UnknownHostException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } }
响应过来的消息就是:
HTTP/1.1 200 OK Date: Wed, 11 May 2016 10:56:05 GMT Server: VWebServer X-Frame-Options: SAMEORIGIN Last-Modified: Wed, 11 May 2016 06:04:55 GMT ETag: "6060-5328ad214afc0" Accept-Ranges: bytes Content-Length: 24672 Vary: User-Agent,Accept-Encoding Cache-Control: private, max-age=600 Expires: Wed, 11 May 2016 11:06:05 GMT Connection: close Content-Type: text/html Content-Language: zh-CN //后面的就是网站的html代码了~~~ ?<!DOCTYPE html><html><head><title>湖南城市学院</title> ... ... ... ... </body></html>
编程练习3:
3、网络蜘蛛,收集网页中的邮箱地址信息。
package cn.hncu.bs; import java.io.BufferedReader; import java.io.FileNotFoundException; import java.io.FileReader; import java.io.IOException; import java.io.InputStreamReader; import java.net.MalformedURLException; import java.net.URL; import java.util.regex.Matcher; import java.util.regex.Pattern; import org.junit.Test; import com.sun.org.apache.xerces.internal.impl.xpath.regex.Match; /** * @author 陈浩翔 * * @version 1.0 2016-5-11 */ public class SpiderDemo { //读取网页的邮箱 public static void main(String[] args) { try { URL url = new URL("http://www.sina.com.cn/"); System.out.println(url.getHost()); BufferedReader br = new BufferedReader(new InputStreamReader(url.openStream())); String regex = "\\w+@\\w+(\\.\\w+)+"; Pattern p = Pattern.compile(regex); String line = null; while((line=br.readLine())!=null){ Matcher m = p.matcher(line); while(m.find()){ System.out.println(m.group()); } } System.out.println("读取完!"); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } /** * 先测试本地的邮箱地址看能不能读取出来---成功 */ @Test public void Test() { try { BufferedReader br = new BufferedReader(new FileReader("mail.html")); // 正则表达式 String regex= "\\w+@\\w+(\\.\\w+)+"; Pattern p = Pattern.compile(regex); String line=null; while((line=br.readLine())!=null){ Matcher m = p.matcher(line); while(m.find()){ System.out.println(m.group()); } } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } }