InetAddress类

  • 此类表示Internet协议(IP)地址

    InetAddress API如下:

  1. IP地址的代表类是谁?
    1. InetAddress类
  2. 如何获取本机IP对象
    1. public static InetAddress getLocalHost();
  3. 如何判断与该IP地址对象是否互通
    1. public boolean isReachable(int timeout);

测试代码:

  1. package com.h;
  2. import java.io.IOException;
  3. import java.net.InetAddress;
  4. import java.net.UnknownHostException;
  5. /*
  6. * 网络编程
  7. * */
  8. public class InetAddressDemo01 {
  9. public static void main(String[] args) throws IOException {
  10. //1、 获取本机地址对象
  11. InetAddress address = InetAddress.getLocalHost();
  12. //2、获取本地主机名与地址
  13. System.out.println(address.getHostName());
  14. System.out.println(address.getHostAddress());
  15. //3、获取域名ip地址,前提是电脑联网或者有dns缓存
  16. InetAddress address1 = InetAddress.getByName("www.baidu.com");
  17. System.out.println(address1.getHostAddress());
  18. System.out.println(address1.getHostName());
  19. //4、获取公网IP对象
  20. InetAddress address2 = InetAddress.getByName("110.242.68.3");
  21. System.out.println(address2.getHostName());
  22. System.out.println(address2.getHostAddress());
  23. //5、判断是否能通:ping 5s之前测试是否可通
  24. System.out.println(address2.isReachable(5000)); //返回true
  25. }
  26. }

端口:

端口号:范围0~65535
端口号的作用

  • 唯一标识正在计算机设备上运行的进程(程序)

一个设备中,能否出现2个应用程序的端口号一样,为什么?

  • 不可以,如果一样会出现端口冲突错误

传输层的2个常见协议

  • TCP:传输控制协议
  • UDP:用户数据报协议

TCP协议特点

  • 使用TCP协议,必须双方先建立连接,它是一种面向连接的可靠通信协议
  • 传输前,采用“三次握手”方式建立连接,所以是可靠的
  • 在连接中可进行大数据量的传输
  • 连接,发送数据都需要确认,且传输完毕后,还需释放已建立的连接,通信效率较低

TCP协议通信场景

  • 对信息安全要求较高的场景,例如:文件下载、金融等数据通信

三次握手
客户端 ——> 服务器:客户端向服务器发出连接请求,等待服务器确认
服务器 ——> 客户端:服务器向客户端返回一个响应,告诉客户端收到了请求
客户端 ——> 服务器:客户端向服务器再次发出确认信息,连接建立
TCP四次挥手断开连接
客户端 ——> 服务器:客户端向服务器发出取消连接请求
…………..
UDP协议

  • UDP是一种无连接、不可靠传输的协议
  • 语音通话,视频会话等。

TCP必须先建立连接才能通信,UDP不需要建立连接

UDP

DatagramPacket:数据包对象(韭菜盘子)

  • 构造器:public DatagramPacket(byte[] buf, int length, InetAddress address, int port);
    • 创建发送端数据包对象;
    • buf:要发送的内容,字节数组;
    • length:要发送内容的字节长度;
    • address:接收端的IP地址对象;
    • port:接收端口号

DatagramSocket:发送端和接收端对象(人)

  • 构造器:public DatagramSocket();创建发送端的Socket对象,系统会随机分配一个端口号
  • 构造器2:public DatagramSocket(int port):创建接收端的Socket对象并指定端口号
  • send(DatagramPacket dp);发送数据包
  • receive(DatagramPacket p);接收数据包

