Java网络编程以及部分协议的使用说明

网络编程

概述

  • Java 是 Internet 上的语言,它从语言级上提供了对网络应用程序的支持,程序员能够很容易开发常见的网络应用程序。
  • Java 提供的网络类库,可以实现无痛的网络连接,联网的底层细节被隐藏在 Java 的本机安装系统里,由 JVM 进行控制。并且 Java 实现了一个跨平台的网络库,程序员面对的是一个统一的网络编程环境。

网络基础

计算机网络

把分布在不同地理区域的计算机与专门的外部设备用通信线路互连成一个规模大、功能强的网络系统,从而使众多的计算机可以方便地互相传递信息、 共享硬件、软件、数据信息等资源。

网络编程的目的

直接或间接地通过网络协议与其它计算机实现数据交换,进行通讯。

网络编程中有两个主要的问题

  1. 如何准确地定位网络上一台或多台主机;定位主机上的特定的应用

  2. 找到主机后如何可靠高效地进行数据传输

网络通信要素概述

  • IP和端口号
  • 网络通信协议

如何实现网络中的主机互相通信

  • 通信双方地址

    IP

    端口号

  • 一定的规则(即:网络通信协议。有两套参考模型)

    OSI 参考模型:模型过于理想化,未能在因特网上进行广泛推广

    TCP/IP 参考模型(或 TCP/IP 协议):事实上的国际标准。

网络通信协议

在这里插入图片描述

通信要素1:IP和端口号

IP 地址:InetAddress

  • 唯一的标识 Internet 上的计算机(通信实体)

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

  • IP 地址分类方式1:IPV4 和 IPV6

    IPV4:4 个字节组成,4 个 0-255。大概 42 亿,30 亿都在北美,亚洲 4 亿。2011 年初已经用尽。以点分十进制表示,如192.168.0.1

    IPV6:128 位(16 个字节),写成 8 个无符号整数,每个整数用四个十六进制位表示,数之间用冒号(:)分开,如:3ffe:3201:1401:1280:c8ff:fe4d:db39:1984

  • IP地址分类方式2公网地址(万维网使用)私有地址(局域网使用)。192.168. 开头的就是私有地址,范围即为192.168.0.0–192.168.255.255,专门为组织机构内部使用

  • 特点:不易记忆

说明

  1. IP:唯一的表示 Internet 上的计算机(通信实体)

  2. 在 Java 中使用 InetAddress 类代表 IP

  3. IP 分类:IPv4 和 IPv6;万维网和局域网

  4. 域名:比如 www.baidu.com、www.mi.com 等

  5. 本地回路地址:127.0.0.1 对应着 localhost

  6. 如何实例化 InetAddress:两个方法:getByName(String host)、getLocalHost()

    两个常用方法:getHostName()、getHostAddress()

端口号:标识正在计算机上运行的进程(程序)

  • 不同的进程有不同的端口号

  • 被规定为一个 16 位的整数 0~65535。

  • 端口分类:

    公认端口:0~1023。被预先定义的服务通信占用(如:HTTP 占用端口 80,FTP 占用端口 21,Telnet 占用端口 23)

    注册端口:1024~49151。分配给用户进程或应用程序。(如:Tomcat 占用端口 8080,MySQL 占用端口 3306,Oracle 占用端口 1521 等)。

    动态/私有端口:49152~65535。

  • 端口号与IP地址的组合得出一个网络套接字:Socket。

说明

  1. 端口号:标识正在计算机上运行的进程
    要求:不同的进程有不同的端口号
    范围:被规定为一个 16 位的整数, 0~65535
  2. 端口号和 IP 地址的组合得出一个网络套接字:Socket

InetAddress类

  1. Internet 上的主机有两种方式表示地址:

    域名(hostName):www.baidu.com

    IP 地址(hostAddress):202.108.35.210

  2. InetAddress 类主要表示 IP 地址,两个子类:Inet4Address、Inet6Address。

  3. InetAddress 类对象含有一个 Internet 主机地址的域名和 IP 地址: www.baidu.com 和 202.108.35.210。

  4. 域名容易记忆,当在连接网络时输入一个主机的域名后,域名服务器(DNS) 负责将域名转化成IP地址,这样才能和主机建立连接。 -------这种过程被称为:域名解析

  5. InetAddress 类没有提供公共的构造器,而是提供了如下几个静态方法来获取 InetAddress 实例

    public static InetAddress getLocalHost()
    public static InetAddress getByName(String host)
    
  6. InetAddress 提供了如下几个常用的方法

    public String getHostAddress():返回 IP 地址字符串(以文本表现形式)。
    public String getHostName():获取此 IP 地址的主机名
    public boolean isReachable(int timeout):测试是否可以达到该地址
    

