【Spring】IoCDI详解

1. IoC详解

前面提到过IoC就是将对象的控制权交由Spring的IoC容器进行管理,由Spring的IoC容器创建和销毁bean,那么既然涉及到容器,就一定包含以下两方面功能:

  1. bean的存储
  2. bean的获取

1.1 类注解

Spring框架为了更好地服务应用程序,提供了非常丰富的注解用于存储bean,大致可以分为如下两类:

  1. 类注解:例如@Controller@Service@Repository@Configuration@Component
  2. 方法注解:例如:@Bean

接下来我们分别来看各类注解的使用:

1.1.1 @Controller(控制器存储)

1.1.1.1 bean的存储

使用@Controller存储bean的测试代码如下:

@Controller // 将bean存储到Spring容器中
public class TestController {
    public void testController() {
        System.out.println("test controller");
    }
}

那么我们如何才能够获取到Spring容器中的bean呢?

1.1.1.2 **bean的获取 **

我们可以在使用@SpringBootApplication的启动类中获取到ApplicationContext对象,该上下文对象就保存着程序启动时的一些信息,可以看做是IoC容器实例

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取bean
        TestController controllerBean = context.getBean(TestController.class);
        // 3. 测试bean的testController方法
        controllerBean.testController();
    }
}

此时运行结果如下:
image.png
如果再把@Controller注解去掉,观察结果:
image.png
此时就会抛出这样一个异常:No qualifying bean of type 'com.example.ioc_di_demo.controller.TestController' available表明没有找到类型为TestController的bean实例

1.1.1.3 获取bean对象的其他方式

上述方式我们使用了通过类型的方式获取Bean,但是如果Spring容器中有多个同一类型的bean呢,我们应该如何进行获取?ApplicationContext也提供了其他的方式获取bean,这些方式都是其父接口BeanFactory提供的
image.png
其中第一种、第二种、第四种方式是比较常用的!其中1、2两种方式都涉及到name即根据名称来获取,那么bean的名称是什么呢?实质上,Spring容器在管理bean的时候会为每一个bean分配不同的唯一的名称来标识该bean实例,bean的命令规则如下(可观察源码java.beans.Introspector的decapitalize方法求证):

一般来说,如果程序员没有手动指定bean的名称,那么就会自动按照类名首字母小写,小驼峰形式,例如UserController => userController、AccountService => accountService

但是也会有特殊情况,当有多个字符,并且第一个和第二个字符都是大写字符是就保留原始的类名称

例如IService的bean名称就是IService

示例代码

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2.1 获取bean(通过类型)
        TestController controllerBean1 = context.getBean(TestController.class);
        // 2.2 获取bean(通过名称)
        TestController controllerBean2 = (TestController) context.getBean("testController");
        // 2.3 获取bean(通过名称 + 类型)
        TestController controllerBean3 = context.getBean("testController", TestController.class);
        // 3.1 测试controllerBean1的testController方法
        controllerBean1.testController();
        // 3.2 测试controllerBean2的testController方法
        controllerBean2.testController();
        // 3.3 测试controllerBean3的testController方法
        controllerBean3.testController();
    }
}

1.1.2 @Service(业务逻辑存储)

1.1.2.1 bean的存储

我们也可以使用@Service注解将bean交由Spring容器进行管理

@Service // 将TestService作为bean交由Spring容器管理
public class TestService {
    public void testService() {
        System.out.println("test service");
    }
}
1.1.2.2 bean的获取

示例代码如下:

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestService的bean对象
        TestService beanService = (TestService) context.getBean("testService");
        // 3. 调用bean对象的testService方法
        beanService.testService();
    }
}

运行结果
image.png

1.1.3 @Repository(仓库存储)

1.1.3.1 bean的存储
@Repository // 将bean存储到Spring容器
public class TestMapper {
    public void testRepository() {
        System.out.println("test repository");
    }
}
1.1.3.2 bean的获取
@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestMapper的bean对象
        TestMapper beanRepository = (TestMapper) context.getBean("testMapper");
        // 3. 调用bean对象的testRepository方法
        beanRepository.testRepository();
    }
}

1.1.4 @Component(组件存储)

1.1.4.1 bean的存储
@Component
public class TestComponent {
    public void testComponent() {
        System.out.println("test component");
    }
}
1.1.4.2 bean的获取
@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestComponent的bean对象
        TestComponent beanComponent = (TestComponent) context.getBean("testComponent");
        // 3. 调用bean对象的testComponent方法
        beanComponent.testComponent();
    }
}

1.1.5 @Configuration(配置存储)

