当前位置: 首页 > news >正文

搭建Tomcat(四)---Servlet容器

目录

引入

Servlet容器

一、优化MyTomcat

①先将MyTomcat的main函数搬过来:

②将getClass()函数搬过来

 ③创建容器

 ④连接ServletConfigMapping和MyTomcat

连接:

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping

b.MyTomcat

二、优化Server

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

方法二: 直接嵌套

三、获取类对象

处理请求:

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

结果:


引入

在先前的tomcat搭建学习中,已经对tomcat的雏形做了基本的实现,即如下的过程:

接下来继续tomcat的搭建。

Servlet容器

目前Servlet容器:

 接下来,我们来优化一下MyTomcat:

一、优化MyTomcat

首先在tomcat包下创建一下config包,然后在里面创建文件ServletConfigMapping:

那么ServletConfigMapping里面编写什么呢?就是从MyTomcat中编写的内容,现在需要copy到这个文件中,实现MyTomcat的解放:

【ServletConfigMapping在tomcat中就是为了获取配置信息;而MyTomcat中写的,正是扫描的过程,即未来扫描动态资源映射表的过程,所以现在将它放进合适的地方,即ServletConfigMapping里面】

注意: 不是原封不动的照搬,需要做一些小小的调整。

①先将MyTomcat的main函数搬过来:

注意这里做了个小调整,将MyTomcat的main函数里面的内容放进了代码块中;

static代码块的特点:在main函数执行之前先执行

