lukaliou123 / lukaliou123.github.io

lukaliou123在2022年的面试用知识点总结
Other
5 stars 0 forks source link

Spring基础知识 #21

Open lukaliou123 opened 2 years ago

lukaliou123 commented 2 years ago

1.什么是Spring 框架

Spring是一种轻量级开发框架,旨在提高开发人员的开发效率以及系统的可维护性

我们一般说Spring框架指的都是spring framwrofk,它是很多模块的集合,使用这些模块可以很方便地协助我们进行开发。这些模块是:核心容器Core Container,数据访问/集成Data Access/Integration,Spring Web,AOP(面向切面编程),消息和测试模块Spring Test and message。Beans组件和Context组件是实现IOC和依赖注入的基础,AOP组件用来实现面向切面编程

Spring 6 个特征: 核心技术:依赖注入(DI),AOP,事件,资源 测试 数据访问 Web支持:Spring MVC 和 Spring WebFlux Web框架 集成 语言

2.重要的Spring模块

image

Core:基础,可以说Spring其它所有的功能都需要依赖于该库类。主要提供IoC依赖注入功能。 Aspects:该模块为与AspectJ的集成提供支持 AOP:提供了面向切面的编程实现 JDBC:Java数据库连接 JMS:Java消息服务 ORM:用于支持Hibernate等ORM工具 Web:为创建Web应用程序提供支持

lukaliou123 commented 2 years ago

3.@RestController vs @Controller

Controller 返回一个页面 单独使用@Controller 不加 @ResponseBody 一般是用在要返回一个视图的情况,这种情况属于传统的Spring MVC的应用,对应于前后端不分离的情况 image

@RestController 返回JSON或XML形式数据 @RestController只返回对象,对象数据直接以JSON或XML形式写入HTTP相应(Response)中,这种情况属于Restful Web服务,这也是目前日常开发所接触的最常用的情况(前后端分离) image

@Controller +@ResponseBody 返回JSON 或 XML 形式数据 如果你需要在Spring4之前开发 RESTful Web服务的话,你需要使用@Controller 并结合@ResponseBody注解,也就是说@Controller +@ResponseBody= @RestController(Spring 4 之后新加的注解) @ResponseBody 注解的作用是将 Controller 的方法返回的对象通过适当的转换器转换为指定的格式之后,写入到HTTP 响应(Response)对象的 body 中,通常用来返回 JSON 或者 XML 数据,返回 JSON 数据的情况比较多。 image

4.Spring IOC & AOP

IOC

IoC(Inverse of Control:控制反转)是一种设计思想,就是将原本在程序中手动创建对象的控制权,交由Spring框架来管理。 IoC在其他语言中也有应用,并非Spring特有。IoC容器是Spring用来实现IoC的载体,IoC容器实际就是个Map(key,value),Map中存放的是各种对象。

将对象之间的相互依赖关系交给IoC容器来管理,并由IoC容器完成对象的注入。这样可以很大程度上简化应用的的开发,把应用从复杂的依赖关系中解放出来。IoC容器就像是一个工厂一样,当我们需要创建一个对象的时候,只需要配置好文件/注解即可,完全不用考虑对象是如何被创建出来的。在实际项目中,一个Service类可能有几百甚至上千个类作为它的底层函数,加入我们需要实例化这个Service,你可能要每次都搞清这个Service所有底层类的构造函数,这可能会把人逼疯。如果利用IoC的话,只需要配置好,然后再需要的地方引用就行了,这大大增加了项目的可维护性且降低了开发难度。

Spring 时代我们一般通过 XML 文件来配置 Bean,后来开发人员觉得 XML 文件来配置不太好,于是 SpringBoot 注解配置就慢慢开始流行起来。 image

AOP