1.1.5.1 存储bean
@Configuration
public class TestConfiguration {
    public void testConfiguration() {
        System.out.println("test configuration");
    }
}
1.1.5.2 获取bean
@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestConfiguration的bean对象
        TestConfiguration beanConfiguration = (TestConfiguration) context.getBean("testConfiguration");
        // 3. 调用bean对象的testConfiguration方法
        beanConfiguration.testConfiguration();
    }
}

1.1.6 为什么要使用那么多注解?

Spring为我们提供了各式各样的注解,其实与三层架构这样的应用分层思想不谋而合,可以让程序员方便的看到类的注解就明白该类的含义与作用:

  • @Controller:用户标识控制层(表现层),用于接收请求,返回处理完毕的响应结果
  • @Service:用于标识业务逻辑层,用于处理具体的业务逻辑
  • @Repository:用于标识数据层(持久层),负责数据访问操作即与数据库交互
  • @Configuration:用于标识配置层,管理程序中一些配置信息

各个注解之间的关系
观察五大注解的详细信息如下:
image.pngimage.png
image.pngimage.png
可以发现他们都是@Component的衍生注解,而@Component是一个元注解,可以被其他注解所使用

常见面试题:五大注解是否可以相互替换?

  1. 从功能上来说,大部分注解可以相互替换,但是唯独@Controller注解不可以使用别的注解替换,因为该注解用于表现层进行路由映射,使用其他注解代替有极小概率发生意想不到的错误
  2. 从语义性来说,不可以相互替换,@Controller用于标识控制层,@Service注解用于标识业务逻辑层,@Repository用于标识这是一个持久层

1.2 方法注解

上述我们介绍了五大类注解的使用,但是如果面对以下情况就无法使用类注解:

  1. 使用第JDK原生类/第三方包中的类,无法在类上使用注解
  2. 一个类需要多个对象,比如需要多个数据源对象

1.2.1 方法注解的使用

1.2.1.1 bean的配置
public class BeanConfig {

    @Bean
    public LocalDateTime localDateTime() {
        return LocalDateTime.now();
    }
}
1.2.1.2 bean的获取

上述代码我们使用@Bean方法注解配置了一个对象,交由Spring容器进行管理,下面我们尝试获取该实例

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取LocalDateTime的bean对象
        LocalDateTime bean = context.getBean(LocalDateTime.class);
        // 3. 打印获取到的bean
        System.out.println(bean.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
    }
}

image.png
但是此时却抛出了异常!这是因为 方法注解需要配合类注解使用!

1.2.1.3 方法注解需要配合类注解使用

将上述配置bean代码更改如下:

@Configuration
public class BeanConfig {

    @Bean
    public LocalDateTime localDateTime() {
        return LocalDateTime.now();
    }
}

此时程序正常运行:
image.png

1.2.1.4 定义多个对象

比如在多数据源的场景下,我们需要使用多个@Bean注解,如果依旧按照之前通过类型的方式获取,会出现什么问题呢?

@Configuration
public class BeanConfig {

    @Bean
    public LocalDateTime localDateTime() {
        return LocalDateTime.now();
    }

    @Bean LocalDateTime localDateTime2() {
        return LocalDateTime.of(2024, 4, 1, 12, 0, 0);
    }

    @Bean LocalDateTime localDateTime3() {
        return LocalDateTime.of(2024, 5, 1, 8, 0, 0);
    }
}

image.png
此时出现该异常,表示如果按照类型匹配,期望只得到一个实例,但是却匹配到了三个!这个时候我们就需要使用名称来获取bean了!

