Java——TCP UDP Socket编程

news2024/11/24 15:00:51

目录

一、网络的相关概念

(一)网络通信

(二)网络

(三)ip地址

(四)ipv4地址分类

(五)域名

(六)网络通信协议

(七)TCP和UDP

二、InetAddress类

三、Socket

四、TCP网络编程

(一)案例一——使用字节流

(二)案例二——使用字节流 

(三)案例三——使用字符流

(四)案例四——使用字符流

五、UDP网络编程

(一)基本介绍

(二)基本流程

(三)案例

六、案例

(一)案例一

(二)案例二


一、网络的相关概念

(一)网络通信

1.概念:两台设备之间通过网络实现数据传输

2.网络通信:将数据通过网络从一台设备传输到另一台设备

3.java.net包下提供了一系列的类或接口,供程序员使用,完成网络通信

(二)网络

1.概念:两台或多台设备通过一定物理设备连接起来构成了网络
2.根据网络的覆盖范围不同,对网络进行分类:

  • 局域网: 覆盖范围最小,仅仅覆盖一个教室或一个机房
  • 城域网:覆盖范围较大,可以覆盖一个城市
  • 广域网:覆盖范围最大,可以覆盖全国,甚至全球,万维网是广域网的代表

(三)ip地址

1.概念: 用于唯一标识网络中的每台计算机/主机

2.查看ip地址: ipconfig

3.ip地址的表示形式:点分十进制 xx.xx.xx.xx

4.每一个十进制数的范围: 0~255
5.ip地址的组成=网络地址+主机地址,比如: 192.168.16.69

6.lPv6是互联网工程任务组设计的用于替代IPv4的下一代IP协议,其地址数量号称可以为全世界的每一粒沙子编上一个地址

7.由于IPv4最大的问题在于网络地址资源有限,严重制约了互联网的应用和发展。IPv6的使用,不仅能解决网络地址资源数量的问题,而且也解决了多种接入设备连入互联网的障碍

(四)ipv4地址分类

(五)域名

1.比如www.baidu.com,将ip映射到域名上,访问域名就是访问ip
2.好处:为了方便记忆,解决记ip的困难


端口号

1.概念: 用于标识计算机上某个特定的网络程序

2.表示形式: 以整数形式,端口范围0~65535 [2个字节表示端口 0~2^16-1]

3.0~1024已经被占用, 比如 ssh 22, ftp 21, smtp 25 http 80

4.常见的网络程序端口号:

  • tomcat :8080
  • mysql:3306
  • oracle:1521
  • sqlserver:1433

(六)网络通信协议

协议(tcp/ip)

TCP/IP (Transmission Control Protocol/Internet Protocol)的简写, 中文译名为传输控制协议/因特网互联协议,又叫网络通讯协议,这个协议是Internet最基本的协议、Internet国际互联网络的基础,简单地说,就是由网络层的IP协议和传输层的TCP协议组成的

(七)TCP和UDP

TCP协议: 传输控制协议
1.使用TCP协议前,须先建立TCP连接,形成传输数据通道

2.传输前,采用“三次握手"方式,是可靠的

3.TCP协议进行通信的两个应用进程: 客户端、服务端

4.在连接中可进行大数据量的传输

5.传输完毕,需释放已建立的连接,效率低

UDP协议: 用户数据协议
1.将数据、源、目的封装成数据包,不需要建立连接

2.每个数据报的大小限制在64K内,不适合传输大量数据

3.因无需连接,故是不可靠

4.发送数据结束时无需释放资源(因为不是面向连接的),速度快

5.举例: 发短信

二、InetAddress类

方法:

  1. 获取本机InetAddress对象 getLocalHost
  2. 根据指定主机名/域名获取ip地址对象 getByName
  3. 获取InetAddress对象的主机名 getHostName
  4. 获取InetAddress对象的地址 getHostAddress
/*
 * 演示InetAddress类的使用
 */
public class API_ {
    public static void main(String[] args) throws UnknownHostException {
        // 1.获取本机的InetAddress对象
        InetAddress localHost = InetAddress.getLocalHost();
        System.out.println(localHost); // LAPTOP-RVFFB7FM/192.168.23.1

        // 2.根据机器的名字获取InetAddress对象
        InetAddress inetAddress = InetAddress.getByName("LAPTOP-RVFFB7FM");
        System.out.println(inetAddress);

        // 3.根据域名返回InetAddress对象,比如www.baidu.com
        InetAddress inetAddress1 = InetAddress.getByName("www.baidu.com");
        System.out.println(inetAddress1);

        // 4.根据InetAddress对象,获取对应的ip地址
        String address = inetAddress1.getHostAddress();
        System.out.println(address);

        // 5.通过InetAddress对象,获取对应的主机名
        String hostName = inetAddress1.getHostName();
        System.out.println(hostName);
    }
}

