Java--Socket通信

下面内容是Java开发内容的高级知识点,需要对Java中的面向对象、IO、多线程、以及网络相关知识有一定的基础。(知识永远都有深度,本章节长期更新内容)

 

1、网络基础知识

 

网络通信的条件:1、两个通信的端都要有各自的IP地址作为唯一标识,简单的来说IP地址用来区分不同机器(计算机)。2、语言要相通。3、如何辨别不同程序的通信需要端口号来区别,简单的来说端口号就是用来标识不同的应用程序。

TCP/IP是目前世界上使用最广泛的协议,是以TCP和IP为基础的不同层次上多个协议的集合,也称TCP/IP协议族 或 TCP/IP 协议栈。

TCP:Transmission Control Protocol 传输控制协议
IP:Internet Protocol 互联网协议

TCP/IP模型

 

>>IP和端口<<
1、用于区分不同应用程序。
2、端口号范围为0~65535,其中0~1023为系统所保留。如果自定义端口号,所以建议用1024之后的端口号。
3、IP地址和端口号组成了所谓的Socket,Socket是网络上运行程序之间双向通信链路的终结点,是TCP和UDP的基础。


常用的端口号需要记一下:http:80    ftp:21    telnet:23

 

——————————Java中的网络支持—————————
针对网络通信的不同层次,Java提供的网络功能有四大类:
>>1、InetAddress:用于标识网络上的硬件资源
>>2、URL:统一资源定位符 通过URL可以直接读取或写入网络上的数据
>>3、Socket:使用TCP协议实现网络通信的Socket相关的类。
>>4、Datagram:使用UDP协议,将数据保存在数据报中,通过网络进行通信。

 


2、InetAddress类

查看I-net-Address的API文档,发现没有构造方法,也就是不能通过new来创建。所以肯定有静态的方法来创建。

 1 import java.net.InetAddress;
 2 import java.net.UnknownHostException;
 3 import java.util.Arrays;
 4 
 5 public class Test1{
 6     public static void main(String[] args) throws UnknownHostException{
 7         // 获取本机的InetAdresss实例
 8         InetAddress address = InetAddress.getLocalHost();
 9         System.out.println("计算机名:"+address.getHostName()+"\nIP地址:"+address.getHostAddress());
10 
11         // 获取字节数组形式的IP地址
12         byte[] bytes = address.getAddress();
13         System.out.println("字节数组形式的IP:"+Arrays.toString(bytes));
14         System.out.println(address);
15 
16         // 也可以通过机器名来获取InewAdress
17         InetAddress address2 = InetAddress.getByName("MacBook-Air-2.local");
18         System.out.println("通过计算机名字创建的InetAddress对象:"+address2);
19         System.out.println("计算机名:"+address2.getHostName());
20         System.out.println("IP地址:"+address2.getHostAddress());
21 
22         // 也可以通过IP地址来获取InewAdress
23         InetAddress address3 = InetAddress.getByName("192.168.1.102");
24         System.out.println("通过计算机IP地址创建的InetAddress对象:"+address3);
25         System.out.println("计算机名:"+address3.getHostName());
26         System.out.println("IP地址:"+address3.getHostAddress());
27 
28 
29     }
30 }

输出结果:

 

 
3、URL

 URL(Uniform Resource Locator)统一资源定位符,表示Internet上某一资源的地址。 俗称就是网址。

 URL由两部分组成:协议名称+资源名称。

 在Java.net包中,提供了URL类来表示URL。

 1 import java.net.MalformedURLException;
 2 import java.net.URL;
 3 
 4 public class Test1{
 5     public static void main(String[] args){
 6         
 7         try {
 8             // 创建一个URL实例
 9             URL imoocURL = new URL("http://www.imooc.com");
10             URL url = new URL(imoocURL,"/index.html?username=tom#test");
11             // ?后面表示参数,#后面表示的是锚点
12 
13             // 创建URL对象之后,可以根据这个对象获取相关的信息
14             System.out.println("协议:"+url.getProtocol());
15             System.out.println("主机:"+url.getHost());
16             // 如果未指定端口号,则使用默认的端口号,此时getPort()方法返回值为-1
17             System.out.println("端口:"+url.getPort());
18             System.out.println("文件路径:"+url.getPath());
19             System.out.println("文件名:"+url.getFile());
20             System.out.println("相对路径:"+url.getRef());// 实际上就是#锚点后面的内容
21             System.out.println("查询字符串:"+url.getQuery());
22 
23         } catch (MalformedURLException e) {
24             e.printStackTrace();
25         }
26 
27 
28     }
29 }

