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

《从零构建一个简易的IOC容器,理解Spring的核心思想》

大家好呀!今天我们要一起探索Java开发中最神奇的魔法之一 —— Spring框架的IOC容器!🧙‍♂️ 我会用最最最简单的方式,让你彻底明白这个看似高深的概念。准备好了吗?Let’s go! 🚀

一、什么是IOC容器?🍯

想象你有一个超级大的玩具箱🎁,里面装满了各种玩具。以前你要玩某个玩具时,得自己伸手进去找(new 对象())。现在有了IOC容器,就像有个智能机器人🤖帮你管理玩具箱,你只需要说:“我要玩小汽车!🚗”,机器人就会自动找到并递给你 —— 这就是IOC(控制反转)!

专业点说:IOC(Inversion of Control)控制反转就是把创建和管理对象的控制权从程序员手中"反转"给了容器。

二、为什么要用IOC?🤔

举个生活中的例子🌰:

没有IOC时:

// 你要喝咖啡,得自己种咖啡豆、磨粉、冲泡...
Coffee coffee = new Coffee();
coffee.drink();

有IOC时:

// 只需要去咖啡店说"我要一杯咖啡"
@Autowired
Coffee coffee;  // 咖啡自动送到你面前
coffee.drink();

IOC的好处:

  1. 不用自己new对象了,省事!😌
  2. 方便统一管理对象
  3. 降低代码耦合度(类之间不那么依赖了)
  4. 更容易测试和维护

三、手写迷你IOC容器实战 ✍️

现在,让我们从零开始造一个超简易IOC容器!分三步走:

第1步:创建容器类 🏗️

public class MyMiniContainer {// 用来存放所有bean的Map,key是名字,value是对象private Map beans = new HashMap<>();// 注册bean的方法public void registerBean(String name, Object bean) {beans.put(name, bean);}// 获取bean的方法public Object getBean(String name) {return beans.get(name);}
}

第2步:测试我们的容器 🧪

public class Test {public static void main(String[] args) {// 1. 创建容器MyMiniContainer container = new MyMiniContainer();// 2. 创建对象并放入容器UserService userService = new UserServiceImpl();container.registerBean("userService", userService);// 3. 需要时从容器获取UserService service = (UserService) container.getBean("userService");service.sayHello();  // 输出: Hello World!}
}

第3步:实现自动依赖注入 🎯

上面的容器太简单了,我们来升级它,实现自动"@Autowired"功能!

public class EnhancedContainer {private Map beans = new HashMap<>();// 新增:根据类型自动注入依赖public void autowire(Object bean) throws Exception {Field[] fields = bean.getClass().getDeclaredFields();for (Field field : fields) {if (field.isAnnotationPresent(Autowired.class)) {// 获取字段类型Class fieldType = field.getType();// 从容器找对应类型的实例Object dependency = findBeanByType(fieldType);// 设置字段值field.setAccessible(true);field.set(bean, dependency);}}}// 根据类型查找beanprivate Object findBeanByType(Class type) {for (Object bean : beans.values()) {if (type.isAssignableFrom(bean.getClass())) {return bean;}}throw new RuntimeException("找不到类型为 " + type.getName() + " 的bean");}
}

四、Spring IOC容器的完整实现思路 🧩

现在让我们看看真正的Spring IOC是怎么做的(简化版):

  1. 配置读取阶段 📖

    • 读取XML配置或扫描注解
    • 识别哪些类需要被管理
  2. 实例化阶段 🏭

    • 通过反射创建Bean实例
    • 放到一个叫"BeanFactory"的大Map里
  3. 依赖注入阶段 💉

    • 检查每个Bean的@Autowired注解
    • 把依赖的其他Bean注入进去
  4. 初始化阶段 🎉