static{try {// 1. 扫描包路径 (com.wzh.tomcat.myweb)String packageName = "com.qcby.tomcat.myweb";List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中// 2. 遍历所有类,检查是否有@WebServlet注解for (Class<?> clazz : classes) {if (clazz.isAnnotationPresent(WebServlet.class)) {// 3. 获取@WebServlet注解的值WebServlet webServlet = clazz.getAnnotation(WebServlet.class);System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);}}} catch (Exception e) {e.printStackTrace();}}

②将getClass()函数搬过来

 这里直接copy就好,放到刚才的static代码块之下:

/*** 获取指定包下的所有类** @param packageName 包名,例如 "com.qcby.tomcat.myweb"* @return 类对象列表* @throws Exception*///将MyTomcat中的getClass方法也粘贴过来private static List<Class<?>> getClasses(String packageName) throws Exception {List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中String path = packageName.replace('.', '/'); // 将包名转换为文件路径// 通过类加载器获取包的资源路径ClassLoader classLoader = Thread.currentThread().getContextClassLoader();Enumeration<URL> resources = classLoader.getResources(path);while (resources.hasMoreElements()) {URL resource = resources.nextElement();File directory = new File(resource.toURI());// 扫描文件夹下的所有类文件if (directory.exists()) {for (File file : directory.listFiles()) {if (file.getName().endsWith(".class")) {    //获得.class文件// 获取类的完整类名String className = packageName + "." + file.getName().replace(".class", "");classes.add(Class.forName(className));}}}}return classes;}

 基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可:

public static void main(String[] args) {//当然也可以直接将static中的内容放进main函数里面(不常用)}

 ③创建容器

真正的servlet容器:

public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();

/*
* 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态
* 多态--父类的引用指向子类的对象
* 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象
* 这些类对象都继承了HttpServlet
* 即:
* MyFirstServlet的类对象--->是HttpServlet的子类
* MySecondServlet的类对象--->是HttpServlet的子类
* MyThirdServlet的类对象--->是HttpServlet的子类
*【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】
* */

 ④连接ServletConfigMapping和MyTomcat

我们只是想优化MyTomcat,而不是彻底换掉MyTomcat,所以现在我们要做的就是将创建的ServletConfigMapping和先前的MyTomcat连接起来:

当然由于主要代码已经放进ServletConfigMapping中了,所以原先的MyTomcat可以清空了。

连接:

注意②中的执行方法---放main;那么想要连接ServletConfigMapping和MyTomcat,我们依旧可以采用这种方法:让MyTomcat调用ServletConfigMapping中的方法,迫使ServletConfigMapping中的static代码块运行。

public class MyTomcat {public static void main(String[] args) {ServletConfigMapping.init();//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接}
}

当然这个写在ServletConfigMapping中的方法,可以是空的。

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping
package com.qcby.tomcat.config;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.webservlet.WebServlet;import java.io.File;
import java.net.URL;
import java.util.*;/*
* servlet容器
* */public class ServletConfigMapping {//真正的servlet容器/** 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态* 多态--父类的引用指向子类的对象* 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象* 这些类对象都继承了HttpServlet* 即:* MyFirstServlet的类对象--->是HttpServlet的子类* MySecondServlet的类对象--->是HttpServlet的子类* MyThirdServlet的类对象--->是HttpServlet的子类*【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】* */public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();//扫描遍历(即MyTomcat之中的内容)//static代码块在main方法执行之前执行//将MyTomcat中的main方法放进代码块中static{try {// 1. 扫描包路径 (com.wzh.tomcat.myweb)String packageName = "com.qcby.tomcat.myweb";List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中// 2. 遍历所有类,检查是否有@WebServlet注解for (Class<?> clazz : classes) {if (clazz.isAnnotationPresent(WebServlet.class)) {// 3. 获取@WebServlet注解的值WebServlet webServlet = clazz.getAnnotation(WebServlet.class);System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);}}} catch (Exception e) {e.printStackTrace();}}/*** 获取指定包下的所有类** @param packageName 包名,例如 "com.qcby.tomcat.myweb"* @return 类对象列表* @throws Exception*///将MyTomcat中的getClass方法也粘贴过来private static List<Class<?>> getClasses(String packageName) throws Exception {List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中String path = packageName.replace('.', '/'); // 将包名转换为文件路径// 通过类加载器获取包的资源路径ClassLoader classLoader = Thread.currentThread().getContextClassLoader();Enumeration<URL> resources = classLoader.getResources(path);while (resources.hasMoreElements()) {URL resource = resources.nextElement();File directory = new File(resource.toURI());// 扫描文件夹下的所有类文件if (directory.exists()) {for (File file : directory.listFiles()) {if (file.getName().endsWith(".class")) {    //获得.class文件// 获取类的完整类名String className = packageName + "." + file.getName().replace(".class", "");classes.add(Class.forName(className));}}}}return classes;}//基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可public static void main(String[] args) {//当然也可以直接将static中的内容放进main函数里面(不常用)}public static void init(){}}
b.MyTomcat
package com.qcby.tomcat;import com.qcby.tomcat.config.ServletConfigMapping;public class MyTomcat {public static void main(String[] args) {ServletConfigMapping.init();//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接}
}

二、优化Server

用同样的方法,首先把server的main函数解放掉:

【将main函数中的内容取出,并放进新创的serverInit函数中,删除原来的main函数即可。】

//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

接着用同样的操作,在MyTomcat中调用这新创建的方法:

package com.qcby.tomcat;import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;public class MyTomcat {public static void main(String[] args) {try {//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接ServletConfigMapping.init();//初始化servlet容器Server.serverInit();//启动server服务} catch (Exception e) {e.printStackTrace();}}
}

【看着代码变多了很多,实际上就加了一行,然后为了安全性,抛出了异常】

方法二: 直接嵌套

这里我们用另一种方法去优化Server,即将Server和MyTomcat连接起来:

即直接将方才做过修改的代码粘贴到MyTomcat中,并且在MyTomcat的main函数中调用Server的serverInit(即过去Server类的main函数)

如下是这一小阶段完成后的MyTomcat代码示意:

package com.qcby.tomcat;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;public class MyTomcat {//实例化Requestpublic static Request request = new Request();public static void main(String[] args) throws Exception {ServletConfigMapping.init();serverInit();}//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}public static void handler(Socket socket) throws Exception {//读取请求的数据InputStream inputStream = socket.getInputStream();requestContext(inputStream);}public static void requestContext(InputStream inputStream) throws IOException { //获取全部信息//将bit流转为文字信息int count = 0;while (count == 0) {count = inputStream.available();}byte[] bytes = new byte[count];inputStream.read(bytes);String Context = new String(bytes);System.out.println(Context);//解析数据if (Context.equals("")) {System.out.println("你输入了一个空请求");} else {String firstLine = Context.split("\\n")[0];String method = firstLine.split("\\s")[0];String path = firstLine.split("\\s")[1];System.out.println(method + " " + path);//任何请求都会被打到这个类中,随后就会被解析//将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)request.setMethod(method);request.setPath(path);}}
}

三、获取类对象

前面已经对MyTomcat做了优化,下一步就是去获取类对象;

如何获取类对象呢?简单来说谁存储着类对象的信息呢?---Request

那么就要接着对server类做出优化---因为目前Server和MyTomcat类还是存在问题,两者打不通;

既然想把两者联系起来,那么就要有嵌套调用或是直接嵌套。

在这里,我们依旧是用直接嵌套的方法写:

处理请求:

处理接收到的请求,首先在Map映射中寻找是否能够匹配上key值,即传送的path值

能查到则取获取Map的value值,即类对象: 

 //处理请求public static void dis(Request request) throws Exception {if(!request.getPath().equals("")){if(ServletConfigMapping.classMap.get(request.getPath())!=null){//能进到这里,则说明能够获取Map中对应的key值,能够匹配上Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)servlet.doGet(request,response);}}}

上面的代码最终是要被requestContext(InputStream inputStream)函数调用的。

即最终的MyTomcat代码:

package com.qcby.tomcat;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;public class MyTomcat {//实例化Requestpublic static Request request = new Request();public static Response response = new Response();public static void main(String[] args) throws Exception {//两项调用连接ServletConfigMapping.init();serverInit();}//处理请求public static void dis(Request request) throws Exception {if(!request.getPath().equals("")){if(ServletConfigMapping.classMap.get(request.getPath())!=null){//能进到这里,则说明能够获取Map中对应的key值,能够匹配上Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)servlet.doGet(request,response);}}}//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}public static void handler(Socket socket) throws Exception {//读取请求的数据InputStream inputStream = socket.getInputStream();requestContext(inputStream);}public static void requestContext(InputStream inputStream) throws Exception { //获取全部信息//将bit流转为文字信息int count = 0;while (count == 0) {count = inputStream.available();}byte[] bytes = new byte[count];inputStream.read(bytes);String Context = new String(bytes);System.out.println(Context);//解析数据if (Context.equals("")) {System.out.println("你输入了一个空请求");} else {String firstLine = Context.split("\\n")[0];String method = firstLine.split("\\s")[0];String path = firstLine.split("\\s")[1];System.out.println(method + " " + path);//任何请求都会被打到这个类中,随后就会被解析//将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)request.setMethod(method);request.setPath(path);}//调用上面的处理函数dis(request);}
}

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

 输入对应key值的:

结果:

取看输出这里:
 可以看出,上述代码被顺利执行了。

相关文章:

搭建Tomcat(四)---Servlet容器

目录 引入 Servlet容器 一、优化MyTomcat ①先将MyTomcat的main函数搬过来&#xff1a; ②将getClass()函数搬过来 ③创建容器 ④连接ServletConfigMapping和MyTomcat 连接&#xff1a; ⑤完整的ServletConfigMapping和MyTomcat方法&#xff1a; a.ServletConfigMappin…...

PT2044A 单触控单输出IC

1 产品概述 ● PT2044A 是一款单通道触摸检测芯片。该芯片内建稳压电路&#xff0c;提供稳定电压给触摸感应电路使用。同时内部集成高效完善的触摸检测算法&#xff0c;使得芯片具有稳定的触摸检测效果。该芯片专为取代传统按键而设计&#xff0c;具有宽工作电压与低功耗的特性…...

docker安装mysql5.7

1、宿主机创建映射目录 mkdir -p /data/mysql/log mkdir -p /data/mysql/data mkdir -p /data/mysql/conf这里我放在了/data/mysql目录下 2、拉取mysql镜像 docker pull mysql:5.7注意是5.7版本&#xff0c;如果是8版本操作会略有不同&#xff0c;下篇文章介绍安装8版本的操…...

安卓 文件管理相关功能记录

文件管理细分为图片、视频、音乐、文件四类 目录 权限 静态声明权限 动态检查和声明权限方法 如何开始上述动态申请的流程 提示 图片 获取图片文件的对象列表 展示 删除 视频 获取视频文件的对象列表 获取视频file列表 按日期装载视频文件列表 展示 播放 删除…...

GB28181系列三:GB28181流媒体服务器ZLMediaKit

我的音视频/流媒体开源项目(github) GB28181系列目录 目录 一、ZLMediaKit介绍 二、 ZLMediaKit安装、运行(Ubuntu) 1、安装 2、运行 3、配置 三、ZLMediaKit使用 一、ZLMediaKit介绍 ZLMediaKit是一个基于C11的高性能运营级流媒体服务框架&#xff0c;项目地址&#xf…...

ScottPlot学习的常用笔记

ScottPlot学习的常用笔记 写在前面版本的选择第一个障碍&#xff1a;版本问题。 ScottPlot4.0的官方网站与示例官方起始页cookbook5.0Demo4.1 demo以4.1为例&#xff0c;解压和运行如下&#xff1a; 下载源代码和编译先说结论&#xff1a; 写在前面 之前调研的TraceCompass&am…...

二、mapbox-gl实现白膜立体建筑

有时候我们只有二维的面数据&#xff0c;怎么实现类似高德地图中的白膜立体建筑呢&#xff1f;在mapbox-gl中很容易实现&#xff0c;具体如下。 要在Vue中结合Mapbox显示自定义的GeoJSON数据&#xff0c;并实现建筑物的白膜效果&#xff0c;我们需要执行以下步骤&#xff1a; …...

mybatisplus 分库查询

mybatisplus 分库查询 比如我们的项目有两个数据库 不同的表在不同的库 我们是可以使用mybatisplus来实现 首选引入pom <dependency><groupId>com.baomidou</groupId><artifactId>dynamic-datasource-spring-boot-starter</artifactId><vers…...

计算属性 (vue3)

二 实例 1route.query.type...

RabbitMQ 安装、配置和使用介绍 使用前端js直接调用方式

1. 安装 RabbitMQ 1.1 安装 Erlang RabbitMQ 是基于 Erlang 语言开发的&#xff0c;因此首先需要安装 Erlang。 在 Ubuntu 上安装 Erlang&#xff1a; bash sudo apt-get update sudo apt-get install erlang 在 CentOS 上安装 Erlang&#xff1a; bash sudo yum insta…...

电脑显示器选购指南2024

选择显示器是五花八门的显示参数&#xff0c;如何选择&#xff0c;以下给出参数说明&#xff0c;及部分参考&#xff1a; 1. 尺寸和分辨率 尺寸&#xff08;英寸&#xff09; 根据使用距离和用途选择合适的屏幕尺寸&#xff1a; 21-24 英寸&#xff1a;适合小桌面空间、日常…...

vue2中如何实现自定义指令

实现自动聚焦功能 1.不用自定义指令 使用生命周期钩子mounted 2.使用自定义指令 1. 使用全局组件 首先在main.js中注册 然后在组件中直接使用v-指令名 2. 使用局部注册 局部注册和全局注册类似 如果很多组件需要自定义&#xff0c;建议使用自定义指令注册在全局...

QT从入门到精通(一)——Qlabel介绍与使用

1. QT介绍——代码测试 Qt 是一个跨平台的应用程序开发框架&#xff0c;广泛用于开发图形用户界面&#xff08;GUI&#xff09;应用程序&#xff0c;也支持非图形应用程序的开发。Qt 提供了一套工具和库&#xff0c;使得开发者能够高效地构建高性能、可移植的应用程序。以下是…...

【jpa】springboot使用jpa示例

目录 1. 请求示例2. pom依赖3. application.yaml4.controller5. service6. repository7. 实体8. 启动类 1. 请求示例 curl --location --request POST http://127.0.0.1:8080/user \ --header User-Agent: Apifox/1.0.0 (https://apifox.com) \ --header Content-Type: applic…...

Python Flask Web框架快速入门

Flask 入门Demo Flask 开发环境搭建&#xff0c;执行如下指令&#xff1a; pip install flask# 第一节: Flask 快速入门from flask import Flask app Flask(__name__)app.route(/flask) def hello_flask():return Hello Flaskapp.run()核心代码剖析&#xff1a; 从flask包导…...

ansible自动化运维(五)roles角色管理

目录 Roles角色管理 创建nginx的角色目录 定义任务配置文件&#xff1a; 定义jinjia2模板&#xff1a; 定义nginx yml文件 检查语法&#xff0c;执行命令 检验结果&#xff1a; Roles角色管理 角色&#xff08;roles&#xff09;是ansible自1.2版本开始引入的新特性&…...

前端学习一

一 进程与线程 线程是进程执行的最小单位&#xff0c;进程是系统分配任务的最小单位。 一个进程可执行最少一个线程。线程分为子线程和主线程。 主线程关闭则子线程关闭。 二 浏览器进程 浏览器是多进程多线程应用。 进程包括&#xff1a; 浏览器进程 负责程序交互渲染…...

【OSS】php使用oss存储

阿里云oss官方文档&#xff1a;文档 1、前期工作 创建阿里云账号&#xff0c;登录创建bucket&#xff0c;注意修改权限&#xff0c;要不然可能读取不到 申请accessKeyId和accessKeySecret accessKey 2、项目中安装OSS扩展 composer require aliyuncs/oss-sdk-php3、基础使…...

UE5 C+、C++、C# 构造方法区别示例

我们对比一下UE C、C 、C#的构造方法&#xff1a; 1. UE4 C例子&#xff1a; // 声明和构造合并在一起static ConstructorHelpers::FObjectFinder<UTexture2D> CrosshairTexObj(TEXT("/Game/Path"));// 使用加载的资源UTexture2D* Texture CrosshairTexObj.…...

leetcode-146.LRU缓存(易理解)

为了实现一个满足 LRU&#xff08;最近最少使用&#xff09;缓存约束的数据结构&#xff0c;我们需要在 (O(1)) 时间复杂度内完成 get 和 put 操作。这通常可以通过结合使用哈希表和双向链表来实现&#xff1a; 哈希表&#xff1a;用于在 (O(1)) 时间复杂度内实现对缓存中元素…...

JavaSe部分总结

我们先来了解一下Java语言,JavaSE是Java编程语言的标准版,主要是来学习Java的基本语法,书写方式,以及一些简单的逻辑循环和判断,包括一些关键字,特殊类(抽象类),特殊的方法(static修饰的方法,final修饰的方法)等等,最重要的是Java语言是比较C语言和C语言是比较简单的,Java是面向…...

iPhone批量删除照片的方法

对于每一个iPhone用户来说&#xff0c;照片管理是一项日常而重要的任务。随着时间的积累&#xff0c;无数的照片快速填满了我们的存储空间&#xff0c;从美丽的风景到重要的家庭聚会&#xff0c;每一张照片都记录着我们生活中的瞬间。然而&#xff0c;当存储空间即将耗尽时&…...

红日靶场vulnstack 7靶机的测试报告[细节](一)

目录 一、测试环境 1、系统环境 2、注意事项 3、使用工具/软件 二、测试目的 三、操作过程 1、信息搜集 2、Redis未授权访问漏洞获取web1靶机系统权限 3、获取docker靶机系统权限 ①Laravel框架漏洞利用getshell ②Laravel主机的提权&&docker容器逃逸 提权…...

ubuntu+ros新手笔记(二):古月·ROS2入门21讲学习笔记

系统ubuntu22.04 ros2 humble 按照如下视频教程学习的&#xff1a;【古月居】古月ROS2入门21讲 | 带你认识一个全新的机器人操作系统 此处仅记录我报错的地方&#xff0c;以及相应的解决方案&#xff0c;没有出错的略过&#xff01; 对应的古月居ROS2入门21讲源码下载地址&a…...

Harmonyos之深浅模式适配

Harmonyos之换肤功能 概述实现原理颜色适配颜色资源配置工具类编写界面代码编写适配效果 概述 深色模式&#xff08;Dark Mode&#xff09;又称之为暗色模式&#xff0c;是与日常应用使用过程中的浅色模式&#xff08;Light Mode&#xff09;相对应的一种UI主题。 换肤功能应…...

牛客网 SQL2查询多列

SQL2查询多列 select device_id,gender,age,university //查询哪些字段 from user_profile //从哪个表中查找 每日问题 C 中面向对象编程如何处理异常&#xff1f; 在C中&#xff0c;面向对象编程&#xff08;OOP&#xff09;处理异常主要通过异常处理机制来实现。C 提供了…...

Angular由一个bug说起之十二:网页页面持续占用CPU过高

随着网络日益发达&#xff0c;网页的内容也更加丰富&#xff0c;形式也更加多样化。而随之而来的性能问题也不容小觑。这篇文章我会根据我在实践中遇到的一个问题来总结&#xff0c;我在面对性能问题的一些解决步骤&#xff0c;希望能对大家有所启发。 查找问题原因 我接触的…...

【从零开始入门unity游戏开发之——C#篇05】转义字符、@处理多行文本或者不使用转义字符、随机数

文章目录 一、转义字符1、什么是转义字符&#xff1f;2、常见的转义字符3、总结 二、使用处理多行文本或者不使用转义字符1、多行字符串2、不使用转义字符 三、随机数1、Random.Next()生成随机整数示例&#xff1a;生成一个随机整数生成指定范围内的随机整数 2、Random.NextSin…...

我们来对接蓝凌OA --报文格式

题记 数智化办公专家、国家高新技术企业、知识管理国家标准制定者、信创供应商10强…等等&#xff0c;这些和咱们有关系吗&#xff01;&#xff01;不好意思&#xff0c;走错片场了&#xff0c;刚和项目经理在甲方那边吹牛B想想刚刚的大饼&#xff0c;看看支付宝余额&#xff…...

旅游系统旅游小程序PHP+Uniapp

旅游门票预订系统&#xff0c;支持景点门票、导游产品便捷预订、美食打卡、景点分享、旅游笔记分享等综合系统 更新日志 V1.3.0 1、修复富文本标签 2、新增景点入驻【高级版本】3、新增门票核销【高级版】4、新增门票端口【高级版】...