三、Socket

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

2.通信的两端都要有Socket,是两台机器间通信的端点,网络通信其实就是Socket间的通信

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

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

四、TCP网络编程

1.基于客户端一服务端的网络通信, 底层使用的是TCP/IP协议

2.应用场景举例: 客户端发送数据,服务端接受并显示控制台

3.基于Socket的TCP编程

(一)案例一——使用字节流

 

服务端

public class SocketTCP01Server {
    public static void main(String[] args) throws IOException {
        // 1.在9999端口监听,等待链接
        //  细节:要求该端口没有其他人监听
        ServerSocket serverSocket = new ServerSocket(9999);

        System.out.println("服务端,在9999端口监听,等待连接...");
        // 2.如果没有客户端连接9999端口,该方法会阻塞等待连接
        // 如果有连接,返回一个socket
        Socket socket = serverSocket.accept();

        System.out.println(socket.getClass());
        // 3.通过socket.getInputStream() 读取
        // 客户端写入到数据通道的数据,显示
        InputStream inputStream = socket.getInputStream();
        byte[] bytes = new byte[1024];
        int readLen = 0;
        while((readLen = inputStream.read(bytes)) != -1) {
            System.out.println(new String(bytes, 0, readLen));
        }
        // 4.关闭流
        inputStream.close();
        socket.close();
        serverSocket.close();
        System.out.println("服务端退出...");

    }
}

客户端 

public class SocketTCP01Client {
    public static void main(String[] args) throws IOException {
        // 1.连接9999端口,连接成功返回一个socket
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        // 2.连接上后,生成socket,通过socket.getOutputStream()
        OutputStream outputStream = socket.getOutputStream();
        // 3.输出信息
        outputStream.write("hello, socket".getBytes());
        // 4.关闭流和连接,节省资源
        outputStream.close();
        socket.close();
        System.out.println("客户端退出...");
    }
}

(二)案例二——使用字节流 

服务端

public class SocketTCP02Server {
    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(9999);
        Socket socket = serverSocket.accept();

        InputStream inputStream = socket.getInputStream();
        byte[] buf = new byte[1024];
        int readLen = 0;
        while((readLen = inputStream.read(buf)) != -1) {
            System.out.println(new String(buf, 0, readLen));
        }

        OutputStream outputStream = socket.getOutputStream();
        outputStream.write("hello Client".getBytes());
        // 设置结束标记
        socket.shutdownOutput();
        outputStream.close();
        inputStream.close();
        socket.close();
        serverSocket.close();
    }
}

客户端 

public class SocketTCP02Client {
    public static void main(String[] args) throws Exception{
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        OutputStream outputStream = socket.getOutputStream();
        outputStream.write("hello Server".getBytes());
        // 设置结束标记, socket.close()也做了同样的事情
        // 但是因为我们下面还有获取数据通道信息,因此不能关闭,只能shutdownOutput()
        socket.shutdownOutput();
        InputStream inputStream = socket.getInputStream();
        byte[] buf = new byte[1024];
        int readLen = 0;
        while((readLen = inputStream.read(buf)) != -1) {
            System.out.println(new String(buf, 0, readLen));
        }

        outputStream.close();
        inputStream.close();
        socket.close();
    }
}

(三)案例三——使用字符流

服务端

public class SocketTCP03Server {
    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(9999);
        Socket socket = serverSocket.accept();

        InputStream inputStream = socket.getInputStream();
        // IO读取,使用字符流输入,使用InputStreamReader将InputStream转化为字符流
        BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
        String s = reader.readLine();
        System.out.println(s);

        // 字符流方式的输出
        OutputStream outputStream = socket.getOutputStream();
        BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(outputStream));
        writer.write("hello Client 字符流");
        writer.newLine(); // 插入一个换行符,代表写入的内容结束
        writer.flush(); // 字符流需要手动刷新,否则数据不会写入数据通道

        reader.close();
        writer.close();
        socket.close();
        serverSocket.close();
    }
}

客户端 

