4.Spring IoCDI

文章目录

  • 1.Ioc - 控制反转(解耦)
    • 1.1传统开发
    • 1.2批量生产车轮(修改代码) - 传统方式,繁琐
    • 1.3解耦
      • 1.3.1使用Ioc方法后
      • 1.3.2添加变量颜色 只需要修改Tire即可
    • 1.4Bean的存储
      • 1.4.1@Controller(控制器存储)
      • 1.4.2@Service(服务存储)
        • 1.4.2.1根据context来获取bean
        • 1.4.2.2根据名称来获取bean
        • 1.4.2.3根据名称和类型获取bean
        • 1.4.2.4特殊情况-类名前两位都大写,bean的名称为类名
      • 1.4.3@Repository (仓库存储)
      • 1.4.4@Component (组件存储)
      • 1.4.5@Configuration (配置存储)
      • 1.4.6为什么要这么多类注解? - 五大注解
      • 1.4.7@Controller和其他注解的区别
      • 1.4.8@Bean 使用
        • 1.4.8.1@Bean正常使用举例
        • 1.4.8.2测试@Bean传递参数是如何选择的
      • 1.4.9启动类位置关系影响
    • 1.5 Bean命名
    • 1.6@Bean取
  • 2.DI - 依赖注入
    • 2.1属性注入
      • 2.1.1属性注入流程
      • 2.1.2属性注入问题
    • 2.2构造方法注入
      • 2.2.1只存在一个构造函数时
      • 2.2.2存在多个构造函数时
    • 2.3Setter方法注入
    • 2.4三种注入优缺点分析
    • 2.5@Autowired存在问题
      • 2.5.1.属性名和你需要使用的对象名保持一致
      • 2.5.2.使用@Primary注解标识默认的对象
      • 2.5.3.使用@Qualifier
      • 2.5.4 使用 @Resource 注解
    • 2.6常见面试题 - @Autowird 与 @Resouce的区别
  • 3.Ioc、DI、AOP的关系
    • 2.6常见面试题 - @Autowird 与 @Resouce的区别
  • 3.Ioc、DI、AOP的关系

大家好,我是晓星航。今天为大家带来的是 Ioc和DI 相关的讲解!😀

1.Ioc - 控制反转(解耦)

1.1传统开发

软件设计原则:高内聚,低耦合。

高内聚:一个模块内部的关系

低耦合:各个模块之间的关系

我们的实现思路是这样的:

先设计轮子(Tire),然后根据轮子的大小设计底盘(Bottom),接着根据底盘设计车身(Framework),最后根据车身设计好整个汽车(Car)。这里就出现了一个"依赖"关系: 汽车依赖车身,车身依赖底盘,底盘依赖轮子.

image-20240315202415950

最终程序的实现代码如下:

Main.java

package com.example.demo.ioc;/*** Created with IntelliJ IDEA* Description* User: 晓星航* Date: 2024 -03 -15* Time: 20:16*/
public class Main {public static void main(String[] args) {Car car = new Car(17);car.run();Car car2 = new Car(19);car2.run();}
}

Car.java

package com.example.demo.ioc;/*** Created with IntelliJ IDEA* Description* User: 晓星航* Date: 2024 -03 -15* Time: 20:16*/
public class Car {private Framework framework;public Car(int size) {framework = new Framework(size);System.out.println("car init...");}public void run() {System.out.println("car run");}
}

FrameWork.java

package com.example.demo.ioc;/*** Created with IntelliJ IDEA* Description* User: 晓星航* Date: 2024 -03 -15* Time: 20:17*/
public class Framework {private  Bottom bottom;public Framework(int size) {bottom = new Bottom(size);System.out.println("framework init...");}
}

Bottom.java

package com.example.demo.ioc;/*** Created with IntelliJ IDEA* Description* User: 晓星航* Date: 2024 -03 -15* Time: 20:19*/
public class Bottom {private Tire tire;public Bottom(int size) {tire = new Tire(size);System.out.println("bottom init...");}
}

Tire.java

package com.example.demo.ioc;/*** Created with IntelliJ IDEA* Description* User: 晓星航* Date: 2024 -03 -15* Time: 20:20*/
public class Tire {private int size;public Tire(int size) {this.size =size;System.out.println("tire init...size" + size);}
}