InetAdress 的使用

package com.laoyang.test.day1;

import java.net.InetAddress;
import java.net.UnknownHostException;

/**
 * @ClassName InetAddresstest
 * @Description: 网络通信
 * @Author Laoyang
 * @Date 2021/10/27 10:11
 */
public class InetAddressTest {
    public static void main(String[] args) {
        try {
            // 创建 InetAddress 对象,类似于 File,File参数表示对应的文件地址,而InetAddress的参数表示对应的主机地址
            InetAddress byNameA = InetAddress.getByName("192.168.10.14");
            System.out.println(byNameA);    // /127.0.0.1

            InetAddress byNameB = InetAddress.getByName("www.baidu.com");
            System.out.println(byNameB);    // www.baidu.com/14.215.177.38

            InetAddress byNameC = InetAddress.getByName("127.0.0.1");
            System.out.println(byNameC);

            // getLocalHost():获取本机的 IP
            InetAddress localHost = InetAddress.getLocalHost();
            System.out.println(localHost);

            // getHostName():获取域名
            String hostName = localHost.getHostName();
            System.out.println(hostName);

            // getHostAddress():获取 IP 地址
            String hostAddress = localHost.getHostAddress();
            System.out.println(hostAddress);
        } catch (UnknownHostException e) {
            e.printStackTrace();
        }
    }
}

通信要素2:网络协议

网络通信协议

计算机网络中实现通信必须有一些约定,即通信协议,对速率、传输代码、代码结构、传输控制步骤、出错控制等制定标准。

问题:网络协议太复杂

计算机网络通信涉及内容很多,比如指定源地址和目标地址,加密解密,压缩解压缩,差错控制,流量控制,路由控制,如何实现如此复杂的网络协议呢?

通信协议分层的思想

在制定协议时,把复杂成份分解成一些简单的成份,再将它们复合起来。最常用的复合方式是层次方式,即同层间可以通信、上一层可以调用下一层,而与再下一层不发生关系。各层互不影响,利于系统的开发和扩展。

TCP/IP 协议簇

  1. 传输层协议中有两个非常重要的协议:

    传输控制协议 TCP (Transmission Control Protocol)

    用户数据报协议 UDP (User Datagram Protocol)。

  2. TCP/IP 以其两个主要协议:传输控制协议(TCP)和网络互联协议(IP) 而得名,实际上是一组协议,包括多个具有不同功能且互为关联的协议。

  3. IP(Internet Protocol) 协议是网络层的主要协议,支持网间互连的数据通信。

  4. TCP/IP 协议模型从更实用的角度出发,形成了高效的四层体系结构,即物理链路层、IP层、传输层和应用层

TCP 和 UDP

TCP 协议

  • 使用 TCP 协议前,须先建立 TCP 连接,形成传输数据通道
  • 传输前,采用 “三次握手” 方式,点对点通信,是可靠的
  • TCP 协议进行通信的两个应用进程:客户端、服务端。
  • 在连接中可进行大数据量的传输
  • 传输完毕,需释放已建立的连接,效率较低

UDP 协议

  • 将数据、源、目的地封装成数据包(报),不需要建立连接
  • 每个数据报的大小限制在 64KB 内
  • 发送不管对方是否准备好,接收方收到也不确认,故是不可靠的
  • 可以广播发送
  • 发送数据结束时无需释放资源,开销小,速度快

TCP 三次握手的简单理解

第一次:告诉服务器,我要开始连接了

第二次:服务器收到连接请求,反馈给主机,并进行连接

第三次:主机收到反馈,连接成功

一般来说,三次握手连接成功的概率高达百分之九十九,就好比人与人交流,如果有人存在且跟你说话,别人肯定会回复你,而如果没有人存在并跟你说话,则无法进行正常交流

在这里插入图片描述

TCP 四次挥手的简单理解

第一次:告诉服务器,我要关闭连接了

第二次:服务器收到要关闭的信息,告诉主机说已收到

