网络编程的常识

目前主流的网络通讯软件有:微信、QQ、飞信、阿里旺旺、陌陌、探探、…

七层网络模型

  • OSI(Open System Interconnect),即开放式系统互联,是ISO(国际标准化组织)组织在1985年研究的网络互连模型。
  • OSI七层模型和TCP/IP五层模型的划分如下:image.png
  • 当发送数据时,需要对发送的内容按照上述七层模型进行层层加包后发送出去。
  • 当接收数据时,需要对接收的内容按照上述七层模型相反的次序层层拆包并显示出来。

    相关的协议(笔试题)

    协议的概念

    计算机在网络中实现通信就必须有一些约定或者规则,这种约定和规则就叫做通信协议,通信协议可以对速率、传输代码、代码结构、传输控制步骤、出错控制等制定统一的标准。

    TCP协议

  • 传输控制协议(Transmission Control Protocol),是一种面向连接的协议,类似于打电话。

    • 建立连接 => 进行通信 => 断开连接
    • 在传输前采用”三次握手”方式。
    • 在通信的整个过程中全程保持连接,形成数据传输通道。
    • 保证了数据传输的可靠性和有序性。
    • 是一种全双工的字节流通信方式,可以进行大数据量的传输。
    • 传输完毕后需要释放已建立的连接,发送数据的效率比较低

image.png
image.png

UDP协议

  • 用户数据报协议(User Datagram Protocol),是一种非面向连接的协议,类似于写信。

    • 在通信的整个过程中不需要保持连接,其实是不需要建立连接。
    • 不保证数据传输的可靠性和有序性。
    • 是一种全双工的数据报通信方式,每个数据报的大小限制在64K内。
    • 发送数据完毕后无需释放资源,开销小,发送数据的效率比较高,速度快。

      IP地址(重点)

  • 192.168.1.1 - 是绝大多数路由器的登录地址,主要配置用户名和密码以及Mac过滤。

  • IP地址是互联网中的唯一地址标识,本质上是由32位二进制组成的整数,叫做IPv4,当然也有128位二进制组成的整数,叫做IPv6,目前主流的还是IPv4。
  • 日常生活中采用点分十进制表示法来进行IP地址的描述,将每个字节的二进制转化为一个十进制整数,不同的整数之间采用小数点隔开。
  • 如:

0x01020304 => 1.2.3.4

  • 查看IP地址的方式:

Windows系统:在dos窗口中使用ipconfig或ipconfig/all命令即可
Unix/linux系统:在终端窗口中使用ifconfig或/sbin/ifconfig命令即可

  • 特殊的地址

本地回环地址(hostAddress):127.0.0.1 主机名(hostName):localhost

端口号(重点)

  • IP地址 - 可以定位到具体某一台设备。
  • 端口号 - 可以定位到该设备中具体某一个进程。
  • 端口号本质上是16位二进制组成的整数,表示范围是:0 ~ 65535,其中0 ~ 1024之间的端口号通常被系统占用,建议编程从1025开始使用。
  • 特殊的端口:

HTTP:80 FTP:21 Oracle:1521 MySQL:3306 Tomcat:8080

  • 网络编程需要提供:IP地址 + 端口号,组合在一起叫做网络套接字:Socket。

    基于tcp协议的编程模型(重点)

    C/S架构的简介

  • 在C/S模式下客户向服务器发出服务请求,服务器接收请求后提供服务。

  • 例如:在一个酒店中,顾客找服务员点菜,服务员把点菜单通知厨师,厨师按点菜单做好菜后让服务员端给客户,这就是一种C/S工作方式。如果把酒店看作一个系统,服务员就是客户端,厨师就是服务器。这种系统分工和协同工作的方式就是C/S的工作方式。
  • 客户端部分:为每个用户所专有的,负责执行前台功能。
  • 服务器部分:由多个用户共享的信息与功能,招待后台服务。

image.png

编程模型

  • 服务器:

(1)创建ServerSocket类型的对象并提供端口号;
(2)等待客户端的连接请求,调用accept()方法;
(3)使用输入输出流进行通信;
(4)关闭Socket;

  • 客户端:

(1)创建Socket类型的对象并提供服务器的IP地址和端口号;
(2)使用输入输出流进行通信;
(3)关闭Socket;