输出结果:

image-20240315203338137

1.2批量生产车轮(修改代码) - 传统方式,繁琐

我们为了批量化生产各种型号的车轮胎,继而修改了Tire类中的size变量为默认值,传参到Tire中,这就导致了Bottom的关联问题。

image-20240315203403081

在将参数int size传到Bottom中之后,Tire不报错了,但是Bottom又出现了关联问题

image-20240315203412088

在将参数int size传到Framework中之后,Bottom不报错了,但是Framework又出现了关联问题

在将参数int size传到Car中之后,Framework不报错了,此时Car中可以正常批量化生产各种型号的车轮的车了

使用@Component和@Autowired来简化代码

image-20240316132512242

@Component: 标注Spring管理的Bean,使用@Component注解在一个类上,表示将此类标记为Spring容器中的一个Bean

@Autowired: 是一个注释,它可以对类成员变量、方法及构造函数进行标注,让 spring 完成 bean 自动装配的工作

image-20240316134424056

我们之前都用@Controller管理,为什么现在用@Component

image-20240316134358595

因为@Controller就是用@Component来实现的。

1.3解耦

1.3.1使用Ioc方法后

image-20240315211617940

传统方法耦合性高,我们为了提高代码编写效率就引入了Ioc可以将代码进行解耦操作,提高代码编写的效率

1.2中我们只为了更改轮胎的尺寸,但是我们改了轮胎Tire之后,导致我们还要修改底盘Bottom,车身Framework,以及汽车Car,那么有没有什么方法可以只修改轮胎Tire一个代码呢?

传统方法:

image-20240315205246577

使用了Ioc之后的方法:

image-20240315205219055

通过上述两个图的对比,我们可以明显发现使用完Ioc之后,无论我们在车上加任意种新的元素,我们代码都不会产生关联报错。

使用Ioc的车代码:

Car.java:

image-20240315205642046

Framwork.java:

image-20240315205607623

Bottom.java:

image-20240315205728762

Tire.java:

image-20240315205831388

Main.java:

image-20240315210615758

运行结果

image-20240315210857033

1.3.2添加变量颜色 只需要修改Tire即可

  1. 在Tire中增加color变量
image-20240315210941681
  1. 在传参时多传一个颜色参数即可
image-20240315211023075

运行结果:

image-20240315211136604

image-20240315211711046

1.4Bean的存储

共有两种注解类型可以实现:

1.注解: @Controller、 @Service、 @Repository、 @Component、 @Configuration

2.方法注解: @Bean.

1.4.1@Controller(控制器存储)

启动类和我们的容器类一定要同级或者启动类高于其他类,这样启动类中的bean才能拿到其他类中的值

image-20240316152357281

启动类:

image-20240316150200308

spring上下文

image-20240316150244800

使用getBean获取spring容器中的对象

image-20240316151156594

可以看到,此时我们使用bean来获取spring容器中的值已经成功!

image-20240316151307210

如果删除@Controller注解,就会报如下错误

image-20240316151352935

1.4.2@Service(服务存储)

1.4.2.1根据context来获取bean

再写一个userService类,我们还是在启动类中使用bean来获取spring容器中的类中的元素

image-20240316152510172

输出结果:

image-20240316152623272

注:启动类在文件夹中的存放位置一定要等于或高于其他类的位置,不然启动类的作用会失效。

image-20240316152614063

1.4.2.2根据名称来获取bean

image-20240316153625344

image-20240316153613749

image-20240316153926600
1.4.2.3根据名称和类型获取bean

image-20240316153625344

image-20240316153839780

image-20240316153900407
1.4.2.4特殊情况-类名前两位都大写,bean的名称为类名

image-20240316154300883

image-20240316154341541

1.4.3@Repository (仓库存储)

代码:

image-20240316184309930

image-20240316184330618

运行结果图:

image-20240316184339258

1.4.4@Component (组件存储)

代码:

image-20240316184654458

image-20240316184709503

运行结果:

image-20240316184723357

1.4.5@Configuration (配置存储)

代码:

image-20240316185032106

image-20240316184948307

运行结果:

image-20240316184938053