AOP(Aspect-Oriented Programming:面向切面编程)能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任(例如事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可拓展性和可维护性

Spring AOP就是基于动态代理的,如果要代理的对象,实现了某个接口,那么Spring AOP会使用JDK Proxy,去创建代理对象,而对于没有实现接口的对象,就无法使用 JDK Proxy 去进行代理了,这时候Spring AOP会使用Cglib ,这时候Spring AOP会使用 Cglib 生成一个被代理对象的子类来作为代理,如下图所示: image 使用 AOP 之后我们可以把一些通用功能抽象出来,在需要用到的地方直接使用即可,这样大大简化了代码量。我们需要增加新功能时也方便,这样也提高了系统扩展性。日志功能、事务管理等等场景都用到了 AOP 。

lukaliou123 commented 2 years ago

5.Spring Bean

Spring 中的 bean 的作用域有哪些? singleton : 唯一 bean 实例,Spring 中的 bean 默认都是单例的。 prototype : 每次请求都会创建一个新的 bean 实例。 request : 每一次HTTP请求都会产生一个新的bean,该bean仅在当前HTTP request内有效。 session : 每一次HTTP请求都会产生一个新的 bean,该bean仅在当前 HTTP session 内有效。 global-session: 全局session作用域,仅仅在基于portlet的web应用中才有意义,Spring5已经没有了。Portlet是能够生成语义代码(例如:HTML)片段的小型Java Web插件。它们基于portlet容器,可以像servlet一样处理HTTP请求。但是,与 servlet 不同,每个 portlet 都有不同的会话

6.Spring 中的单例 bean 的线程安全问题了解吗?

大部分时候我们并没有在系统中使用多线程,所以很少有人会关注这个问题。单例 bean 存在线程问题,主要是因为当多个线程操作同一个对象的时候,对这个对象的非静态成员变量的写操作会存在线程安全问题

常见的有两种解决办法: 在Bean对象中尽量避免定义可变的成员变量(不太现实)。 在类中定义一个ThreadLocal成员变量,将需要的可变成员变量保存在 ThreadLocal 中(推荐的一种方式)。

lukaliou123 commented 2 years ago

7.@Component 和 @Bean 的区别是什么?

1.作用对象不同: @Component 注解作用于,而@Bean注解作用于方法。 2.@Component通常是通过类路径扫描来自动侦测以及自动装配到Spring容器中(我们可以使用 @ComponentScan 注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 容器中)。@Bean 注解通常是我们在标有该注解的方法中定义产生这个 bean,@Bean告诉了Spring这是某个类的示例,当我需要用它的时候还给我。 3.@Bean 注解比 Component 注解的自定义性更强,而且很多地方我们只能通过 @Bean 注解来注册bean。比如当我们引用第三方库中的类需要装配到 Spring容器时,则只能通过 @Bean来实现。 @Bean注解使用示例: 1650598889(1) 上面的代码相当于下面的 xml 配置 1650598914(1)

*下面这个例子是通过 @Component 无法实现的。 1650598939(1)

8.将一个类声明为Spring的 bean 的注解有哪些?

我们一般使用 @Autowired 注解自动装配 bean,要想把类标识成可用于 @Autowired 注解自动装配的 bean 的类,采用以下注解可实现:

@Component :通用的注解,可标注任意类为 Spring 组件。如果一个Bean不知道属于哪个层,可以使用@Component 注解标注@Repository : 对应持久层即 Dao 层,主要用于数据库相关操作。 @Service : 对应服务层,主要涉及一些复杂的逻辑,需要用到 Dao层。 @Controller : 对应 Spring MVC 控制层,主要用户接受用户请求并调用 Service 层返回数据给前端页面。

lukaliou123 commented 2 years ago

9.Spring 中的 bean 生命周期?

人的一生和bean的一生对比

image

Bean详细生命周期

image

例子:PersonBean的一生

image

用文字描述一下这个过程

1.Bean容器在配置文件中找到Person Bean的定义,这个可以说是妈妈怀上了。 2.Bean容器使用Java 反射API创建Bean的实例,孩子出生了。 3.Person声明了属性no、name,它们会被设置,相当于注册身份证号和姓名。如果属性本身是Bean,则将对其进行解析和设置。 4.Person类实现了BeanNameAware接口,通过传递Bean的名称来调用setBeanName()方法,相当于起个学名。 5.Person类实现了BeanFactoryAware接口,通过传递BeanFactory对象的实例来调用setBeanFactory()方法,就像是选了一个学校。 6..PersonBean实现了BeanPostProcessor接口,在初始化之前调用用postProcessBeforeInitialization()方法,相当于入学报名。 7.PersonBean类实现了InitializingBean接口,在设置了配置文件中定义的所有Bean属性后,调用afterPropertiesSet()方法,就像是入学登记。 8.配置文件中的Bean定义包含init-method属性,该属性的值将解析为Person类中的方法名称,初始化的时候会调用这个方法,成长不是走个流程,还需要自己不断努力。 9.Bean Factory对象如果附加了Bean 后置处理器,就会调用postProcessAfterInitialization()方法,毕业了,总得拿个证。 10.Person类实现了DisposableBean接口,则当Application不再需要Bean引用时,将调用destroy()方法,简单说,就是人挂了。 11 .配置文件中的Person Bean定义包含destroy-method属性,所以会调用Person类中的相应方法定义,相当于选好地儿,埋了。 1691420943975

BeanPostProcessor(可以在初始化之前和之后给bean再添加一些方法或者属性)

1691420977714 1691421165505

1691421001046

lukaliou123 commented 2 years ago

10. Spring MVC

谈到这个问题,我们不得不提提之前 Model1Model2 这两个没有 Spring MVC 的时代。 Model1 时代 : 很多学 Java 后端比较晚的朋友可能并没有接触过 Model1 模式下的 JavaWeb 应用开发。在 Model1 模式下,整个 Web 应用几乎全部用 JSP 页面组成,只用少量的 JavaBean 来处理数据库连接、访问等操作。这个模式下 JSP 即是控制层又是表现层。显而易见,这种模式存在很多问题。比如①将控制逻辑和表现逻辑混杂在一起,导致代码重用率极低;②前端和后端相互依赖,难以进行测试并且开发效率极低;

Model2 时代 :学过 Servlet 并做过相关 Demo 的朋友应该了解“Java Bean(Model)+ JSP(View,)+Servlet(Controller) ”这种开发模式,这就是早期的 JavaWeb MVC 开发模式。Model:系统涉及的数据,也就是 dao 和 bean。View:展示模型中的数据,只是用来展示。Controller:处理用户请求都发送给 ,返回数据给 JSP 并展示给用户。

MVC 是一种设计模式,Spring MVC 是一款很优秀的 MVC 框架。Spring MVC 可以帮助我们进行更简洁的Web层的开发,并且它天生与 Spring 框架集成。Spring MVC 下我们一般把后端项目分为 Service层(处理业务)、Dao层(数据库操作)、Entity层(实体类)、Controller层(控制层,返回数据给前台页面)

Spring MVC 的简单原理图如下image

11.Spring MVC的工作原理

image 上图的一个笔误的小问题:Spring MVC 的入口函数也就是前端控制器 DispatcherServlet 的作用是接收请求,响应结果

流程说明(重要): 1.客户端(浏览器)发送请求,直接请求到 DispatcherServlet。 2.DispatcherServlet 根据请求信息调用 HandlerMapping,解析请求对应的 Handler。 3.解析到对应的 Handler(也就是我们平常说的 Controller 控制器)后,开始由 HandlerAdapter 适配器处理。 4.HandlerAdapter 会根据 Handler 来调用真正的处理器开处理请求,并处理相应的业务逻辑。 5.处理器处理完业务后,会返回一个 ModelAndView 对象,Model 是返回的数据对象,View 是个逻辑上的 View。 6.ViewResolver 会根据逻辑 View 查找实际的 View。 7.DispaterServlet 把返回的 Model 传给 View(视图渲染)。 8.把 View 返回给请求者(浏览器)

lukaliou123 commented 2 years ago

12.Spring 框架中用到了哪些设计模式?

工厂设计模式 : Spring使用工厂模式通过 BeanFactory、ApplicationContext 创建 bean 对象。 代理设计模式 : Spring AOP 功能的实现。 单例设计模式 : Spring 中的 Bean 默认都是单例的。 包装器设计模式 : 我们的项目需要连接多个数据库,而且不同的客户在每次访问中根据需要会去访问不同的数据库。这种模式让我们可以根据客户的需求能够动态切换不同的数据源。 观察者模式: Spring 事件驱动模型就是观察者模式很经典的一个应用。 适配器模式 :Spring AOP 的增强或通知(Advice)使用到了适配器模式、spring MVC 中也是用到了适配器模式适配Controller。

lukaliou123 commented 1 year ago

补充:IOC

1.什么是IOC?

IOC,全名为控制反转(Inversion of Control),是面向对象编程中的一种设计原则,用于降低计算机程序的耦合度。它是一种编程思想,其中对象之间的控制关系是被反转的,由原先在程序中手动创建对象,改为由程序自动创建

在没有应用IOC的传统程序设计中,当一个对象需要和另一个对象交互时,通常会自行创建或者查找所需的对象然后进行交互。但是这种方式使得对象之间紧密耦合,影响程序的灵活性和可扩展性。

当应用了IOC后,对象的创建和查找会交给一个叫做IOC容器的东西来完成,对象只需要关心自己的核心逻辑,其他的都由IOC容器来负责,比如对象之间的依赖关系、对象的生命周期等。这样做的结果就是,对象与对象之间实现了松耦合,每个对象可以独立地完成自己的任务,只要知道与其交互的对象的接口,而不需要知道具体实现

小疑问:IOC容器是什么?长什么样?感觉有点抽象。。。

Spring框架就是一个非常著名的应用了IOC原则的框架。在Spring中,IOC容器负责实例化、配置和装配对象。开发者只需要配置声明依赖对象,无需手动创建和管理对象。

1.2.IOC的例子

1.一个没有使用IOC的例子。 假设我们有一个汽车类(Car)和一个司机类(Driver)。在不使用IOC的情况下,Driver类可能是这样的 1689656965487 在上面的例子中,司机类(Driver)直接依赖于汽车类(Car)。如果汽车类发生变化,例如需要添加新的构造函数参数,那么司机类也需要改变。同样,如果我们想在测试环境下使用一个特殊的汽车类(例如MockCar),我们就需要改变司机类的代码。

2.使用了IOC的例子

1689657208145 在上面的例子中,汽车类的实例是从外部传入的。司机类不需要关心汽车类是如何创建的,也不需要关心汽车类的具体实现是什么。这就是控制反转——创建汽车类的控制权从司机类反转到了外部

2.1.使用了spring注解的例子

假设我们有一个汽车类Car和一个司机类Driver,我们希望在Driver中使用Car的实例,但是我们不希望Driver类自己去创建Car的实例,因为这样会导致Driver类和Car类的耦合度过高。

那么,在Spring中,我们可以这样做: image 在Spring中,使用@Autowired注解就是实现IOC的一种方式,我们称之为依赖注入(DI)

在上面的例子中,我们使用了@Component注解告诉Spring这两个类都是需要Spring管理的Bean。然后,在Driver类中,我们使用@Autowired注解告诉Spring我们希望它能自动注入Car的实例

这样,Driver类就不需要自己去创建Car的实例,而是由Spring来负责创建和注入。这就是控制反转(IOC)的概念:Driver类不再控制Car实例的创建和生命周期,这个控制权已经反转给了Spring

2.2.进一步的DI例子

问:DI是IOC的一个实现方式,这个的好处是司机需要调用车这个函数,不需要在司机这个内部自己新建,也不需要在构建函数时创建进去。这个车可以是一个接口,我如果想换车,只需要换一个车载司机进行注入,是吗?这样有什么好处,能举例跟我讲讲吗 答: 车(Car)可以是一个接口,然后我们可以有不同的实现类,例如电动车(ElectricCar)和汽油车(GasCar)。这样的设计确实让我们可以轻易地在不同的车型之间进行切换。

例如,我们可以这样设计: image

在这个例子中,我们使用了@Qualifier注解来指定要注入的Bean的名字。当我们想要改变司机开的车的类型时,我们只需要改变@Qualifier的参数就可以了。

这样的设计可以带来许多好处

1.可插拔性:如果我们希望使用一种新类型的车,我们只需要添加一个新的实现类,然后改变@Qualifier的参数就可以了,而不需要修改Driver类的代码。

2.可测试性:当我们在进行单元测试时,我们可以注入一个模拟的Car实例,这样可以方便我们进行测试。

3.可维护性和可扩展性:通过降低类之间的耦合度,使我们的代码更加易于维护和扩展。

3.DI和IOC的关系

DI和IOC的关系,DI(Dependency Injection,依赖注入)是IOC(Inversion of Control,控制反转)的一种实现方式

IOC的基本理念是将对象的创建和维护工作交给容器,而不是由对象自己来负责。这样一来,我们可以将对象之间的依赖关系从硬编码转移到容器中进行管理,从而实现解耦。

DI是实现IOC理念的一种手段。通过DI,我们可以将依赖对象(例如上述例子中的Car)从外部注入到需要它的对象(例如Driver)中,而不是由这个对象自己去创建依赖对象。这样可以使我们的代码更加模块化和可测试。

总的来说,IOC和DI都是为了提高代码的可测试性、可维护性和可重用性。IOC是一种设计理念,DI是这种理念的一种实现方式。

4.在面试中如何回答什么是IOC,是否写过?

当面试官问你是否写过IOC(Inversion of Control,控制反转)时,他们实际上是想知道你是否使用过像Spring这样的框架,因为Spring框架的核心就是IOC

你可以这样回答:

"是的,我使用过Spring框架,这个框架的核心概念就是控制反转。通过控制反转,我可以将对象的创建、配置和管理的工作交给Spring容器,而不需要自己在代码中显式地进行这些工作。这样可以减少代码之间的耦合,提高代码的可重用性和可维护性

我使用Spring的注解,比如@Autowired和@Qualifier,来实现依赖注入,这是一种实现控制反转的方法。通过依赖注入,我可以将需要的依赖从外部注入到我的类中,而不是在类内部自己创建。这样可以使我的类更加松耦合,更容易进行单元测试。

例如,在一个我曾经的项目中,我有一个'司机(Driver)'类需要一个'汽车(Car)'对象。我没有在'司机(Driver)'类中直接创建'汽车(Car)'对象,而是通过@Autowired注解将'汽车(Car)'注入到'司机(Driver)'类中。当我需要改变'司机(Driver)'使用的'汽车(Car)'类型时,我只需要改变注入的'汽车(Car)'的定义,而不需要修改'司机(Driver)'类的代码。这就是控制反转和依赖注入带来的好处。

此外,我还了解到控制反转不仅仅可以用在依赖注入上,也可以用在其他方面,比如事件处理,模板方法的实现等。总的来说,控制反转是我在写代码时会考虑的一个重要原则,它帮助我写出更好、更易维护的代码。"

lukaliou123 commented 1 year ago

补充:AOP

1.什么是AOP?

AOP是一种编程范式,其目标是提高模块化编程,特别是业务处理中的横切关注点。切面(Aspect)表示的是横跨多个对象的关注点,切面可以包含通知(Advice)和切入点(Pointcut)。通知定义了在切入点选定的位置应用的代码,而切入点则确定在何处应用通知

使用AOP的好处是能将各个部分解耦,使核心业务逻辑从周边的系统级服务中分离出来。例如,日志记录、事务管理、安全检查等都可以通过AOP进行模块化。

在Spring中,AOP是通过动态代理实现的**。Spring AOP默认使用JDK的动态代理来实现AOP,如果被代理的类实现了至少一个接口,则使用JDK动态代理**。如果被代理的类没有实现任何接口,那么Spring会使用CGLIB来创建动态代理。

所以,你可以说AOP和动态代理有很大的关系,动态代理是AOP实现的基础之一。

2.Springboot自动实现AOP的例子

SpringBoot支持通过注解来简化AOP的实现。我会给你举一个日志记录的例子,它是AOP常见的一个应用场景。

1.定义一个切面 1689660250015

2.定义一个注解: 1689660293366

3.方法上使用@Loggable注解 1689660554655 注解和切面的联系主要体现在这一行代码:@Around("@annotation(Loggable)")。在这里,@Around是一个AspectJ的注解,它表明这个方法将被用作一个环绕通知,即在目标方法执行前后都会执行一些操作@annotation(Loggable)则表明这个环绕通知应用到所有被@Loggable注解的方法上。所以当你在方法上使用@Loggable注解,那么这个方法就成了AOP的目标(Join Point),当这个方法被调用时,logExecutionTime这个环绕通知就会被触发。

3.更具体的例子

1689662046375 每次在执行被 **@Loggable 注解标记的方法时,都会先打印 "Method [foo] start",然后打印 "Method [foo] executed in

如果我们有一个被 @Loggable 注解标记的方法 A,这个方法本身打印出 "A",那么当我们使用这个环绕增强时,最终的输出会是: 1689662119989 joinPoint.proceed() 是 AspectJ 中的一个关键方法,它负责触发切入点方法的执行。

在环绕通知(Around advice)中,joinPoint.proceed() 方法的位置就决定了切入点方法在何时执行。如果它在通知方法的开始处调用,那么在通知方法中的其他代码就可以在切入点方法之后执行。反之,如果在通知方法的末尾处调用,那么其他代码就可以在切入点方法之前执行。

在我们的例子中,joinPoint.proceed() 在计算方法执行时间的前后都被调用了,所以我们能够测量并输出方法的执行时间

4.常见的AOP5种通知

1.前置通知(Before advice): 在目标方法被调用之前调用通知功能。

2.后置通知(After returning advice): 在目标方法完成之后调用通知,此时可以获得目标方法的返回值。

3.环绕通知(Around advice): 包围目标方法的通知,可以在调用前后自定义行为。环绕通知方法需要提供一个ProceedingJoinPoint类型的参数,该参数可以决定是否执行目标方法。

4.异常通知(After throwing advice): 当目标方法抛出异常时调用通知功能,可以拦截到异常信息。 1690557343217

5.最终通知(After (finally) advice): 无论目标方法是否抛出异常,该通知都会执行。

lukaliou123 commented 1 year ago

补充:常见的一些spring注解

@SpringBootApplication:这是Spring Boot的核心注解,用于开启自动配置。

@Autowired 重点!:这是一个实现依赖注入的注解,Spring会自动将合适的bean注入到被该注解标记的字段或者方法。

@Component,@Service,@Repository,@Controller 重点!:这些注解用于标记Spring应该管理的类,可以让Spring知道应该为这些类创建bean。

@RestController:这是一个特殊的@Controller注解,用于创建RESTful web服务。

@RequestMapping和它的派生注解,比如@GetMapping,@PostMapping:这些注解用于映射web请求到对应的处理方法。

@Configuration 重点:用于标记配置类,表示这个类包含了应用的一些额外配置。

@Bean:用在@Configuration类中的方法上,表示这个方法会返回一个对象,这个对象应该被Spring注册为一个bean。

@Profile:表示某个组件只在某个或某些特定的Profile中才会创建。

@Value:用于注入外部的值到字段中,比如配置文件的值。

@PathVariable和@RequestParam:用于从请求中获取参数。 @qualifier 喜欢!:表示使用哪个component

lukaliou123 commented 1 year ago

补充:如何自己写一个注解?

首先,创建一个新的注解是相当简单的。注解通常用 @interface 关键字来定义。在下面的例子中,我将创建一个名为 MyAnnotation 的注解: image

在这个例子中,MyAnnotation 是一个非常简单的注解,它只有一个属性 value,默认值为一个空字符串。

然后,我们就可以在代码中使用这个注解了 image

在上面的代码中,我在 TestClass 上使用了 MyAnnotation 并设置了 value 为 "This is a test"。

这只是最基本的使用,实际上,Java注解可以更复杂。你可以定义带有多个属性的注解,甚至可以定义可以注解其他注解的注解(也就是所谓的元注解)。

例如,我们经常会在Spring或J2EE这样的框架中见到的 @Override、@Entity、@Autowired、@Path 等就是注解

lukaliou123 commented 1 year ago

补充:@Resource和@Autowire区别

@Autowired和@Resource都是用于依赖注入的,但它们之间有一些不同:

1.出处:@Autowired是由Spring提供的,而@Resource是由Java提供的,来自于javax.annotation.Resource。

2.注入方式:@Autowired是按类型自动注入,而@Resource默认是按名称自动注入。

3.如果找不到匹配的bean进行注入:@Autowired会抛出异常,而@Resource则不会。

G}{E3J`K_9(XQW432A%E)P 在上述代码中,因为@Autowired按类型注入,当遇到有多个同类型的bean时,Spring不知道应该注入哪一个,所以会抛出异常。

(autowire+qualifier)成功 DZY8PM9E~VDL82107T8)(BK O%S8X6Q`~FIYWJ_33K2P}N6

