spring之基于注解管理Bean

本文涉及的产品
应用实时监控服务-用户体验监控,每月100OCU免费额度
Serverless 应用引擎免费试用套餐包,4320000 CU,有效期3个月
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
简介: 【1月更文挑战第14天】从 Java 5 开始,Java 增加了对注解(Annotation)的支持,它是代码中的一种特殊标记,可以在编译、类加载和运行时被读取,执行相应的处理。开发人员可以通过注解在不改变原有代码和逻辑的情况下,在源代码中嵌入补充信息。Spring 从 2.5 版本开始提供了对注解技术的全面支持,我们可以使用注解来实现自动装配,简化 Spring 的 XML 配置。Spring 通过注解实现自动装配的步骤如下:1. 引入依赖2. 开启组件扫描3. 使用注解定义 Bean4. 依赖注入

文章目录

前言

一、搭建子模块spring6-ioc-annotation

二、开启组件扫描

三、使用注解定义 Bean

四、实验一:@Autowired注入

①场景一:属性注入

②场景二:set注入

③场景三:构造方法注入

④场景四:形参上注入

⑤场景五:只有一个构造函数,无注解

⑥场景六:@Autowired注解和@Qualifier注解联合

五、实验二:@Resource注入

①场景一:根据name注入

②场景二:name未知注入

③场景三 其他情况

六、Spring全注解开发

总结


前言

从 Java 5 开始,Java 增加了对注解(Annotation)的支持,它是代码中的一种特殊标记,可以在编译、类加载和运行时被读取,执行相应的处理。开发人员可以通过注解在不改变原有代码和逻辑的情况下,在源代码中嵌入补充信息。

Spring 从 2.5 版本开始提供了对注解技术的全面支持,我们可以使用注解来实现自动装配,简化 Spring 的 XML 配置。

Spring 通过注解实现自动装配的步骤如下:

  1. 引入依赖
  2. 开启组件扫描
  3. 使用注解定义 Bean
  4. 依赖注入

一、搭建子模块spring6-ioc-annotation

①搭建模块

搭建方式如:spring6-ioc-xml

②引入配置文件

引入spring-ioc-xml模块日志log4j2.xml

③添加依赖

<dependencies><!--spring context依赖--><!--当你引入Spring Context依赖之后,表示将Spring的基础依赖引入了--><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>6.0.3</version></dependency><!--junit5测试--><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter-api</artifactId><version>5.3.1</version></dependency><!--log4j2的依赖--><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-core</artifactId><version>2.19.0</version></dependency><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-slf4j2-impl</artifactId><version>2.19.0</version></dependency></dependencies>

二、开启组件扫描

Spring 默认不使用注解装配 Bean,因此我们需要在 Spring 的 XML 配置中,通过 context:component-scan 元素开启 Spring Beans的自动扫描功能。开启此功能后,Spring 会自动从扫描指定的包(base-package 属性设置)及其子包下的所有类,如果类上使用了 @Component 注解,就将该类装配到容器中。

<?xmlversion="1.0" encoding="UTF-8"?><beansxmlns="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-3.0.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd"><!--开启组件扫描功能--><context:component-scanbase-package="com.gedeshidai.spring6"></context:component-scan></beans>

注意:在使用 context:component-scan 元素开启自动扫描功能前,首先需要在 XML 配置的一级标签 中添加 context 相关的约束。

情况一:最基本的扫描方式

<context:component-scanbase-package="com.gedeshidai.spring6"></context:component-scan>

情况二:指定要排除的组件

<context:component-scanbase-package="com.gedeshidai.spring6"><!-- context:exclude-filter标签:指定排除规则 --><!-- type:设置排除或包含的依据type="annotation",根据注解排除,expression中设置要排除的注解的全类名type="assignable",根据类型排除,expression中设置要排除的类型的全类名--><context:exclude-filtertype="annotation"expression="org.springframework.stereotype.Controller"/><!--<context:exclude-filter type="assignable" expression="com.gedeshidai.spring6.controller.UserController"/>--></context:component-scan>

情况三:仅扫描指定组件