一发一收

  1. 发送端
  2. package com.h.UDP;
  3. import java.net.*;
  4. import java.nio.charset.StandardCharsets;
  5. /*
  6. * 发送端
  7. * */
  8. public class ClientDemo1 {
  9. public static void main(String[] args) throws Exception {
  10. //1、创建发送端对象:发送端自带默认端口号。不填7777就是默认端口填7777是指定端口
  11. DatagramSocket socket = new DatagramSocket(7777);
  12. //2、创建一个数据包对象封装数据(韭菜盘子)
  13. //参数
  14. //参数一:封装要发送的数据(韭菜) byte[]
  15. //参数二:数据大小 int length
  16. //参数三:服务端IP地址 对方的ip
  17. //参数四:服务端的端口 对方的端口
  18. byte[] buff = "我是韭菜".getBytes();
  19. //这个8888端口是将数据包发送给服务端的socket端口必须一样
  20. DatagramPacket packet = new DatagramPacket(buff,4,
  21. InetAddress.getLocalHost() ,8888);
  22. //3、发送数据出去
  23. socket.send(packet);
  24. socket.close();
  25. }
  26. }
  27. 接收端
  28. package com.h.UDP;
  29. import java.net.DatagramPacket;
  30. import java.net.DatagramSocket;
  31. import java.net.SocketException;
  32. /*
  33. * 接收端 服务端
  34. * */
  35. public class ServerDemo2 {
  36. public static void main(String[] args) throws Exception {
  37. System.out.println("===服务端启动===");
  38. //1、创建接收端对象:注册端口(人) 这里的Socket8888端口必须和数据包里面的端口一致
  39. DatagramSocket socket = new DatagramSocket(8888);
  40. //2、创建一个数据包对象接数据
  41. byte[] buff = new byte[1024 * 64];
  42. DatagramPacket packet = new DatagramPacket(buff,buff.length);
  43. //3、接收数据
  44. socket.receive(packet);
  45. //4、取出数据
  46. //读多少倒多少
  47. int len = packet.getLength();
  48. String str = new String(buff,0,len);
  49. //获取发送端的ip
  50. String address = packet.getSocketAddress().toString();
  51. //这个端口是对方默认端口
  52. int port = packet.getPort();
  53. System.out.println(str+" "+port+" "+address);
  54. socket.close();
  55. }
  56. }

广播:
当前主机与所在网络中所有主机通信,广播消息是可以发给当前所在网络中的所有服务器
广播ip:255.255.255.255

  1. package com.h.udp;
  2. import java.net.*;
  3. /*
  4. * 发送端
  5. * */
  6. public class ClientDemo1 {
  7. public static void main(String[] args) throws Exception {
  8. //1、创建发送端对象:发送端自带默认端口号。
  9. DatagramSocket socket = new DatagramSocket();
  10. //2、创建一个数据包对象封装数据(韭菜盘子)
  11. //参数
  12. //参数一:封装要发送的数据(韭菜) byte[]
  13. //参数二:数据大小 int length
  14. //参数三:服务端IP地址 对方的ip
  15. //参数四:服务端的端口 对方的端口
  16. byte[] buff = "我是韭菜".getBytes();
  17. //这个8888端口是将数据包发送给服务端的socket端口必须一样
  18. //广播IP 255.255.255.255
  19. DatagramPacket packet = new DatagramPacket(buff,4,
  20. InetAddress.getByName("255.255.255.255") ,8888);
  21. //3、发送数据出去
  22. socket.send(packet);
  23. socket.close();
  24. }
  25. }
  26. //接收端
  27. package com.h.udp;
  28. import java.net.DatagramPacket;
  29. import java.net.DatagramSocket;
  30. /*
  31. * 接收端 服务端
  32. * */
  33. public class ServerDemo2 {
  34. public static void main(String[] args) throws Exception {
  35. System.out.println("===服务端启动===");
  36. //1、创建接收端对象:注册端口(人) 这里的Socket8888端口必须和数据包里面的端口一致
  37. DatagramSocket socket = new DatagramSocket(8888);
  38. //2、创建一个数据包对象接数据
  39. byte[] buff = new byte[1024 * 64];
  40. DatagramPacket packet = new DatagramPacket(buff,buff.length);
  41. //3、接收数据
  42. socket.receive(packet);
  43. //4、取出数据
  44. //读多少倒多少
  45. int len = packet.getLength();
  46. String str = new String(buff,0,len);
  47. //获取发送端的ip
  48. String address = packet.getSocketAddress().toString();
  49. //这个端口是对方默认端口
  50. int port = packet.getPort();
  51. System.out.println(str+" "+port+" "+address);
  52. socket.close();
  53. }
  54. }

组播:

  • 使用组播地址:224.0.0.0 ~ 239.255.255.255
  • 具体操作
    • 发送端的数据包的目的地是组播IP(例如:224.0.1.1,端口9999)
    • 接收端必须绑定该组播IP(224.0.1.1),端口要对应发送端的端口9999,这样即可接收消息
    • DatagramSocket的子类MulticastSocket可以在接收端绑定组播IP