使用@Bean方法注解声明的bean的名称默认就是方法名称!

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestConfiguration的bean对象
        LocalDateTime bean = (LocalDateTime) context.getBean("localDateTime");
        LocalDateTime bean2 = (LocalDateTime) context.getBean("localDateTime2");
        LocalDateTime bean3 = (LocalDateTime) context.getBean("localDateTime3");
        // 3. 打印获取到的bean
        System.out.println(bean.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        System.out.println(bean2.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        System.out.println(bean3.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
    }
}

此时就可以正确获取结果!
image.png

1.3 配置扫描路径

Q:假设我们按照规则正确地使用注解配置bean,那么一定会被Spring容器进行管理吗?
A:不一定,我们还需要配置bean的扫描路径
image.pngimage.png
我们修改启动类的路径,再次运行程序!
image.png
此时程序运行错误了!其实原因就在于启动类中包含了注解@ComponentScan默认会扫描启动类所在的包及其子包,只有在指定路径下的bean才会被Spring容器管理,解决方式就是使用注解@ComponentScan配置扫描路径!

@ComponentScan(basePackages = {"com.example.ioc_di_demo"})
@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestConfiguration的bean对象
        LocalDateTime bean = (LocalDateTime) context.getBean("localDateTime");
        LocalDateTime bean2 = (LocalDateTime) context.getBean("localDateTime2");
        LocalDateTime bean3 = (LocalDateTime) context.getBean("localDateTime3");
        // 3. 打印获取到的bean
        System.out.println(bean.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        System.out.println(bean2.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        System.out.println(bean3.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
    }
}

2. DI详解

下面我们就来详细介绍下DI(Dependency Injection),中文翻译就是依赖注入,它是一种设计模式,用来实现IoC编程思想,它的核心思想就是将对象之间的依赖关系进行解耦合。通过依赖注入,可以为类的属性、构造方法、setter方法中提供所依赖的对象和值,是一种可重用、低耦合的思想
关于依赖注入,Spring提供了三种方式:

  1. 属性注入(Field Injection)
  2. 构造方法注入(Constructor Injection)
  3. setter注入(Setter Injection)

2.1 属性注入

Controller层代码如下:

@Controller // 将bean存储到Spring容器中
public class TestController {

    @Autowired
    private TestService testService; // 属性注入

    public void testDI() {
        // 调用属性方法
        testService.testService();
    }
}

此时在启动类中获取bean实例并调用testDI方法

@SpringBootApplication
public class IocDiDemoApplication {

    public static void main(String[] args) {
        // 1. 获取IoC容器
        ApplicationContext context = SpringApplication.run(IocDiDemoApplication.class, args);
        // 2. 获取TestController的bean对象
        TestController bean = (TestController) context.getBean("testController");
        // 3. 调用testDI方法
        bean.testDI();
    }
}

image.png
此时程序正常运行!说明Spring容器通过@Autowired注解已经自动使用 属性注入 的方式给TestService进行赋值了

2.2 构造方法注入

2.2.1 构造注入示例

此时如果我们不使用属性注入的方式,而是使用构造方法进行注入!代码示例如下:

@Controller // 将bean存储到Spring容器中
public class TestController {
    private TestService testService;

    public TestController(TestService testService) {
        this.testService = testService;
    }

    public void testDI() {
        // 调用属性方法
        testService.testService();
    }


    public void testController() {
        System.out.println("test controller");
    }
}

再次执行启动类代码,发现程序正常运行!

2.2.2 多个构造方法

此时我们将代码进行修改,提供多个构造方法:

@Controller // 将bean存储到Spring容器中
public class TestController {

    private TestService testService;

    private TestMapper testMapper;

    public TestController(TestService testService) {
        this.testService = testService;
    }

    public TestController(TestService testService, TestMapper testMapper) {
        this.testService = testService;
    }

    public void testDI() {
        // 调用属性方法
        testService.testService();
    }
}

image.png
此时就会发现程序报错了!

这是因为如果提供了多个构造器,那么Spring容器无法确定使用哪个构造器,因此默认会注入到无参构造器中,如果想要自己指定构造器,可以在构造方法上加上@Autowired注解

解决方式如下:

@Controller // 将bean存储到Spring容器中
public class TestController {
    
    private TestService testService;

    private TestMapper testMapper;

    @Autowired
    public TestController(TestService testService) {
        this.testService = testService;
    }

    public TestController(TestService testService, TestMapper testMapper) {
        this.testService = testService;
    }

    public void testDI() {
        // 调用属性方法
        testService.testService();
    }
}

2.3 setter注入

我们还可以提供setter方法注入类所需要的依赖对象

@Controller // 将bean存储到Spring容器中
public class TestController {

    private TestService testService;

    @Autowired
    public void setTestService(TestService testService) {
        this.testService = testService;
    }

    public void testDI() {
        // 调用属性方法
        testService.testService();
    }
}

启动程序后,正常打印结果,说明setter方法同样可以进行依赖注入

2.4 三种注入方式优缺点分析

  • 属性注入:
    • 优点:比较简单,使用方便
    • 缺点:只能在IoC容器中进行使用;有可能出现NPE(空指针异常);而且不能注入Final修饰的属性
  • 构造注入:
    • 优点:可以注入Final修饰的属性;注入的对象不会被改变较为安全;依赖对象在使用之前一定会被完全初始化;通用性好(是JDK支持的)
    • 缺点:如果需要注入多个对象,代码比较冗长
  • setter注入:
    • 优点:方便在类实例创建完后,重新赋值或者注入
    • 缺点:不能注入Final修饰的属性;对象有可能被外部类访问调用,不安全

2.5 @Autowired存在的问题

下面来演示@Autowired注解存在的问题!
Config/BeanConfig.java

@Configuration
public class BeanConfig {

    @Bean
    public LocalDateTime localDateTime() {
        return LocalDateTime.now();
    }

    @Bean
    LocalDateTime localDateTime2() {
        return LocalDateTime.of(2024, 4, 1, 12, 0, 0);
    }

    @Bean
    LocalDateTime localDateTime3() {
        return LocalDateTime.of(2024, 5, 1, 8, 0, 0);
    }
}

我们将三个相同类型的LocalDateTime对象交由Spring容器进行管理!此时在TestConrtoller中尝试使用@Autowired注解就会发现编译失败!
image.png
这是因为@Autowired注解按照bean的类型进行自动装配,此时Spring容器内部有三个同一类型的bean,因此无法进行注入,解决方式有如下三种:

  1. 使用@Primary注解,告知Spring容器在获取该类型bean时优先选择该对象

image.png

  1. 使用@Autowired + @Qualifier搭配使用,配置value属性为bean的名称

image.png

  1. 使用@Resource注解声明bean的名称

image.png

常见面试题:@Autowired和@Resource注解的区别

  • @Autowired是Spring框架提供的注解,而@Resource是JDK提供的注解
  • @Autowired默认按照类型进行依赖注入,而@Resource是按照名称进行注入,相比较于@Autowired,@Resource提供了更多的注解参数设置

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

GIT开发中的使用

GIT 什么是Git? Git是一个版本控制器:可以记录工程的每一次改动和版本迭代的一个管理系统 注意事项: 所有的版本控制系统,其实只能跟踪文本文件的改动(如TXT文件、网页、所有的程序代码等),…

数据库系统概论(超详解!!!) 第四节 关系数据库标准语言SQL(Ⅱ)

1.数据查询 SELECT [ ALL | DISTINCT] <目标列表达式>[&#xff0c;<目标列表达式>] … FROM <表名或视图名>[&#xff0c; <表名或视图名> ] … [ WHERE <条件表达式> ] [ GROUP BY <列名1> [ HAVING <条件表达式> ] ] [ ORDER BY…

Mac 装 虚拟机 vmware、centos7等

vmware&#xff1a; https://www.vmware.com/products/fusion.html centos7 清华镜像&#xff1a; 暂时没有官方的 m1 arm架构镜像 centos7 链接: https://pan.baidu.com/s/1oZw1cLyl6Uo3lAD2_FqfEw?pwdzjt4 提取码: zjt4 复制这段内容后打开百度网盘手机App&#xff0c;操…

2015年认证杯SPSSPRO杯数学建模C题(第二阶段)荒漠区动植物关系的研究全过程文档及程序

2015年认证杯SPSSPRO杯数学建模 C题 荒漠区动植物关系的研究 原题再现&#xff1a; 环境与发展是当今世界所普遍关注的重大问题, 随着全球与区域经济的迅猛发展, 人类也正以前所未有的规模和强度影响着环境、改变着环境, 使全球的生命支持系统受到了严重创伤, 出现了全球变暖…

生物信息学 GO、KEGG

文章目录 北大基因本体论分子通路KEGGGO注释分子通路鉴定 关于同源 相似性 b站链接&#xff1a;北大课程 概述了当前生物信息学领域中几个重要的概念和工具&#xff0c;介绍基因本体论&#xff08;Gene Ontology, GO&#xff09;、分子通路知识库KEGG&#xff08;Kyoto Encyclo…

Redis进阶

缓存雪崩 缓存穿透 缓存击穿 Redis在项目中常用作缓存来使用&#xff0c;主要用两大作用&#xff1a; 1.提升系统的性能 Redis基于内存&#xff0c;IO效率远高于MySql数据库 2.减少数据库压力 Redis处理很多请求&#xff0c;使用Redis作为缓存可以减少数据库的请求量&…

9.2024

使用冒泡排序给{10 ,1,35,61,89,36,55}排序 代码&#xff1a; public class 第九题 {public static void main(String[] args) {int a[]{10,1,35,61,89,36,55};for (int i0;i<a.length-1;i){for (int j0;j<a.length-1;j){if (a[j]>a[j1]){int temp0;tempa[j];a[j]a[…

数字量化值Digital Number, 辐射亮度Radiance, 反射率Reflectance,发射率Emissive

我们经常听到有人困惑于图像的像素值储存的是什么信息&#xff0c;以及如何获取所需的值。这里我们总结以下几个概念。 数字量化值&#xff08;Digital Number &#xff1a;DN&#xff09; 像素值的通用术语是数字量化值或DN值&#xff0c;它通常被用来描述还没有校准到具有意…

hbase启动错误-local host is“master:XXXX“ destination is:master

博主的安装前提&#xff1a; zookeeper安装完成&#xff0c;且启动成功 hdfs高可用安装&#xff0c;yarn高可用安装&#xff0c;且启动成功 报错原因&#xff1a;端口配置不对 解决方案&#xff1a; 输入&#xff1a;hdfs getconf -confKey fs.default.name 然后把相应的…

Spring Cloud 网关Gateway + 配置中心

网关 网络的接口&#xff0c;负责请求的路由、转发、身份校验 路由&#xff1a;告诉请求去哪找 转发&#xff1a;请求找不到直接带请求过去 路由及转发 判断前端请求的规则就这么配 当前情况下只需要访问8080端口 就可以完成对全部微服务的访问 路由属性 登录校验 没必要在每…

如果有意外,这个窗口就会弹出,希望你们能够看到!——夜读(逆天打工人爬取热门微信文章解读)

第一个日二更 引言Python 代码第一篇 定时任务运行结果 第二篇 人民日报 【夜读】最好的教养&#xff0c;是对家人和颜悦色结尾 时间不会无缘无故增加 也不会无缘无故减少 我们唯一能够控制就是 加大时间的密度 引言 为了不让我在大庭广众下大喊我是沙比 我来更新文章啦 这次带…

网络七层模型之物理层:理解网络通信的架构(一)

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

【漏洞复现】商混ERP系统 DictionaryEdit.aspx接口处存在SQL注入漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…

并发编程之的虚假唤醒和精准唤醒的详细解析

虚假唤醒 例子 wait&#xff08;&#xff09;是object类自带的方法&#xff0c;在jdk有介绍&#xff0c;有可能出现中断、虚假唤醒 也就是在下面的例子中 if(number ! 0){this.wait(); } 当线程成功进入if语句块中&#xff0c;发生了中断&#xff0c;cpu跑去调度别的进程了&am…

厨余垃圾处理设备工业监控PLC连接APP小程序智能软硬件开发之功能结构篇

厨余垃圾处理设备工业监控PLC连接APP小程序智能软硬件开发之功能结构篇 好几年前&#xff0c;应朋友之邀&#xff0c;为其工厂的厨余垃圾处理设备研发一套用于对现场的生产及维护进行远程查看、管理和质量监控的厨余垃圾处理设备工业监控PLC连接APP小程序智能软硬件系统。 因为…

MVC框架里的几种对象

Java语言是一门面向对象的编程语言&#xff0c;所有都用类表达&#xff0c;入口都是一个类&#xff0c;没有独立的main&#xff08;&#xff09;函数&#xff0c;类的实例化就是对象。 简单来讲类包括数据和方法&#xff0c;方法就是操作&#xff0c;是实现业务逻辑的地方&…

获取高德安全码SHA1

高德开发者平台上给的三种方法 获取安全码SHA1&#xff0c;这里我自己使用的是第三种方法。 1、通过Eclipse编译器获取SHA1 使用 adt 22 以上版本&#xff0c;可以在 eclipse 中直接查看。 Windows&#xff1a;依次在 eclipse 中打开 Window -> Preferances -> Androi…

C++动态内存管理:new/delete与malloc/free的对比

在C中&#xff0c;动态内存管理是一个至关重要的概念。它允许我们在程序运行时根据需要动态地分配和释放内存&#xff0c;为对象创建和销毁提供了灵活性。在C中&#xff0c;我们通常会用到两对工具&#xff1a;new/delete 和 malloc/free。虽然它们都能够完成类似的任务&#x…

「10」文本(GDI+):添加文字,可设置背景添加移动效果

「10」文本&#xff08;GDI&#xff09;添加文字&#xff0c;可设置背景添加移动效果 在OBS软件里&#xff0c;通过来源组件「文本&#xff08;GDI&#xff09;」&#xff0c;您可以添加任意您想要呈现的文字&#xff0c;在直播窗口中显示&#xff0c;它可以是提示语、广告词、…

【双指针】Leetcode 盛最多水的容器

题目解析 11. 盛水最多的容器 木桶效应&#xff0c;寻找一个区间使得这个区间的体积最大 算法讲解 1. 暴力枚举 遍历这个容器&#xff0c;将每一个区间的体积求出来&#xff0c;然后找出最大的 class Solution { public:int maxArea(vector<int>& height){int n…