自己手写tomcat项目
一:Servlet的原理
在Servlet(接口中)有:
1.init():初始化servlet
2.getServletConfig():获取当前servlet的配置信息
3.service():服务器(在HttpServlet中实现,目的是为了更好的匹配http的请求方式)
4.getServletInfo():获取servlet当前运行过程中的信息
5.destroy():销毁,回收内存
在I/O中应包含:1.请求头(f12+网络) 2.请求方式(get/post) 3.请求内容
get请求:将请求的内容放在url中(相对不安全),url长度有限(导致发送的内容不能太长),get请求做查询
post请求:请求内容放在请求体当中,无法看到(相对安全),一般用来做文件上传、下载,post请求做增删改操作
在ServletRequest中应该有:1.method:请求方式 2.编码方式 3.parmater 4.url 5.cookie
在ServletResponse中应该有:1.状态码 2.编码方式 3.字符集 4.data数据
每一个和外界进行通讯的进程(端口号0~65535)
端口号区分当前的进程
serverSocket.accept():阻塞监听(停在这里等到程序的到来)
二:tomcat原理
下面用一个图给大家展示
注解:给程序看的(@webservlet等等)
注释:给人看得(//)
实现自定义注解:需要使用jdk提供元注解(主要使用前两个)
1.@Target注解(用来描述注解的使用范围)
2.@Retention注解(表示这个注解在什么时候还有效,用于描述注解的生命周期)
3.@Documented注解
4.@Inherited注解 这四个帮助实现自定义注解
前两个的使用方法
1.
2.
利用上面两个就能够实现自定义注解(下面就是实现)
三、手写tomcat
首先先创建idea项目(名称和位置可以随意)
在创建好项目之后会有一个.java文件(可删可不删)
之后在创建下面几个软件包和MyTomcat文件(MyTomcat和刚开始有的那个java文件不一样)
之后在每个软件包中创建好java文件和注解文件、接口文件(其中zj软件包中的注解文件可用自己的姓名起,因为是自己手写的tomcat文件)
创建完成之后,首先在HttpServletRequest和HttpServletResponse中分别写
package com.qcby.request;public class HttpServletRequest {private String method;private String url;public String getMethod() {return method;}public void setMethod(String method) {this.method = method;}public String getUrl() {return url;}public void setUrl(String url) {this.url = url;}
}
package com.qcby.response;import com.qcby.request.HttpServletRequest;import java.io.IOException;
import java.io.OutputStream;public class HttpServletResponse {private OutputStream outputStream;public HttpServletResponse(OutputStream outputStream){this.outputStream = outputStream;}public void writeServlet(String context) throws IOException {outputStream.write(context.getBytes());}
}
写好之后就可以在写servlet软件包中的内容了
这是接口
package com.qcby.servlet;import com.qcby.request.HttpServletRequest;
import com.qcby.response.HttpServletResponse;import java.io.IOException;public interface servlet {public void service(HttpServletRequest request, HttpServletResponse response) throws IOException;
}
这是java
package com.qcby.servlet;import com.qcby.request.HttpServletRequest;
import com.qcby.response.HttpServletResponse;import java.io.IOException;public abstract class HttpServlet implements servlet{public void doPost(HttpServletRequest request, HttpServletResponse response) {}public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {}@Overridepublic void service(HttpServletRequest request, HttpServletResponse response) throws IOException {if(request.getMethod().equals("GET")){doGet(request,response);}else if(request.getMethod().equals("POST")){doPost(request,response);}}
}
写关于自己的注解
package com.qcby.zj;import java.lang.annotation.*;@Target(ElementType.TYPE)//该注解用在类上面
@Retention(RetentionPolicy.RUNTIME)//在运行期间表达
public @interface LRHServlet {String url();
}
之后写myweb中的
package com.qcby.myweb;import com.qcby.request.HttpServletRequest;
import com.qcby.response.HttpServletResponse;
import com.qcby.servlet.HttpServlet;
import com.qcby.util.ResponseUtil;
import com.qcby.zj.LRHServlet;import java.io.IOException;@LRHServlet(url="/myservlet")
public class MyFirstServlet extends HttpServlet {@Overridepublic void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {System.out.println("hello word");response.writeServlet(ResponseUtil.getResponseHeader200("First hello word"));}@Overridepublic void doPost(HttpServletRequest request,HttpServletResponse response){}
}
package com.qcby.myweb;import com.qcby.request.HttpServletRequest;
import com.qcby.response.HttpServletResponse;
import com.qcby.servlet.HttpServlet;
import com.qcby.zj.LRHServlet;@LRHServlet(url="/insert")
public class insertServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response){System.out.println("I am insert");}@Overridepublic void doPost(HttpServletRequest request,HttpServletResponse response){}
}
util中放的是工具类,可直接复制
package com.qcby.util;import java.io.File;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;/*** 扫描指定包,获取该包下所有的类的全路径信息*/
public class SearchClassUtil {/*** 扫描指定包路径下的所有类* @param basePack 需要扫描的包名* @return 类的全路径列表*/public static List<String> searchClass(String basePack) {//创建存储类路径的列表List<String> classPaths = new ArrayList<>();try {// 获取类加载器(用于查找类路径资源)ClassLoader classLoader = Thread.currentThread().getContextClassLoader();// 将包名转换为路径格式(如 com.qcby.test)转换为路径格式(com/qcby/test)String path = basePack.replace('.', '/');// 获取资源枚举通过类加载器获取该路径下的所有资源(可能来自文件系统或 JAR 包)Enumeration<URL> resources = classLoader.getResources(path);//resources 是通过类加载器获取的资源枚举(包含文件系统和 JAR 包中的资源)while (resources.hasMoreElements()) {//每次循环处理一个资源 URLURL resource = resources.nextElement();// 判断资源类型(文件系统或JAR包)if (resource.getProtocol().equalsIgnoreCase("file")) {// 处理文件系统中的类//获取 URL 中的路径部分String filePath = resource.getPath();//解码特殊字符filePath = java.net.URLDecoder.decode(filePath, "UTF-8");//调用 findAndAddClassesInPackageByFile() 递归扫描目录findAndAddClassesInPackageByFile(basePack, filePath, classPaths);} else if (resource.getProtocol().equalsIgnoreCase("jar")) {// 处理JAR包中的类//从 JAR 资源 URL 中提取实际的 JAR 文件路径。跳过前缀 jar:file:(长度为 5)。//indexOf("!"):找到 ! 的位置,截取到此前的部分,得到 JAR 文件的路径String jarPath = resource.getPath().substring(5, resource.getPath().indexOf("!"));//解码 URL 编码的特殊字符jarPath = java.net.URLDecoder.decode(jarPath, "UTF-8");//打开 JAR 文件,扫描指定包下的所有类文件。findAndAddClassesInPackageByJar(basePack, jarPath, classPaths);}}} catch (Exception e) {e.printStackTrace();}return classPaths;}/*** 从文件系统中查找类*/private static void findAndAddClassesInPackageByFile(String packageName, String packagePath, List<String> classPaths) {//创建文件对象将传入的路径字符串转换为 File 对象,用于文件系统操作。File dir = new File(packagePath);//确保当前路径是一个存在且有效的目录。if (!dir.exists() || !dir.isDirectory()) {return;}// 获取目录下的所有文件和子目录,并过滤出以下两类File[] dirfiles = dir.listFiles(file ->// 文件夹或.class文件//使用 Lambda 表达式作为 listFiles 的过滤器,简洁地指定筛选条件。file.isDirectory() || file.getName().endsWith(".class"));//目录无法访问(如权限不足)或为空时,listFiles 可能返回 null,此处直接跳过。if (dirfiles == null) {return;}// 遍历所有文件for (File file : dirfiles) {if (file.isDirectory()) {// 递归处理子目录findAndAddClassesInPackageByFile(packageName + "." + file.getName(), //包名file.getAbsolutePath(), //文件或目录的绝对路径classPaths //类路径集合(用于储存结果));} else {// 处理class文件String className = file.getName().substring(0, file.getName().length() - 6);classPaths.add(packageName + '.' + className);}}}/*** 从JAR包中查找类*/private static void findAndAddClassesInPackageByJar(String packageName, String jarPath, List<String> classPaths) {//尝试打开 JAR 文件(资源自动关闭)try (JarFile jar = new JarFile(jarPath)) {//获取 JAR 文件中所有条目的枚举Enumeration<JarEntry> entries = jar.entries();//逐个处理 JAR 中的每个条目。while (entries.hasMoreElements()) {JarEntry entry = entries.nextElement();String name = entry.getName();// 判断是否是类文件并且在指定包路径下//确保只提取指定包及其子包下的类文件。if (name.endsWith(".class") && name.startsWith(packageName.replace('.', '/'))) {//提取类全限定名去除 .class 后缀:String className = name.substring(0, name.length() - 6).replace('/', '.');//添加类名到结果列表classPaths.add(className);}}} catch (Exception e) {e.printStackTrace();}}public static void main(String[] args) {// 测试示例//测试扫描指定包List<String> classes = searchClass("com.qcby.myweb");//遍历并打印结果for (String className : classes) {System.out.println(className);}}public static List<String> searchClass() {return null;}
}
package com.qcby.util;public class ResponseUtil {public static final String responseHeader200 = "HTTP/1.1 200 \r\n"+"Content-Type:text/html; charset=utf-8 \r\n"+"\r\n";public static String getResponseHeader404(){return "HTTP/1.1 404 \r\n"+"Content-Type:text/html; charset=utf-8 \r\n"+"\r\n" + "404";}public static String getResponseHeader200(String context){return "HTTP/1.1 200 \r\n"+"Content-Type:text/html; charset=utf-8 \r\n"+"\r\n" + context;}
}
在config中写入
package com.qcby.config;import com.qcby.servlet.HttpServlet;
import com.qcby.util.SearchClassUtil;
import com.qcby.zj.LRHServlet;import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;/*** tomcat路由*/
public class TomcatRoute {//HashMap<String, HttpServlet> HttpServlet向上转型(使用多态),因为myweb中的父类全是HttpServlet//将public static HashMap<String, HttpServlet> routes = new HashMap<>();写入static{}中会导致无法调用public static HashMap<String, HttpServlet> routes = new HashMap<>();static{// 传入要扫描的包名List<String> paths = SearchClassUtil.searchClass("com.qcby.myweb");for(String path : paths){try{//urlClass clazz = Class.forName(path);LRHServlet webServlet = (LRHServlet) clazz.getDeclaredAnnotation(LRHServlet.class);routes.put(webServlet.url(), (HttpServlet) clazz.getDeclaredConstructor().newInstance());System.out.println(webServlet.url());
// if (webServlet != null) {
// System.out.println(webServlet.url());
// }//对象clazz.getDeclaredConstructor().newInstance();}catch(ClassNotFoundException e){e.printStackTrace();} catch (InvocationTargetException e) {throw new RuntimeException(e);} catch (InstantiationException e) {throw new RuntimeException(e);} catch (IllegalAccessException e) {throw new RuntimeException(e);} catch (NoSuchMethodException e) {throw new RuntimeException(e);}}}
}
最后就是直接的MyTomcat
package com.qcby;import com.qcby.config.TomcatRoute;
import com.qcby.request.HttpServletRequest;
import com.qcby.response.HttpServletResponse;
import com.qcby.servlet.HttpServlet;import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.HashMap;/*** tomcat主启动类*/
public class MyTomcat {static HashMap<String, HttpServlet> routes = TomcatRoute.routes;static HttpServletRequest request = new HttpServletRequest();/*** 分发器*/public static void dispatch(HttpServletResponse response) throws IOException {HttpServlet servlet = routes.get(request.getUrl());System.out.println(servlet);if (servlet != null) {servlet.service(request, response);}}public static void start() throws IOException {System.out.append("服务器端启动...");//1.定义ServerSocket对象进行服务器端的端口注册ServerSocket serverSocket = new ServerSocket(8080);while (true) {//2.监听客户端的Socket链接程序Socket socket = serverSocket.accept(); // 阻塞监听//3.打开输入流,解析客户端发来的内容InputStream inputStream = socket.getInputStream();//输入流BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));//将字节流转换成字符流String str = reader.readLine();request.setMethod(str.split("\\s")[0]);request.setUrl(str.split("\\s")[1]);//4.打开输出流OutputStream outputStream = socket.getOutputStream();HttpServletResponse response = new HttpServletResponse(outputStream);dispatch(response);}}public static void main(String[] args) throws IOException {start();}
}
这样就完成三分之二了
之后运行MyTomcat程序后,回显示com.qcby.myweb下的所有软件包名,和服务器端启动的字样
在浏览器的地址栏里输入(localhost:8080/myservlet)这里的8080要根据你实际写的端口号是多少写多少(myservlet要写自己定义的url)
在浏览器中就能够看到下面写在MyFirstServlet中的内容
在控制台能够看到MyFirstservlet中打印的内容
这样一个简单的tomcat就写好了
相关文章:
自己手写tomcat项目
一:Servlet的原理 在Servlet(接口中)有: 1.init():初始化servlet 2.getServletConfig():获取当前servlet的配置信息 3.service():服务器(在HttpServlet中实现,目的是为了更好的匹配http的请求方式) 4.g…...
C语言—再学习(结构体)
一、建立结构体 用户自己建立由不同类型数据组成的组合型的数据结构,它称为结构体。 struct Student { int num; //学号char name[20]; //名字为字符串char sex; //性别int age; //年纪float score; //分数char addr[30]; 地址为字符…...
SpringBoot--自动配置原理详解
为什么要学习自动配置原理? 原因:在实际开发中,我们经常会定义一些公共的组件,提供各个团队来使用,为了使用方便,我们经常会将公共的组件自定义成starter,如果想自定义starter,必须…...
MiInsertPageInFreeList函数分析和MmFreePagesByColor数组的关系
第一部分: Color MI_GET_COLOR_FROM_LIST_ENTRY(PageFrameIndex, Pfn1); ColorHead &MmFreePagesByColor[ListName][Color]; 第二部分: #define MI_GET_COLOR_FROM_LIST_ENTRY(index,pfn) \ ((ULONG)(((pfn)->…...
Windows/MacOS WebStorm/IDEA 中开发 Uni-App 配置
文章目录 前言1. 安装 HBuilder X2. WebStorm/IDEA 安装 Uniapp Tool 插件3. 配置 Uniapp Tool 插件4. 运行 Uni-App 项目 前言 前端开发人员对 WebStorm 一定不陌生,但有时需要开发 Uni-App 的需求,就必须要采用 HBuilder X,如果不习惯 HBu…...
redisson分布式锁实现原理归纳总结
Redisson 分布式锁的实现原理主要依赖于 Redis 的 Hash 数据结构、Lua 脚本、发布订阅机制以及看门狗(Watchdog)机制,以下是核心要点总结: 1. 核心原理 • 互斥性与可重入性: 通过 Redis 的 Hash 数据结构保存锁的持…...
Ubuntu 添加系统调用
实验内容 通过内核编译法添加一个不用传递参数的系统调用,其功能可自定义。 (1)添加系统调用号,系统会根据这个号找到syscall_table中的相应表项。具体做法是在syscall_64.tbl文件中添加系统调用号和调用函数的对应关系。 &#…...
Olib 2.2.0 | 免费开源软件,无需注册登录即可从ZLibrary下载多语言电子书
Olib是一款专为书籍爱好者设计的免费开源软件,它允许用户无需注册或登录即可从ZLibrary高速下载各种语言的电子书。该软件支持上百种语言的电子书下载,非常适合需要多语言资源的读者和研究人员使用。Olib的操作界面非常直观,使得书籍的搜索与…...
c++动态链接库
1. 生成动态链接库 首先实现一个动态链接库的代码 // example.cpp #include <iostream> void sayHello() {std::cout << "Hello from shared library!" << std::endl; }int add(int a, int b) {return a b; }// example.h #pragma once void sa…...
HelloWorld
HelloWorld 新建一个java文件 文件后缀名为 .javahello.java【注意】系统可能没有显示文件后缀名,我们需要手动打开 编写代码 public class hello {public static void main(String[] args) {System.out.print(Hello,World)} }编译 javac java文件,会生…...
SVGPlay:一次 CodeBuddy 主动构建的动画工具之旅
我正在参加CodeBuddy「首席试玩官」内容创作大赛,本文所使用的 CodeBuddy 免费下载链接:腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 背景与想法 我一直对 SVG 图标的动画处理有浓厚兴趣,特别是描边、渐变、交互等效果能为图标增添许…...
SLAM定位常用地图对比示例
序号 地图类型 概述 1 格栅地图 将现实环境栅格化,每一个栅格用 0 和 1 分别表示空闲和占据状态,初始化为未知状态 0.5 2 特征地图 以点、线、面等几何特征来描绘周围环境,将采集的信息进行筛选和提取得到关键几何特征 3 拓扑地图 将重要部分抽象为地图,使用简单的图形表示…...
强化学习中,frames(帧)和 episodes(回合)
在强化学习中,frames(帧)和 episodes(回合)是两个不同的概念: 1. 定义差异 Frame(帧): 表示智能体与环境交互的单个时间步(step),例如…...
HCIP第六次作业
一、拓扑图 二、需求 1、使用PreVal策略,确保R4通过R2到达192.168.10.0/24 2、使用AS_Path策略,确保R4通过R3到达192.168.11.0/24 3、配置MED策略,确保R4通过R3到达192.168.12.0/24 4、使用Local Preference策略,确保R1通过R2…...
高频面试题(含笔试高频算法整理)基本总结回顾110
干货分享,感谢您的阅读! (暂存篇---后续会删除,完整版和持续更新见高频面试题基本总结回顾(含笔试高频算法整理)) 备注:引用请标注出处,同时存在的问题请在相关博客留言…...
数据湖与数据仓库融合:Hudi、Iceberg、Delta Lake 实践对比
在实时与离线一体化的今天,数据湖与数据仓库边界不断融合,越来越多企业选用如 Hudi、Iceberg、Delta Lake 等开源方案实现统一的数据存储、计算、分析平台。本篇将围绕以下关键点,展开实战对比与解决方案分享: ✅ 实时写入能力 ✅ ACID 保证 ✅ 增量数据处理能力 ✅ 流批一…...
OGG 更新表频繁导致进程中断,见鬼了?非也!
大家好,这里是 DBA学习之路,专注于提升数据库运维效率。 目录 前言问题描述问题分析解决方案后续 前言 最近几周一直遇到一个 OGG 问题,有一张表已更新就会中断 OGG 同步进程,本文记录一下分析过程以及解决方案。 问题描述 昨天…...
C++学习-入门到精通-【7】类的深入剖析
C学习-入门到精通-【7】类的深入剖析 类的深入剖析 C学习-入门到精通-【7】类的深入剖析一、Time类的实例研究二、组成和继承三、类的作用域和类成员的访问类作用域和块作用域圆点成员选择运算符(.)和箭头成员选择运算符(->)访问函数和工具函数 四、具有默认实参的构造函数重…...
非易失性存储技术综合对比:EEPROM、NVRAM、NOR Flash、NAND Flash和SD卡
非易失性存储技术综合对比:EEPROM、NVRAM、NOR Flash、NAND Flash和SD卡 读写性能对比 存储类型读取速度写入速度随机访问能力最小操作单位NVRAM极快(~10ns)极快(~10ns)极优(字节级)字节EEPROM中等(~100ns)慢(~5ms/字节)优(字节级)字节NOR Flash快(~50ns)慢(~5ms/…...
数字化转型- 数字化转型路线和推进
数字化转型三个阶段 百度百科给出的企业的数字化转型包括信息化、数字化、数智化三个阶段 信息化是将企业在生产经营过程中产生的业务信息进行记录、储存和管理,通过电子终端呈现,便于信息的传播与沟通。数字化通过打通各个系统的互联互通,…...
ARM (Attention Refinement Module)
ARM模块【来源于BiSeNet】:细化特征图的注意力,增强重要特征并抑制不重要的特征。 Attention Refinement Module (ARM) 详解 ARM (Attention Refinement Module) 是 BiSeNet 中用于增强特征表示的关键模块,它通过注意力机制来细化特征图&…...
符合Python风格的对象(对象表示形式)
对象表示形式 每门面向对象的语言至少都有一种获取对象的字符串表示形式的标准方 式。Python 提供了两种方式。 repr() 以便于开发者理解的方式返回对象的字符串表示形式。str() 以便于用户理解的方式返回对象的字符串表示形式。 正如你所知,我们要实现_…...
AtCoder AT_abc406_c [ABC406C] ~
前言 除了 A 题,唯一一道一遍过的题。 题目大意 我们定义满足以下所有条件的一个长度为 N N N 的序列 A ( A 1 , A 2 , … , A N ) A(A_1,A_2,\dots,A_N) A(A1,A2,…,AN) 为波浪序列: N ≥ 4 N\ge4 N≥4(其实满足后面就必须满足这…...
多指标组合策略
该策略(MultiConditionStrategy)是一种基于多种技术指标和市场条件的交易策略。它通过综合考虑多个条件来生成交易信号,从而决定买入或卖出的时机。 以下是对该策略的详细分析: 交易逻辑思路 1. 条件1:星期几和价格变化判断 - 该条件根据当前日期是星期几以及价格的变化…...
系统架构-大数据架构设计
基础介绍 三大挑战: 如何处理非结构化和半结构化数据如何探索大数据复杂性、不确定性特征描述的刻画方法及大数据的系统建模数据异构性与决策异构性的关系对大数据知识发现与管理决策的影响 架构特征: 鲁棒性(稳定性)和容错性…...
R语言空间数据处理入门教程
我的课程《R语言空间数据处理入门教程》已重新恢复课程售卖,有需要的读者可以学习。 👇点击下方链接(文末“阅读原文”可直达),立即开启你的空间数据之旅: https://www.bilibili.com/cheese/play/ss13775…...
QT+EtherCAT 主站协议库—SOEM主站
SOEM 是 Simple Open EtherCAT Master Library 的缩写,是瑞典 rt-lab 提供 的一个开源 EtherCAT 主站协议库 。 SOEM 库使用 C 语言编写,可以在 windows 以及 Linux 平台上运行,并也可以方便地移植到嵌入式平台上。 SOEM 支持 CoE ࿰…...
Java-反射(Reflection)
一:概述 (1)出现背景 (2)解决方案 (3)使用场景 业务开发用的少,框架使用的多,业务反射被认为是动态语言的关键 (4)与原方法对比 (5…...
第一次经历项目上线
这几天没写csdn,因为忙着项目上线的问题,我这阶段改了非常多的前端bug哈哈哈哈,说几个比较好的bug思想! 这个页面算是我遇到的比较大的bug,因为我一开始的逻辑都写好了,询价就是在点击快递公司弹出弹框的时…...
基于C#的MQTT通信实战:从EMQX搭建到发布订阅全解析
MQTT(Message Queueing Telemetry Transport) 消息队列遥测传输,在物联网领域应用的很广泛,它是基于Publish/Subscribe模式,具有简单易用,支持QoS,传输效率高的特点。 它被设计用于低带宽,不稳定或高延迟的…...
DeepSeek超大模型的高效训练策略
算力挑战 训练DeepSeek此类千亿乃至万亿级别参数模型,对算力资源提出了极高要求。以DeepSeek-V3为例,其基础模型参数量为67亿,采用专家混合(MoE)架构后实际激活参数可达几百亿。如此规模的模型远超单张GPU显存容量极限,必须借助分布式并行才能加载和训练。具体挑战主要包…...
【论文阅读】人脸修复(face restoration ) 不同先验代表算法整理
转眼做人脸复原(face restoration)算法也一段时间了,根据自己的记忆整理一下自己的一些看法,算作个人记录,当然如果有人愿意分享自己的看法也是极好的。先挂下文章链接,下一篇在写总结。 一、前述 人脸修复(face restoration)任…...
最小二乘法拟合平面(线性回归法、梯度下降、PCA法)
参考笔记: Open3D 最小二乘拟合平面(直接求解法)【2025最新版】_python open3d已知平面方程绘制平面-CSDN博客 目录 1.前言 2.线性回归法 2.1 模型假设 2.2 定义误差函数 2.3 求偏导并解方程 2.4 解方程 2.5 案例演示 2.5.1 手工计…...
数组名既可作为指针也可作为变量名
在C语言中,数组名在不同的上下文中既可以作为指向数组首个元素的指针,也可以代表整个数组,这是由C语言的设计和语法规则决定的,下面我来详细解释一下。 1. 数组名作为指向首元素的指针 在大多数情况下,当数组名出现在…...
MySQL相关
1.多表查询关键点在哪 📖 1️⃣ 明确关联关系 先搞清楚多表之间的关联关系: 一对一(1:1) 一对多(1:N) 多对多(M:N) 比如: 一个课程对应一个教室(1:1&am…...
Axure制作可视化大屏动态滚动列表教程
在可视化大屏设计中,动态滚动列表是一种常见且实用的展示方式,能够有效地展示大量信息。本文将详细介绍如何使用Axure制作一个动态滚动的列表展示模块。 一、准备工作 打开Axure软件:确保你已经安装并打开了Axure RP软件。创建新项目&#x…...
计算机网络(1)——概述
1.计算机网络基本概念 1.1 什么是计算机网络 计算机网络的产生背景 在计算机网络出现之前,计算机之间都是相互独立的,每台计算机只能访问自身存储的数据,无法与其他计算机进行数据交换和资源共享。这种独立的计算机系统存在诸多局限性&#…...
融智学视域下的系统性认知增强框架——基于文理工三类AI助理赋能HI四阶跃迁路径
融智学视域下的系统性认知增强框架 ——基于文理工三类AI助理赋能HI四阶跃迁路径 一、如何排除50个认知偏差:消除50类偏差的精准矫正系统 1. 技术架构 文科AI: 构建文化语义场(Cultural Semantic Field, CSF),通过…...
C++ - 仿 RabbitMQ 实现消息队列(2)(Protobuf 和 Muduo 初识)
C - 仿 RabbitMQ 实现消息队列(2)(Protobuf 和 Muduo 初识) Protobuf1. 序列化/反序列化方法(最核心)_InternalSerialize()_InternalParse() 2. 内存管理方法SharedCtor()/SharedDtor()InternalSwap() 3. 字…...
FTP与NFS服务实战:从配置到应用
一、FTP服务进阶:客户端工具与访问控制 1. FTP客户端工具对比 在Linux中,ftp和lftp是常用的FTP客户端工具,功能各有侧重: 工具特点适用场景ftp基础命令交互,需手动输入用户名/密码简单文件传输lftp支持多协议、批量…...
高考AI试题查询系统
高考AI试题查询系统 gitee:https://gitee.com/ltyyyds26/GaoKao_AI 数据 来源:OpenLMLab/GAOKAO-Bench: GAOKAO-Bench is an evaluation framework that utilizes GAOKAO questions as a dataset to evaluate large language models. (github.com) 数…...
记录算法笔记(2025.5.17)验证二叉搜索树
给你一个二叉树的根节点 root ,判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义如下: 节点的左子树只包含 小于 当前节点的数。节点的右子树只包含 大于 当前节点的数。所有左子树和右子树自身必须也是二叉搜索树。 示例 1: 输入&…...
DataX:一个开源的离线数据同步工具
DataX 是一个异构数据源离线同步(ETL)工具,实现了包括关系型数据库(MySQL、Oracle 等)、HDFS、Hive、ODPS、HBase、FTP 等各种异构数据源之间稳定高效的数据同步功能。它也是阿里云 DataWorks 数据集成功能的开源版本。 为了解决异构数据源同…...
剑指offer第一周
目录 二维数组中的查找 旋转数组的最小数字 调整数组顺序使奇数位于偶数前面 数组中出现次数超过一半的数字 替换空格 从尾到头打印链表 重建二叉树 矩形覆盖 链表中倒数最后k个结点 二进制中1的个数 合并两个排序的链表 树的子结构 二叉树的镜像 二…...
素数筛(欧拉筛算法)
#include<bits/stdc.h> using namespace std; #define maxn 100000 int vis[maxn]; int prime[maxn]; //欧拉筛函数 int Euler_sieve(int n) { int i,j,k; k0;//保存素数的个数 memset(vis,0,sizeof(int)*maxn);//初始化数组 for(i2;i<n;i) { if(vis[i]0)//i是素数…...
遨游科普:三防平板是什么?有什么功能?
清晨的露珠还挂在帐篷边缘,背包里的三防平板却已开机导航;工地的尘土飞扬中,工程师正通过它查看施工图纸;暴雨倾盆的救援现场,应急队员用它实时回传灾情数据……这些看似科幻的场景,正因三防平板的普及成为…...
CSS 浮动与定位以及定位中z-index的堆叠问题
CSS 浮动与定位以及定位中z-index的堆叠问题 一、浮动布局的特点与应用 1. 浮动核心特性 脱离标准流:浮动元素会脱离文档流。环绕特性:后续内容会环绕浮动元素排列自动换行:多个浮动元素在容器宽度不足时自动换行 .float-box {float: lef…...
在Maven中替换文件内容的插件和方法
在Maven中替换文件内容的插件和方法 Maven提供了几种方式来替换文件内容,以下是常用的插件和方法: 1. maven-replacer-plugin (推荐) 这是专门用于文件内容替换的插件,功能强大且灵活。 基本配置 <plugin><groupId>com.goog…...
C# lock
在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时,其他线程无法访问同一实例的该代码块。这是一种简单的同步机制,用来防止多个线程同时访问共享资源或执行需要独占访问的代码段(临界区),从而…...
OGGMA 21c 微服务 (MySQL) 安装避坑指南
前言 这两天在写 100 天实战课程 的 OGG 微服务课程: 在 Oracle Linux 8.10 上安装 OGGMA 21c MySQL 遇到了一点问题,分享给大家一起避坑! 环境信息 环境信息: 主机版本主机名实例名MySQL 版本IP 地址数据库字符集Goldengate …...