组播是可以选择性的发送给指定ip的服务器

  1. //发送端
  2. package com.h.udp_multicast;
  3. import java.net.*;
  4. public class Cline {
  5. public static void main(String[] args) {
  6. try {
  7. //创建发送器
  8. DatagramSocket d = new DatagramSocket();
  9. byte[] buffer = "组播".getBytes();
  10. //创建数据包
  11. //组播ip为224.0.0.0 ~ 239.255.255.255
  12. DatagramPacket dp = new DatagramPacket(buffer,buffer.length, InetAddress.getByName("224.0.1.1"),8888);
  13. d.send(dp);
  14. d.close();
  15. } catch (Exception e) {
  16. e.printStackTrace();
  17. }
  18. }
  19. }
  20. //接收端
  21. package com.h.udp_multicast;
  22. import java.net.*;
  23. /*
  24. * 接收端 服务端
  25. * */
  26. public class Server {
  27. public static void main(String[] args) throws Exception {
  28. System.out.println("===服务端启动===");
  29. //组播主要是需要MulticastSocket这个对象
  30. //创建接收端对象 注册端口
  31. MulticastSocket socket = new MulticastSocket(8888);
  32. //当前接收端加入到一个组播组里面去:绑定对应的组播ip
  33. // socket.joinGroup(InetAddress.getByName("224.0.1.1")); 过时的写法
  34. //使用新的写法
  35. socket.joinGroup(new InetSocketAddress(InetAddress.getByName("224.0.1.1"),8888),
  36. NetworkInterface.getByInetAddress(InetAddress.getLocalHost()));
  37. //2、创建一个数据包对象接数据
  38. byte[] buff = new byte[1024 * 64];
  39. DatagramPacket packet = new DatagramPacket(buff,buff.length);
  40. //3、接收数据
  41. socket.receive(packet);
  42. //4、取出数据
  43. //读多少倒多少
  44. int len = packet.getLength();
  45. String str = new String(buff,0,len);
  46. //获取发送端的ip
  47. String address = packet.getSocketAddress().toString();
  48. //这个端口是对方默认端口
  49. int port = packet.getPort();
  50. System.out.println(str+" "+port+" "+address);
  51. socket.close();
  52. }
  53. }

多发多收

  1. package com.h.udp2;
  2. import java.net.DatagramPacket;
  3. import java.net.DatagramSocket;
  4. import java.net.InetAddress;
  5. import java.util.Scanner;
  6. /*
  7. * 发送端 多发多收
  8. * */
  9. public class ClientDemo1 {
  10. public static void main(String[] args) throws Exception {
  11. //1、创建发送端对象:发送端自带默认端口号。
  12. DatagramSocket socket = new DatagramSocket();
  13. //2、创建一个数据包对象封装数据(韭菜盘子)
  14. //参数
  15. //参数一:封装要发送的数据(韭菜) byte[]
  16. //参数二:数据大小 int length
  17. //参数三:服务端IP地址 对方的ip
  18. //参数四:服务端的端口 对方的端口
  19. Scanner sc = new Scanner(System.in);
  20. while (true) {
  21. System.out.println("请说:");
  22. String s = sc.nextLine();
  23. if(s.equals("exit")){
  24. System.out.println("离线成功");
  25. socket.close();
  26. break;
  27. }
  28. byte[] buff = s.getBytes();
  29. //这个8888端口是将数据包发送给服务端的socket端口必须一样
  30. DatagramPacket packet = new DatagramPacket(buff, buff.length,
  31. InetAddress.getLocalHost(), 8888);
  32. //3、发送数据出去
  33. socket.send(packet);
  34. }
  35. }
  36. }
  37. 接收端
  38. package com.h.udp2;
  39. import java.net.DatagramPacket;
  40. import java.net.DatagramSocket;
  41. import java.util.Scanner;
  42. /*
  43. * 接收端 服务端 多发多收
  44. * */
  45. public class ServerDemo2 {
  46. public static void main(String[] args) throws Exception {
  47. System.out.println("===服务端启动===");
  48. //1、创建接收端对象:注册端口(人) 这里的Socket8888端口必须和数据包里面的端口一致
  49. DatagramSocket socket = new DatagramSocket(8888);
  50. byte[] buff = new byte[1024 * 64];
  51. DatagramPacket packet = new DatagramPacket(buff, buff.length);
  52. while (true) {
  53. socket.receive(packet);
  54. //收到多收数据
  55. int len = packet.getLength();
  56. System.out.println("收到了来自:"+packet.getPort()+
  57. packet.getSocketAddress()+"的数据"+
  58. new String(buff,0,len));
  59. }
  60. }
  61. }