    • 调用初始化方法
    • 处理AOP代理等增强功能

五、完整手写IOC容器代码 🖥️

下面是一个相对完整的简易IOC容器实现:

// 自定义Autowired注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAutowired {
}// 自定义Component注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyComponent {String value() default "";
}// IOC容器核心类
public class MyIOCContainer {private Map beans = new ConcurrentHashMap<>();// 初始化容器public void init(String basePackage) throws Exception {// 1. 扫描包路径下的所有类Set> classes = scanPackage(basePackage);// 2. 创建所有带有@MyComponent注解的类的实例createBeans(classes);// 3. 自动注入依赖autowireBeans();}// 扫描包路径下的所有类private Set> scanPackage(String basePackage) {// 实现略,可以使用反射工具包return new HashSet<>();}// 创建Bean实例private void createBeans(Set> classes) throws Exception {for (Class clazz : classes) {if (clazz.isAnnotationPresent(MyComponent.class)) {MyComponent component = clazz.getAnnotation(MyComponent.class);String beanName = component.value().isEmpty() ? clazz.getSimpleName() : component.value();Object instance = clazz.getDeclaredConstructor().newInstance();beans.put(beanName, instance);}}}// 自动注入依赖private void autowireBeans() throws Exception {for (Object bean : beans.values()) {autowireBean(bean);}}// 为单个Bean注入依赖private void autowireBean(Object bean) throws Exception {Field[] fields = bean.getClass().getDeclaredFields();for (Field field : fields) {if (field.isAnnotationPresent(MyAutowired.class)) {Object dependency = findBeanByType(field.getType());field.setAccessible(true);field.set(bean, dependency);}}}// 根据类型查找Beanprivate Object findBeanByType(Class type) {for (Object bean : beans.values()) {if (type.isAssignableFrom(bean.getClass())) {return bean;}}throw new RuntimeException("找不到类型为 " + type.getName() + " 的bean");}// 获取Beanpublic Object getBean(String name) {return beans.get(name);}
}

六、Spring IOC的更多魔法 ✨

真正的Spring IOC容器比我们的简易版强大得多,它还有:

  1. Bean作用域 🔍

    • Singleton:单例(默认)
    • Prototype:每次获取新实例
    • Request/Session/Application:Web相关作用域
  2. 生命周期回调

    • @PostConstruct:初始化方法
    • @PreDestroy:销毁前方法
  3. 条件化Bean ☑️

    • @Conditional:满足条件才创建Bean
  4. Bean后处理器 🔧

    • BeanPostProcessor:对Bean进行额外处理

七、面试常问的IOC问题 💼

  1. IOC和DI有什么区别?

    • IOC是思想(控制反转)
    • DI是实现方式(依赖注入)
    • 好比:IOC是"不用自己做饭",DI是"外卖送到家" 🍔
  2. Spring容器启动流程是怎样的?

    1. 加载配置
    2. 解析成BeanDefinition
    3. 注册到BeanFactory
    4. 实例化非懒加载的单例Bean
    5. 发布容器启动事件
  3. 循环依赖怎么解决?

    • Spring使用三级缓存:
      • 一级缓存:完整Bean
      • 二级缓存:早期暴露的Bean(还没注入属性)
      • 三级缓存:Bean工厂(能创建Bean)

八、实际项目中的应用案例 🏢

假设我们在开发一个电商系统🛒:

@MyComponent
public class OrderService {@MyAutowiredprivate PaymentService paymentService;@MyAutowired private InventoryService inventoryService;public void placeOrder(Order order) {inventoryService.checkStock(order);paymentService.processPayment(order);// 创建订单...}
}// 使用时:
public class Main {public static void main(String[] args) throws Exception {MyIOCContainer container = new MyIOCContainer();container.init("com.ecommerce");OrderService orderService = (OrderService) container.getBean("orderService");orderService.placeOrder(new Order());}
}

九、性能优化小贴士 ⚡

  1. 合理使用作用域

    • 无状态服务用Singleton
    • 有状态服务考虑Prototype
  2. 延迟加载

    • @Lazy注解减少启动时间
  3. 避免过度依赖注入

    • 一个类最好不要超过5个依赖
  4. 使用构造器注入

    • 比字段注入更利于测试和不变性

十、常见错误排查 🚨

  1. NoSuchBeanDefinitionException

    • 检查是否加了@Component
    • 扫描包路径是否正确
  2. BeanCurrentlyInCreationException(循环依赖):

    • 使用@Lazy打破循环
    • 重构代码解耦
  3. 注入的Bean为null

    • 检查是否在容器外使用@Autowired
    • 字段是否是private

十一、总结 🎓

今天我们从小白的角度,一步步揭开了Spring IOC容器的神秘面纱:

  1. IOC就像智能玩具箱🤖,帮你管理所有对象
  2. 核心思想是"控制反转"和"依赖注入"
  3. 自己动手实现了一个迷你IOC容器
  4. 了解了Spring容器的更多高级特性

记住,理解IOC的关键是明白:不要来找我,我会去找你 —— 这就是控制反转的精髓!💡

希望这篇文章能让你对Spring IOC有全新的认识!如果有任何问题,欢迎留言讨论~ 😊

思考题:如果让你给这个迷你容器添加AOP功能,你会怎么设计呢?🤔

推荐阅读文章

  • 由 Spring 静态注入引发的一个线上T0级别事故(真的以后得避坑)

  • 如何理解 HTTP 是无状态的,以及它与 Cookie 和 Session 之间的联系

  • HTTP、HTTPS、Cookie 和 Session 之间的关系

  • 什么是 Cookie?简单介绍与使用方法

  • 什么是 Session?如何应用?

  • 使用 Spring 框架构建 MVC 应用程序:初学者教程

  • 有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误

  • 如何理解应用 Java 多线程与并发编程?

  • 把握Java泛型的艺术:协变、逆变与不可变性一网打尽

  • Java Spring 中常用的 @PostConstruct 注解使用总结

  • 如何理解线程安全这个概念?

  • 理解 Java 桥接方法

  • Spring 整合嵌入式 Tomcat 容器

  • Tomcat 如何加载 SpringMVC 组件

  • “在什么情况下类需要实现 Serializable,什么情况下又不需要(一)?”

  • “避免序列化灾难:掌握实现 Serializable 的真相!(二)”

  • 如何自定义一个自己的 Spring Boot Starter 组件(从入门到实践)

  • 解密 Redis:如何通过 IO 多路复用征服高并发挑战!

  • 线程 vs 虚拟线程:深入理解及区别

  • 深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别

  • 10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!

  • “打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”

  • Java 中消除 If-else 技巧总结

  • 线程池的核心参数配置(仅供参考)

  • 【人工智能】聊聊Transformer,深度学习的一股清流(13)

  • Java 枚举的几个常用技巧,你可以试着用用

  • 由 Spring 静态注入引发的一个线上T0级别事故(真的以后得避坑)

  • 如何理解 HTTP 是无状态的,以及它与 Cookie 和 Session 之间的联系

  • HTTP、HTTPS、Cookie 和 Session 之间的关系

  • 使用 Spring 框架构建 MVC 应用程序:初学者教程

  • 有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误

  • Java Spring 中常用的 @PostConstruct 注解使用总结

  • 线程 vs 虚拟线程:深入理解及区别

  • 深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别

  • 10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!

  • 探索 Lombok 的 @Builder 和 @SuperBuilder:避坑指南(一)

  • 为什么用了 @Builder 反而报错?深入理解 Lombok 的“暗坑”与解决方案(二)

相关文章:

《从零构建一个简易的IOC容器,理解Spring的核心思想》

大家好呀&#xff01;今天我们要一起探索Java开发中最神奇的魔法之一 —— Spring框架的IOC容器&#xff01;&#x1f9d9;‍♂️ 我会用最最最简单的方式&#xff0c;让你彻底明白这个看似高深的概念。准备好了吗&#xff1f;Let’s go! &#x1f680; 一、什么是IOC容器&…...

差分与位移算子

差分与位移算子是数值分析和离散数学中处理序列或离散函数的重要工具。它们通过算子代数简化差分的计算和分析&#xff0c;以下是关键概念和关系的总结&#xff1a; 1. 位移算子&#xff08;Shift Operator&#xff09; 定义&#xff1a; 位移算子 ( E ) 将函数 ( f(x) ) 沿自变…...

Robot之VideoMimic:《Visual Imitation Enables Contextual Humanoid Control》翻译与解读

Robot之VideoMimic&#xff1a;《Visual Imitation Enables Contextual Humanoid Control》翻译与解读 导读&#xff1a;这篇论文介绍了VIDEOMIMIC&#xff0c;一个基于视觉模仿的真实到模拟到真实流水线&#xff0c;用于训练人形机器人执行上下文相关的全身动作。该方法通过分…...

【Java学习日记34】:this关键字和成员变量

为什么不需要加 this&#xff1f; 作用域规则&#xff1a; Java编译器在查找变量时遵循“就近原则”。 先在当前方法内查找局部变量或参数。 若找不到&#xff0c;则去类的成员变量中查找。 getName() 的上下文&#xff1a; 该方法没有参数或局部变量名为 name&#xff0c;因…...

包名查看器APP:高效管理手机应用的实用工具

包名查看器APP是一款功能强大的文件查看软件&#xff0c;专为安卓用户设计&#xff0c;能够帮助用户快速了解手机上安装和未安装的APK包信息。作为酷安首发的APK信息查看工具&#xff0c;它提供了比系统设置更详细的信息&#xff0c;如版本号、包名、MD5等&#xff0c;帮助用户…...

左右括号的最小处理次数

1、题目描述 多多君在处理一个由左结号(和右语号&#xff09;组成的字符串&#xff0c;多多君每次处理时可以顺序读取一个字符或者一个有效括号子串&#xff0c;求问多多的最小处理次数。 输入描述&#xff1a; 第一行为一个整数N&#xff0c;表示字符串长度&#xff08;1<…...

22.第二阶段x64游戏实战-分析周围对象类型

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 上一个内容&#xff1a;21.第二阶段x64游戏实战-分析采集物偏移 上一个内容里发现采集物的名字通过我们…...

【C/C++】无符号调试:GDB解栈实战指南

文章目录 无符号调试&#xff1a;GDB解栈实战指南1 生成并加载 Core Dump2 查看原始堆栈信息&#xff08;地址形式&#xff09;3 确认加载的共享库地址范围4 手动转换地址为函数名5 反汇编关键代码段6 加载外部符号文件&#xff08;如有&#xff09;7 结合系统库文档分析8 示例…...

梦熊联盟:202505基础语法-题解

202505基础语法-题解 T1 - 九的倍数 解法&#xff1a; 对于 9 的倍数&#xff0c;只需要判定其各位的数码和是否为 9 的倍数即可。 例如判断一个数是不是 9 的倍数&#xff0c;只要判断其各位数字之和是不是 9 的倍数&#xff0c;因为一个数能被 9 整除当且仅当它的各位数字之和…...

Java SE(11)——内部类

1.内部类 定义&#xff1a;Java中的内部类(Inner Class)是指在一个类的内部定义的类。 使用场景&#xff1a;当一个类的内部&#xff0c;存在一个部分需要完成的结构进行描述&#xff0c;而该内部结构只为外部类提供服务&#xff0c;那么这个内部结构就可以使用内部类&#xff…...

优化审核模块响应时间从8s降至1.2s的数据库解决方案

优化审核模块响应时间从8s降至1.2s的数据库解决方案 要优化审核模块的数据库性能&#xff0c;需要从多个层面进行分析和优化。以下是具体的SQL语句设计和优化方案&#xff1a; 1. 分析当前性能瓶颈 首先需要找出慢查询&#xff1a; -- 查看慢查询日志中的审核模块相关查询 …...

YOLO-World:基于YOLOv8的开放词汇目标检测

文章目录 前言1、出发点2、方法2.1.TextEncoder2.2.ReparmVLPAN2.3.输出头 3、实验3.1.数据集3.2.LVIS测试集 总结 前言 本文介绍一篇来自腾讯的开放词汇检测工作&#xff0c;发表自CVPR2024&#xff0c;论文链接&#xff0c;开源地址。 1、出发点 GroundingDINO在开放词汇检测…...

NX989NY104美光科技芯片NY109NY113

NX989NY104美光科技芯片NY109NY113 存储市场新势力&#xff1a;美光科技的崛起与技术突围 在半导体行业波澜壮阔的浪潮中&#xff0c;美光科技宛如一颗璀璨的明珠&#xff0c;以其独特的技术实力和敏锐的市场洞察力&#xff0c;在存储领域占据了重要的一席之地。尤其是其旗下…...

LabVIEW的PID参数自适应控制

在工业控制领域&#xff0c;PID 控制凭借结构简单、稳定性好、工作可靠等优点被广泛应用。然而&#xff0c;传统固定参数的 PID 控制在面对复杂多变的工况时&#xff0c;控制效果往往难以达到最优。基于 LabVIEW 实现 PID 控制根据情况选择参数&#xff08;即参数自适应调整&am…...

Quartus与Modelsim-Altera使用手册

目录 文章内容&#xff1a; 视频内容&#xff1a; Quartus&#xff1a; ModelSim&#xff1a; 顶层设计与子模块&#xff1a; 只是对所查阅的相关文章的总结与视频总结 文章内容&#xff1a; 这篇对基础操作很详细&#xff1a; 一、Quartus II软件的使用_quartus2软件上…...

设计模式之工厂模式(二):实际案例

设计模式之工厂模式(一) 在阅读Qt网络部分源码时候&#xff0c;发现在某处运用了工厂模式&#xff0c;而且编程技巧也用的好&#xff0c;于是就想分享出来&#xff0c;供大家参考&#xff0c;理解的不对的地方请多多指点。 以下是我整理出来的类图&#xff1a; 关键说明&#x…...

数据可视化大屏——智慧社区内网比对平台

综述分析&#xff1a; 智慧社区内网数据比对信息系统 这段代码实现了一个智慧社区内网数据比对信息系统的前端界面&#xff0c;采用三栏式布局展示各类社区安全相关数据。界面主要由左侧数据统计、中间地图展示和右侧数据分析三部分组成&#xff0c;使用了多种图表可视化技术…...

Spark任务调度流程详解

1. 核心调度组件 DAGScheduler&#xff1a;负责将Job拆分为Stage&#xff0c;处理Stage间的依赖关系。 TaskScheduler&#xff1a;将Task分配到Executor&#xff0c;监控任务执行。 SchedulerBackend&#xff1a;与集群管理器&#xff08;如YARN、K8s&#xff09;通信&#x…...

LeetCode 215题解 | 数组中的第K个最大元素

数组中的第K个最大元素 一、题目链接二、题目三、算法原理四、编写代码 一、题目链接 数组中的第K个最大元素 二、题目 三、算法原理 法一&#xff1a;排序 法二&#xff1a;优先级队列&#xff08;堆&#xff09; 重点看法二&#xff1a; 默认建大堆&#xff0c;意味着以…...

探秘 Cursor 核心:解锁系统提示词的进阶之路

在 AI 编程领域&#xff0c;Cursor 无疑是一颗耀眼的明星&#xff0c;其母公司 Anysphere 在短短三个月内&#xff0c;估值从 25 亿美元狂飙至 100 亿美元&#xff0c;这样的发展速度令人咋舌。而 Cursor 强大功能背后的核心 —— 系统提示词&#xff0c;始终笼罩着一层神秘的面…...

ElasticSearch入门详解

1.ElasticSearch 1.1 ElasticSearch(简称es) Elasticsearch是用Java开发并且是当前最流行的开源的企业级搜索引擎。 能够达到实时搜索&#xff0c;稳定&#xff0c;可靠&#xff0c;快速&#xff0c;安装使用方便。 客户端支持Java、.NET&#xff08;C#&#xff09;、PHP、Py…...

【计算机网络01】 网络组成与三种交换方式

【参考资料】 《自顶向下的计算机网络第八版》湖科大计算机网络&#xff08;b站&#xff09;王道考研&#xff08;b站&#xff09; 文章目录 一、网络基础概念解析1.1 网络、互联网与因特网 二、因特网发展三阶段&#xff08;了解&#xff09;三、ISP3.1 ISP基本概念3.2 基于I…...

计算机网络——以太网交换机

目录 交换机的作用 以太网交换机的自学习功能 因为以太网交换机有自学习功能&#xff0c;所以以太网交换机支持即插即用 交换机的作用 它工作在数据链路层&#xff0c;为结点转发帧&#xff0c;并且可以根据一个帧的目的MAC地址去进行相应的转发&#xff0c;以及交换机的每…...

机器视觉开发教程——C#如何封装海康工业相机SDK调用OpenCV/YOLO/VisionPro/Halcon算法

目录 引言前期准备Step1 创建工程Step2 创建接口2.1定义操作相机实例接口方法2.2定义设置相机参数接口方法&#xff08;部分&#xff09; Step3 创建基类3.1定义操作相机实例&&设置相机参数的抽象层3.2定义操作相机实例&&设置相机参数的公用方法1.获取当前帧图…...

c++STL-string的模拟实现

cSTL-string的模拟实现 string的模拟实现string的模拟线性表的实现构造函数析构函数获取长度&#xff08;size&#xff09;和获取容量&#xff08;capacity&#xff09;访问 [] 和c_str迭代器&#xff08;iterator&#xff09;交换swap拷贝构造函数赋值重载&#xff08;&#x…...

HTTP 和 WebSocket 的区别

✅ 一、定义对比 协议简要定义HTTP一种基于请求-响应模式的、无状态的应用层协议&#xff0c;通常用于客户端与服务器之间的数据通信。WebSocket一种全双工通信协议&#xff0c;可以在客户端和服务器之间建立持久连接&#xff0c;实现实时、低延迟的数据传输。 ✅ 二、通信方式…...

【Tools】Visual Studio使用经验介绍(包括基本功能、远程调试、引入第三方库等等)

这里写目录标题 1. VS基本使用1.1. 快捷键1.2. 查看变量地址1.3. 查看代码汇编1.4. visual studio 热重载功能的使用1.5. vs远程服务器调试1.6. 引入第三方库VLD1.7. release debug模式 1. VS基本使用 1.1. 快捷键 ctrl c :复制光标所在行 注意&#xff1a;只需要光标在这…...

一周内学完计算机网络课程之二:计算机网络物理层的理解

消失人口回归&#xff0c;重新开始学习新知识。再次伟大。 物理层详解 需要理解的几个概念&#xff1a; 曼彻斯特编码、差分曼彻斯特编码 码元&#xff1a;构成信号的基本单元 调制&#xff1a; 通信中的调制是一种将原始信号&#xff08;如音频、视频、数据等&#xff09;转…...

Python OpenCV性能优化与部署实战指南

在计算机视觉领域&#xff0c;OpenCV作为开源视觉库的标杆&#xff0c;其性能表现直接影响着从工业检测到AI模型推理的各类应用场景。本文结合最新技术趋势与生产实践&#xff0c;系统性梳理Python环境下OpenCV的性能优化策略与部署方案。 一、性能优化核心技术矩阵 1.1 内存…...

深度解析:可视化如何重塑销售策略制定与执行

为什么你的销售策略总是“听起来挺对&#xff0c;做起来却没用”&#xff1f; 你有没有遇到过这样的情况&#xff1a; 销售团队天天跑客户&#xff0c;但业绩还是上不去&#xff1b;市场部说数据在增长&#xff0c;销售部却觉得“根本没转化”&#xff1b;高层开会时信心满满…...

opencv关键点检测

python 使用opencv进行图片关键点检测 功能&#xff1a; 在一张图片中裁剪出一块小图 使用cv2中 cv2.SIFT_create() SIFT检测器检测关键点 匹配原图和小图的关键点 import cv2 import numpy as np # 读取图像 img1 cv2.imread(rE:\234947.jpg, cv2.IMREAD_GRAYSCALE) img…...

C#游戏开发中的注意事项

目录 一、性能优化:提升游戏运行效率 1. 避免不必要的循环和迭代 2. 减少字符串拼接 3. 利用Unity的生命周期函数 4. 使用对象池(Object Pooling) 二、内存管理:避免内存泄漏和资源浪费 1. 及时释放非托管资源 2. 避免空引用异常 3. 合理使用引用类型和值类型 4. …...

MySQL的锁

锁 概述&#xff1a;锁是计算机协调多个线程或进程并发访问某一资源的机制。如何保证数据库中并发的一致性&#xff0c;有效性&#xff0c;这就是锁的作用。 分类&#xff1a; 全局锁 对数据库实例加锁&#xff0c;加锁之后&#xff0c;处于只读状态&#xff0c;后续的DML语句…...

学习黑客5 分钟小白弄懂Windows Desktop GUI

5 分钟小白弄懂Windows Desktop GUI &#x1f5a5;️ 大家好&#xff01;今天我们将深入浅出地探索Windows桌面图形用户界面(GUI)——这是我们每天与计算机交互的"门面"。无论你是刚开始接触计算机&#xff0c;还是想在TryHackMe等平台上提升安全技能&#xff0c;理…...

机器人运动控制原理浅析-UC Berkeley超视觉模态模型

加州伯克利发布的超视觉多感知模态融合(FuSe, Fuse Heterogeneous Sensory Data)模型&#xff0c;基于视觉、触觉、听觉、本体及语言等模态&#xff0c;利用自然语言跨模态对齐(Cross-Modal Grounding)优调视觉语言动作等通用模型&#xff0c;提高模型任务成功率。 总体框架 …...

【计算机网络】网络IP层

&#x1f4da; 博主的专栏 &#x1f427; Linux | &#x1f5a5;️ C | &#x1f4ca; 数据结构 | &#x1f4a1;C 算法 | &#x1f152; C 语言 | &#x1f310; 计算机网络 上篇文章&#xff1a;传输层协议TCP 下篇文章&#xff1a;数据链路层 文章摘要&#xff1…...

Nginx重写功能

目录 一 . 简介 二. if指令 2.1基本语法 2.2 举例说明 2.3 配置实例 三. return 3.1 基本语法 3.2 配置实例 四. set指令 4.1 基本语法 4.2 举例说明 4.3 配置实例 五.break指令 5.1 作用 5.2 举例说明 5.3 配置实例 六.rewrite指令 6.1 基本语法 6.2 配…...

2025-05-11 项目绩效域记忆逻辑管理

好的&#xff0c;我们可以用一个故事来帮助记忆这些规划绩效域的要素&#xff0c;同时通过逻辑关系来串联它们。以下是一个故事化的版本&#xff1a; 《项目管理的奇幻之旅》 在一个遥远的王国里&#xff0c;有一个勇敢的项目经理名叫小K。小K被国王赋予了一个艰巨的任务&…...

全模态具身智能:从 VLM 到 MLLM

写在前面 人工智能的感知边界正在以前所未有的速度扩展。最初,我们惊叹于大型语言模型(LLM)对文本的深刻理解和流畅生成。很快,视觉语言模型(Vision-Language Models, VLM) 登场,让 AI 第一次真正“看见”了世界,能够理解图像内容并将其与语言关联,实现了“看图说话”…...

C++入门小馆: 二叉搜索树

嘿&#xff0c;各位技术潮人&#xff01;好久不见甚是想念。生活就像一场奇妙冒险&#xff0c;而编程就是那把超酷的万能钥匙。此刻&#xff0c;阳光洒在键盘上&#xff0c;灵感在指尖跳跃&#xff0c;让我们抛开一切束缚&#xff0c;给平淡日子加点料&#xff0c;注入满满的pa…...

C++.IP协议通信

C++IP协议通信 1. TCP协议通信1.1 服务端实现创建套接字绑定地址监听连接接受连接数据传输关闭连接1.2 客户端实现创建套接字连接服务器数据传输关闭连接1.3 示例代码服务端代码示例客户端代码示例绑定地址接收数据发送数据关闭套接字2.2 客户端实现创建套接字发送数据接收数据…...

虚幻引擎5-Unreal Engine笔记之UE编辑器退出时的保存弹框

虚幻引擎5-Unreal Engine笔记之UE编辑器退出时的保存弹框 code review! 文章目录 虚幻引擎5-Unreal Engine笔记之UE编辑器退出时的保存弹框1. 退出编辑器时弹出的“Save Content”窗口2. File 菜单中的保存选项3. 区别总结 1. 退出编辑器时弹出的“Save Content”窗口 退出时…...

【KEIL】更新AC6编译器

看过部分的文章&#xff0c;Arm Compiler 6&#xff08;AC6&#xff09;编译器&#xff0c;相比AC5在编译速度和代码优化上提升了。因此&#xff0c;笔者决定升级到AC6的最新版本。可以更新keil5到最新版本&#xff0c;上面集成AC6编译器的版本&#xff0c;与最新版相差不远。假…...

Mosquitto MQTT库实战指南

目录 1. MQTT协议简介2. Mosquitto概述3. 开源MQTT实现对比4. 为什么选择Mosquitto5. Mosquitto的交叉编译6. MQTT发布订阅实战7. 进阶应用与最佳实践8. 总结 1. MQTT协议简介 MQTT&#xff08;Message Queuing Telemetry Transport&#xff09;是一种基于发布/订阅模式的轻…...

C语音学习---函数指针

目录 1. 函数指针解析 2. 自定义实现&#xff08;函数指针赋值&#xff09; 利用下面一段例子来解析&#xff1a; int (*set_slave)(modbus_t *ctx, int slave); 1. 函数指针解析 set_slave 是一个 函数指针&#xff0c;指向一个函数。 该函数接受两个参数&#xff1a; mo…...

04.three官方示例+编辑器+AI快速学习webgl_animation_skinning_additive_blending

本实例主要讲解内容 这个示例展示了Three.js中**骨骼动画的叠加混合(Additive Animation Blending)**技术。通过加载一个机器人模型&#xff0c;演示了如何在基础动画(如站立、行走、跑步)之上叠加额外的动画效果(如潜行姿态、悲伤表情、点头同意等)&#xff0c;实现更丰富的角…...

WSL配置docker启动nacos容器load derby-schema.sql error.问题解决方案

nacos配置问题 问题再现查看错误logsQWEN的解答不挂载/data结论 问题再现 本来想要本地跑一下nacos,之前都是直接在Linux环境下面&#xff0c;现在就使用windows的wsl跑一下nacos,之前是需要先配置/conf文件下面的porperties文件以及构建对应的nacos-config数据库。所以我使用…...

游戏引擎学习第272天:显式移动转换

回顾并为今天的内容铺垫背景 我们刚开始为游戏主角编写一些程序逻辑&#xff0c;因为我们之前已经完成了大部分引擎方面的开发&#xff0c;现在可以专注在角色身上。这个角色的移动方式会有些特别&#xff0c;与大多数游戏角色的运动机制不太一样。我们当前正在实现的控制方式…...

AVL树解析

插入操作 // 插入操作 bool insert(const pair<K, V>& kv) {// 若树为空&#xff0c;直接构造&#xff0c;new一个if (_root nullptr) {_root new Node(kv);return true;}// 用于遍历树的当前节点Node* cur _root;// 用于记录当前节点的父节点Node* parent n…...

vue 中的数据代理

在 Vue 中&#xff0c;数据代理&#xff08;Data Proxy&#xff09; 是 Vue 实现 MVVM 模式 的关键技术之一。Vue 使用数据代理让你可以通过 this.message 访问 data.message&#xff0c;而不需要写 this.data.message —— 这大大简化了模板和逻辑代码。 我们来深入理解它的本…...