第三次:第二次反馈完之后,第三次服务器就会直接断开连接

第四次:再次发送请求至服务器,看服务器是否还能接收,如果没有服务器接收,则表示已经断开连接了,反之则还没有断开连接

在这里插入图片描述

Socket

说明

  1. 利用套接字(Socket)开发网络应用程序早已被广泛的采用,以至于成为事实上的标准。

  2. 网络上具有唯一标识的 IP 地址和端口号组合在一起才能构成唯一能识别的标识符套接字。

  3. 通信的两端都要有 Socket,是两台机器间通信的端点。

  4. 网络通信其实就是 Socket 间的通信。

  5. Socket 允许程序把网络连接当成一个流,数据在两个 Socket 间通过 IO 传输。

  6. 一般主动发起通信的应用程序属客户端,等待通信请求的为服务端。

  7. Socket分类:

    流套接字(stream socket):使用 TCP 提供可依赖的字节流服务

    数据报套接字(datagram socket):使用UDP提供 “尽力而为” 的数据报服务

  8. Socket 类的常用构造器:

    public Socket(InetAddress address,int port):创建一个流套接字并将其连接到指定 IP 地址的指定端口号。
    public Socket(String host,int port):创建一个流套接字并将其连接到指定主机上的指定端口号。
    

常用方法

方法作用
public InputStream getInputStream()返回此套接字的输入流。可以用于接收网络消息
public OutputStream getOutputStream()返回此套接字的输出流。可以用于发送网络消息
public InetAddress getInetAddress()此套接字连接到的远程 IP 地址;如果套接字是未连接的,则返回 null
public InetAddress getLocalAddress()获取套接字绑定的本地地址。 即本端的IP地址
public int getPort()此套接字连接到的远程端口号;如果尚未连接套接字,则返回 0
public int getLocalPort()返回此套接字绑定到的本地端口。 如果尚未绑定套接字,则返回 -1。即本端的端口号
public void close()关闭此套接字。套接字被关闭后,便不可在以后的网络连接中使用(即无法重新连接或重新绑定)。需要创建新的套接字对象。 关闭此套接字也将会关闭该套接字的 InputStream 和OutputStream。
public void shutdownInput()如果在套接字上调用 shutdownInput() 后从套接字输入流读取内容,则流将返回 EOF(文件结束符)。 即不能在从此套接字的输入流中接收任何数据
public void shutdownOutput()禁用此套接字的输出流。对于 TCP 套接字,任何以前写入的数据都将被发送,并且后跟 TCP 的正常连接终止序列。 如果在套接字上调用 shutdownOutput() 后写入套接字输出流,则该流将抛出 IOException。 即不能通过此套接字的输出流发送任何数据

在这里插入图片描述

TCP 网络编程

基于 Socket 的 TCP 编程

Java 语言的基于套接字编程分为服务端编程和客户端编程,其通信模型如图所示

在这里插入图片描述

客户端 Socket 的工作过程包含以下四个基本的步骤

  1. 创建 Socket:根据指定服务端的 IP 地址或端口号构造 Socket 类对象。若服务器端响应,则建立客户端到服务器的通信线路。若连接失败,会出现异常。
  2. 打开连接到 Socket 的输入/出流: 使用 getInputStream() 方法获得输入流,使用 getOutputStream() 方法获得输出流,进行数据传输
  3. 按照一定的协议对 Socket 进行读/写操作:通过输入流读取服务器放入线路的信息 (但不能读取自己放入线路的信息),通过输出流将信息写入线程。
  4. 关闭 Socket:断开客户端到服务器的连接,释放线路

客户端创建Socket对象

  • 客户端程序可以使用 Socket 类创建对象,创建的同时会自动向服务器方发起连接。Socket 的构造器是:

    Socket(String host,int port) throws UnknownHostException,IOException:向服务器 (域名是host。端口号为port) 发起 TCP 连接,若成功,则创建 Socket 对象,否则抛出异常。
     
    Socket(InetAddress address,int port) throws IOException:根据 InetAddress 对象所表示的 IP 地址以及端口号 port 发起连接。
    
  • 客户端建立 socketAtClient 对象的过程就是向服务器发出套接字连接请求

案例代码

Socket s = new Socket(192.168.40.165,9999);	// 对应服务端的端口号
OutputStream out = s.getOutputStream();
out.write(" hello".getBytes());
s.close();