1.4.6为什么要这么多类注解? - 五大注解

  • @Controller: 控制层,接收请求,对请求进行处理,并进行响应
  • @Servie: 业务逻辑层,处理具体的业务逻辑
  • @Repository: 数据访问层,也称为持久层。负责数据访问操作
  • @Configuration: 配置层,处理项目中的一些配置信息

image-20240316185450624

程序的应用分层,调用流程如下:

image-20240316185124591 image-20240316190512217

image-20240316191900580

1.4.7@Controller和其他注解的区别

使用@Controller

image-20240316191440577

image-20240316191430758

可正常访问

使用@Service

image-20240316191357259

image-20240316191329434

不能正常返回

证明@Controller可以作为程序的入口实现,而其他注解不行

1.4.8@Bean 使用

1.4.8.1@Bean正常使用举例

image-20240316193319796

image-20240316191906238

UserInfo.java:

image-20240316193851969

BeanConfig.java:

image-20240316193920806

启动类代码:

image-20240316193658479

此时代码没有正常运行,产生报错信息

image-20240316193811563

修改代码,将根据context获取bean变为根据名称来获取bean

image-20240316194029138

此时程序正常输出结果:

image-20240316194106553

总结:

image-20240316194230805

可以使用 名称获取bean 或者 根据名称和类型来获取bean 这两种方式都可以成功获取到bean。

1.4.8.2测试@Bean传递参数是如何选择的

当name和name2都存在时,Bean会取走对应的name值 zhangsan。(与name和name2先后顺序无关)

image-20240316195319989

当name被注释掉,只剩name2存在时,Bean会取走的name2值 wangwu。(bean会只能取走相关的值,而不会不取值,除非没有相关定义的name)

image-20240316195439130

image-20240316194940379

结论:如果需要的Bean的类型,对应的对象只有一个时,就直接赋值,如果有多个时,通过名称去匹配

1.4.9启动类位置关系影响

SpringBoot 特点;约定大于配置

其中之一体现: 就是扫描路径
默认扫描路径是: 启动类所在的目录及其子孙目录

image-20240316200356144

1.5 Bean命名

image-20240318151304230

如何修改BeanName
五大注解

image-20240318151333944 image-20240318151429945 image-20240318151436277

1.6@Bean取

1.属性注入 @AutoWired

2.构造方法注入

如果只有一个构造方法, @AutoWired可以省略

如果有多个构造方法,必须使用@Qualifier指定(参考2.5.3)一个对象,或使用@Primary注解标识默认的对象(参考2.5.2)

3.Setter方法注入

@Autowired出现的问题 - 参考2.5

2.DI - 依赖注入

image-20240316144543403

2.1属性注入

2.1.1属性注入流程

代码:

image-20240316203728679

image-20240316203845855

运行结果图:

image-20240316203759294

image-20240316203705508

2.1.2属性注入问题

启动类:

image-20240316211409557

UserController.java:

image-20240316211603667

BeanConfig.java:

image-20240316211336017

运行结果

image-20240316211459443

当我们使用类UserInfo属性来定义一个新的值时,程序没有发生报错,一切成功运行

我们修改一下代码,将userInfo改为user,再次运行

image-20240316211258866

我们可以看到报错了,报错信息为我们命名为user后,idea找到了一个userInfo和一个userInfo2,此时idea不知道应该使用哪一个导致报错

image-20240316211826039

如果只有一个对象,属性注入以类型进行匹配,与注入的属性名称无关

但是如果一个类型存在多个对象时,优先名称匹配,如果名称
都匹配不上,那就报错~

image-20240316212449211

2.2构造方法注入

2.2.1只存在一个构造函数时

代码:

image-20240316204753150

image-20240316203845855

运行结果图:

image-20240316204815125

2.2.2存在多个构造函数时

我们增加一个值后,相应的增加了构造函数:

image-20240316205048087

此时我们运行报了空指针异常。

image-20240316205123860

通过分析上述代码得知,我们idea默认调用了第一个无参的构造函数,导致us空指针异常

我们将无参的构造函数注解掉,再次运行

image-20240316205318885

报错信息大致为,无参构造函数注解掉后,idea不知道该使用哪个构造函数,导致找不到构造函数,从而报错

image-20240316205510939

