Spring 基于注解的IoC配置
1. 常用的IoC注解按作用分类
<!-- 非注解的bml配置 -->
<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"
scope="" init-method="" destroy-method="">
<property name="" value="" | ref=""></property>
</bean>
1.1 用于创建对象的
作用就和在XML配置文件中编写一个<bean>
标签实现的功能是一样的
-
@Component
:- 作用:用于把当前类对象存入spring容器中
- 属性:
value
:用于指定bean的id。当我们不写时,它的默认值是当前类名,且首字母改小写。
以下三个注解的作用和属性与Component是一模一样。spring框架提供明确的三层使用的注解,使用三层对象更加清晰
@Controller
:一般用在表现层@Service
:一般用在业务层@Repository
:一般用在持久层
1.2 用于注入数据的
作用就和在xml配置文件中的bean标签中写一个<property>
标签的作用是一样的
位置:可以是变量上,也可以是方法上
细节:在使用注解注入时,set方法就不是必须的了。
-
@Autowired
:- 作用:自动按照类型注入。只要容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功; 如果ioc容器中没有任何bean的类型和要注入的变量类型匹配,则报错。
-
@Qualifier
:- 如果Ioc容器中有多个类型匹配时,需要使用
@Qualifier
注解: - 作用:在按照类中注入的基础之上再按照名称注入。它在给类成员注入时不能单独使用,必须和
@Autowired
注解一起使用。但是在给方法参数注入时可以。 - 属性:
value
:用于指定注入bean的id。
- 如果Ioc容器中有多个类型匹配时,需要使用
-
@Resource
- 作用:直接按照bean的id注入。它可以独立使用
- 属性:
name
:用于指定bean的id。
注意:以上三个注入都只能注入其他bean类型的数据,而基本类型和String类型无法使用上述注解实现。另外,集合类型的注入只能通过XML来实现。
-
@Value
- 作用:用于注入基本类型和String类型的数据
value
:用于指定数据的值。它可以使用spring中SpEL(也就是spring的el表达式); SpEL的写法:${表达式}
1.3 用于改变作用范围的
作用就和在bean标签中使用scope属性实现的功能是一样的
@Scope
:- 作用:用于指定bean的作用范围
value
:指定范围的取值。常用取值:singleton prototype
1.4 和生命周期相关
(了解)作用就和在bean标签中使用init-method和destroy-methode的作用是一样的
@PreDestroy
用于指定销毁方法@PostConstruct
用于指定初始化方法
2. 配置注解
实例代码
@Configuration
@ComponentScan(basePackages = "com.lwq")
@Import(JdbcConfiguration.class)
@PropertySource("classpath:jdbc.properties")
public class SpringConfiguration {
}
public class JdbcConfiguration {
@Bean(value = "runner")
public QueryRunner createQueryRunner(DataSource dataSource) {
return new QueryRunner(dataSource);
}
@Bean(value = "dataSource")
public DataSource createDatasource() {
try {
ComboPooledDataSource ds = new ComboPooledDataSource();
ds.setDriverClass("com.mysql.jdbc.Driver");
ds.setJdbcUrl("jdbc:mysql://localhost:3306/eesy?useSSL=false");
ds.setUser("root");
ds.setPassword("root");
return ds;
} catch (PropertyVetoException e) {
throw new RuntimeException(e);
}
}
}
示例代码中的SpringConfiguration
类,是一个配置类,它的作用和applicationContext.xml是一样的,其中的注解:
@Configuration
- 作用: 指定当前类是一个配置类
- 注意: 当配置类作为AnnotationConfigApplicationContext对象创建的参数时,该注解可以不写。
@CompontScan
- 作用: 通过注解指定spring在创建容器时要扫描的包
- 属性:
value
和basePackages
的作用是一样的,都是用于指定创建容器时要扫描的包。
@Bean
- 作用: 用于把当前方法的返回值作为bean对象存入spring的ioc容器中
- 注意: 当我们使用注解配置方法时,如果方法有参数,spring框架会去容器中查找有没有可用的bean对象。查找的方式和Autowired注解的作用是一样的
@Import
- 作用: 用于导入其他的配置类
- 属性:
value
用于指定其他配置类的字节码。当使用@Import
的注解之后,有Import注解的类就父配置类,而导入的都是子配置类
使用jdbc.properties配置文件
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/eesy?useSSL=false
jdbc.user=root
jdbc.password=root
@Configuration
@ComponentScan(basePackages = "com.lwq")
@Import(JdbcConfiguration.class)
@PropertySource("classpath:jdbc.properties")
public class SpringConfiguration {
}
@Configuration
public class JdbcConfiguration {
@Value("${jdbc.driver}")
private String driver;
@Value("${jdbc.url}")
private String url;
@Value("${jdbc.user}")
private String user;
@Value("${jdbc.password}")
private String password;
@Bean(value = "runner")
public QueryRunner createQueryRunner(DataSource dataSource) {
return new QueryRunner(dataSource);
}
@Bean(value = "dataSource")
public DataSource createDatasource() {
try {
ComboPooledDataSource ds = new ComboPooledDataSource();
ds.setDriverClass(driver);
ds.setJdbcUrl(url);
ds.setUser(user);
ds.setPassword(password);
return ds;
} catch (PropertyVetoException e) {
throw new RuntimeException(e);
}
}
}
@PropertySource
- 作用:用户指定properties文件的位置
- value:指定文件的名称和路径
3. Spring整合junit
-
导入Spring整合junit的jar包
<!-- pom.xml --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>4.3.13.RELEASE</version> </dependency>
-
使用Junit提供的
@Runwith
注解 -
使用
@ContextConfiguration
注解,告知spring的运行器,spring和ioc创建是基于xml还是注解的,并且说明位置 -
当我们使用spring 5.x版本的时候,要求junit的jar必须是4.12及以上
@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = SpringConfiguration.class) public class AccountServiceTest { @Autowired private AccountService accountService; @Test public void testFindAll() { List<Account> accounts = accountService.findAllAccount(); for (Account account : accounts) { System.out.println(account); } } }