<context:component-scanbase-package="com.gedeshidai"use-default-filters="false"><!-- context:include-filter标签:指定在原有扫描规则的基础上追加的规则 --><!-- use-default-filters属性:取值false表示关闭默认扫描规则 --><!-- 此时必须设置use-default-filters="false",因为默认规则即扫描指定包下所有类 --><!-- type:设置排除或包含的依据type="annotation",根据注解排除,expression中设置要排除的注解的全类名type="assignable",根据类型排除,expression中设置要排除的类型的全类名--><context:include-filtertype="annotation"expression="org.springframework.stereotype.Controller"/><!--<context:include-filter type="assignable" expression="com.gedeshidai.spring6.controller.UserController"/>--></context:component-scan>

三、使用注解定义 Bean

Spring 提供了以下多个注解,这些注解可以直接标注在 Java 类上,将它们定义成 Spring Bean。

注解 说明
@Component 该注解用于描述 Spring 中的 Bean,它是一个泛化的概念,仅仅表示容器中的一个组件(Bean),并且可以作用在应用的任何层次,例如 Service 层、Dao 层等。 使用时只需将该注解标注在相应类上即可。
@Repository 该注解用于将数据访问层(Dao 层)的类标识为 Spring 中的 Bean,其功能与 @Component 相同。
@Service 该注解通常作用在业务层(Service 层),用于将业务层的类标识为 Spring 中的 Bean,其功能与 @Component 相同。
@Controller 该注解通常作用在控制层(如SpringMVC 的 Controller),用于将控制层的类标识为 Spring 中的 Bean,其功能与 @Component 相同。

四、实验一:@Autowired注入

单独使用@Autowired注解,默认根据类型装配。【默认是byType】

查看源码:

packageorg.springframework.beans.factory.annotation;
importjava.lang.annotation.Documented;
importjava.lang.annotation.ElementType;
importjava.lang.annotation.Retention;
importjava.lang.annotation.RetentionPolicy;
importjava.lang.annotation.Target;
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documentedpublic@interfaceAutowired {
booleanrequired() defaulttrue;
}

源码中有两处需要注意:

  • 第一处:该注解可以标注在哪里?
  • 构造方法上
  • 方法上
  • 形参上
  • 属性上
  • 注解上
  • 第二处:该注解有一个required属性,默认值是true,表示在注入的时候要求被注入的Bean必须是存在的,如果不存在则报错。如果required属性设置为false,表示注入的Bean存在或者不存在都没关系,存在的话就注入,不存在的话,也不报错。

①场景一:属性注入

创建UserDao接口

packagecom.gedeshidai.spring6.dao;
publicinterfaceUserDao {
publicvoidprint();
}

创建UserDaoImpl实现

packagecom.gedeshidai.spring6.dao.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importorg.springframework.stereotype.Repository;
@RepositorypublicclassUserDaoImplimplementsUserDao {
@Overridepublicvoidprint() {
System.out.println("Dao层执行结束");
    }
}

创建UserService接口

packagecom.gedeshidai.spring6.service;
publicinterfaceUserService {
publicvoidout();
}

创建UserServiceImpl实现类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@AutowiredprivateUserDaouserDao;
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

创建UserController类

packagecom.gedeshidai.spring6.controller;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Controller;
@ControllerpublicclassUserController {
@AutowiredprivateUserServiceuserService;
publicvoidout() {
userService.out();
System.out.println("Controller层执行结束。");
    }
}

测试一

packagecom.gedeshidai.spring6.bean;
importcom.gedeshidai.spring6.controller.UserController;
importorg.junit.jupiter.api.Test;
importorg.slf4j.Logger;
importorg.slf4j.LoggerFactory;
importorg.springframework.context.ApplicationContext;
importorg.springframework.context.support.ClassPathXmlApplicationContext;
publicclassUserTest {
privateLoggerlogger=LoggerFactory.getLogger(UserTest.class);
@TestpublicvoidtestAnnotation(){
ApplicationContextcontext=newClassPathXmlApplicationContext("Beans.xml");
UserControlleruserController=context.getBean("userController", UserController.class);
userController.out();
logger.info("执行成功");
    }
}