再次修改代码,加上@Autowired,表示我们给idea指定了要用哪个构造函数,再次运行:

image-20240316205705302

此时可以发现,我们程序运行又回归正常了

image-20240316205854103

总结:如果存在多个构造函数时,需要加上@AutoWired注明使用哪个构造函数。
如果只有一个构造函数时,@AutoWired可以省略掉

image-20240316203947505

2.3Setter方法注入

当我们直接使用构造函数里面的Setter方法自动生成,此时我们的UserService有没有注入进来呢?

image-20240316210321839

我们运行一下,发现报错了

image-20240316210427649

报错信息为us空指针异常

我们修改一下代码,在Setter方法上面加上@Autowired

image-20240316210530888

再次运行,可以看到我们的元素再次注入成功!

image-20240316210611503

从上代码结果可以看出我们在使用Setter方法注入时,也需要在Setter方法上加上 @Autowired 才能运行成功

2.4三种注入优缺点分析

  • 属性注入

    • 优点:简洁,使用方便。
    • 缺点:
      • 只能用于 IoC 容器,如果是非 oC 容器不可用,并且只有在使用的时候才会出现 NPE(空指针异常)
      • 不能注入一个Final修饰的属性
  • 构造函数注入(Spring 4.X推荐)

    • 优点:
      • 可以注入final修饰的属性
      • 注入的对象不会被修改
      • 依赖对象在使用前一定会被完全初始化,因为依赖是在类的构造方法中执行的,而构造方法是在类加载阶段就会执行的方法
      • 通用性好,构造方法是JDK支持的,所以更换任何框架,他都是适用的
    • 缺点:注入多个对象时,代码会比较繁琐
  • Setter注入(Spring 3.X推荐)

    • 优点:方便在类实例之后,重新对该对象进行配置或者注入
    • 缺点:
      • 不能注入一个Final修饰的属性
      • 注入对象可能会被改变,因为setter方法可能会被多次调用,就有被修改的风险

2.5@Autowired存在问题

当程序中同一个类型有多个对象是,使用@Autowired会报错(一些情况下)

2.5.1.属性名和你需要使用的对象名保持一致

image-20240318142156845 image-20240318142231794 image-20240318142357726

可以看到只要名称保持一致,idea就可以正常找到对应的属性名,从而打印出正确的结果。

2.5.2.使用@Primary注解标识默认的对象

image-20240318141537121

2.5.3.使用@Qualifier

image-20240318141810602 image-20240318141914710

idea运行结果:

image-20240318141938193

可以看到在加上 @Qualifier("userInfo2") 指定属性名后,idea在运行时就能按照加的属性名指定使用,而不会导致idea不知道使用useInfo还是userInfo2。

2.5.4 使用 @Resource 注解

image-20240318142906929 image-20240318142939805 image-20240318143041304

可以看到在我们加上了 @Resource@Resource(name = "userInfo2") 指定属性名为 userInfo2 后,我们idea也可以正常打印出结果

2.6常见面试题 - @Autowird 与 @Resouce的区别

  • @Autowired 是spring框架提供的注解,而@Resource是JDK提供的注解
  • **@Autowired 默认是按照类型注入,而@Resource是按照名称注入.**相比于 @Autowired 来说,@Resource 支持更多的参数设置,例如 name 设置,根据名称获取 Bean。

image-20240318143626518

一个类型存在多个对象:

image-20240318143822736

按照名称来匹配:

image-20240318143848451

3.Ioc、DI、AOP的关系

di依赖注入 AOP面向切片 IOC控制反转
spring是一个轻量级控制反转Ioc和面向切片Aop的容器
控制反转IOC是一种设计思想,DI依赖注入是实现IOC的一种方法

性名为 userInfo2 后,我们idea也可以正常打印出结果

2.6常见面试题 - @Autowird 与 @Resouce的区别

  • @Autowired 是spring框架提供的注解,而@Resource是JDK提供的注解
  • **@Autowired 默认是按照类型注入,而@Resource是按照名称注入.**相比于 @Autowired 来说,@Resource 支持更多的参数设置,例如 name 设置,根据名称获取 Bean。

[外链图片转存中…(img-M4s0g0ws-1712818818828)]

一个类型存在多个对象:

image-20240318143822736

按照名称来匹配:

image-20240318143848451

3.Ioc、DI、AOP的关系

di依赖注入 AOP面向切片 IOC控制反转
spring是一个轻量级控制反转Ioc和面向切片Aop的容器
控制反转IOC是一种设计思想,DI依赖注入是实现IOC的一种方法

感谢各位读者的阅读,本文章有任何错误都可以在评论区发表你们的意见,我会对文章进行改正的。如果本文章对你有帮助请动一动你们敏捷的小手点一点赞,你的每一次鼓励都是作者创作的动力哦!😘

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SSM整合----第一个SSM项目

文章目录 前言一、使用步骤1.引入库2.建表3 项目结构4 web.xml的配置5 配置数据源6 SpringMVC配置7 配置MyBatis Mapper8 书写控制类 总结 前言 提示:这里可以添加本文要记录的大概内容: SSM整合是指Spring、SpringMVC和MyBatis这三个框架的整合使用。…

【汇编语言实战】对输入的数组实现快速排序

C语言描述&#xff1a; #include <stdio.h>// 交换数组中两个元素的位置 void swap(int *a, int *b) {int temp *a;*a *b;*b temp; }// 分区函数&#xff0c;将数组按照基准值划分为两部分 int partition(int arr[], int low, int high) {int pivot arr[high]; // 选…

JavaWeb中的Servlet是什么?怎么使用?

文章目录 一、什么是Servlet二、Servlet的基本内容1、Servlet的作用2、Servlet接口3、Servlet接口实现类4、Servlet接口实现类开发步骤5、Servlet对象生命周期6、HttpServletResquest接口7、HttpServletResponse接口8、请求对象和响应对象流程图9、请求对象和响应对象生命周期1…

阿里面试总结 一

写了这些还是不够完整&#xff0c;阿里 字节 卷进去加班&#xff01;奥利给 ThreadLocal 线程变量存放在当前线程变量中&#xff0c;线程上下文中&#xff0c;set将变量添加到threadLocals变量中 Thread类中定义了两个ThreadLocalMap类型变量threadLocals、inheritableThrea…

迷宫 — — 蓝桥杯(动态规划)

迷宫 题目&#xff1a; 输入样例&#xff1a; 3 1 1 1 2 3 4 5 6 7 8 9 2 2 1 3 1 R输出样例&#xff1a; 21思路&#xff1a; 题目大意&#xff1a;给定一个n x m的平面网格&#xff0c;并且每一个格子都有一定的代价&#xff0c;并且设有障碍物和陷阱&#xff0c;障碍物的意…

c++的友元函数,详细笔记,细说三种友元用法

解释友元 友元用通俗易懂的话来说&#xff0c;就是&#xff1a;当有人来到你家里&#xff0c;他就只能呆在客厅里面&#xff0c;你是不可能让他来到你的卧室之中的。但是如果这个人是你的朋友&#xff0c;那么你是默许他可以进入你的卧室的。 此时呢&#xff1f;我告诉你&…

网络IO模型以及实际应用

网络IO模型 本文主要介绍了几种不同的网络IO模型&#xff0c;以及实际应用中使用到的Reactor模型等。 我们常说的网络IO模型&#xff0c;主要包含阻塞IO、非阻塞IO、多路复用IO、信号驱动IO、异步IO。 根据第一个阶段&#xff1a;是否需要阻塞&#xff0c;分为阻塞和非阻塞IO。…

【从浅学到熟知Linux】进程状态与进程优先级(含进程R/S/T/t/D/X/Z状态介绍、僵尸进程、孤儿进程、使用top及renice调整进程优先级)

&#x1f3e0;关于专栏&#xff1a;Linux的浅学到熟知专栏用于记录Linux系统编程、网络编程及数据库等内容。 &#x1f3af;每天努力一点点&#xff0c;技术变化看得见 文章目录 进程状态进程状态查看R运行状态&#xff08;running&#xff09;S睡眠状态&#xff08;sleeping&a…

GT收发器64B66B协议(2)自定义PHY设计

文章目录 前言一、设计框图二、GT_module三、PHY_module3.1、PHY_tx模块3.2、PHY_rx_bitsync模块3.3、PHY_rx模块 四、上板测试总结 前言 有了对64B66B协议的认识以及我们之前设计8B10B自定义PHY的经验&#xff0c;本文开始对64B66B自定义PHY的设计 一、设计框图 二、GT_modu…

