一.通过Socket实现TCP编程

1.1 TCP编程

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

1.2 服务器端套路

  1.创建ServerSocket对象,绑定监听端口。

  2.通过accept()方法监听客户端请求。

  3.连接建立后,通过输入流读取客户端发送的请求信息。

  4.通过输出流向客户端发送响应信息。

  5.关闭响应的资源。

1.3 客户端套路

  1.创建Socket对象,指明需要连接的服务器的地址和端口号。

  2.连接建立后,通过输出流向服务器发送请求信息。

  3.通过输入流获取服务器响应的信息。

  4.关闭相应资源。

1.4 多线程实现服务器与多客户端之间通信步骤

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

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

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

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

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

1.5 创建处理线程类ServerThread

  这里选择实现runnable接口而不是继承Thread是因为一个类只能继承一个父类,当我需要继承其他类的时,父类就就不好处理了。

  

  1. package com.tzzh;
  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. public class ServerThread implements Runnable{
  12.  
  13. Socket socket = null;//和本线程相关的Socket
  14.  
  15. public ServerThread(Socket socket) {
  16. this.socket = socket;
  17. }
  18.  
  19. @Override
  20. public void run() {
  21. InputStream is = null;
  22. InputStreamReader isr = null;
  23. BufferedReader br = null;
  24. OutputStream os = null;
  25. PrintWriter pw = null;
  26. try {
  27. //与客户端建立通信,获取输入流,读取取客户端提供的信息
  28. is = socket.getInputStream();
  29. isr = new InputStreamReader(is,"GBK");
  30. br = new BufferedReader(isr);
  31. String data = null;
  32. while((data=br.readLine()) != null){//循环读取客户端的信息
  33. System.out.println("我是服务器,客户端提交信息为:"+data);
  34. }
  35. socket.shutdownInput();//关闭输入流
  36.  
  37. //获取输出流,响应客户端的请求
  38. os = socket.getOutputStream();
  39. pw = new PrintWriter(os);
  40. pw.write("服务器端响应成功!");
  41. pw.flush();
  42. } catch (IOException e) {
  43. e.printStackTrace();
  44. }finally {
  45. //关闭资源即相关socket
  46. try {
  47. if(pw!=null)
  48. pw.close();
  49. if(os!=null)
  50. os.close();
  51. if(br!=null)
  52. br.close();
  53. if(isr!=null)
  54. isr.close();
  55. if(is!=null)
  56. is.close();
  57. if(socket!=null)
  58. socket.close();
  59. } catch (IOException e) {
  60. e.printStackTrace();
  61. }
  62.  
  63. }
  64.  
  65. }
  66.  
  67. }

1.6 创建服务器端类

  使用while以达到可以循环侦听不同客户端的连接请求。因为这是一个死循环,所以不用关闭也没有机会去关闭serverSocket。设置count值,用于记录服务器端被连接过的次数并显示客户端所在ip值。如果线程处理类是继承Thread类,那么创建新线程代码可以改为ServerThread serverThread = new ServerThread(socket);serverThread.start();

  1. package com.tzzh;
  2.  
  3. import java.io.IOException;
  4. import java.net.InetAddress;
  5. import java.net.ServerSocket;
  6. import java.net.Socket;
  7.  
  8. public class Server {
  9. public static void main(String[] args) {
  10. try {
  11. //创建一个服务器端的Socket,即ServerSocket,绑定需要监听的端口
  12. ServerSocket serverSocket = new ServerSocket(8888);
  13. Socket socket = null;
  14. //记录连接过服务器的客户端数量
  15. int count = 0;
  16. System.out.println("***服务器即将启动,等待客户端的连接***");
  17. while(true){//循环侦听新的客户端的连接
  18. //调用accept()方法侦听,等待客户端的连接以获取Socket实例
  19. socket = serverSocket.accept();
  20. //创建新线程
  21. Thread thread = new Thread(new ServerThread(socket));
  22. thread.start();
  23.  
  24. count++;
  25. System.out.println("服务器端被连接过的次数:"+count);
  26. InetAddress address = socket.getInetAddress();
  27. System.out.println("当前客户端的IP为:"+address.getHostAddress());
  28. }
  29. //serverSocket.close();一直循环监听,不用关闭连接
  30. } catch (IOException e) {
  31. e.printStackTrace();
  32. }
  33. }
  34. }