相关类和方法的解析

ServerSocket类

java.net.ServerSocket类主要用于描述服务器套接字信息(大插排)。
常用的方法如下:
image.png

Socket类

java.net.Socket类主要用于描述客户端套接字,是两台机器间通信的端点(小插排)。
常用的方法如下:
image.png

  1. package com.lagou.task19;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStreamReader;
  5. import java.io.PrintStream;
  6. import java.net.Socket;
  7. import java.util.Scanner;
  8. /**
  9. * @author lijing
  10. * @date 2020/10/16 14:11
  11. * @description
  12. */
  13. public class ClientStringTest {
  14. public static void main(String[] args) {
  15. Socket s= null;
  16. PrintStream ps=null;
  17. Scanner sc=null;
  18. BufferedReader br=null;
  19. try {
  20. //1.创建Socket类型的对象并提供服务器的主机名和端口号
  21. s = new Socket("127.0.0.1",8888);
  22. System.out.println("连接服务器成功");
  23. //2.使用输入输出流进行通信
  24. sc=new Scanner(System.in);
  25. ps=new PrintStream(s.getOutputStream());
  26. br=new BufferedReader(new InputStreamReader(s.getInputStream()));
  27. while(true){
  28. //Thread.sleep(1000);
  29. //实现客户端发送的内容由用户从键盘输入
  30. System.out.println("请输入要发送的数据内容:");
  31. String str1=sc.next();
  32. //实现客户端向服务器发送字符串内容"hello"
  33. // ps.println("hello");
  34. ps.println(str1);
  35. System.out.println("客户端发送成功");
  36. //当发送的数据为"bye"的时候,则聊天结束
  37. if("bye".equalsIgnoreCase(str1)){
  38. System.out.println("聊天结束");
  39. break;
  40. }
  41. //实现接受服务器发来的字符串内容并打印
  42. // br.readLine();
  43. String str2=br.readLine();
  44. System.out.println("服务器回发的消息为:"+str2);
  45. }
  46. } catch (IOException e) {
  47. e.printStackTrace();
  48. } /*catch (InterruptedException e) {
  49. e.printStackTrace();
  50. }*/ finally {
  51. //3.关闭Socket并释放有关资源
  52. if(null!=br){
  53. try {
  54. br.close();
  55. } catch (IOException e) {
  56. e.printStackTrace();
  57. }
  58. }
  59. if(null!=sc){
  60. sc.close();
  61. }
  62. if (null!=s){
  63. try {
  64. s.close();
  65. } catch (IOException e) {
  66. e.printStackTrace();
  67. }
  68. }
  69. if(null!=ps){
  70. ps.close();
  71. }
  72. }
  73. }
  74. }
  1. package com.lagou.task19;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStreamReader;
  5. import java.io.PrintStream;
  6. import java.net.ServerSocket;
  7. import java.net.Socket;
  8. /**
  9. * @author lijing
  10. * @date 2020/10/16 14:06
  11. * @description
  12. */
  13. public class ServerStringTest {
  14. public static void main(String[] args) {
  15. ServerSocket ss= null;
  16. Socket s = null;
  17. try {
  18. //1.创建ServerSocket类型的对象并提供端口号
  19. ss = new ServerSocket(8888);
  20. //2.等待客户端的连接请求.调用accpet方法
  21. while(true) {
  22. System.out.println("等待客户端的连接请求...");
  23. s = ss.accept();
  24. //当客户端没有连接时,则服务器阻塞在accept方法的调用这里
  25. System.out.println("客户端"+s.getInetAddress()+"连接成功");
  26. //3.使用输入输出流进行通信
  27. //每当有一个客户端连接成功,则需要启动一个新的线程为之服务
  28. new ServerThread(s).start();
  29. }
  30. } catch (IOException e) {
  31. e.printStackTrace();
  32. } finally {
  33. //4.关闭Socket并释放有关的资源
  34. if(null!=s){
  35. try {
  36. s.close();
  37. } catch (IOException e) {
  38. e.printStackTrace();
  39. }
  40. }
  41. if(null!=ss){
  42. try {
  43. ss.close();
  44. } catch (IOException e) {
  45. e.printStackTrace();
  46. }
  47. }
  48. }
  49. }
  50. }
  1. package com.lagou.task19;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStreamReader;
  5. import java.io.PrintStream;
  6. import java.net.InetAddress;
  7. import java.net.ServerSocket;
  8. import java.net.Socket;
  9. import java.util.Scanner;
  10. /**
  11. * @author lijing
  12. * @date 2020/10/16 15:22
  13. * @description
  14. */
  15. public class ServerThread extends Thread{
  16. private Socket s;
  17. public ServerThread(Socket s){
  18. this.s=s;
  19. }
  20. @Override
  21. public void run() {
  22. BufferedReader br=null;
  23. PrintStream ps=null;
  24. try {
  25. //3.使用输入输出流进行通信
  26. br=new BufferedReader(new InputStreamReader(s.getInputStream()));
  27. ps=new PrintStream(s.getOutputStream());
  28. while(true){
  29. //实现对客户端发来字符串内容的接受
  30. String s1=br.readLine();
  31. InetAddress inetAddress = s.getInetAddress();
  32. System.out.println("客户单"+inetAddress+"发来的字符串内容是:"+s1);
  33. //当客户端发来的内容是"bye"是,则聊天结束
  34. if (s1.equalsIgnoreCase("bye")){
  35. System.out.println("客户端"+inetAddress+"已下线");
  36. break;
  37. }
  38. //实现服务器向客户端回发字符串内容"I received!"
  39. ps.println("I received!");
  40. System.out.println("服务器发送数据成功");
  41. }
  42. } catch (IOException e) {
  43. e.printStackTrace();
  44. } finally {
  45. //4.关闭Socket并释放有关的资源
  46. if(null!=ps){
  47. ps.close();
  48. }
  49. if(null!=br){
  50. try {
  51. br.close();
  52. } catch (IOException e) {
  53. e.printStackTrace();
  54. }
  55. }
  56. if(null!=s){
  57. try {
  58. s.close();
  59. } catch (IOException e) {
  60. e.printStackTrace();
  61. }
  62. }
  63. }
  64. }
  65. }