输出:

 下面再通过URL读取网页内容:

 1 import java.net.MalformedURLException;
 2 import java.net.URL;
 3 import java.io.InputStream;
 4 import java.io.InputStreamReader;
 5 import java.io.BufferedReader;
 6 import java.io.IOException;
 7 
 8 /*
 9  * 使用URL读取网页页面内容
10  */
11 public class Test1{
12     public static void main(String[] args){
13         
14         try {
15             // 创建一个URL实例
16             URL url = new URL("http://www.baidu.com");
17             // 通过URL的openStream方法获取URL对象所表示的资源的字节输入流
18             InputStream is = url.openStream();
19             // 将字节输入流转换为字符输入流
20             InputStreamReader isr = new InputStreamReader(is,"utf-8");// 如果没有指明编码可能会出现中文乱码
21             // 为字符输入流添加缓冲
22             BufferedReader br = new BufferedReader(isr);
23             String data = br.readLine();// 读取数据
24             while(data != null){
25                 System.out.println(data);// 输出数据
26                 data = br.readLine();
27             }
28             // 最后按照上面对象倒序关闭
29             br.close();
30             isr.close();
31             is.close();
32         } catch (MalformedURLException e) {
33             e.printStackTrace();
34         } catch (IOException e) {
35             e.printStackTrace();
36         } 
37     }
38 }

输入:

 


4、TCP编程

4-1、Socket简介

TCP协议是面向连接、可靠的、有序的,以字节流的方式发送数据
基于TCP协议实现网络通信的类:
>>1、客户端的Socket类
>>2、服务器端的ServerSocket类

基于Socket的TCP通信模型

 

 Socket通信实现步骤:1、创建ServerSocket和Socket。2、打开连接到Socket的输入/输出流。3、按照协议对Socket进行读/写操作。4、关闭输入输出流、关闭Socket。
通过在线API文档:http://tool.oschina.net/apidocs/apidoc?api=jdk-zh 中查询到的结果:

4-2、编程实现基于TCP的Socket服务器端和客户端的通信

服务器端:
    1、创建ServerSocket对象,绑定监听端口。
    2、通过accept()方法监听客户端请求。
    3、链接建立后,通过输入流读取客户端发送的请求信息。
    4、通过输出流向客户端发送响应信息。
    5、关闭相关资源。

 1 package com.heyang;
 2 
 3 import java.io.BufferedReader;
 4 import java.io.IOException;
 5 import java.io.InputStream;
 6 import java.io.InputStreamReader;
 7 import java.net.ServerSocket;
 8 import java.net.Socket;
 9 
10 /*
11  * 基于TCP协议的Socket通信,实现用户登录
12  * 服务器端
13  */
14 public class Server {
15     public static void main(String[] args) {
16         try {
17             // 1、创建一个服务器Socket,即ServerSocket,指定绑定的端口,并监听此端口
18             ServerSocket serverSocket = new ServerSocket(8888);
19             // 2、调用()方法开始监听,等待客户端的连接
20             System.out.println("***服务器即将启动,等待客户端的连接***");
21             Socket socket = serverSocket.accept();// 就会处于阻塞的状态,等待监听
22             // 3、获取输入流,病读取客户端信息
23             InputStream is = socket.getInputStream();// 字节输入流
24             // 将字节流转换为字符流
25             InputStreamReader isr = new InputStreamReader(is);
26             // 为输入流添加缓冲
27             BufferedReader br = new BufferedReader(isr);
28             String info = null;
29             while((info = br.readLine())!=null){
30                 System.out.println("我是服务器,读取客户端发过来的信息:"+info);
31             }
32             socket.shutdownInput();//关闭输入流
33             
34             // 关闭资源
35             br.close();
36             isr.close();
37             is.close();
38             socket.close();
39             serverSocket.close();
40             
41         } catch (IOException e) {
42             // TODO Auto-generated catch block
43             e.printStackTrace();
44         }
45     }
46 }