1.7 创建客户端类

  在后面的关闭资源中,我把输入输出相关的流关闭注释了,是因为对于同一个Socket,关闭socket的时候也会把输入输出流关闭,直接关闭socket就行,当然保留也是可以的。

  1. package com.tzzh;
  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. public class Client {
  13. public static void main(String[] args) {
  14. try {
  15. //创建客户端Socket,指定服务器地址和端口
  16. Socket socket = new Socket("localhost", 8888);
  17. //建立连接后,获取输出流,向服务器端发送信息
  18. OutputStream os = socket.getOutputStream();
  19. //输出流包装为打印流
  20. PrintWriter pw = new PrintWriter(os);
  21. //向服务器端发送信息
  22. pw.write("用户名:zzh;密码:123");//写入内存缓冲区
  23. pw.flush();//刷新缓存,向服务器端输出信息
  24. socket.shutdownOutput();//关闭输出流
  25.  
  26. //获取输入流,接收服务器端响应信息
  27. InputStream is = socket.getInputStream();
  28. BufferedReader br = new BufferedReader(new InputStreamReader(is, "GBK"));
  29. String data = null;
  30. while((data=br.readLine())!= null){
  31. System.out.println("我是客户端,服务器端提交信息为:"+data);
  32. }
  33.  
  34. //关闭其他资源
  35. // br.close();
  36. // is.close();
  37. // pw.close();
  38. // os.close();
  39. socket.close();
  40.  
  41. } catch (UnknownHostException e) {
  42. e.printStackTrace();
  43. } catch (IOException e) {
  44. e.printStackTrace();
  45. }
  46. }
  47. }

1.8 先运行服务器端,在运行客户端

此时在看服务器控制台:服务器端一直在循环侦听客户端的连接

1.9 进行第二个客户端的连接

  修改相应信息将用户名zzh改为admin。运行客户端,打开服务端控制台

  

输出客户端的ip都为127.0.0.1,是因为服务器和客户端都是本机,在真实的环境中会显示客户端的ip地址信息。

二. 通过Socket实现UDP编程

2.1 UDP编程

  UDP协议又叫用户数据报协议,是无连接,不可靠的,无序的。特点是传输速度相对要快,UDP协议以数据报作为数据传输的载体。当进行数据传输时,首先需要将要传输的数据定义成数据报(Datagram),在数据报中指明数据所要达到的Socket(主机地址和端口号),然后再将数据报发送出去。相关操作类有:DatagramPacket数据报包,DatagramSocket进行端到端通信的类。

2.2 服务器端实现套路

  1.创建DatagramSocket,指定端口号。2.创建DatagramPacket。3.接收客户端发送的数据信息。4.读取数据。

2.3 客户端实现套路

  1.定义发送信息,比如发送地址,端口号和内容。2. 创建DatagramPacket,包含将要发送的信息。3.创建DatagramSocket。4.发送数据。

2.4 多线程实现服务器与多客户端之间通信步骤

  1.服务器端创建DatagramSocket的实例socket,循环调用receive()方法,此方法在接收到数据报之前会一直阻塞。

  2.客户端创建DatagramSocket,将含有地址,端口号和内容的数据报包发送出去。

  3. 服务器端收到数据报包packet,通过DatagramSocket和packet与客户端建立一个线程

  4. 服务器端继续等待新的数据报包。

  5. 发送方的DatagramPacket构造方法传递四个参数包含数据内容,数据大小,地址和端口号。接收方的DatagramPacket构造方法有两个参数接收数据和数据大小。

2.5 创建服务器线程处理类UDPThread

  注意,DatagramSocket的实例socket不能关闭,会出现SocketException。读取数据用到的new String(packet.getData(), 0, packet.getLength()),参数表示数据报中的字节数组,位置和长度。

  1. package com.uzzh;
  2.  
  3. import java.io.IOException;
  4. import java.net.DatagramPacket;
  5. import java.net.DatagramSocket;
  6. import java.net.InetAddress;
  7.  
  8. public class UDPThread implements Runnable{
  9.  
  10. DatagramSocket socket = null;
  11. DatagramPacket packet = null;
  12.  
  13. public UDPThread(DatagramSocket socket,DatagramPacket packet) {
  14. this.socket = socket;
  15. this.packet = packet;
  16. }
  17.  
  18. @Override
  19. public void run() {
  20. String info = null;
  21. InetAddress address = null;
  22. int port = 8800;
  23. byte[] data2 = null;
  24. DatagramPacket packet2 = null;
  25. try {
  26. info = new String(packet.getData(), 0, packet.getLength());
  27. System.out.println("我是服务器,客户端说:"+info);
  28.  
  29. address = packet.getAddress();
  30. port = packet.getPort();
  31. data2 = "我在响应你!".getBytes();
  32. packet2 = new DatagramPacket(data2, data2.length, address, port);
  33. socket.send(packet2);
  34. } catch (IOException e) {
  35. e.printStackTrace();
  36. }
  37. //socket.close();不能关闭
  38. }
  39.  
  40. }