②场景二:set注入

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
privateUserDaouserDao;
@AutowiredpublicvoidsetUserDao(UserDaouserDao) {
this.userDao=userDao;
    }
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

修改UserController类

packagecom.gedeshidai.spring6.controller;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Controller;
@ControllerpublicclassUserController {
privateUserServiceuserService;
@AutowiredpublicvoidsetUserService(UserServiceuserService) {
this.userService=userService;
    }
publicvoidout() {
userService.out();
System.out.println("Controller层执行结束。");
    }
}

测试:成功调用

③场景三:构造方法注入

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
privateUserDaouserDao;
@AutowiredpublicUserServiceImpl(UserDaouserDao) {
this.userDao=userDao;
    }
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

修改UserController类

packagecom.gedeshidai.spring6.controller;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Controller;
@ControllerpublicclassUserController {
privateUserServiceuserService;
@AutowiredpublicUserController(UserServiceuserService) {
this.userService=userService;
    }
publicvoidout() {
userService.out();
System.out.println("Controller层执行结束。");
    }
}

测试:成功调用

④场景四:形参上注入

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
privateUserDaouserDao;
publicUserServiceImpl(@AutowiredUserDaouserDao) {
this.userDao=userDao;
    }
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

修改UserController类

packagecom.gedeshidai.spring6.controller;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Controller;
@ControllerpublicclassUserController {
privateUserServiceuserService;
publicUserController(@AutowiredUserServiceuserService) {
this.userService=userService;
    }
publicvoidout() {
userService.out();
System.out.println("Controller层执行结束。");
    }
}

测试:成功调用

⑤场景五:只有一个构造函数,无注解

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.beans.factory.annotation.Qualifier;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@AutowiredprivateUserDaouserDao;
publicUserServiceImpl(UserDaouserDao) {
this.userDao=userDao;
    }
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

测试通过

当有参数的构造方法只有一个时,@Autowired注解可以省略。

说明:有多个构造方法时呢?大家可以测试(再添加一个无参构造函数),测试报错

⑥场景六:@Autowired注解和@Qualifier注解联合

添加dao层实现

packagecom.gedeshidai.spring6.dao.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importorg.springframework.stereotype.Repository;
@RepositorypublicclassUserDaoRedisImplimplementsUserDao {
@Overridepublicvoidprint() {
System.out.println("Redis Dao层执行结束");
    }
}

测试:测试异常

错误信息中说:不能装配,UserDao这个Bean的数量等于2

怎么解决这个问题呢?当然要byName,根据名称进行装配了。

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@Autowired@Qualifier("userDaoImpl") // 指定bean的名字privateUserDaouserDao;
@Overridepublicvoidout() {
userDao.print();
System.out.println("Service层执行结束");
    }
}

总结

  • @Autowired注解可以出现在:属性上、构造方法上、构造方法的参数上、setter方法上。
  • 当带参数的构造方法只有一个,@Autowired注解可以省略。()
  • @Autowired注解默认根据类型注入。如果要根据名称注入的话,需要配合@Qualifier注解一起使用。

五、实验二:@Resource注入

@Resource注解也可以完成属性注入。那它和@Autowired注解有什么区别?

  • @Resource注解是JDK扩展包中的,也就是说属于JDK的一部分。所以该注解是标准注解,更加具有通用性。(JSR-250标准中制定的注解类型。JSR是Java规范提案。)
  • @Autowired注解是Spring框架自己的。
  • @Resource注解默认根据名称装配byName,未指定name时,使用属性名作为name。通过name找不到的话会自动启动通过类型byType装配。
  • @Autowired注解默认根据类型装配byType,如果想根据名称装配,需要配合@Qualifier注解一起用。
  • @Resource注解用在属性上、setter方法上。
  • @Autowired注解用在属性上、setter方法上、构造方法上、构造方法参数上。

@Resource注解属于JDK扩展包,所以不在JDK当中,需要额外引入以下依赖:【如果是JDK8的话不需要额外引入依赖。高于JDK11或低于JDK8需要引入以下依赖。