客户端:
    1、创建Socket对象,指明需要连接的服务器的地址和端口号。
    2、连接建立后,通过输出流向服务器端发送请求信息。
    3、通过输入流获取服务器响应的信息。
    4、关闭相关资源。

 1 package com.heyang;
 2 
 3 import java.io.IOException;
 4 import java.io.OutputStream;
 5 import java.io.PrintWriter;
 6 import java.net.Socket;
 7 import java.net.UnknownHostException;
 8 
 9 /*
10  * 客户端
11  */
12 public class Client {
13     public static void main(String[] args) {
14         // 1、创建客户端Socket,指定服务器地址和端口
15         try {
16             Socket  socket = new Socket("localhost", 8888);
17             // 2、获取输出流,向服务器端发送信息
18             OutputStream os = socket.getOutputStream();// 获取字节输出流
19             // 将输出流包装为打印流
20             PrintWriter pw = new PrintWriter(os);
21             pw.write("用户名:admin 密码:123");
22             pw.flush();
23             socket.shutdownInput();//关闭输出流
24             
25             // 3、关闭资源
26             pw.close();
27             os.close();
28             socket.close();
29             
30         } catch (UnknownHostException e) {
31             // TODO Auto-generated catch block
32             e.printStackTrace();
33         } catch (IOException e) {
34             // TODO Auto-generated catch block
35             e.printStackTrace();
36         }
37     }
38 }

输出:

 

 4-3、完善客户端登陆之后服务器响应客户端

代码逻辑深化分析:

  在前面简单的代码的基础上,我们需要完善客户端登陆之后服务器响应客户端的逻辑。

  事实上,站在客户端的角度,对外(这里指的是服务器)发出数据流,也就是需要用输出流来输出数据流。

  反过来,站在服务器端的角度,就要不断的监听外部(这里指的是外部客户端)输入进来的数据流,所以就需要输入流来接收输入进来的数据流。

  那么,当服务器端收到客户端传输过来的数据流之后,就应该响应,那么这时候,站在服务器端的角度,要对外进行响应,就需要用输出流来输出响应回馈的信息,这时候就和客户端一开始的输出流代  码一致了。

  回到客户端,由于需要接收和读取服务器端的发出的响应,就需要输入流来接收服务器发过来的回馈信息了,这时候就和服务器端一开始的输入流代码一致了。

 

服务器代码:

 1 package com.heyang;
 2 
 3 import java.io.BufferedReader;
 4 import java.io.IOException;
 5 import java.io.InputStream;
 6 import java.io.InputStreamReader;
 7 import java.io.OutputStream;
 8 import java.io.PrintWriter;
 9 import java.net.ServerSocket;
10 import java.net.Socket;
11 
12 /*
13  * 基于TCP协议的Socket通信,实现用户登录
14  * 服务器端
15  */
16 public class Server {
17     public static void main(String[] args) {
18         try {
19             // 1、创建一个服务器Socket,即ServerSocket,指定绑定的端口,并监听此端口
20             ServerSocket serverSocket = new ServerSocket(8888);
21             // 2、调用()方法开始监听,等待客户端的连接
22             System.out.println("***服务器即将启动,等待客户端的连接***");
23             Socket socket = serverSocket.accept();// 就会处于阻塞的状态,等待监听
24             // 3、获取输入流,病读取客户端信息
25             InputStream is = socket.getInputStream();// 字节输入流
26             // 将字节流转换为字符流
27             InputStreamReader isr = new InputStreamReader(is);
28             // 为输入流添加缓冲
29             BufferedReader br = new BufferedReader(isr);
30             String info = null;
31             while((info = br.readLine())!=null){
32                 System.out.println("我是服务器,读取客户端发过来的信息:"+info);
33             }
34             socket.shutdownInput();//关闭输入流
35             
36             // 4、作为服务器端,就需要响应客户端的请求,使用输出流来响应
37             OutputStream os = socket.getOutputStream();
38             PrintWriter pw = new PrintWriter(os);
39             pw.write("欢迎您!");
40             pw.flush();//调用flush()方法将缓冲输出
41             
42             
43             // 5、关闭资源
44             pw.close();
45             os.close();
46             br.close();
47             isr.close();
48             is.close();
49             socket.close();
50             serverSocket.close();
51             
52         } catch (IOException e) {
53             // TODO Auto-generated catch block
54             e.printStackTrace();
55         }
56     }
57 }