2.6 创建服务器端类

  

  1. package com.uzzh;
  2.  
  3. import java.io.IOException;
  4. import java.net.DatagramPacket;
  5. import java.net.DatagramSocket;
  6. import java.net.InetAddress;
  7.  
  8. public class UDPServer {
  9. public static void main(String[] args) throws IOException {
  10. DatagramSocket socket = new DatagramSocket(8800);
  11. DatagramPacket packet = null;
  12. byte[] data = null;
  13. int count = 0;
  14. System.out.println("***服务器端启动,等待发送数据***");
  15. while(true){
  16. data = new byte[1024];//创建字节数组,指定接收的数据包的大小
  17. packet = new DatagramPacket(data, data.length);
  18. socket.receive(packet);//此方法在接收到数据报之前会一直阻塞
  19. Thread thread = new Thread(new UDPThread(socket, packet));
  20. thread.start();
  21. count++;
  22. System.out.println("服务器端被连接过的次数:"+count);
  23. InetAddress address = packet.getAddress();
  24. System.out.println("当前客户端的IP为:"+address.getHostAddress());
  25.  
  26. }
  27.  
  28. }
  29. }

之前我将new DatagramSocket放入了while循环中,报了java.net.BindException: Address already in use: Cannot bind,才知道不能在while中连续创建新的DatagramSocket对象。

2.7 创建客户端类

  1. package com.uzzh;
  2.  
  3. import java.io.IOException;
  4. import java.net.DatagramPacket;
  5. import java.net.DatagramSocket;
  6. import java.net.InetAddress;
  7.  
  8. public class UDPClient {
  9. public static void main(String[] args) throws IOException {
  10. //定义服务器的地址,端口号,数据
  11. InetAddress address = InetAddress.getByName("localhost");
  12. int port = 8800;
  13. byte[] data = "用户名:admin;密码:123".getBytes();//将字符串转换为字节数组
  14. //创建数据报
  15. DatagramPacket packet = new DatagramPacket(data, data.length, address, port);
  16. //创建DatagramSocket,实现数据发送和接收
  17. DatagramSocket socket = new DatagramSocket();
  18. //向服务器端发送数据报
  19. socket.send(packet);
  20.  
  21. //接收服务器响应数据
  22. byte[] data2 = new byte[1024];
  23. DatagramPacket packet2 = new DatagramPacket(data2, data2.length);
  24. socket.receive(packet2);
  25. String info = new String(data2, 0, packet2.getLength());
  26. System.out.println("我是客户端,服务器说:"+info);
  27. socket.close();
  28. }
  29. }

2.8 先运行服务器端,在运行客户端

2.9 修改客户端信息,再次运行客户端

服务器控制台:服务器端一直在循环等待接收客户端的数据。

三. 总结

  这两个例子只是简单的实现了基于TCP和UDP的socket编程,其中像多线程的优先级等都暂且没做考虑,不过依然要强调一下,服务器与多个客户端进行通信,因为是死循环,不设置多线程优先级,可能会导致运行时速度非常慢,优先级的范围1-10,默认为5,我们可以适当降低线程的优先级,比如thread.setPriority(4);

  对于同一个socket,如果关闭了输出流比如(pw.close()),则与该输出流关联的socket也会关闭,所以一般不需要关闭输出流,当关闭socket的时候,输出流也会关闭,直接关闭socket就行。

  在使用TCP通信传输信息时,更多是使用对象的形式来传输,可以使用ObjectOutputStream对象序列化流来传递对象,比如ObjectOutputStream os = new ObjectOutputStream(socket.getOutputStream());User user = new User("admin","123"); os.writeObject(user);

  希望这篇文章能让你有所获,麻烦点赞或关注我,谢谢观看!