基础版互相聊天:使用线程来来完成

  1. package com.h.udp2;
  2. import java.net.DatagramPacket;
  3. import java.net.DatagramSocket;
  4. import java.net.InetAddress;
  5. import java.net.SocketException;
  6. import java.util.Scanner;
  7. /*
  8. * 发送端 多发多收
  9. * */
  10. public class ClientDemo1 {
  11. public static void main(String[] args) throws Exception {
  12. //1、创建发送端对象:发送端自带默认端口号。
  13. //2、创建一个数据包对象封装数据(韭菜盘子)
  14. //参数
  15. //参数一:封装要发送的数据(韭菜) byte[]
  16. //参数二:数据大小 int length
  17. //参数三:服务端IP地址 对方的ip
  18. //参数四:服务端的端口 对方的端口
  19. new Thread() {
  20. @Override
  21. public void run() {
  22. Scanner sc = new Scanner(System.in);
  23. DatagramSocket socket = null;
  24. try {
  25. socket = new DatagramSocket();
  26. while (true) {
  27. System.out.println("请说:");
  28. String s = sc.nextLine();
  29. if (s.equals("exit")) {
  30. System.out.println("离线成功");
  31. socket.close();
  32. break;
  33. }
  34. byte[] buff = s.getBytes();
  35. //这个8888端口是将数据包发送给服务端的socket端口必须一样
  36. DatagramPacket packet = new DatagramPacket(buff, buff.length,
  37. InetAddress.getLocalHost(), 7777);
  38. //3、发送数据出去
  39. socket.send(packet);
  40. }
  41. } catch (Exception e) {
  42. e.printStackTrace();
  43. }
  44. }
  45. }.start();
  46. new Thread(){
  47. @Override
  48. public void run() {
  49. System.out.println("===服务端启动===");
  50. //1、创建接收端对象:注册端口(人) 这里的Socket8888端口必须和数据包里面的端口一致
  51. DatagramSocket socket = null;
  52. try {
  53. socket = new DatagramSocket(8888);
  54. byte[] buff = new byte[1024 * 64];
  55. DatagramPacket packet = new DatagramPacket(buff, buff.length);
  56. while (true) {
  57. //等待接收的数据
  58. socket.receive(packet);
  59. //收到多收数据
  60. int len = packet.getLength();
  61. System.out.println("收到了来自:"+packet.getPort()+
  62. packet.getSocketAddress()+"的数据"+
  63. new String(buff,0,len));
  64. }
  65. } catch (Exception e) {
  66. e.printStackTrace();
  67. }
  68. }
  69. }.start();
  70. }
  71. }
  72. 第二个端
  73. package com.h.udp2;
  74. import java.net.DatagramPacket;
  75. import java.net.DatagramSocket;
  76. import java.net.InetAddress;
  77. import java.net.SocketException;
  78. import java.util.Scanner;
  79. /*
  80. * 接收端 服务端 多发多收
  81. * */
  82. public class ServerDemo2 {
  83. public static void main(String[] args) throws Exception {
  84. new Thread() {
  85. @Override
  86. public void run() {
  87. Scanner sc = new Scanner(System.in);
  88. DatagramSocket socket = null;
  89. try {
  90. socket = new DatagramSocket();
  91. while (true) {
  92. System.out.println("请说:");
  93. String s = sc.nextLine();
  94. if (s.equals("exit")) {
  95. System.out.println("离线成功");
  96. socket.close();
  97. break;
  98. }
  99. byte[] buff = s.getBytes();
  100. //这个8888端口是将数据包发送给服务端的socket端口必须一样
  101. DatagramPacket packet = new DatagramPacket(buff, buff.length,
  102. InetAddress.getLocalHost(), 8888);
  103. //3、发送数据出去
  104. socket.send(packet);
  105. }
  106. } catch (Exception e) {
  107. e.printStackTrace();
  108. }
  109. }
  110. }.start();
  111. new Thread(){
  112. @Override
  113. public void run() {
  114. System.out.println("===服务端启动===");
  115. //1、创建接收端对象:注册端口(人) 这里的Socket8888端口必须和数据包里面的端口一致
  116. DatagramSocket socket = null;
  117. try {
  118. socket = new DatagramSocket(7777);
  119. byte[] buff = new byte[1024 * 64];
  120. DatagramPacket packet = new DatagramPacket(buff, buff.length);
  121. while (true) {
  122. //等待接收的数据
  123. socket.receive(packet);
  124. //收到多收数据
  125. int len = packet.getLength();
  126. System.out.println("收到了来自:"+packet.getPort()+
  127. packet.getSocketAddress()+"的数据"+
  128. new String(buff,0,len));
  129. }
  130. } catch (Exception e) {
  131. e.printStackTrace();
  132. }
  133. }
  134. }.start();
  135. }
  136. }