客户端代码:

 1 package com.heyang;
 2 
 3 import java.io.BufferedReader;
 4 import java.io.IOException;
 5 import java.io.InputStream;
 6 import java.io.InputStreamReader;
 7 import java.io.OutputStream;
 8 import java.io.PrintWriter;
 9 import java.net.Socket;
10 import java.net.UnknownHostException;
11 
12 /*
13  * 客户端
14  */
15 public class Client {
16     public static void main(String[] args) {
17         // 1、创建客户端Socket,指定服务器地址和端口
18         try {
19             Socket  socket = new Socket("localhost", 8888);
20             // 2、获取输出流,向服务器端发送信息
21             OutputStream os = socket.getOutputStream();// 获取字节输出流
22             // 将输出流包装为打印流
23             PrintWriter pw = new PrintWriter(os);
24             pw.write("用户名:admin 密码:123");
25             pw.flush();
26             socket.shutdownOutput();//关闭输出流
27             
28             // 3、获取输入流,并读取服务器端的响应信息
29             InputStream is = socket.getInputStream();
30             BufferedReader br = new BufferedReader(new InputStreamReader(is));
31             String info = null;
32             while((info = br.readLine())!=null){
33                 System.out.println("我是客户端,服务器跟我说:"+info);
34             }
35             
36             // 4、关闭资源
37             br.close();
38             is.close();
39             pw.close();
40             os.close();
41             socket.close();
42             
43         } catch (UnknownHostException e) {
44             // TODO Auto-generated catch block
45             e.printStackTrace();
46         } catch (IOException e) {
47             // TODO Auto-generated catch block
48             e.printStackTrace();
49         }
50     }
51 }

具体运行结果就不展示了。

 

4-4、使用多线程实现多客户端的通信

 事实上,以上只实现了单个客户端和单个服务器端进行socket通信。那么问题来了,实际应用程序是由一个服务器持续不断的运行中,然后由多个客户端异步通过服务器进行客户端之间的收发信息,这该如何实现呢?

 基本步骤:

  1、服务器端创建ServerSocket,循环调用accept()等待客户端连接。

  2、客户端创建一个socket并请求和服务器端连接。

  3、服务器端接受客户端请求,创建socket与该客户端建立专线连接。

  4、建立连接的两个socket在一个单独的线程上对话。

  5、服务器端继续等待新的连接。

服务器端的代码:

 1 package com.heyang;
 2 
 3 
 4 import java.io.IOException;
 5 import java.net.InetAddress;
 6 import java.net.ServerSocket;
 7 import java.net.Socket;
 8 import com.heyang.ServerThread;;
 9 /*
10  * 基于TCP协议的Socket通信,实现用户登录
11  * 服务器端
12  */
13 public class Server {
14     public static void main(String[] args) {
15         try {
16             // 1、创建一个服务器Socket,即ServerSocket,指定绑定的端口,并监听此端口
17             ServerSocket serverSocket = new ServerSocket(8888);
18             // 2、调用()方法开始监听,等待客户端的连接
19             
20             // 记录客户端的数量
21             int count = 0;
22             System.out.println("***服务器即将启动,等待客户端的连接***");
23             
24             while(true){
25                 // 调用accept()方法开始监听,等待客户端的链接
26                 Socket socket = serverSocket.accept();
27                 // 创建一个新的线程
28                 ServerThread serverThread = new ServerThread(socket);
29                 // 启动线程·
30                 serverThread.start();
31                 
32                 count++;
33                 System.out.println("客户端连接的数量:"+count+"个");
34                 
35                 // 获取客户端的IP地址等信息
36                 InetAddress address = socket.getInetAddress();
37                 System.out.println("当前客户端的IP:"+address.getHostAddress());
38                 
39             }
40             
41             // 需要死循环持续监听客户端的信息发送
42 //            serverSocket.close();
43             
44         } catch (IOException e) {
45             // TODO Auto-generated catch block
46             e.printStackTrace();
47         }
48     }
49 }

服务器线程代码:

 1 package com.heyang;
 2 
 3 import java.io.BufferedReader;
 4 import java.io.IOException;
 5 import java.io.InputStream;
 6 import java.io.InputStreamReader;
 7 import java.io.OutputStream;
 8 import java.io.PrintWriter;
 9 import java.net.Socket;