Java Socket实现基于TCP和UDP多线程通信的更多相关文章

  1. java Socket编程-基于TCP

    package com.wzy.Test; import java.io.BufferedReader; import java.io.IOException; import java.io.Inpu ...

  2. 基于TCP与UDP协议的socket通信

    基于TCP与UDP协议的socket通信 C/S架构与初识socket 在开始socket介绍之前,得先知道一个Client端/服务端架构,也就是 C/S 架构,互联网中处处充满了 C/S 架构(Cl ...

  3. 基于TCP和UDP的socket

    为什么学习socket 你自己现在完全可以写一些小程序了,但是前面的学习和练习,我们写的代码都是在自己的电脑上运行的,虽然我们学过了模块引入,文件引入import等等,我可以在程序中获取到另一个文件的 ...

  4. Python网络编程02 /基于TCP、UDP协议的socket简单的通信、字符串转bytes类型

    Python网络编程02 /基于TCP.UDP协议的socket简单的通信.字符串转bytes类型 目录 Python网络编程02 /基于TCP.UDP协议的socket简单的通信.字符串转bytes ...

  5. Java网络编程之TCP、UDP

    Java网络编程之TCP.UDP 2014-11-25 15:23 513人阅读 评论(0) 收藏 举报 分类: java基础及多线程(28) 版权声明:本文为博主原创文章,未经博主允许不得转载.   ...

  6. Java 网络编程 -- 基于TCP 模拟多用户登录

    Java TCP的基本操作参考前一篇:Java 网络编程 – 基于TCP实现文件上传 实现多用户操作之前先实现以下单用户操作,假设目前有一个用户: 账号:zs 密码:123 服务端: public c ...

  7. 计算机网络(十三),Socket编程实现TCP和UDP

    十三.Socket编程实现TCP和UDP 1.TCP (1)TCPServer.java类 package com.interview.javabasic.socket; import com.int ...

  8. socket通讯,TCP,UDP,HTTP的区别

    socket编程有TCP和UDP, TCP:传送控制协议(Transmission Control Protocol) 传输控制协议TCP是TCP/IP协议栈中的传输层协议,它通过序列确认以及包重发机 ...

  9. 实现TCP、UDP相互通信及应用

    实验名称  Socket编程综合实验(1) 一.实验目的: 1.理解进程通信的原理及通信过程 2.掌握基于TCP和UDP的工作原理 3.掌握基本的Socket网络编程原理及方法 二.实验内容 1.掌握 ...

随机推荐

  1. JS截取字符串常用方法详细整理&&MYSQL

    截取字符串的使用比较广泛,有很多中方法,本文粗略的整理了一些,感兴趣的额朋友可以才参考下 使用 substring()或者slice() 函数:split() 功能:使用一个指定的分隔符把一个字符串分 ...

  2. Ngnix常用的操作

    Nginx的常用参数如下: # /usr/local/nginx/sbin/nginx -h nginx version: nginx/0.7.63 Usage: nginx [-?hvVt] [-s ...

  3. 爬虫之requests与bautifullSoup

    requests Python标准库中提供了:urllib.urllib2.httplib等模块以供Http请求,但是,它的 API 太渣了.它是为另一个时代.另一个互联网所创建的.它需要巨量的工作, ...

  4. Educational Codeforces Round 37-G.List Of Integers题解

    一.题目 二.题目链接 http://codeforces.com/contest/920/problem/G 三.题意 给定一个$t$,表示有t次查询.每次查询给定一个$x$, $p$, $k$,需 ...

  5. memcached配置 (初级)以及测试

    一.memcached安装 memcached依赖 $ sudo apt-get install libevent-dev   安装memcached服务 $ sudo apt-get install ...

  6. oracle账户密码更新

    oracle忘记用户名密码怎样恢复 首先cmd - sqlplusw 普通用户登陆:用户名:scott(普通用户名)      密码:tiger(普通用户密码) 管理员登陆:用户名:system 密码 ...

  7. ES6系列_11之Set和WeakSet数据结构

    一.Set 1.Set是什么? Set是ES6 提供的一种新的数据结构.类似于数组. 2.Set能解决什么问题 Set和Array 的区别是Set不允许内部有重复的值,如果有只显示一个,相当于去重. ...

  8. C# JSON 序列化

    1.JavaScriptSerializer System.Web.Extensions.dll System.Web.Script.Serialization命名空间 Serialize Deser ...

  9. TI davinci DM6467通过串口0将UBL和u-boot写入NAND flash

    TI的davinci系列一般支持好几种启动模式,如下图TMS320DM6467的datasheet可以查到所有的BOOTMODE LVS301和LW9226的开发板上有一个选择bootmode的拨码开 ...

  10. artZoom 图片可放大旋转

    资源下载:http://www.sucaijiayuan.com/Js/TuPianDaiMa/774.html 第一步:引入文件 <link rel="stylesheet" ...