UDP广播,组播

TCP:

三次握手,
客户端和服务端通信之间,双方会建立一个连接管道,Socket
TCP是通过流来进行实现消息的传递
image.png
Socket :
构造器:public Socket(String host,int port);创建一个发送端的Socket对象与服务端连接,参数为服务端程序的ip地址和端口
成员方法:

  • OutputStream getOutputStream();获取字节输出流对象
  • InputStream getInputStream();获取输入流对象

    客户端发送信息

    需求:客户端实现步骤
  1. 创建客户端的Socket对象,请求与服务端的连接
  2. 使用socket对象调用getOutputStream()方法得到一个字节输出流
  3. 使用字节输出流完成数据的发送
  4. 释放资源:关闭资源

现在服务端为什么不可以接收多个客户端

  • TCP服务端是单线程一次只能处理一个客户端所以不能实现一个服务端接收多个客户端

TCP代码实现
一发一收:

  1. //服务端
  2. package com.h.tcp;
  3. import java.io.BufferedReader;
  4. import java.io.InputStream;
  5. import java.io.InputStreamReader;
  6. import java.io.PrintStream;
  7. import java.net.ServerSocket;
  8. import java.net.Socket;
  9. /*
  10. * 服务端
  11. * */
  12. public class TcpServer {
  13. public static void main(String[] args) {
  14. System.out.println("服务端启动");
  15. try (
  16. ServerSocket ss = new ServerSocket(7777);
  17. ) {
  18. //阻塞等待客户端连接
  19. Socket socket = ss.accept();
  20. //如果阻塞在IO后面的话就会出现编译错误,因为阻塞是在等客户端的流
  21. InputStream in = socket.getInputStream();
  22. //使用高级字符流来获取文本信息 使用了转换流
  23. BufferedReader br = new BufferedReader(new InputStreamReader(in));
  24. String str ;
  25. //对方挂了server也挂 实现了一发一收
  26. if ((str = br.readLine())!=null) {
  27. System.out.println(socket.getRemoteSocketAddress()+"说了:"+str);
  28. }
  29. } catch (Exception e) {
  30. e.printStackTrace();
  31. }
  32. }
  33. }
  34. //接收端
  35. package com.h.tcp;
  36. import java.io.FileOutputStream;
  37. import java.io.OutputStream;
  38. import java.io.PrintStream;
  39. import java.net.InetAddress;
  40. import java.net.Socket;
  41. import java.net.UnknownHostException;
  42. /*
  43. * 客户端
  44. * */
  45. public class Tcpclient {
  46. public static void main(String[] args) {
  47. try{
  48. //参数一、服务端ip地址
  49. //参数二、服务端端口号
  50. // 10.50.128.211
  51. Socket socket = new Socket("10.50.128.211",7777);
  52. //从socket中得到一个字节输出流,负责发送消息
  53. //把低级的字节流包装成打印流
  54. PrintStream ps = new PrintStream(out);
  55. // //发送消息
  56. ps.println("你好TCP服务器,我是客户端");
  57. ps.flush();
  58. //关闭资源
  59. socket.close();
  60. }catch (Exception e){
  61. }
  62. }
  63. }