服务器程序的工作过程包含以下四个基本的步骤

  1. 调用 ServerSocket(int port) :创建一个服务器端套接字,并绑定到指定端口上。用于监听客户端的请求。
  2. 调用 accept():监听连接请求,如果客户端请求连接,则接受连接,返回通信套接字对象。
  3. 调用该 Socket 类对象的 getOutputStream() 和 getInputStream ():获取输出流和输入流,开始网络数据的发送和接收。
  4. 关闭 ServerSocket 和 Socket 对象:客户端访问结束,关闭通信套接字。

服务器建立 ServerSocket 对象

  • ServerSocket 对象负责等待客户端请求建立套接字连接,类似邮局某个窗口中的业务员。也就是说,服务器必须事先建立一个等待客户请求建立套接字连接的 ServerSocket 对象。
  • 所谓 “接收” 客户的套接字请求,就是 accept() 方法会返回一个 Socket 对象。

案例代码

ServerSocket ss = new ServerSocket(9999);
Socket s = ss.accept ();
InputStream in = s.getInputStream();
byte[] buf = new byte[1024];
int num = in.read(buf);
String str = new String(buf, 0, num);
System.out.println(s.getInetAddress().toString()+:+str);
s.close();
ss.close();

Ps:以上只是使用 Socket 和 ServerSocket 实现网络编程的基本步骤,而不是只有那几步,如果想要实现一些功能可以在添加一些其他步骤进行实现,可以根据下面的案例进行参考

案例

案例一、客户端发送信息给服务端,服务端接收

运行时先启动服务端,在启动客户端

package com.laoyang.test.day1;

import org.junit.Test;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @ClassName TCPTest
 * @Description: 实现 TCP 网络编程
 * @Author Laoyang
 * @Date 2021/10/27 11:16
 */