<dependency><groupId>jakarta.annotation</groupId><artifactId>jakarta.annotation-api</artifactId><version>2.1.1</version></dependency>

源码:

packagejakarta.annotation;
importjava.lang.annotation.ElementType;
importjava.lang.annotation.Repeatable;
importjava.lang.annotation.Retention;
importjava.lang.annotation.RetentionPolicy;
importjava.lang.annotation.Target;
@Target({ElementType.TYPE, ElementType.FIELD, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(Resources.class)
public@interfaceResource {
Stringname() default"";
Stringlookup() default"";
Class<?>type() defaultObject.class;
Resource.AuthenticationTypeauthenticationType() defaultResource.AuthenticationType.CONTAINER;
booleanshareable() defaulttrue;
StringmappedName() default"";
Stringdescription() default"";
publicstaticenumAuthenticationType {
CONTAINER,
APPLICATION;
privateAuthenticationType() {
        }
    }
}

①场景一:根据name注入

修改UserDaoImpl类

packagecom.gedeshidai.spring6.dao.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importorg.springframework.stereotype.Repository;
@Repository("myUserDao")
publicclassUserDaoImplimplementsUserDao {
@Overridepublicvoidprint() {
System.out.println("Dao层执行结束");
    }
}

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importjakarta.annotation.Resource;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.beans.factory.annotation.Qualifier;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@Resource(name="myUserDao")
privateUserDaomyUserDao;
@Overridepublicvoidout() {
myUserDao.print();
System.out.println("Service层执行结束");
    }
}

测试通过

②场景二:name未知注入

修改UserDaoImpl类

packagecom.gedeshidai.spring6.dao.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importorg.springframework.stereotype.Repository;
@Repository("myUserDao")
publicclassUserDaoImplimplementsUserDao {
@Overridepublicvoidprint() {
System.out.println("Dao层执行结束");
    }
}

修改UserServiceImpl类

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importjakarta.annotation.Resource;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.beans.factory.annotation.Qualifier;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@ResourceprivateUserDaomyUserDao;
@Overridepublicvoidout() {
myUserDao.print();
System.out.println("Service层执行结束");
    }
}

测试通过

当@Resource注解使用时没有指定name的时候,还是根据name进行查找,这个name是属性名。

③场景三 其他情况

修改UserServiceImpl类,userDao1属性名不存在

packagecom.gedeshidai.spring6.service.impl;
importcom.gedeshidai.spring6.dao.UserDao;
importcom.gedeshidai.spring6.service.UserService;
importjakarta.annotation.Resource;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.beans.factory.annotation.Qualifier;
importorg.springframework.stereotype.Service;
@ServicepublicclassUserServiceImplimplementsUserService {
@ResourceprivateUserDaouserDao1;
@Overridepublicvoidout() {
userDao1.print();
System.out.println("Service层执行结束");
    }
}

测试异常

根据异常信息得知:显然当通过name找不到的时候,自然会启动byType进行注入,以上的错误是因为UserDao接口下有两个实现类导致的。所以根据类型注入就会报错。

@Resource的set注入可以自行测试

总结:

@Resource注解:默认byName注入,没有指定name时把属性名当做name,根据name找不到时,才会byType注入。byType注入时,某种类型的Bean只能有一个

六、Spring全注解开发

全注解开发就是不再使用spring配置文件了,写一个配置类来代替配置文件。

packagecom.gedeshidai.spring6.config;
importorg.springframework.context.annotation.ComponentScan;
importorg.springframework.context.annotation.Configuration;
@Configuration//@ComponentScan({"com.gedeshidai.spring6.controller", "com.gedeshidai.spring6.service","com.gedeshidai.spring6.dao"})@ComponentScan("com.gedeshidai.spring6")
publicclassSpring6Config {
}

测试类

@TestpublicvoidtestAllAnnotation(){
ApplicationContextcontext=newAnnotationConfigApplicationContext(Spring6Config.class);
UserControlleruserController=context.getBean("userController", UserController.class);
userController.out();
logger.info("执行成功");
}