TCP多发多收

  1. package com.h.tcp2;
  2. import java.io.BufferedReader;
  3. import java.io.InputStream;
  4. import java.io.InputStreamReader;
  5. import java.net.ServerSocket;
  6. import java.net.Socket;
  7. /*
  8. * 服务端
  9. * */
  10. public class TcpServer {
  11. public static void main(String[] args) {
  12. System.out.println("服务端启动");
  13. try (
  14. ServerSocket ss = new ServerSocket(7777);
  15. ) {
  16. //阻塞等待客户端连接
  17. Socket socket = ss.accept();
  18. //如果阻塞在IO后面的话就会出现编译错误,因为阻塞是在等客户端的流
  19. InputStream in = socket.getInputStream();
  20. //使用高级字符流来获取文本信息 使用了转换流
  21. BufferedReader br = new BufferedReader(new InputStreamReader(in));
  22. String str;
  23. //每次接收到了客户端后就会一直在这等消息
  24. while ((str = br.readLine()) != null) {
  25. System.out.println(socket.getRemoteSocketAddress() + "说了:" + str);
  26. }
  27. } catch (Exception e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }
  32. package com.h.tcp2;
  33. import java.io.OutputStream;
  34. import java.io.PrintStream;
  35. import java.net.Socket;
  36. import java.util.Scanner;
  37. /*
  38. 多发多收
  39. * 客户端
  40. * */
  41. public class Tcpclient {
  42. public static void main(String[] args) {
  43. try {
  44. //创建用户输入类
  45. Scanner sc = new Scanner(System.in);
  46. //参数一、服务端ip地址
  47. //参数二、服务端端口号
  48. // 10.50.128.211
  49. Socket socket = new Socket("10.50.128.214", 7777);
  50. //从socket中得到一个字节输出流,负责发送消息
  51. OutputStream out = socket.getOutputStream();
  52. //把低级的字节流包装成打印流 这里只是把低级的流进行了包装后再发过去
  53. PrintStream ps = new PrintStream(out);
  54. while (true) {
  55. //发送消息
  56. System.out.println("请输入你要发送的内容");
  57. String s = sc.nextLine();
  58. //发送内容
  59. ps.println(s);
  60. ps.flush();
  61. }
  62. } catch (Exception e) {
  63. }
  64. }
  65. }

完成一个服务端同时处理多个客户端消息

  • 使用线程实现了一个服务端接收多个客户端
  • Socket.accept()这里阻塞直到获取一个客户端后代码才会继续向下执行
  • 获取到了客户端后接下来就会在read()方法哪里等待可以客户端输入内容过来,会一直在获取客户端内容
  • 所以用while来创建多个来接收多个客户端的,一个线程处理一个客户端
  • 这样一来就实现了一个服务端接收多个客户端信息

代码:

  1. //服务端
  2. System.out.println("=====服务端启动=====");
  3. try {
  4. ServerSocket ss = new ServerSocket(7777);
  5. while (true) {
  6. //每次都会接收到客户端电的Socket管道,交给一个独立线程,会在这里阻塞
  7. Socket socket = ss.accept();
  8. //开始创建独立线程处理socket
  9. new ThreadT(socket).start();
  10. }
  11. //线程
  12. public class ThreadT extends Thread{
  13. private Socket socket;
  14. public ThreadT(Socket socket){
  15. this.socket = socket;
  16. }
  17. public void run() {
  18. try {
  19. InputStream inputStream = socket.getInputStream();
  20. BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));
  21. String str ;
  22. while ((str = br.readLine())!=null){
  23. System.out.println(socket.getRemoteSocketAddress()+"说:"+str);
  24. }
  25. } catch (Exception e) {
  26. e.printStackTrace();
  27. }
  28. }
  29. }
  30. //客户端
  31. package com.h.tcp2;
  32. import java.io.OutputStream;
  33. import java.io.PrintStream;
  34. import java.net.Socket;
  35. import java.util.Scanner;
  36. /*
  37. 多发多收
  38. * 客户端
  39. * */
  40. public class Tcpclient {
  41. public static void main(String[] args) {
  42. try {
  43. //创建用户输入类
  44. Scanner sc = new Scanner(System.in);
  45. //参数一、服务端ip地址
  46. //参数二、服务端端口号
  47. // 10.50.128.211
  48. Socket socket = new Socket("10.50.128.214", 7777);
  49. //从socket中得到一个字节输出流,负责发送消息
  50. OutputStream out = socket.getOutputStream();
  51. //把低级的字节流包装成打印流 这里只是把低级的流进行了包装后再发过去
  52. PrintStream ps = new PrintStream(out);
  53. while (true) {
  54. //发送消息
  55. System.out.println("请输入你要发送的内容");
  56. String s = sc.nextLine();
  57. if (s.equals("exit")) {
  58. break;
  59. }
  60. //发送内容
  61. ps.println(s);
  62. ps.flush();
  63. }
  64. } catch (Exception e) {
  65. }
  66. }
  67. }

使用线程池实现:

  • 服务端可以复用线程处理多个客户端,可以避免系统瘫痪 ```java //客户端 package com.h.tcp_many_majorization;

import com.h.tcp_many.ThreadT;

import java.net.ServerSocket; import java.net.Socket; import java.util.concurrent.*;

/

  • 使用线程池进行优化
  • */ public class TcpServerDuo优化 { //使用静态变量记住一个线程池对象 private static ExecutorService tpe = new ThreadPoolExecutor(3, 5, 6,

    1. TimeUnit.SECONDS, new ArrayBlockingQueue<>(2),
    2. Executors.defaultThreadFactory(),
    3. new ThreadPoolExecutor.AbortPolicy());

    public static void main(String[] args) {

    1. System.out.println("=====服务端启动=====");
    2. try {
    3. //注册端口
    4. ServerSocket ss = new ServerSocket(7777);
    5. while (true) {
    6. //每次都会接收到客户端电的Socket管道,交给一个独立线程,会在这里阻塞
    7. Socket socket = ss.accept();
    8. System.out.println(socket.getRemoteSocketAddress() + "上线了");
    9. //创建线程类
    10. ServerRunnable s = new ServerRunnable(socket);
    11. tpe.execute(s);
    12. }
    13. } catch (Exception e) {
    14. e.printStackTrace();
    15. }

    } } //线程 package com.h.tcp_many_majorization;

import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; import java.net.Socket;

public class ServerRunnable implements Runnable{

  1. private Socket socket;
  2. public ServerRunnable(Socket socket){
  3. this.socket = socket;
  4. }
  5. @Override
  6. public void run() {
  7. try {
  8. InputStream inputStream = socket.getInputStream();
  9. BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));
  10. String str ;
  11. while ((str = br.readLine())!=null){
  12. System.out.println(socket.getRemoteSocketAddress()+"说:"+str);
  13. }
  14. } catch (Exception e) {
  15. System.out.println(socket.getRemoteSocketAddress()+"这个吊毛已经离线了");
  16. }
  17. }

} //接收端跟上面一样

  1. <a name="QpktB"></a>
  2. ### TCP即时通信
  3. 即时通信是什么含义,要实现怎么样的设计
  4. - 发送消息个服务端,服务端在转发给所有客户端
  5. - 即时通信,是指一个客户端的消息发出去,其他客户端可以接收到
  6. - 即时通信需要进行端口转发的设计思想
  7. - 服务端需要把在线的Socket管道存储起来
  8. - 一旦收到一个消息要推送给其他管道
  9. 代码实现:
  10. ```java
  11. package com.h.tcpim;
  12. import com.h.tcp_many_majorization.ServerRunnable;
  13. import java.io.*;
  14. import java.net.ServerSocket;
  15. import java.net.Socket;
  16. import java.util.ArrayList;
  17. import java.util.List;
  18. import java.util.concurrent.*;
  19. /*
  20. *
  21. * 端口转发 即时通信
  22. * */
  23. public class TcpServerIM {
  24. //定义一个静态的List集合存储当前全部在线的Socket管道
  25. public static List<Socket> list = new ArrayList<>();
  26. //使用静态变量记住一个线程池对象
  27. private static ExecutorService tpe = new ThreadPoolExecutor(3, 5, 6,
  28. TimeUnit.SECONDS, new ArrayBlockingQueue<>(2),
  29. Executors.defaultThreadFactory(),
  30. new ThreadPoolExecutor.AbortPolicy());
  31. public static void main(String[] args) {
  32. System.out.println("=====服务端启动=====");
  33. try {
  34. //注册端口
  35. ServerSocket ss = new ServerSocket(7777);
  36. while (true) {
  37. //每次都会接收到客户端电的Socket管道,交给一个独立线程,会在这里阻塞
  38. Socket socket = ss.accept();
  39. //每次有一个客户端上线就把他加入到这个集合中去
  40. list.add(socket);
  41. System.out.println(socket.getRemoteSocketAddress() + "上线了");
  42. //创建线程类
  43. ThreadIM s = new ThreadIM(socket);
  44. tpe.execute(s);
  45. }
  46. } catch (Exception e) {
  47. e.printStackTrace();
  48. }
  49. }
  50. }
  51. class ThreadIM extends Thread{
  52. private Socket socket;
  53. public ThreadIM(Socket socket){
  54. this.socket = socket;
  55. }
  56. @Override
  57. public void run() {
  58. try {
  59. InputStream inputStream = socket.getInputStream();
  60. BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));
  61. String str ;
  62. while ((str = br.readLine())!=null){
  63. System.out.println(socket.getRemoteSocketAddress()+"说:"+str);
  64. //服务端收到消息后 转发出去 进行端口转发发送给全部的Socket管道
  65. sendMsgToAll(str);
  66. }
  67. } catch (Exception e) {
  68. System.out.println(socket.getRemoteSocketAddress()+"这个吊毛已经离线了");
  69. //当客户端挂掉后 就会从集合中删除这个socket 这样程序就不会报错
  70. TcpServerIM.list.remove(socket);
  71. }
  72. }
  73. private void sendMsgToAll(String str) throws Exception {
  74. //list里面有所有的上线了的客户端,所以我们就把这些客户端遍历出来把消息通过流在发出去
  75. for (Socket socket: TcpServerIM.list ) {
  76. PrintStream pr = new PrintStream(socket.getOutputStream());
  77. pr.println(str);
  78. pr.flush();
  79. }
  80. }
  81. }
  82. package com.h.tcpim;
  83. import java.io.*;
  84. import java.net.Socket;
  85. import java.net.SocketAddress;
  86. import java.util.HashMap;
  87. import java.util.Random;
  88. import java.util.Scanner;
  89. import java.util.Set;
  90. /*
  91. 多发多收
  92. * 客户端
  93. * */
  94. public class TcpClient {
  95. public static void main(String[] args) throws Exception {
  96. //创建用户输入类
  97. Scanner sc = new Scanner(System.in);
  98. //参数一、服务端ip地址
  99. //参数二、服务端端口号
  100. // 10.50.128.211
  101. Socket socket = new Socket("10.50.128.214", 7777);
  102. //创建一个独立的线程专门负责这个客户端的读消息,服务器随时会转发消息过来
  103. new ClientReaderThread(socket).start();
  104. //从socket中得到一个字节输出流,负责发送消息
  105. OutputStream out = socket.getOutputStream();
  106. //把低级的字节流包装成打印流 这里只是把低级的流进行了包装后再发过去
  107. PrintStream ps = new PrintStream(out);
  108. while (true) {
  109. //发送消息
  110. System.out.println("请输入你要发送的内容");
  111. String s = sc.nextLine();
  112. if (s.equals("exit")) {
  113. break;
  114. }
  115. //发送内容
  116. ps.println(s);
  117. ps.flush();
  118. }
  119. }
  120. }
  121. class ClientReaderThread extends Thread {
  122. private Socket socket;
  123. public ClientReaderThread(Socket socket) {
  124. this.socket = socket;
  125. }
  126. @Override
  127. public void run() {
  128. try {
  129. InputStream inputStream = socket.getInputStream();
  130. BufferedReader br = new BufferedReader(new InputStreamReader(inputStream));
  131. String str;
  132. while ((str = br.readLine()) != null) {
  133. System.out.println(getName() + "发送消息:" + str);
  134. }
  135. } catch (Exception e) {
  136. System.out.println("服务端把你踢出去了");
  137. }
  138. }
  139. }