public class TCPTestA {
    /**
     * 客户端
     */
    @Test
    public void testClient() {
        Socket socket = null;
        OutputStream outputStream = null;
        try {
            // 1. 创建 Soket 对象,指明服务器端的 IP 和 端口号
            InetAddress inetAddress = InetAddress.getByName("127.0.0.1");
            // 设置要连接到的服务端 IP 地址和端口号
            socket = new Socket(inetAddress, 3344);

            // 2. 获取一个输出流,用于发送数据
            outputStream = socket.getOutputStream();

            //3. 发送数据的操作
            outputStream.write("你好,我是客户端testClient".getBytes());
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            // 4. 关闭流资源
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 服务端
     */
    @Test
    public void testServer() {
        ServerSocket serverSocket = null;
        Socket socket = null;
        InputStream inputStream = null;
        ByteArrayOutputStream baos = null;
        try {
            // 1. 创建服务器端的 ServerSocket,指明自己的端口号
            serverSocket = new ServerSocket(3344);

            // 2. 调用 accept() 方法表示接收来自于客户端的 socket
            socket = serverSocket.accept();

            // 3. 获取输入流,用于接收数据
            inputStream = socket.getInputStream();

            // 4. 读取输入流中的数据
            byte[] buffer = new byte[10];
            int read = inputStream.read(buffer);

//            while (read != -1) {
            // 会出现乱码
//                String str = new String(buffer, 0, read);
//                System.out.println(str);
//                read = inputStream.read(buffer);
//            }

            baos = new ByteArrayOutputStream();
            while (read != -1) {
                // ByteArrayOutputStream 内部会有一个数组,用于存储添加的数据
                baos.write(buffer, 0, read);
                read = inputStream.read(buffer);
            }
            System.out.println(baos.toString());
            System.out.println("收到了来自 " + socket.getInetAddress().getHostAddress() + " 的数据");
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 5. 关闭流资源
            if (baos != null) {
                try {
                    baos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (serverSocket != null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

案例二、客户端发送文件给服务端,服务端将文件保存在本地。

这里以发送图片为例

package com.laoyang.test.day1;

import org.junit.Test;
import java.io.*;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @ClassName TCPTestA
 * @Description: 实现 TCP 网络编程
 * @Author Laoyang
 * @Date 2021/10/27 17:15
 */
public class TCPTestB {
    /**
     * 客户端
     */
    @Test
    public void testClient() {
        Socket socket = null;
        OutputStream outputStream = null;
        BufferedInputStream bis = null;
        try {
            // 1. 创建 Socket 对象,指明要连接的服务端的 IP 和端口号
            socket = new Socket(InetAddress.getByName("127.0.0.1"), 5566);

            // 2. 创建对应的流
            outputStream = socket.getOutputStream();

            // 3. 获取指定文件
            bis = new BufferedInputStream(new FileInputStream("神奇宝贝.jpg"));

            // 4. 具体的读写操作,实现发送功能
            byte[] buffer = new byte[1024];
            int read = bis.read(buffer);
            while (read != -1) {
                outputStream.write(buffer, 0, read);
                read = bis.read(buffer);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 5. 关闭流资源
            if (bis != null) {
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 服务端
     */
    @Test
    public void testServer() {
        ServerSocket serverSocket = null;
        Socket socket = null;
        InputStream inputStream = null;
        BufferedOutputStream bos = null;
        try {
            // 1. 创建 ServerSocket 服务端对象,指明 端口号
            serverSocket = new ServerSocket(5566);

            // 2. 创建 Socket 对象,调用 accept() 方法获取客户端的 socket
            socket = serverSocket.accept();

            // 3. 读取客户端的数据
            inputStream = socket.getInputStream();

            // 4. 具体的读写操作,实现接收功能
            bos = new BufferedOutputStream(new FileOutputStream("口袋妖怪.jpg"));
            byte[] buffer = new byte[1024];
            int read = inputStream.read(buffer);
            while (read != -1) {
                bos.write(buffer, 0, read);
                read = inputStream.read(buffer);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 5. 关闭流资源
            if (bos != null) {
                try {
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (serverSocket != null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

案例三、从客户端发送文件给服务端,服务端保存到本地。并返回 “发送成功” 给客户端。并关闭相应的连接

这里以文本文件为例

package com.laoyang.test.day1;

import org.junit.Test;
import java.io.*;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @ClassName TCPTestC
 * @Description: 实现 TCP 网络编程
 * @Author Laoyang
 * @Date 2021/10/28 10:26
 */
public class TCPTestC {
    /**
     * 客户端
     */
    @Test
    public void testClient() {
        Socket socket = null;
        OutputStream outputStream = null;
        BufferedInputStream bis = null;

        InputStream inputStream = null;
        ByteArrayOutputStream baos = null;

        try {
            // 1. 创建 Socket 对象,指定要连接的服务器 IP 地址和端口号
            socket = new Socket(InetAddress.getByName("127.0.0.1"), 7788);

            // 2. 创建对应的流
            outputStream = socket.getOutputStream();

            // 3. 获取要发送的文件
            bis = new BufferedInputStream(new FileInputStream("hello.txt"));

            // 4. 具体的发送操作
            byte[] buffer = new byte[10];
            int read = bis.read(buffer);
            while (read != -1) {
                outputStream.write(buffer, 0, read);
                read = bis.read(buffer);
            }

            // 关闭数据的输出(如果不执行此操作,则会一直在 while 循环中出不来)
            socket.shutdownOutput();

            // 5. 接收来自于服务器daunt的数据,并显示到控制台上
            inputStream = socket.getInputStream();
            baos = new ByteArrayOutputStream();
            byte[] bytes = new byte[10];
            int read1 = inputStream.read(bytes);
            while (read1 != -1) {
                baos.write(bytes, 0, read1);
                read1 = inputStream.read(bytes);
            }
            System.out.println(baos.toString());
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 6. 关闭流资源
            if (baos != null) {
                try {
                    baos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (bis != null) {
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 服务端
     */
    @Test
    public void testServer() {
        ServerSocket serverSocket = null;
        Socket accept = null;
        InputStream inputStream = null;
        BufferedOutputStream bos = null;

        OutputStream outputStream = null;

        try {
            // 1. 创建 ServerSocket 对象,声明服务端的端口号
            serverSocket = new ServerSocket(7788);

            // 2. 调用 accept() 方法,获取客户端的 Socket
            accept = serverSocket.accept();

            // 3. 创建对应的流
            inputStream = accept.getInputStream();

            // 4. 获取要接收的文件(复制后的文件)
            bos = new BufferedOutputStream(new FileOutputStream("新的文件.txt"));

            // 5. 具体的接收操作
            byte[] buffer = new byte[10];
            int read = inputStream.read(buffer);
            while (read != -1) {
                bos.write(buffer, 0, read);
                read = inputStream.read(buffer);
            }

            // 6. 服务器端给予客户端反馈
            outputStream = accept.getOutputStream();
            outputStream.write("发送成功!".getBytes());
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 7. 关闭流资源
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (bos != null) {
                try {
                    bos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (accept != null) {
                try {
                    accept.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (serverSocket != null) {
                try {
                    serverSocket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

UDP 网络编程

UDP 网络通信

  • DatagramSocketDatagramPacket 实现了基于 UDP 协议网络程序。
  • UDP 数据报通过数据报套接字 DatagramSocket 发送和接收,系统不保证 UDP 数据报一定能够安全送到目的地,也不能确定什么时候可以抵达。
  • DatagramPacket 对象封装了 UDP 数据报,在数据报中包含了发送端的 IP 地址和端口号以及接收端的 IP 地址和端口号。
  • UDP 协议中每个数据报都给出了完整的地址信息,因此无须建立发送方和接收方的连接。如同发快递包裹一样。

DatagramSocket 类的常用方法

方法作用
public DatagramSocket(int port)创建数据报套接字并将其绑定到本地主机上的指定端口。套接字将被绑定到通配符地址,IP 地址由内核来选择
public DatagramSocket(int port,InetAddress laddr)创建数据报套接字,将其绑定到指定的本地地址。本地端口必须在 0 到 65535 之间(包括两者)。如果 IP 地址为 0.0.0.0,套接字将被绑定到通配符地址,IP 地址由内核选择。
public void close()关闭此数据报套接字
public void send(DatagramPacket p)从此套接字发送数据报包。DatagramPacket 包含的信息指示:将要发送的数据、其长度、远程主机的 IP 地址和远程主机的端口号
public void receive(DatagramPacket p)从此套接字接收数据报包。当此方法返回时,DatagramPacket的缓冲区填充了接收的数据。数据报包也包含发送方的 IP 地址和发送方机器上的端口号。 此方法在接收到数据报前一直阻塞。数据报包对象的 length 字段包含所接收信息的长度。如果信息比包的长度长,该信息将被截短
public InetAddress getLocalAddress()获取套接字绑定的本地地址
public int getLocalPort()返回此套接字绑定的本地主机上的端口号
public InetAddress getInetAddress()返回此套接字连接的地址。如果套接字未连接,则返回 null
public int getPort()返回此套接字的端口。如果套接字未连接,则返回 -1

在这里插入图片描述

DatagramPacket 类的常用方法

方法作用
public DatagramPacket(byte[] buf,int length)构造 DatagramPacket,用来接收长度为 length 的数据包。 length 参数必须小于等于 buf.length
public DatagramPacket(byte[] buf,int length,InetAddress address,int port)构造数据报包,用来将长度为 length 的包发送到指定主机上的指定端口号。length参数必须小于等于 buf.length
public InetAddress getAddress()返回某台机器的 IP 地址,此数据报将要发往该机器或者是从该机器接收到的
public int getPort()返回某台远程主机的端口号,此数据报将要发往该主机或者是从该主机接收到的
public byte[] getData()返回数据缓冲区。接收到的或将要发送的数据从缓冲区中的偏移量 offset 处开始,持续 length 长度
public int getLength()返回将要发送或接收到的数据的长度

在这里插入图片描述

UDP 网络通信基本流程

  1. 创建 DatagramSocket 与 DatagramPacket 对象

  2. 建立发送端,接收端

  3. 建立数据包

  4. 调用Socket的发送、接收方法

  5. 关闭Socket

发送端与接收端是两个独立的运行程序

案例

package com.laoyang.test.day2;

import org.junit.Test;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;

/**
 * @ClassName UDPTestA
 * @Description: UDP 协议的网络编程
 * @Author Laoyang
 * @Date 2021/10/28 17:40
 */
public class UDPTestA {
    /**
     * 发送端
     */
    @Test
    public void testSender() {
        DatagramSocket socket = null;
        DatagramPacket packet = null;
        try {
            socket = new DatagramSocket();

            String str = "我是UDP方式发送的信息";
            byte[] data = str.getBytes();
            InetAddress inetAddress = InetAddress.getByName("127.0.0.1");
            packet = new DatagramPacket(data, 0, data.length, inetAddress, 9090);

            socket.send(packet);
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (socket != null) {
                socket.close();
            }
        }
    }

    /**
     * 接收端
     */
    @Test
    public void testReceiver() {
        DatagramSocket socket = null;
        DatagramPacket packet = null;
        try {
            socket = new DatagramSocket(9090);

            byte[] buffer = new byte[1024];
            packet = new DatagramPacket(buffer, 0, buffer.length);

            socket.receive(packet);

            System.out.println(new String(packet.getData(), 0, packet.getLength()));
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (socket != null) {
                socket.close();
            }
        }
    }
}

URL 网络编程

URL 类

  • URL(Uniform Resource Locator):统一资源定位符,它表示 Internet(互联网)上某一 资源的地址。

  • 它是一种具体的 URI,即 URL 可以用来标识一个资源,而且还指明了如何 locate 这个资源。

  • 通过 URL 我们可以访问 Internet 上的各种网络资源,比如最常见的 www,ftp 站点。浏览器通过解析给定的 URL 可以在网络上查找相应的文件或其他资源。

  • URL 的基本结构由 5 部分组成:

    <传输协议>://<主机名>:<端口号>/<文件名>#片段名?参数列表

    例如: http://localhost:8080/examples/beauty.jpg?username=root

    http:协议
    localhost:主机(IP地址)
    8080:端口号
    examples/beauty.jpg:具体资源地址
    username=root:参数列表
        
    #片段名:即锚点,例如看小说,直接定位到章节
    参数列表格式:参数名=参数值&参数名=参数值....
    

URL类构造器

  • 为了表示URL,java.net 中实现了类 URL。我们可以通过下面的构造器来初始化一个 URL 对象:

    public URL (String spec):通过一个表示 URL 地址的字符串可以构造一个 URL 对象。
    例如:URL url = new URL ("http://www.baidu.com"); 
    
    public URL(URL context, String spec):通过基 URL 和相对 URL 构造一个 URL 对象。 
    例如:URL downloadUrl = new URL(url, “download.html")
    
    public URL(String protocol, String host, String file); 
    例如:new URL("http",  "www.baidu.com", “download. html");
    
    public URL(String protocol, String host, int port, String file); 
    例如: URL gamelan = new  URL("http", "www.baidu.com", 80, “download.html");
    
  • URL 类的构造器都声明抛出非运行时异常,必须要对这一异常进行处理,通常是用 try-catch 语句进行捕获。

生活案例:我们在使用迅雷下载文件的时候需要填写对于的 URL 地址,才能进行数据的下载

在这里插入图片描述

常用方法

方法作用
public String getProtocol( )获取该 URL 的协议名
public String getHost( )获取该 URL 的主机名
public String getPort( )获取该 URL 的端口号
public String getPath( )获取该 URL 的文件路径
public String getFile( )获取该 URL 的文件名
public String getQuery( )获取该 URL 的查询名

案例

package com.laoyang.test.day3;

import org.junit.Test;
import java.net.MalformedURLException;
import java.net.URL;

/**
 * @ClassName URLTest
 * @Description: URL 网络编程
 * @Author Laoyang
 * @Date 2021/10/29 16:02
 */
public class URLTestA {
    @Test
    public void testOne() {
        try {
            URL url = new URL("http://localhost:8080/examples/beauty.jpg?username=root");

            // public String getProtocol( ) 获取该URL的协议名
            System.out.println(url.getProtocol());  // http

            // public String getHost( ) 获取该URL的主机名
            System.out.println(url.getHost());  // localhost

            // public String getPort( ) 获取该URL的端口号
            System.out.println(url.getPort());  // 8080

            // public String getPath( ) 获取该URL的文件路径
            System.out.println(url.getPath());  // /examples/beauty.jpg

            // public String getFile( ) 获取该URL的文件名
            System.out.println(url.getFile());  // /examples/beauty.jpg?username=root

            // public String getQuery( ) 获取该URL的查询名
            System.out.println(url.getQuery()); // username=root
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }
    }
}

针对 HTTP 协议的 URLConnection 类

  • URL 的方法 openStream():能从网络上读取数据

  • 若希望输出数据,例如向服务器端的 CGI (公共网关接口-Common Gateway Interface-的简称,是用户浏览器和服务器端的应用程序进行连接的接口)程序发送一些数据,则必须先与 URL 建立连接,然后才能对其进行读写,此时需要使用 URLConnection 。

  • URLConnection:表示到 URL 所引用的远程对象的连接。当与一个 URL 建立连接时, 首先要在一个 URL 对象上通过方法openConnection() 生成对应的 URLConnection 对象。如果连接过程失败,将产生 IOException

    URL netchinaren = new URL ("http://www.baidu.com/index.html"); 
    URLConnectonn u = netchinaren.openConnection( );
    
  • 通过 URLConnection 对象获取的输入流和输出流,即可以与现有的 CGI 程序进行交互。

    public Object getContent() throws IOException
    public int getContentLength()
    public String getContentType()
    public long getDate()
    public long getLastModified()
    public InputStream getInputStream() throws IOException
    public OutputSteram getOutputStream() throws IOException
    

案例

package com.laoyang.test.day3;

import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;

/**
 * @ClassName URLTestB
 * @Description:
 * @Author Laoyang
 * @Date 2021/10/29 16:12
 */
public class URLTestB {
    public static void main(String[] args) {
        HttpURLConnection connection = null;
        InputStream inputStream = null;
        BufferedOutputStream bis = null;
        try {
            // 创建 URL 对象,指明要访问的 url 地址
            URL url = new URL("http://localhost:8080/examples/beauty.jpg?username=root");

            // 获取对应的连接(建立连接)
            connection = (HttpURLConnection) url.openConnection();
            connection.connect();

            // 创建输入输出流(用于读取要下载的文件和要下载到的地址)
            inputStream = connection.getInputStream();
            // 因为 main 方法是直接指向工程下的,所以这里需要指明放在工程下的哪一个模块中
            bis = new BufferedOutputStream(new FileOutputStream("Demo_16\\beautyA.jpg"));

            // 具体的下载操作
            byte[] buffer = new byte[1024];
            int read = inputStream.read(buffer);
            while (read != -1) {
                bis.write(buffer, 0, read);
                read = inputStream.read(buffer);
            }
            System.out.println("下载完成!");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            // 关闭资源
            if (bis != null) {
                try {
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            // 断开连接
            if (connection != null) {
                connection.disconnect();
            }
        }
    }
}

URI、URL 和 URN 的区别(了解)

  • URI 是 uniform resource identifier,统一资源标识符,用来唯一的标识一个资源。

  • URL 是 uniform resource locator,统一资源定位符,它是一种具体的 URI,即 URL 可以用来标识一个资源,而且还指明了如何 locate 这个资源。

  • URN uniform resource name,统一资源命名,是通过名字来标识资源, 比如 mailto:java-net@java.sun.com。

也就是说,URI 是以一种抽象的,高层次概念定义统一资源标识,而 URL 和 URN 则是具体的资源标识的方式URL 和 URN 都是一种URI。 在 Java 的 URI 中,一个 URI 实例可以代表绝对的,也可以是相对的,只要它符合 URI 的语法规则。而 URL 类则不仅符合语义,还包含了定位该资源的信息, 因此它不能是相对的。

小结

  1. 位于网络中的计算机具有唯一的 IP 地址,这样不同的主机可以互相区分。
  2. 客户端-服务器是一种最常见的网络应用程序模型。服务器是一个为其客户端提供某种特定服务的硬件或软件。客户机是一个用户应用程序,用于访问某台服务器提供的服务。端口号是对一个服务的访问场所,它用于区分同一物理计算机上的多个服务。套接字用于连接客户端和服务器,客户端和服务器之间的每个通信会话使用一个不同的套接字。TCP 协议用于实现面向连接的会话。
  3. Java 中有关网络方面的功能都定义在 java.net 程序包中。Java 用 InetAddress 对象表示 IP 地址,该对象里有两个字段:主机名(String) 和 IP 地址(int)
  4. 类 Socket 和 ServerSocket 实现了基于 TCP 协议的客户端-服务器程序。Socket 是客户端和服务器之间的一个连接,连接创建的细节被隐藏了。这个连接提供了一个安全的数据传输通道,这是因为 TCP 协议可以解决数据在传送过程中的丢失、损坏、重复、乱序以及网络拥挤等问题,它保证数据可靠的传送。
  5. 类 URL 和 URLConnection 提供了最高级网络应用。URL 的网络资源的位置来同一表示 Internet 上各种网络资源。通过 URL 对象可以创建当前应用程序和 URL 表示的网络资源之间的连接,这样当前程序就可以读取网络资源数据,或者把自己的数据传送到网络上去。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值