注意事项

  • 客户端 Socket 与服务器端 Socket 对应, 都包含输入和输出流。
  • 客户端的socket.getInputStream() 连接于服务器socket.getOutputStream()。
  • 客户端的socket.getOutputStream()连接于服务器socket.getInputStream()

    基于udp协议的编程模型(熟悉)

    编程模型

  • 接收方:

(1)创建DatagramSocket类型的对象并提供端口号;
(2)创建DatagramPacket类型的对象并提供缓冲区;
(3)通过Socket接收数据内容存放到Packet中,调用receive方法;
(4)关闭Socket;

  • 发送方:

(1)创建DatagramSocket类型的对象;
(2)创建DatagramPacket类型的对象并提供接收方的通信地址;
(3)通过Socket将Packet中的数据内容发送出去,调用send方法;
(4)关闭Socket;

相关类和方法的解析

DatagramSocket类

java.net.DatagramSocket类主要用于描述发送和接收数据报的套接字(邮局)。
换句话说,该类就是包裹投递服务的发送或接收点。
常用的方法如下:
image.png

DatagramPacket类

java.net.DatagramPacket类主要用于描述数据报,数据报用来实现无连接包裹投递服务。
常用的方法如下:
image.png

InetAddress类

java.net.InetAddress类主要用于描述互联网通信地址信息。
常用的方法如下:
image.png

  1. package com.lagou.task19;
  2. import java.io.IOException;
  3. import java.net.DatagramPacket;
  4. import java.net.DatagramSocket;
  5. /**
  6. * @author lijing
  7. * @date 2020/10/16 15:46
  8. * @description
  9. */
  10. public class ReceiveTest {
  11. public static void main(String[] args) {
  12. DatagramSocket ds= null;
  13. try {
  14. //1.创建DatagramSocket类型的对象并提供端口号
  15. ds = new DatagramSocket(8888);
  16. //2.创建DateGramPacket类型的对象并提供缓冲区
  17. byte[] arr=new byte[20];
  18. DatagramPacket dp=new DatagramPacket(arr,arr.length);
  19. //3.通过Socket接受数据内容存放到Packet里面,调用receive方法
  20. System.out.println("等待数据的到来...");
  21. ds.receive(dp);
  22. System.out.println("接受到的数据内容是:"+new String(arr,0,dp.getLength())+"!");
  23. //实现将字符串内容"我收到了"回发过去
  24. byte[] arr2="I revice".getBytes();
  25. DatagramPacket dp2=new DatagramPacket(arr2,arr2.length,dp.getAddress(),dp.getPort());
  26. ds.send(dp2);
  27. System.out.println("回发数据成功");
  28. } catch (IOException e) {
  29. e.printStackTrace();
  30. } finally {
  31. //4.关闭Socket并释放有关资源
  32. if(null!=ds){
  33. ds.close();
  34. }
  35. }
  36. }
  37. }
  1. package com.lagou.task19;
  2. import java.io.IOException;
  3. import java.net.DatagramPacket;
  4. import java.net.DatagramSocket;
  5. import java.net.InetAddress;
  6. /**
  7. * @author lijing
  8. * @date 2020/10/16 15:51
  9. * @description
  10. */
  11. public class SendTest {
  12. public static void main(String[] args) {
  13. DatagramSocket ds = null;
  14. try {
  15. //1.创建DatagramSocket类型的对象
  16. ds = new DatagramSocket();
  17. //2.创建DatagramPacket类型的对象并提供接受方的通讯地址和端口号
  18. byte[] arr="hello".getBytes();
  19. DatagramPacket dp = new DatagramPacket(arr,arr.length, InetAddress.getLocalHost(),8888);
  20. //3.发送Socket发送Packet,调用send方法
  21. ds.send(dp);
  22. System.out.println("发送数据成功");
  23. //接受回发的数据内容
  24. byte[] arr2=new byte[20];
  25. DatagramPacket dp2=new DatagramPacket(arr2,arr.length);
  26. ds.receive(dp2);
  27. System.out.println("接受到的回发信息是:"+new String(arr2,0,dp2.getLength()));
  28. } catch (IOException e) {
  29. e.printStackTrace();
  30. } finally {
  31. //4.关闭
  32. if (null!=ds) {
  33. ds.close();
  34. }
  35. }
  36. }
  37. }