10 
11 /*
12  * 服务器端 线程处理类 
13  */
14 public class ServerThread extends Thread {
15     // 创建和本线程相关的socket
16     Socket socket = null;
17     
18     public ServerThread(Socket socket){
19         this.socket = socket;
20     }
21     
22     // 指向线程的操作,响应服务器端的请求
23     public void run(){
24         
25         InputStream is = null;
26         InputStreamReader isr = null;
27         BufferedReader br = null;
28         OutputStream os = null;
29         PrintWriter pw = null;
30         try {
31             // 3、获取输入流,病读取客户端信息
32             is = socket.getInputStream();// 字节输入流
33             // 将字节流转换为字符流
34             isr = new InputStreamReader(is);
35             // 为输入流添加缓冲
36             br = new BufferedReader(isr);
37             String info = null;
38             while ((info = br.readLine()) != null) {
39                 System.out.println("我是服务器,读取客户端发过来的信息:" + info);
40             }
41             socket.shutdownInput();//关闭输入流
42 
43             // 获取输出流
44             os = socket.getOutputStream();
45             pw = new PrintWriter(os);
46             pw.write("欢迎您!");
47             pw.flush();//调用flush()方法将缓冲输出
48         } catch (Exception e) {
49             // TODO: handle exception
50         }finally{
51             try {
52                 // 5、关闭资源
53                 if (pw != null) {
54                     pw.close();
55                 }
56                 if (os != null) {
57                     os.close();
58                 }
59                 if (br != null) {
60                     br.close();
61                 }
62                 if (isr != null) {
63                     isr.close();
64                 }
65                 if (is != null) {
66                     is.close();
67                 }
68                 if (socket != null) {
69                     socket.close();
70                 } 
71             } catch (IOException e2) {
72                 // TODO: handle exception
73             }
74         }
75         
76         
77     }
78     
79 }

客户端代码:

 1 package com.heyang;
 2 
 3 import java.io.BufferedReader;
 4 import java.io.IOException;
 5 import java.io.InputStream;
 6 import java.io.InputStreamReader;
 7 import java.io.OutputStream;
 8 import java.io.PrintWriter;
 9 import java.net.Socket;
10 import java.net.UnknownHostException;
11 
12 /*
13  * 客户端
14  */
15 public class Client {
16     public static void main(String[] args) {
17         // 1、创建客户端Socket,指定服务器地址和端口
18         try {
19             Socket  socket = new Socket("localhost", 8888);
20             // 2、获取输出流,向服务器端发送信息
21             OutputStream os = socket.getOutputStream();// 获取字节输出流
22             // 将输出流包装为打印流
23             PrintWriter pw = new PrintWriter(os);
24             pw.write("用户名:admin 密码:123");
25             pw.flush();
26             socket.shutdownOutput();//关闭输出流
27             
28             // 3、获取输入流,并读取服务器端的响应信息
29             InputStream is = socket.getInputStream();
30             BufferedReader br = new BufferedReader(new InputStreamReader(is));
31             String info = null;
32             while((info = br.readLine())!=null){
33                 System.out.println("我是客户端,服务器跟我说:"+info);
34             }
35             
36             // 4、关闭资源
37             br.close();
38             is.close();
39             pw.close();
40             os.close();
41             socket.close();
42             
43         } catch (UnknownHostException e) {
44             // TODO Auto-generated catch block
45             e.printStackTrace();
46         } catch (IOException e) {
47             // TODO Auto-generated catch block
48             e.printStackTrace();
49         }
50     }
51 }

运行结果不展示。


5、UDP编程

UDP协议(用户数据报协议)是无连接、不可靠的、无序的。

特点:传输速度相对比较快

UDP协议以数据报作为数据传输的载体

进行数据传输时,首先需要将要传输的数据定义成数据报(Datagram),在数据报中指明数据所要达到的Socket(主机地址和端口号),然后在将数据报发送出去。

 

相关操作的Java类

DatagramPacket:表示数据报包

DatagramSocket:进行端到端通信的类

 

 

 

5-1、编程实现基于UDP的Socket通信之服务器端

 

5-2、编程实现基于UDP的Socket通信之客户端

 

posted @ 2017-02-06 22:39  何杨  阅读(4312)  评论(0编辑  收藏  举报