public class SocketTCP03Client {
    public static void main(String[] args) throws Exception{
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        OutputStream outputStream = socket.getOutputStream();
        // 通过输出流,转换为字符流
        BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(outputStream));
        writer.write("hello Server 字符流");
        writer.newLine(); // 插入一个换行符,代表写入的内容结束
        writer.flush(); // 字符流需要手动刷新,否则数据不会写入数据通道


        InputStream inputStream = socket.getInputStream();
        BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
        String s = reader.readLine();
        System.out.println(s);

        reader.close();
        writer.close();
        socket.close();
        System.out.println("客户端退出...");
    }
}

(四)案例四——使用字符流

 StreamUtils

/**
 * 此类用于演示关于流的读写方法
 *
 */
public class StreamUtils {
	/**
	 * 功能:将输入流转换成byte[], 即可以把文件的内容读入到byte[]
	 * @param is
	 * @return
	 * @throws Exception
	 */
	public static byte[] streamToByteArray(InputStream is) throws Exception{
		ByteArrayOutputStream bos = new ByteArrayOutputStream();//创建输出流对象
		byte[] b = new byte[1024];//字节数组
		int len;
		while((len=is.read(b))!=-1){//循环读取
			bos.write(b, 0, len);//把读取到的数据,写入bos	
		}
		byte[] array = bos.toByteArray();//然后将bos 转成字节数组
		bos.close();
		return array;
	}
	/**
	 * 功能:将InputStream转换成String
	 * @param is
	 * @return
	 * @throws Exception
	 */
	
	public static String streamToString(InputStream is) throws Exception{
		BufferedReader reader = new BufferedReader(new InputStreamReader(is));
		StringBuilder builder= new StringBuilder();
		String line;
		while((line=reader.readLine ())!=null){
			builder.append(line+"\r\n");
		}
		return builder.toString();
		
	}

}

服务端

public class TCPFileUploadServer {
    public static void main(String[] args) throws Exception {
        ServerSocket serverSocket = new ServerSocket(8888);
        // 等待连接
        Socket socket = serverSocket.accept();
        InputStream inputStream = socket.getInputStream();
        BufferedInputStream bis = new BufferedInputStream(inputStream);
        byte[] bytes = StreamUtils.streamToByteArray(bis);
        String destFilePath = "src\\a.jpg";
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(destFilePath));
        bos.write(bytes);
        bos.close();
        // 向客户端回复“收到图片”
        BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
        writer.write("收到图片");
        writer.flush();
        writer.close();
        bis.close();
        socket.close();
    }
}

客户端 

public class TCPFileUploadClient {
    public static void main(String[] args) throws  Exception{
        // 客户端连接服务端8888端口,得到socket对象
        Socket socket = new Socket(InetAddress.getLocalHost(), 8888);
        // 读取本地的文件转换为字节数组
        String filePath = "D:\\image\\a.jpg";
        FileInputStream fileInputStream = new FileInputStream(filePath);
        BufferedInputStream bis = new BufferedInputStream(fileInputStream);
        // 该字节数组就是文件
        byte[] bytes = StreamUtils.streamToByteArray(bis);
        OutputStream outputStream = socket.getOutputStream();
//        outputStream.write(bytes);
        BufferedOutputStream bos = new BufferedOutputStream(outputStream);
        bos.write(bytes);
        
        socket.shutdownOutput(); // 写入数据的结束标记
        BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String s = reader.readLine();
        System.out.println(s);
        reader.close();
        bos.close();
        bis.close();
        socket.close();
    }
}

五、UDP网络编程

(一)基本介绍

1.类 DatagramSocket DatagramPacket[数据包/数据报]实现了基于 UDP协议网络程序。

2.UDP数据报通过数据报套接字 DatagramSocket 发送和接收,系统不保证UDP数据报一定能够安全送到目的地,也不能确定什么时候可以抵达。

3.DatagramPacket 对象封装了UDP数据报,在数据报中包含了发送端的IP地址和端口号以及接收端的IP地址和端口号。

4.UDP协议中每个数据报都给出了完整的地址信息,因此无须建立发送方和接收方的连接

(二)基本流程

1.核心的两个类/对象 DatagramSocket与DatagramPacket

2.建立发送端,接收端(没有服务端和客户端概念)

3.发送数据前,建立数据包/报 DatagramPacket对象

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

5.关闭DatagramSocket

(三)案例

发送端