lukaliou123 commented 1 year ago

补充3:怎么从Spring容器中拿到需要的类

Spring容器中获取需要的类,我们通常通过Spring提供的ApplicationContext接口来获取。ApplicationContext接口提供了许多方法,我们可以通过它获取容器中的Bean,例如getBean()方法。 1689692982433 在这个例子中,beans.xml是我们的Spring配置文件,里面定义了所有的bean。然后我们通过创建ClassPathXmlApplicationContext实例并传入配置文件路径,来加载Spring的配置文件并启动Spring容器。然后,我们就可以通过调用getBean()方法,传入bean的名字,来获取容器中的bean。

那springboot中呢?

Spring Boot的一个主要目标就是简化Spring应用的配置和启动过程。在Spring Boot中,我们通常不需要手动创建XML配置文件来定义bean。相反,Spring Boot提倡"约定大于配置",尽可能地通过智能的默认配置、自动配置和注解来替代XML配置。

在Spring Boot中,我们主要通过注解来声明bean。例如,我们可以使用@Component、@Service、@Repository和@Controller等注解来声明bean,然后Spring Boot会自动扫描这些注解,把相应的类加入到Spring容器中。 1S_CX N0HPZ377UL ~HBTRL

至于获取bean,我们仍然可以通过ApplicationContext来获取,但更常见的是通过依赖注入(DI)。我们可以在其他类中通过@Autowired或@Resource等注解,自动注入需要的bean。 }PPKQ T6 NJ 8NENKC8{)`5

lukaliou123 commented 1 year ago

补充4:Spring如何解决跨域问题

在SpringBoot中,解决跨域的常见方法是使用 @CrossOrigin 注解。这个注解可以用在控制器或者具体的处理器方法上,来控制哪些源可以访问这些接口。

假如你有一个控制器,你可以这样做: 1689704437503 在这个例子中,我在控制器上使用了 @CrossOrigin 注解,并设置了 origins 参数为 "http://example.com"。这意味着只有来自 "http://example.com/" 的请求才能访问这个控制器下的接口。

你还可以在具体的处理器方法上使用这个注解,比如: 1689704510825

正常的解决(应该不会遇到) JSONP:这是一种通常用于解决跨域读取数据的方法。它的工作原理是,网页通过添加一个 Githubissues.

  • Githubissues is a development platform for aggregating issues.