Harmony鸿蒙南向驱动开发-UART

UART指异步收发传输器&#xff08;Universal Asynchronous Receiver/Transmitter&#xff09;&#xff0c;是通用串行数据总线&#xff0c;用于异步通信。该总线双向通信&#xff0c;可以实现全双工传输。 两个UART设备的连接示意图如下&#xff0c;UART与其他模块一般用2线&a…

数据结构初阶:栈和队列

栈 栈的概念及结构 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。 进行数据插入和删除操作的一端 称为栈顶&#xff0c;另一端称为栈底。 栈中的数据元素遵守后进先出 LIFO &#xff08; Last In First Out &#xff09;的原则。…

为什么每个人都需要了解这些数据加密技术?

在数字时代&#xff0c;数据加密技术不仅对保护企业的商业秘密至关重要&#xff0c;也是个人隐私安全的重要屏障。随着技术的进步和网络犯罪的增加&#xff0c;数据加密已经成为了信息安全领域的一个热点议题。以下是探讨为什么每个人都需要了解这些数据加密技术的几个主要原因…

SpringBoot启动时banner设置

SpringBoot启动时banner设置 1.操作步骤2.各种banner图像 1.操作步骤 在application.properties文件中设置新的banner对于的文件位置&#xff0c;最好放在resources目录下 spring.banner.locationbanner.txt2.各种banner图像 &#xff08;1&#xff09;经典大佛图 具体txt文…

[通俗易懂]《动手学强化学习》学习笔记2-第2、3、4章

文章目录 前言小总结&#xff08;前文回顾&#xff09;第二章 多臂老虎机2.2.2形式化描述 第三章 马尔可夫决策过程3.6 占用度量 代码3.6 占用度量 定理2 第四章 动态规划算法4.3.3 策略迭代算法 代码 总结 前言 参考&#xff1a; 《动手学强化学习》作者&#xff1a;张伟楠&a…

为什么要部署IP SSL证书?怎么申请?

我们需要知道什么是IP SSL证书。SSL&#xff0c;全称为Secure Sockets Layer&#xff0c;即安全套接层&#xff0c;是为网络通信提供安全及数据完整性的一种安全协议。而IP SSL证书就是基于SSL协议的一种证书&#xff0c;它能够为网站和用户的数据传输提供加密处理&#xff0c;…

Prometheus-Grafana基础篇安装绘图

首先Prometheus安装 1、下载 https://prometheus.io/download/ 官网路径可以去这儿下载 2、如图&#xff1a; 3.解压&#xff1a; tar -xf prometheus-2.6.1.linux-amd64 cd prometheus-2.6.1.linux-amd64 4.配置文件说明&#xff1a; vim prometheus.yml 5.启动Promethe…

OpenHarmony NAPI 框架生成工具实现流程

NAPI 框架生成工具 可以根据用户指定路径下的 ts(typescript)接口文件一键生成 NAPI 框架代码、业务代码框架、GN 文件等。在开发 JS 应用与 NAPI 间接口时&#xff0c;底层框架开发者无需关注 Nodejs 语法、C 与 JS 之间的数据类型转换等上层应用转换逻辑&#xff0c;只关注底…

论文| Convolutional Neural Network-based Place Recognition - 2014

2014-Convolutional Neural Network-based Place Recognition

Qt plugin 开发UI界面插件

目录 1.创建接口 2.创建插件 3.创建插件界面 4.插件实现 5.创建应用工程 6.应用插件 1.创建接口 打开QtCreater&#xff0c;点击左上角“文件”->新建文件或项目&#xff0c;在弹窗中选择C/CHeader File。 输入文件名&#xff0c;选好路径&#xff08;可自行设置名称…

Golang | Leetcode Golang题解之第16题最接近的三数之和

题目&#xff1a; 题解&#xff1a; func threeSumClosest(nums []int, target int) int {sort.Ints(nums)var (n len(nums)best math.MaxInt32)// 根据差值的绝对值来更新答案update : func(cur int) {if abs(cur - target) < abs(best - target) {best cur}}// 枚举 a…