public class UDPSenderB {
    public static void main(String[] args) throws Exception{
        // 1.创建一个DatagramSocket,监听9998端口
        DatagramSocket socket = new DatagramSocket(9998);
        // 2.创建DatagramPacket,接收数据
        byte[] data = "你好,明天吃火锅".getBytes();
        DatagramPacket packet = new DatagramPacket(data, 0, data.length, InetAddress.getByName("192.168.23.1"), 9999);
        // 3.发送数据
        socket.send(packet);

        // 接收消息
        data = new byte[1024];
        packet = new DatagramPacket(data, data.length);
        socket.receive(packet);
        int length = packet.getLength();
        data = packet.getData();
        String msg = new String(data, 0, length);
        System.out.println(msg);
        // 4.关闭资源
        socket.close();
    }
}

接收端

public class UDPReceiverA {
    public static void main(String[] args) throws Exception{
        // 1.创建一个DatagramSocket对象,监听9999端口
        DatagramSocket socket = new DatagramSocket(9999);
        // 2.构建一个DatagramPacket,准备接受数据
        byte[] data = new byte[1024];
        DatagramPacket packet = new DatagramPacket(data, data.length);
        // 3.接收数据,如果没有数据则堵塞,接收到的数据存放在packet中
        System.out.println("接收端A等待接收数据...");
        socket.receive(packet);
        // 4.接收完数据转化为String并输出
        int length = packet.getLength();
        data = packet.getData();
        String msg = new String(data, 0, length);
        System.out.println(msg);

        // 回复消息
        data = "好的,明天见".getBytes();
        packet = new DatagramPacket(data, 0, data.length, InetAddress.getByName("192.168.23.1"), 9998);
        socket.send(packet);
        // 5.关闭资源
        socket.close();

    }
}

六、案例

(一)案例一

服务端

public class HomeWork01Server {
    public static void main(String[] args) throws Exception{
        System.out.println("服务器等待接收消息...");

        DatagramSocket socket = new DatagramSocket(9999);
        byte[] data = new byte[1024];
        DatagramPacket packet = new DatagramPacket(data, data.length);
        socket.receive(packet);

        int length = packet.getLength();
        data = packet.getData();
        String msg = new String(data, 0, length);
        System.out.println(msg);
        String returnMsg = "";
        if("name".equals(msg)) {
            returnMsg = "我是nova";
        } else if("hobby".equals(msg)){
            returnMsg = "编写java程序";
        } else {
            returnMsg = "你说啥呢";
        }
        data = returnMsg.getBytes();
        packet = new DatagramPacket(data, 0, data.length, InetAddress.getByName("192.168.23.1"), 9998);
        socket.send(packet);

        socket.close();
    }
}

客户端

public class HomeWork01Client {
    public static void main(String[] args) throws Exception{
        System.out.println("请输入您的消息:");

        Scanner sc = new Scanner(System.in);
        String msg = sc.next();
        DatagramSocket socket = new DatagramSocket(9998);
        byte[] data = msg.getBytes();
        DatagramPacket packet = new DatagramPacket(data, 0, data.length, InetAddress.getByName("192.168.23.1"), 9999);
        socket.send(packet);

        data = new byte[1024];
        packet = new DatagramPacket(data, data.length);
        socket.receive(packet);
        int length = packet.getLength();
        data = packet.getData();
        System.out.println(new String(data, 0, length));

    }
}

(二)案例二

服务端

public class HomeWork03Server {
    public static void main(String[] args) throws Exception{
        System.out.println("服务端监听9999端口,等待接收文件名...");
        // 1.创建一个socket对象
        ServerSocket serverSocket = new ServerSocket(9999);
        // 2.接收传递过来的文件名
        Socket socket = serverSocket.accept();
        InputStream inputStream = socket.getInputStream();
        // 这里使用了while循环,是考虑客户端可能发送的文件名较大的情况
        byte[] buf = new byte[1024];
        int len = 0;
        String downLoadFileName = "";
        while((len = inputStream.read(buf)) != -1) {
            downLoadFileName += new String(buf, 0, len);
        }

        System.out.println("客户端希望下载的文件名: " + downLoadFileName);
        // 3.确定文件名
        // 如果客户端要求下载高山流水,则获取文件流返回,否则,返回无名mp3
        String filePath = "";
        if("高山流水".equals(downLoadFileName)) {
            filePath = "src//高山流水.mp3";
        } else {
            filePath = "src//无名.mp3";
        }
        // 4.获取文件输入流并转化为字节数组
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream(filePath));
        byte[] data = StreamUtils.streamToByteArray(bis);
        // 5.获取socket输出流并写出数据
        BufferedOutputStream bos = new BufferedOutputStream(socket.getOutputStream());
        bos.write(data);
        socket.shutdownOutput();
        // 6.关闭资源
        bos.close();
        bis.close();
        socket.close();
        serverSocket.close();