总结

UDP使用的是DatagramSocket类DatagramPacket
客户端

  • DatagramSocket socket = new DatagramSocket();
  • DatagramPacket packet = new DatagramPacket(buff, buff.length, InetAddress.getLocalHost(), 7777);
  • 使用socket.send();

服务端

  • DatagramSocket socket = new DatagramSocket(7777);//注册端口
  • DatagramPacket packet = new DatagramPacket(buff,buff.length);
  • socket.receive();

TCP使用
客户端

  • Socket socket = new Socket(“10.50.128.214”, 7777);
  • socket.getOutputStream();

服务端

  • ServerSocket ss = new Socket(777);//注册端口
  • ss.accept();
  • ss.getInputStream();
  1. 使用代码下载音乐
  2. package com.hhm.下载音乐;
  3. import java.io.File;
  4. import java.io.FileOutputStream;
  5. import java.io.InputStream;
  6. import java.net.HttpURLConnection;
  7. import java.net.MalformedURLException;
  8. import java.net.URL;
  9. public class 下载音乐 {
  10. public static void main(String[] args) {
  11. try {
  12. //下载地址
  13. URL url = new URL("https://m801.music.126.net/20211214174656/0047165544898dca518d32057e866e50/jdyyaac/obj/w5rDlsOJwrLDjj7CmsOj/11858827254/839c/479d/c797/5b2e139670c187c5391ef2e22d698d2d.m4a");
  14. //连接到这个资源
  15. HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
  16. InputStream inputStream = urlConnection.getInputStream();
  17. //下载下来的名字自己改
  18. FileOutputStream fos = new FileOutputStream("f.m4a");
  19. byte[] bytes = new byte[1024];
  20. int len ;
  21. while ((len = inputStream.read(bytes)) != -1){
  22. fos.write(bytes,0,len);//写出这个数据
  23. }
  24. fos.close();
  25. inputStream.close();
  26. urlConnection.disconnect();//断开连接
  27. } catch (Exception e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }