Spring依赖注入方式
写在前面:大家好!我是
晴空๓
。如果博客中有不足或者的错误的地方欢迎在评论区或者私信我指正,感谢大家的不吝赐教。我的唯一博客更新地址是:https://ac-fun.blog.csdn.net/。非常感谢大家的支持。一起加油,冲鸭!
用知识改变命运,用知识成就未来!加油 (ง •̀o•́)ง (ง •̀o•́)ง
文章目录
- 前言
- 依赖注入方式
- 基于构造法方法的注入
- 基于Setter方法的注入
- 基于字段注入
- 基于方法注入
- 配置方式
- 示例代码
- 基于XML文件的配置注入
- pom依赖
- 相关标签
- bean标签
- constructor-arg标签
- property标签
- XML实现构造器注入
- XML实现Setter方法注入
- 基于注解的配置注入
- 将一个类声明为Bean的注解
- 可注入依赖的注解
- 基于XML和注解混合方式
- 基于Java类的配置注入
- @ComponentScan
- @Bean
前言
Spring 框架对于 Java 服务端开发无疑是一个举足轻重的存在,它以简洁、高效、灵活的特性为我们开发提供了强大的支持,极大的提高了我们的开发效率和代码质量。Spring 解决了一个非常重要的问题,它可以通过 XML 或者 注解 来管理对象之间的依赖关系,也就是 Spring 的依赖注入机制。
依赖注入(Dependency Injection,DI) 作为 Spring 核心理念之一,贯穿于整个框架的使用之中。通过依赖注入,Spring容器在创建一个对象时,会自动将这个对象的依赖注入进去,就不需要程序员主动通过 new对象 的方式进行对象创建。Spring 通过依赖注入机制打破了传统编程中对象之间紧密耦合的局面,让各个组件能够更加独立、灵活地进行开发、测试和维护,为构建复杂而稳定的软件系统奠定了坚实的基础。Spring 框架提供了多种依赖注入方式,本文主要介绍一下 Spring 框架中各种依赖注入的方式。
依赖注入方式
Spring 主要有三种常见的注入方式,分别是基于构造方法的注入、基于Setter方法的注入、基于字段注入。还有不经常使用的基于方法注入、接口回调注入。
基于构造法方法的注入
所谓基于构造方法注入,就是通过构造方法将依赖项传递给对象。在对象实例化时,Spring IoC 容器会根据构造方法的参数类型,从容器中查找并注入匹配的依赖项。
public class MyService { private final MyDependency myDependency; @Autowired public MyService(MyDependency myDependency) { this.myDependency = myDependency; }
}
相比于其他注入方式,Spring 官方更推荐构造函数注入。官方文档说明如下(https://docs.spring.io/spring-framework/reference/core/beans/dependencies/factory-collaborators.html#beans-setter-injection):
官方推荐使用构造器注入的原因主要有以下几点:
- 依赖完整性:构造器注入可以确保在对象创建时,所有必需的依赖项都已经被注入,从而避免空指针异常。
- 实现不可变对象:构造器注入可以将对象实现为不可变对象,即对象的状态在创建后不能被修改。
不可变对象在多线程环境中更安全,因为它们的状态不会被改变
。 - 初始化保证:通过构造器注入的组件总是以完全初始化的状态返回给客户端(调用方)代码。组件在使用前已完全初始化,减少了潜在的错误。
- 避免过多的构造器参数:构造器注入鼓励将类的设计保持简洁,避免过多的依赖项。如果一个类有过多的构造器参数,这通常是一个糟糕的设计,表明该类承担了过多的职责,应该进行重构。
基于Setter方法的注入
通过类的 Setter 方法来注入依赖项。适用于可选依赖或易于变更的配置属性的场景,因为对象可以先创建一个默认状态,然后再通过 Setter 方法补充注入依赖。基于 Setter 方法注入的一个好处是,Setter 方法使得该类的对象便于以后重新配置或重新注入。
public class MyService { private MyDependency myDependency; @Autowired public void setMyDependency(MyDependency myDependency) { this.myDependency = myDependency; }
}
基于字段注入
基于字段注入直接通过注解(@Autowired、@Resource、@Inject)将依赖项注入到目标类的字段中。这是最简洁的方式,但通常不推荐使用。
public class MyService { @Autowired private MyDependency myDependency;
}
基于方法注入
通过普通方法(非Setter方法)注入依赖项。这种方式较为灵活,但使用较少。
public class MyService { public void performAction(@Autowired MyDependency myDependency) { myDependency.doSomething(); }
}
配置方式
以上这些注入方式主要通过三种方式进行配置,分别是基于XML文件的配置、基于注解的配置和基于Java类的配置。其中基于XML文件的配置是 Spring 早期的配置方式,现在使用的比较少。目前我们常用的配置方式主要是后两种。但是第一种方式也是需要了解的,如果看一些老项目的话还是需要掌握一下。
示例代码
为了方便展示写了一个简单的计算器,该计算器只实现了加减乘除操作,我们需要在计算器的 Controller 类中注入加减乘除服务的实现类 ServiceImpl
// CalculatorApplication类
package com.qingkong.application;import com.qingkong.calculator.CalculatorController;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;public class CalculatorApplication {public static void main(String[] args) {// 加载Spring的配置文件ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");// 获取 CalculatorController BeanCalculatorController calculatorController = context.getBean(CalculatorController.class);// 测试计算器的功能System.out.println(calculatorController.add(1, 2));}
}---
// CalculatorService 接口
package com.qingkong.service;public interface CalculatorService {double add(double a, double b);double subtract(double a, double b);double multiply(double a, double b);double divide(double a, double b);
}---
// CalculatorService 接口实现类
package com.qingkong.service.impl;import com.qingkong.service.CalculatorService;public class CalculatorServiceImpl implements CalculatorService {public CalculatorServiceImpl() {System.out.println("CalculatorServiceImpl无参构造方法被调用了");}@Overridepublic double add(double a, double b) {return a + b;}@Overridepublic double subtract(double a, double b) {return a - b;}@Overridepublic double multiply(double a, double b) {return a * b;}@Overridepublic double divide(double a, double b) {if (b == 0) {throw new IllegalArgumentException("不能除以0!");}return a / b;}
}
基于XML文件的配置注入
pom依赖
要想实现 Spring 的依赖注入功能,需要引入 spring-context 依赖。spring-context 模块是 Spring 框架的核心模块之一,它提供了支持 Spring 应用上下文和事件驱动模型的功能。它构建在 spring-core 和 spring-beans 模块之上,提供了以下功能:
- 依赖注入(DI):支持通过 XML 配置、注解或 Java 配置进行依赖注入。
- 应用上下文(ApplicationContext):提供了更高级的上下文接口,如 ClassPathXmlApplicationContext 和 FileSystemXmlApplicationContext,用于加载配置文件和管理 Bean。
- 事件支持:支持事件发布和监听。
我们只在 pom.xml 中引入 spring-context 依赖,Maven 会根据 依赖传递 自动将 spring-core、spring-bean、spring-aop等spring-context依赖的其他 Spring 模块引入。这些模块是 spring 框架的核心模块,真实的开发场景下还会引入其他模块。
<dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.3.30</version>
</dependency>
相关标签
bean标签
在 XML 中我们可以通过 <bean>
标签来配置一个bean,该标签主要的元素如下所示:
属性 | 作用 |
---|---|
id | Bean 的唯一标识符,用于在 Spring 容器中引用该 Bean |
class | Bean 的全限定类名 |
scope | 定义 Bean 的作用域(如 singleton、prototype 等) |
autowire | 指定自动装配的方式(如 byName、byType 等) |
factory-bean | 指定一个工厂 Bean,用于创建当前 Bean |
factory-method | 指定工厂 Bean 中用于创建当前 Bean 的方法,与 factory-bean 配合使用 |
parent | 指定当前 Bean 的父 Bean |
lazy-init | 指定是否延迟初始化该 Bean |
depends-on | 指定该 Bean 依赖的其他 Bean |
primary | 标记当前 Bean 为优先选择的 Bean |
init-method | 指定 Bean 初始化时执行的方法 |
destroy-method | 指定 Bean 销毁时执行的方法 |
constructor-arg标签
XML 配置文件使用 <constructor-arg/>
标签传入构造方法所需要的内容。该标签主要的属性如下:
属性 | 作用 |
---|---|
ref | 传给构造方法参数的Bean ID |
value | 传给构造方法参数的值 |
type | 构造方法参数对应的类型 |
index | 构造方法参数对应的位置,从0开始计算 |
name | 构造方法参数对应的名称 |
property标签
使用 <property/>
可以为 Bean 的属性赋值或注入其他 Bean。该标签主要的元素如下:
属性 | 作用 |
---|---|
name | 指定要设置的属性名称 |
value | 指定要注入的属性值 |
ref | 引用另一个 Bean 的 Bean ID |
XML实现构造器注入
对于 CalculatorController 类,先生成该类的构造方法,然后将其依赖的 CalculatorService 类作为构造方法的参数注入。完成之后在 XML 文件中进行配置,具体实现如下:
CalculatorController类:
public class CalculatorController {private final CalculatorService calculatorService;// 构造器注入public CalculatorController(CalculatorService calculatorService) {System.out.println("开始进行构造器注入");this.calculatorService = calculatorService;}public double add(double a, double b) {return calculatorService.add(a, b);}public double subtract(double a, double b) {return calculatorService.subtract(a, b);}public double multiply(double a, double b) {return calculatorService.multiply(a, b);}public double divide(double a, double b) {return calculatorService.divide(a, b);}
}
XML配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><!-- 定义 CalculatorService 的 Bean --><bean id="calculatorService" class="com.qingkong.service.impl.CalculatorServiceImpl"/><!-- 定义 CalculatorController 的 Bean,使用构造器注入 --><bean id="calculatorController" class="com.qingkong.calculator.CalculatorController" ><constructor-arg ref="calculatorService"/></bean></beans>
XML实现Setter方法注入
首先在 CalculatorController 类中设置 calculatorService 的类型的属性,然后生成 Set 方法。完成之后在 XML 文件中进行配置,具体实现如下:
CalculatorController类:
public class CalculatorController {private CalculatorService calculatorService;// Set方法public void setCalculatorService(CalculatorService calculatorService) {System.out.println("开始进行Setter方法注入");this.calculatorService = calculatorService;}public double add(double a, double b) {return calculatorService.add(a, b);}public double subtract(double a, double b) {return calculatorService.subtract(a, b);}public double multiply(double a, double b) {return calculatorService.multiply(a, b);}public double divide(double a, double b) {return calculatorService.divide(a, b);}
}
XML配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><!-- 定义 CalculatorService 的 Bean --><bean id="calculatorService" class="com.qingkong.service.impl.CalculatorServiceImpl"/><!-- 定义 CalculatorController 的 Bean,使用Setter方法注入 --><bean id="calculatorController" class="com.qingkong.calculator.CalculatorController" ><!-- name属性的值对应 calculatorController 方法中相关属性的名称 --><property name="calculatorService" ref="calculatorService"/></bean></beans>
基于注解的配置注入
使用注解注入有两种方式,分别是基于XML和注解混合方式 和 纯注解方式。第一种方式需要在 XML 文件中配置 context:component-scan 标签,而基于纯注解方式则不需要进行任何 XML 配置,只需要在配置类中加入 @ComponentScan 注解并执行扫描包路径即可。
将一个类声明为Bean的注解
在 XML配置注入 中我们通过 <bean/>
配置一个 Bean,而在基于注解的配置时我们只需要在类上面添加如下四个注解中的任意一个都能让 Spring 容器把他们配置为 Bean。
注解 | 说明 |
---|---|
@Component | 将类标识为普通的Bean |
@Service | 用于标识服务层的类 |
@Controller | 用于标识控制器层(Controller)的类(后来针对REST服务又增加了一个@RestController注解) |
@Repository | 用于标识数据访问层的类,与数据库进行交互 |
这四个注解本质上没有区别,而且后面三个注解都是 @Component 注解的衍生注解,之所以这么做主要是为了在分层架构中提供更好的语义化支持和分层的明确性,提高代码的可读性和可维护性。
可注入依赖的注解
Spring 内置的 @Autowired 以及 JDK 内置的 @Resource 和 @Inject 都可以用于注入 Bean。这里需要注意的是 @Autowired 默认的注入方式为 byType(根据类型进行匹配),如果一个接口存在多个实现类,Spring 会同时找到多个满足条件的选择,这种情况下,注入方式会变为 byName(根据名称进行匹配),这个名称通常就是类名(首字母小写)。
如果一个接口有多个实现类一般建议使用 @Autowired 和 @Qualifier 注解配合的方式来显式指定名称而不是依赖变量的名称来区分。@Qualifier 注解主要用来在依赖注入时消除歧义。当面对一个接口有多个实现类我们可以使用 @Qualifier(“类名”) 的方式使 Spring 在依赖注入时准确的找到依赖。
@Resource 默认注入方式为 byName。如果无法通过名称匹配到对应的 Bean 时,注入方式会变为 byType。 该注解 name 和 type 有两个重要的属性。如果仅指定 name 属性则注入方式为 byName,如果仅指定 type属性则注入方式为 byType,如果同时指定 name 和 type 属性(不建议这么做)则注入方式为byType + byName。
基于XML和注解混合方式
CalculatorServiceImpl类
@Component
public class CalculatorServiceImpl implements CalculatorService {// 省略重复代码……
}
CalculatorController类
@Component
public class CalculatorController {@Autowiredprivate CalculatorService calculatorService;public double add(double a, double b) {return calculatorService.add(a, b);}public double subtract(double a, double b) {return calculatorService.subtract(a, b);}public double multiply(double a, double b) {return calculatorService.multiply(a, b);}public double divide(double a, double b) {return calculatorService.divide(a, b);}
}
XML文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd"><!-- 开启注解并扫描指定包中带有注解的类 --><context:component-scan base-package = "com.qingkong"/></beans>
基于Java类的配置注入
从 Spring Framework 3.0 开始,我们可以使用 @Configuration 注解定义配置类,可以达到替换 XML 配置文件的效果。使用 @Configuration、@Bean、@ComponentScan 等一系列注解,基本可以满足日常开发所需。
我们只需要将原来的 XML 配置文件删除,再加上带有 @Configuration 注解的配置类即可。
配置类代码:
@Configuration
@ComponentScan("com.qingkong")
public class Config {
}
当使用时需要通过 AnnotationConfigApplicationContext 对象来加载我们定义的配置类。
// 加载Spring的配置类
ApplicationContext context = new AnnotationConfigApplicationContext(Config.class);
// 获取 CalculatorController Bean
CalculatorController calculatorController = context.getBean(CalculatorController.class);
@ComponentScan
@ComponentScan 注解指定了类扫描的包名,作用与 XML 配置文件中的 <context:component-scan/>
类似。如果配置类的 @ComponentScan 没有指定扫描的基础包路径或者类,那么默认从该配置类的包开始扫描(最好还是指定路径,防止漏包)。该注解的 includeFilters 和 excludeFilters 属性可以用来指定包含和排除组件。官网中也给出了相应的示例:
@Configuration
@ComponentScan(basePackages = "org.example",includeFilters = @Filter(type = FilterType.REGEX, pattern = ".*Stub.*Repository"),excludeFilters = @Filter(Repository.class))
public class AppConfig {...
}
该配置类与以下 XML配置 等效:
<beans><context:component-scan base-package="org.example"><context:include-filter type="regex"expression=".*Stub.*Repository"/><context:exclude-filter type="annotation"expression="org.springframework.stereotype.Repository"/></context:component-scan>
</beans>
@Bean
我们也可以在配置类将 @Bean 注解放在方法上面,这样方法的返回对象就会被当做容器的一个 Bean。默认情况下,Bean 名称与方法名称相同。官网给出了使用例子:
@Configuration
public class AppConfig {@Beanpublic TransferService transferService() {return new TransferServiceImpl();}
}
该配置与以下 XML 中的配置是等价的:
<beans><bean id="transferService" class="com.acme.TransferServiceImpl"/>
</beans>
既然有了 @Component 等注解为什么还要用 @Bean 注解呢?@Component 注解作用于类,而 @Bean 注解作用于方法。当我们引用第三方库中的类需要装配到 Spring 容器时,只能用 @Bean 注解来实现,因为通常情况下我们是无法修改第三方类库的代码的。
- https://docs.spring.io/spring-framework/docs/5.2.x/spring-framework-reference/core.html#spring-core
- 《学透Spring 从入门到项目实战》
- JavaGuide
相关文章:
Spring依赖注入方式
写在前面:大家好!我是晴空๓。如果博客中有不足或者的错误的地方欢迎在评论区或者私信我指正,感谢大家的不吝赐教。我的唯一博客更新地址是:https://ac-fun.blog.csdn.net/。非常感谢大家的支持。一起加油,冲鸭&#x…...
【C++八股】什么是函数指针,如何定义和使⽤场景
函数指针是指向函数的指针变量,它存储的是一个函数的地址,允许通过指针间接调用函数。使用函数指针可以动态地选择和调用函数,或者将函数作为参数传递给其他函数。 1. 函数指针的定义 函数指针的定义与普通指针稍有不同。它需要声明指向特定…...
DeepSeek 入驻 Cursor —— 表现能否超越 Claude?
DeepSeek 刚刚在 Cursor 平台上线了它的两款模型:DeepSeek V3 和 R1。目前,许多开发者(包括我们在内)主要依赖 Claude 3.5 Sonnet(最新版本 claude-3-5-sonnet-20241022)作为主要语言模型,因此我…...
skynet-src/atomic.h:25:23: 致命错误:stdatomic.h:没有那个文件或目录
skynet-src/atomic.h:25:23: 致命错误:stdatomic.h:没有那个文件或目录 故事背景,又接了一个新项目,要搞开发环境啊。skyent的框架。ubuntu已经跑起来了,但是内网是centos的跑不起来。只能怀疑是环境问题了࿰…...
docker compose部署nexus
整个工具的代码都在Gitee或者Github地址内 gitee:solomon-parent: 这个项目主要是总结了工作上遇到的问题以及学习一些框架用于整合例如:rabbitMq、reids、Mqtt、S3协议的文件服务器、mongodb github:GitHub - ZeroNing/solomon-parent: 这个项目主要是…...
IBM服务器刀箱Blade安装Hyper-V Server 2019 操作系统
案例:刀箱某一blade,例如 blade 5 安装 Hyper-V Server 2019 操作系统(安装进硬盘) 刀箱USB插入安装系统U盘,登录192.168... IBM BlandeCenter Restart Blande 5,如果Restart 没反应,那就 Power Off Blade 然后再 Power On 重启后进入BIOS界面设置usb存储为开机启动项 …...
Python的顺序结构和循环结构
文章目录 一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(c)多分支 if-elif-elif-...-else (3)注意事…...
[python SQLAlchemy数据库操作入门]-30.掌握 SQLAlchemy:优雅应对数据库异常的艺术
哈喽,大家好,我是木头左! 捕获和处理 SQLAlchemy 异常 在实际应用中,需要根据不同的异常类型采取不同的处理策略。下面是一个示例,展示了如何捕获和处理 SQLAlchemy 的异常。 from sqlalchemy import create_engine, Column, Integer, String, exc from sqlalchemy.ext.…...
编译和链接【三】
文章目录 编译和链接【三】前言系列文章入口编译过程词法分析语法分析语义分析生成中间代码汇编链接 编译和链接【三】 前言 在我大一的时候, 我使用VC6.0对C语言程序进行编译链接和运行 , 然后我接触了VS, Qt creator等众多IDE,…...
11苍穹外卖之Apache ECharts(数据可视化图表、后端传数据)
课程内容 Apache ECharts 营业额统计 用户统计 订单统计 销量排名Top10 功能实现:数据统计 数据统计效果图: 1. Apache ECharts 1.1 介绍 Apache ECharts 是一款基于 Javascript 的数据可视化图表库,提供直观,生动&#…...
Grafana-使用Button修改MySQL数据库
背景 众所周知,Grafana是一个用来展示数据的平台,但是有时候还是会有需求说能不能有一个按钮,点击的时候再对数据库进行修改,从而达到更新数据的效果 经过多方查证,终于实现了一个简单的,点击button执行sq…...
C++11语法及库详解
目录 一、c11语法 1. 自动类型推导 (auto) 2. 基于范围的 for 循环 3. Lambda 表达式 4. 智能指针 5. 右值引用和移动语义 6. nullptr 关键字 7. constexpr 关键字 8. 初始化列表 9. std::thread 多线程支持 10. std::function 和 std::bind 11. std::unordered_map…...
数据库,数据表的增删改查操作
一.数据库的基本操作 (1)创建数据库 创建数据库就是在数据库系统中划分一块存储数据的空间,方便数据的分配、放置和管理。在MySQL中使用CREATE DATABASE命令创建数据库,语法格式如下: CREATE DATABASE数据库名称; 注:…...
Python——批量图片转PDF(GUI版本)
目录 专栏导读1、背景介绍2、库的安装3、核心代码4、完整代码总结专栏导读 🌸 欢迎来到Python办公自动化专栏—Python处理办公问题,解放您的双手 🏳️🌈 博客主页:请点击——> 一晌小贪欢的博客主页求关注 👍 该系列文章专栏:请点击——>Python办公自动化专…...
RocketMQ面试题:原理部分
🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编…...
中间件-安装Minio-集成使用(ubantu-docker)
目录 1、安装docer 2、运行以下命令拉取MinIO的Docker镜像 3、检查当前所有Docker下载的镜像 4、创建目录 5、创建Minio容器并运行 6、SDK操作 FileUploader.java 1、安装docer 参考这篇:Linux安装Docker 2、运行以下命令拉取MinIO的Docker镜像 docker pull…...
vue3: const一个function怎么写呢?
问: 用ref或者reactive定一个一个const refCharts ref(function xxx (config){ xxxxx,xxxxx xxxx,xxxx })请问怎么写? 回答: 在 Vue 3 中,ref 和 reactive 是两种常用的响应式数据定义方式。ref 用于定义单个响应式对象或原始值…...
【JVM详解三】垃圾回收机制
一、对象是否存活 强引用:Object obj new Object(); 只要强引用还在,垃圾收集器永远不会回收掉被引用的对象。在不用对象的时将引用赋值为 null,能够帮助垃圾回收器回收对象。比如 ArrayList 的 clear() 方法实现。软引用(SoftRe…...
dbeaver 安装之后出现mysql连接异常问题
1.手动下载mysql驱动程序 参考文档DBeaver连接mysql驱动下载失败怎么办?-CSDN博客 2.添加对应的下载程序到dbeaver 【DBeaver】缺少mysql驱动_dbeaver连接mysql缺少驱动-CSDN博客 配置mysql 地址端口账户等即可使用...
Android Studio历史版本下载
Android Studio历史版本下载 历史版本所在网站: https://developer.android.google.cn/studio/archive 等待加载 1、(需要点击switch to english)然后出现Terms and conditions 2、滑到最下面,点击I agree to the terms,之后才会…...
【每日一题 | 2025】2.3 ~ 2.9
个人主页:GUIQU. 归属专栏:每日一题 文章目录 1. 【2.3】P8784 [蓝桥杯 2022 省 B] 积木画2. 【2.4】P8656 [蓝桥杯 2017 国 B] 对局匹配3. 【2.5】[ABC365D] AtCoder Janken 34. 【2.6】P8703 [蓝桥杯 2019 国 B] 最优包含5. 【2.7】P8624 [蓝桥杯 2015…...
国产编辑器EverEdit - 迷你查找
1 迷你查找 1.1 应用场景 某些场景下,用户不希望调出复杂的查找对话框,此时可以使用迷你查找窗口。 1.2 使用方法 选择主菜单查找 -> 迷你查找,或使用快捷键Ctrl Alt F,会在右上角弹出迷你查找窗口,如下图所示…...
制药行业 BI 可视化数据分析方案
一、行业背景 随着医药行业数字化转型的深入,企业积累了海量的数据,包括销售数据、生产数据、研发数据、市场数据等。如何利用这些数据,挖掘其价值,为企业决策提供支持,成为医药企业面临的重大挑战。在当今竞争激烈的…...
第40天:Web开发-JS应用VueJS框架Vite构建启动打包渲染XSS源码泄露代码审计
#知识点 1、安全开发-VueJS-搭建启动&打包安全 2、安全开发-VueJS-源码泄漏&代码审计 一、Vue搭建创建项目启动项目 1、Vue 框架搭建->基于nodejs搭建,安装nodejs即可 参考:https://cn.vuejs.org/ 已安装18.3或更高版本的Node.js 2、Vue 创建…...
2.10学习总结
今天接着看了数据结构,但是跟指针有关的看不懂(万恶的指针),写了考试的补题。 #include <stdio.h> #include <stdlib.h> int a[1000005]; int main() {int n,i,x0;scanf("%d",&n);for(i1;i<n;i){x;i…...
MySQL 中可以通过添加主键来节省磁盘空间吗?(译文)
从历史上看,MySQL 不需要在表上定义显式主键,直到今天默认都是这样。不过,这种要求是通过两种复制方法施加的:组复制和 Percona XtraDB 集群 (PXC),默认情况下不允许使用没有主键的表。对于缺少…...
Django在终端创建项目(pycharm Windows)
1.选择目录 选择或新建一个文件夹,作为项目保存的地方 2.右键在终端打开 3.确定django-admin.exe安装位置 找到自己安装django时,django-admin.exe安装的位置,例如 4.运行命令 使用django-admin.exe的绝对路径,在刚才打开的终端…...
IDEA接入DeepSeek
IDEA 目前有多个途径可以接入deepseek,比如CodeGPT或者Continue,这里借助CodeGPT插件接入,CodeGPT目前用的人最多,相对更稳定 一、安装 1.安装CodeGPT idea插件市场找到CodeGPT并安装 2.创建API Key 进入deepseek官网…...
Linux 内核自旋锁spinlock(二)--- ticket spinlock
文章目录 前言一、ticket spinlock二、源码分析2.1 spin_lock_init2.2 spin_lock2.2 spin_unlock 参考资料 前言 自旋锁是 Linux 内核中最底层的互斥机制。因此,它们对内核的安全性和性能有着巨大的影响,因此对各种(特定架构的)自…...
Elixir语言的计算机基础
Elixir语言的计算机基础 引言 Elixir是一种现代的编程语言,建立在Erlang虚拟机(BEAM)上,专注于并发、分布式系统和容错能力。随着互联网的发展,应用程序的需求变得越来越复杂,Elixir凭借其高效的性能、灵…...
MindStudio制作MindSpore TBE算子(二)算子测试
在上一节中,成功制作了Mindspore的Add算子,具体可以查看MindStudio制作MindSpore TBE算子(一)算子制作,这一节,一起看看如何对算子进行测试。 建议参考以下内容一起食用: 算子代码实现 MindSpor…...
深度解读城市地下网管管廊改造要点
引 言 近日国家发改委和住建部联合发布通知(后附),要求各地抓紧编制“城市地下管网和综合管廊建设改造实施方案”并于12月27日前报国家发改委和住建部相关司处,逾期未报的城市(县、区),视同自愿…...
Docker 部署 redis | 国内阿里镜像
一、简易单机版 1、镜像拉取 # docker hub 镜像 docker pull redis:7.0.4-bullseye # 阿里云镜像 docker pull alibaba-cloud-linux-3-registry.cn-hangzhou.cr.aliyuncs.com/alinux3/redis_optimized:20240221-6.2.7-2.3.0 2、运行镜像 docker run -itd --name redis \n …...
Day88:加载游戏图片
在游戏开发中,加载和显示图片是非常常见的需求,尤其是在 2D 游戏 中,角色、背景、道具、敌人等都需要用图片来表示。今天,我们将学习如何在 Python 游戏开发中使用 Pygame 加载并显示图片。 1. 加载游戏图片的基本步骤 在 Pygame 中加载图片通常需要以下几个步骤: 导入 P…...
Elasticsearch:在 Elastic 中玩转 DeepSeek R1 来实现 RAG 应用
在这个春节,如一声春雷,DeepSeek R1 横空出世。现在人人都在谈论 DeepSeek R1。这个大语言模型无疑在中国及世界的人工智能发展史上留下了重要的里程碑。那么我们改如何结合 DeepSeek R1 及 Elasticsearch 来实现 RAG 呢?在之前的文章 “使用…...
SOME/IP报文格式及发现协议详解
在之前的文章中,我们介绍了SOME/IP协议的几种服务接口。在本篇博客中,主要介绍some/ip协议传输的header报文格式以及SOME/IP-SD发现协议。 目录 流程 报文格式 Message ID Length Request ID protocal version/Interface Version Message Type…...
elementplus 使用日期时间选择器,设置可选范围为前后大于2年且只能选择历史时间不能大于当前时间点
需求:时间选择器可选的时间范围进行限制,-2年<a<2年且a<new Date().getTime()核心:这里需要注意plus版没有picker-options换成disabled-date属性了,使用了visible-change和calendar-change属性逻辑:另设一个参…...
C语言·关键字·char关键字
C语言菜鸟入门关键字char关键字_c char-CSDN博客...
Ansible简单介绍及用法
一、简介 Ansible是一个简单的自动化运维管理工具,基于Python语言实现,由Paramiko和PyYAML两个关键模块构建,可用于自动化部署应用、配置、编排task(持续交付、无宕机更新等)。主版本大概每2个月发布一次。 Ansible与Saltstack最大的区别是…...
Mac 本地搭建自己的 DeepSeek
Mac 本地搭建自己的 DeepSeek 安装 Ollama通过Ollama命令安装 DeepSeek 模型安装一个UI客户端,提升体验 注:本文章完全参考网上教程,没有丝毫原创,只是记录一下我本人在安装DeepSeek 的步骤 安装 Ollama https://ollama.com/dow…...
深度学习-交易预测
下面为你详细介绍如何使用Python结合深度学习库TensorFlow和Keras来构建一个简单的交易预测模型。在这个示例中,我们以股票价格预测为例,假设我们要根据过去一段时间的股票价格数据来预测未来的价格走势。 步骤分析 数据准备:获取股票价格数…...
Prompt逆向工程:如何“骗“大模型吐露其Prompt?
提示词的“逆向工程”,让AI大语言模型帮你反推提示词 一、前言 在日常生活中,我们不时会遇到一些令人惊艳的文本,不论是一篇精彩绝伦的小说、一篇深入浅出的科普文章,还是一篇充满热情的音乐推荐,它们都能在我们的心…...
游戏手柄Type-c方案,支持一边充电一边传输数据
乐得瑞推出LDR6023SS,专门针对USB-C接口手机手柄方案,支持手机快充,支持任天堂游戏机,PS4等设备~同时支持手机充电跟数据传输 1、概述 LDR6023SS SSOP16 是乐得瑞科技针对 USB Type-C 标准中的 Bridge 设备而开发的双 USB-C DRP …...
Vue设计模式到底多少种?
Vue设计模式到底多少种? 很多同学问,Vue到底有多少种设计模式??各个模式到底是什么意思??又各自适合什么场景?? 这里我给大家直接说下,Vue的设计模式没有一个固定的数值…...
C++ 中的 std::timed_mutex 和 std::recursive_timed_mutex
1、背景 在多线程编程中,互斥锁(Mutex)是用于保护共享资源的重要工具。C 标准库提供了多种互斥锁类型,其中 std::timed_mutex 和 std::recursive_timed_mutex 是两种支持超时功能的互斥锁。在阅读FastDDS源码时,发现了…...
HAL库外设宝典:基于CubeMX的STM32开发手册(持续更新)
目录 前言 GPIO(通用输入输出引脚) 推挽输出模式 浮空输入和上拉输入模式 GPIO其他模式以及内部电路原理 输出驱动器 输入驱动器 中断 外部中断(EXTI) 深入中断(内部机制及原理) 外部中断/事件控…...
Kotlin实战经验:将接口回调转换成suspend挂起函数
在 Kotlin 协程中, suspendCoroutine 和 suspendCancellableCoroutine 是用于将回调或基于 future 的异步操作转换成挂起函数。 suspendCoroutine 用途:将回调式异步操作转换为可挂起函数 行为: 启动一个新的协程来处理基于回调的操作挂起当前协程,直到调用回调回调负责…...
银行国际结算
银行国结项目,即国际结算项目,是银行业务中的重要组成部分,它涉及跨国界的货币收付和资金转移。 一、银行国结项目的定义 银行国结项目是指银行为国际贸易、投资等活动提供的国际结算服务,包括各种国际支付和资金清算业务。这些…...
java后端开发day13--面向对象综合练习
(以下内容全部来自上述课程) 注意:先有javabean,才能创建对象。 1.文字版格斗游戏 格斗游戏,每个游戏角色的姓名,血量,都不相同,在选定人物的时候(new对象的时候&#…...
Vue解决父子组件传值,子组件改变值后父组件的值也改变的问题
vue开发过程中,父组件通过props传值给子组件,子组件在页面展示父组件的值,在操作子组件值以后,即使不点击确定按钮,父组件中的值也发生了变化,但是需求是操作子组件数据以后,必须点击"确定…...