URL类(熟悉)

基本概念

  • java.net.URL(Uniform Resource Identifier)类主要用于表示统一的资源定位器,也就是指向万维网上“资源”的指针。这个资源可以是简单的文件或目录,也可以是对复杂对象的引用,例如对数据库或搜索引擎的查询等。
  • 通过URL可以访问万维网上的网络资源,最常见的就是www和ftp站点,浏览器通过解析给定的URL可以在网络上查找相应的资源。
  • URL的基本结构如下:

<传输协议>://<主机名>:<端口号>/<资源地址>

常用的方法

image.png

URLConnection类

基本概念

java.net.URLConnection类是个抽象类,该类表示应用程序和URL之间的通信链接的所有类的超
类,主要实现类有支持HTTP特有功能的HttpURLConnection类。
HttpURLConnection类的常用方法
image.png

  1. package com.lagou.task19;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. import java.io.InputStreamReader;
  6. import java.net.HttpURLConnection;
  7. import java.net.MalformedURLException;
  8. import java.net.URL;
  9. /**
  10. * @author lijing
  11. * @date 2020/10/16 16:04
  12. * @description
  13. */
  14. public class URLTest {
  15. public static void main(String[] args) {
  16. //1.使用参数指定的字符串来构造对象
  17. try {
  18. URL url=new URL("https://www.yuque.com/dashboard");
  19. //2.获取相关信息并打印出来
  20. System.out.println(url.getProtocol());
  21. System.out.println(url.getHost());
  22. System.out.println(url.getPort());
  23. //3.建立连接并读取相关信息打印出来
  24. HttpURLConnection urlConnection= (HttpURLConnection) url.openConnection();
  25. InputStream inputStream = urlConnection.getInputStream();
  26. BufferedReader br=new BufferedReader(new InputStreamReader(inputStream));
  27. String str=null;
  28. while((str=br.readLine())!=null){
  29. System.out.println(str);
  30. }
  31. br.close();
  32. urlConnection.disconnect();
  33. } catch (MalformedURLException e) {
  34. e.printStackTrace();
  35. } catch (IOException e) {
  36. e.printStackTrace();
  37. }
  38. }
  39. }