        System.out.println("服务端传输文件完毕,退出");
    }
}

客户端

public class HomeWork03Client {
    public static void main(String[] args) throws Exception{
        System.out.println("请输入下载的文件名:");
        // 1.获取用户输入的文件名
        Scanner sc = new Scanner(System.in);
        String fileName = sc.next();
        // 2.创建连接
        Socket socket = new Socket(InetAddress.getLocalHost(), 9999);
        // 3.获取socket关联的输出流并写出
        OutputStream outputStream = socket.getOutputStream();
        outputStream.write(fileName.getBytes());
        //   设置写入结束的标记
        socket.shutdownOutput();
        // 4.获取输入流并转化为字节数组
        BufferedInputStream bis = new BufferedInputStream(socket.getInputStream());
        byte[] data = StreamUtils.streamToByteArray(bis);

        // 5.获取写出流,将文件写入磁盘
        String fileNamePath = "src//com//hspedu/homework//" + fileName + ".mp3";
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(fileNamePath));
        bos.write(data);

        // 6.关闭资源
        outputStream.close();
        bos.close();
        bis.close();
        socket.close();

        System.out.println("文件下载完毕,退出");

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/575050.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据结构 第四章:串

文章目录 一、串的定义和实现1.1串的定义和基本操作1.1.1串的定义1.1.2串的基本操作1.1.3小结 1.2串的存储结构1.2.1顺序存储1.2.2链式存储1.2.3基于顺序存储实现基本操作1.2.4小结 二、串的模式匹配2.1什么是字符串的模式匹配2.2朴素模式匹配算法2.3KMP算法2.4求next数组2.5KM…

python+vue空巢老人网上药店购药系统9h2k5

本空巢老人购药系统主要包括三大功能模块,即用户功能模块、家属功能模块和管理员功能模块。 (1)管理员模块:系统中的核心用户是管理员,管理员登录后,通过管理员功能来管理后台系统。主要功能有:…

LCD 显示

概述 LCD显示控制模块接收 MCU 送过来的数据,按一定规律储存在显示 RAM 中,并根据显示 RAM 中的数据驱动 LCD 显示屏来实现期望的字符显示功能。 主要特点: ⚫ 最大支持 840 、 642 、 444 的显示段数 ⚫ 1/3bias 、 1/4bia s ⚫ 16 级灰度可…

还在分不清各种IO模型?

IO模型 我们的程序基本上都是对数据的IO操作以及基于CPU的运算。 基于Java的开发大部分是网络相关的编程,不管是基于如Tomcat般的Web容器,或是基于Netty开发的应用间的RPC服务。为了提供系统吞吐量, 降低硬件资源的开销,IO模型也…

chatgpt赋能python:Python:寻找一组数中的最大值和最小值

Python:寻找一组数中的最大值和最小值 在数值计算和数据处理中,常常需要找出一组数中的最大值和最小值。Python提供了一些内置函数和库来实现这个任务。在本文中,我们将介绍如何使用Python来查找一组数的最大值和最小值,并提供一…

从C语言到C++_13(string的模拟实现)深浅拷贝+传统/现代写法

前面已经对 string 类进行了简单的介绍和应用,大家只要能够正常使用即可。 在面试中,面试官总喜欢让学生自己 来模拟实现string类, 最主要是实现string类的构造、拷贝构造、赋值运算符重载以及析构函数。 为了更深入学习STL,下面我…

JetBrains的Python集成开发环境Pycharm 2023版本在Win10系统的下载与安装配置教程

目录 前言一、PyCharm安装二、使用配置总结 前言 PyCharm是一款专为Python语言开发人员设计的集成开发环境(IDE)。它提供了丰富的功能和工具,可以帮助开发人员更高效地编写、调试和部署Python应用程序。 PyCharm的主要特点: —…

chatgpt赋能python:Python按键的SEO优化

Python按键的SEO优化 Python是一种广泛使用的编程语言之一,它在机器学习、数据科学和网络编程等领域中广泛应用。在Python中,按键是一个重要的概念,它可以让你控制程序的流程。在本文中,我们将介绍Python中按键的基本概念&#x…

JetBrains的Python集成开发环境Pycharm 2023版本在Linux系统的下载与安装配置教程

目录 前言一、PyCharm安装二、使用配置总结 前言 PyCharm是一款专为Python语言开发人员设计的集成开发环境(IDE)。它提供了丰富的功能和工具,可以帮助开发人员更高效地编写、调试和部署Python应用程序。注:已在CentOS7.9和Ubuntu…

vue3的api解读-VUE Reactivity

目录 响应式 命令式程序 vs 响应式程序 Be Reactive:将响应式的值绑定到UI 代理模式(CODING) /src/App.tsx /src/main.ts /vite.config.ts /src/demo/ProxyExample.ts Reactivity是Vue3新加的一个库,新的package 响应式 …

【软考系统规划与管理师笔记】第2篇 信息技术知识1

目录 1 软件工程 1.1 软件需求分析与定义 1.2 软件设计、测试与维护 1.3 软件质量保证及质量评价 1.4 软件配置管理 1.5 软件过程管理 1.6 软件复用 2 面向对象系统分析与设计 2.1 面向对象设计的基本概念 2.2统一建模语言与可视化建模 3. 应用集成技术 3.1 数据库与…

低代码平台中的分布式 RPC 框架 (约 3000 行代码)

RPC 是分布式系统设计中不可或缺的一个部分。国内开源的 RPC 框架很多,它们的设计大都受到了 dubbo 框架的影响,核心的抽象概念与 dubbo 类似。从今天的角度上看,dubbo 的设计已经过于繁琐冗长,如果基于现在的技术环境&#xff0c…

Go语言WEB开发[html/template包]

Go语言Web开发 Go语言提供了html/template包来支持模板渲染。Go提供的html/template包对HTML模板提供了丰富的模板语言,主要用于Web应用程序。 模板中的变量 模板中的变量通过{{.}} 来访问。 {{.}} 称为管道和root。 在模板文件内,{{.}}代表当前变量&…

【ROS】ROS2命令行工具详解

1、简介 ROS1中每个功能都使用ros开头的命令行工具&#xff0c;如&#xff1a;roscore、rosrun等 ROS2中只有一个命令行工具ros2&#xff0c;各个功能模块通过参数子命令来实现 ~$ ros2 -h 用法: ros2 [-h] [--use-python-default-buffering] <command> ……选项:-h, …

二叉树的认识

愚昧将使你达不到任何成果&#xff0c;并在失望和忧郁之中自暴自弃。 --达芬奇 目录 &#x1f341;一.二叉树的概念 &#x1f341;二.二叉树的特点&#xff0c;结构 &#x1f341;三.三种特殊的二叉树 &#x1f341;1.斜树 &#x1f341;2.满二叉树 …

Learning C++ No.24 【哈希/散列实战】

引言&#xff1a; 北京时间&#xff1a;2023/5/20/7:30&#xff0c;周六&#xff0c;可惜有课&#xff0c;而且还是早八&#xff0c;说明我们现在没有多少的学习时间啦&#xff01;得抓紧把该博客的引言给写完&#xff0c;我们距离期末考越来越近啦&#xff01;再过一个星期就…

客服系统搭建,这些技能你必须知道

作为客服系统领域的专家&#xff0c;我来为大家介绍一些客服系统搭建的技能。在本文中&#xff0c;你将了解到如何搭建一个高效的客服系统以及如何应对可能出现的问题。 选择合适的客服系统 在选择客服系统时&#xff0c;需要考虑以下几个因素&#xff1a; 处理的工单量用户…

《人月神话》译文修订明细(3)-读者可以对照修改

《人月神话》译文修订明细&#xff08;1&#xff09;-读者可以对照修改 《人月神话》译文修订明细&#xff08;2&#xff09;-读者可以对照修改 《人月神话》译文修订如下&#xff0c;读者可以对照自己手上的书修改。 相关阅读 这回真要动刀子-征集《人月神话》中译本的翻译…

java开发学习框架

Java基础 1.1. Java简介与安装 1.2. Java基本语法 1.3. 数据类型与变量 1.4. 运算符与表达式 1.5. 流程控制&#xff08;分支与循环&#xff09; 1.6. 数组 面向对象编程 2.1. 类与对象 2.2. 继承与多态 2.3. 接口与抽象类 2.4. 封装与访问控制 2.5. 重载与覆盖 2.6. Java内存管…

【面试题】 ES6中将非数组转换为数组的三种方法

大厂面试题分享 面试题库 前后端面试题库 &#xff08;面试必备&#xff09; 推荐&#xff1a;★★★★★ 地址&#xff1a;前端面试题库 web前端面试题库 VS java后端面试题库大全 我们常常想使用数组的方法&#xff0c;比如forEach&#xff0c;filter&#xff0c;又或者so…