总结

以上就是spring之基于注解管理Bean的相关知识点,希望对你有所帮助。

积跬步以至千里,积怠惰以至深渊。时代在这跟着你一起努力哦!

相关文章
|
1月前
|
XML Java 数据格式
SpringBoot入门(8) - 开发中还有哪些常用注解
SpringBoot入门(8) - 开发中还有哪些常用注解
50 0
|
18天前
|
XML 安全 Java
|
2天前
|
XML Java 数据格式
🌱 深入Spring的心脏:Bean配置的艺术与实践 🌟
本文深入探讨了Spring框架中Bean配置的奥秘,从基本概念到XML配置文件的使用,再到静态工厂方式实例化Bean的详细步骤,通过实际代码示例帮助读者更好地理解和应用Spring的Bean配置。希望对你的Spring开发之旅有所助益。
25 3
|
1月前
|
缓存 Java Spring
实战指南:四种调整 Spring Bean 初始化顺序的方案
本文探讨了如何调整 Spring Boot 中 Bean 的初始化顺序,以满足业务需求。文章通过四种方案进行了详细分析: 1. **方案一 (@Order)**:通过 `@Order` 注解设置 Bean 的初始化顺序,但发现 `@PostConstruct` 会影响顺序。 2. **方案二 (SmartInitializingSingleton)**:在所有单例 Bean 初始化后执行额外的初始化工作,但无法精确控制特定 Bean 的顺序。 3. **方案三 (@DependsOn)**:通过 `@DependsOn` 注解指定 Bean 之间的依赖关系,成功实现顺序控制,但耦合性较高。
实战指南:四种调整 Spring Bean 初始化顺序的方案
|
16天前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
27 1
|
21天前
|
前端开发 Java Spring
Spring MVC核心:深入理解@RequestMapping注解
在Spring MVC框架中,`@RequestMapping`注解是实现请求映射的核心,它将HTTP请求映射到控制器的处理方法上。本文将深入探讨`@RequestMapping`注解的各个方面,包括其注解的使用方法、如何与Spring MVC的其他组件协同工作,以及在实际开发中的应用案例。
36 4
|
21天前
|
前端开发 Java 开发者
Spring MVC中的请求映射:@RequestMapping注解深度解析
在Spring MVC框架中,`@RequestMapping`注解是实现请求映射的关键,它将HTTP请求映射到相应的处理器方法上。本文将深入探讨`@RequestMapping`注解的工作原理、使用方法以及最佳实践,为开发者提供一份详尽的技术干货。
63 2
|
21天前
|
前端开发 Java Spring
探索Spring MVC:@Controller注解的全面解析
在Spring MVC框架中,`@Controller`注解是构建Web应用程序的基石之一。它不仅简化了控制器的定义,还提供了一种优雅的方式来处理HTTP请求。本文将全面解析`@Controller`注解,包括其定义、用法、以及在Spring MVC中的作用。
40 2
|
24天前
|
消息中间件 Java 数据库
解密Spring Boot:深入理解条件装配与条件注解
Spring Boot中的条件装配与条件注解提供了强大的工具,使得应用程序可以根据不同的条件动态装配Bean,从而实现灵活的配置和管理。通过合理使用这些条件注解,开发者可以根据实际需求动态调整应用的行为,提升代码的可维护性和可扩展性。希望本文能够帮助你深入理解Spring Boot中的条件装配与条件注解,在实际开发中更好地应用这些功能。
31 2
|
25天前
|
JSON Java 数据格式
springboot常用注解
@RestController :修饰类,该控制器会返回Json数据 @RequestMapping(“/path”) :修饰类,该控制器的请求路径 @Autowired : 修饰属性,按照类型进行依赖注入 @PathVariable : 修饰参数,将路径值映射到参数上 @ResponseBody :修饰方法,该方法会返回Json数据 @RequestBody(需要使用Post提交方式) :修饰参数,将Json数据封装到对应参数中 @Controller@Service@Compont: 将类注册到ioc容器