Spring(二)基于注解的IOC配置

Spring(二)基于注解的IOC配置

常用注解

用于创建对象的

相当于: <bean id="" class=""> @Component作用:把资源让 spring 来管理。相当于在 xml 中配置一个 bean。属性:value:指定 bean 的 id。如果不指定 value 属性,默认 bean 的 id 是当前类的类名。首字母小写。 @Controller @Service @Repository他们三个注解都是针对一个的衍生注解,他们的作用及属性都是一模一样的。他们只不过是提供了更加明确的语义化。@Controller: 一般用于表现层的注解。@Service: 一般用于业务层的注解。@Repository: 一般用于持久层的注解。细节:如果注解中有且只有一个属性要赋值时,且名称是 value, value 在赋值是可以不写。

用于注入数据的

相当于: <property name="" ref=""> <property name="" value=""> @Autowired作用:自动按照类型注入。当使用注解注入属性时, set 方法可以省略。它只能注入其他 bean 类型。当有多个类型匹配时,使用要注入的对象变量名称作为 bean 的 id,在 spring 容器查找,找到了也可以注入成功。找不到就报错。 @Qualifier作用:在自动按照类型注入的基础之上,再按照 Bean 的 id 注入。它在给字段注入时不能独立使用,必须和@Autowire 一起使用;但是给方法参数注入时,可以独立使用。

属性:value:指定 bean 的 id。 @Resource作用:直接按照 Bean 的 id 注入。它也只能注入其他 bean 类型。属性:name:指定 bean 的 id。 @Value作用:注入基本数据类型和 String 类型数据的属性:value:用于指定值 用于改变作用范围的:相当于: <bean id="" class="" scope=""> @Scope作用:指定 bean 的作用范围。属性:value:指定范围的值。取值: singleton prototype request session globalsession

和生命周期相关的:

相当于: <bean id="" class="" init-method="" destroy-method="" /> @PostConstruct作用:用于指定初始化方法。

@PreDestroy作用:用于指定销毁方法。

XML和注解开发

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="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/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
   
   <!-- 配置 spring 创建容器时要扫描的包 -->
   <context:component-scan base-package="fun.chenqi"></context:component-scan>
   <!-- 配置 QueryRunner -->
   <bean id="runner" class="org.apache.commons.dbutils.QueryRunner" scope="prototype">
       <constructor-arg name="ds" ref="dataSource"/>
   </bean>
   <!-- 配置数据源 -->
   <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
       <property name="driverClass" value="com.mysql.jdbc.Driver"/>
       <property name="jdbcUrl" value="jdbc:mysql:///chenqi"/>
       <property name="user" value="root"/>
       <property name="password" value="111"/>
   </bean>
</beans>
@Service("accountService")
public class AccountServiceImpl implements IAccountService {  
   @Autowired
   IAccountDao dao; // 无依赖
   @Override
   public List<Account> select() throws SQLException {
       return dao.select();
   }
 // 。。。。。。
}

@Repository("accountDao")
public class AccountDaoImpl implements IAccountDao {
   @Autowired
   private QueryRunner qr;
 
   @Override
   public List<Account> select() throws SQLException {
       List<Account> list = null;
       try {
           String sql = "select * from account";
           list = qr.query(sql, new BeanListHandler<Account>(Account.class));
           for (Account account : list) {
               System.out.println(account);
           }
           return list;
       } catch (SQLException e) {
           e.printStackTrace();
       }
       return null;
   }
}
@Test
public void sel() throws SQLException {
   ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
   IAccountService service = (IAccountService) ac.getBean("accountService");
   service.select();
}

纯注解开发

@Configuration

作用:用于指定当前类是一个 spring 配置类, 当创建容器时会从该类上加载注解。 获取容器时需要使用AnnotationApplicationContext(有@Configuration 注解的类.class)。属性:value:用于指定配置类的字节码

@ComponentScan

作用:用于指定 spring 在初始化容器时要扫描的包。 作用和在 spring 的 xml 配置文件中的:<context:component-scan base-package="fun.chenqi"/>是一样的。属性:basePackages:用于指定要扫描的包。和该注解中的 value 属性作用一样。@Bean作用:该注解只能写在方法上,表明使用此方法创建一个对象,并且放入 spring 容器。属性:name:给当前@Bean 注解方法创建的对象指定一个名称(即 bean 的 id)

@PropertySource

作用:用于加载.properties 文件中的配置。例如我们配置数据源时,可以把连接数据库的信息写到properties 配置文件中,就可以使用此注解指定 properties 配置文件的位置。属性:value[]:用于指定 properties 文件位置。如果是在类路径下,需要写上 classpath:

@Import

作用:用于导入其他配置类,在引入其他配置类时,可以不用再写@Configuration 注解。当然,写上也没问题。属性:value[]:用于指定其他配置类的字节码。

通过注解获取容器:

ApplicationContext ac =new AnnotationConfigApplicationContext(SpringConfiguration.class);

代码:

此时已经不需要xml配置文件了
//@Configuration  spring 的配置类,相当于 bean.xml 文件
@ComponentScan("fun.chenqi") // 相当于<context:component-scan base-package="fun.chenqi"/>
@Import({JdbcConfig.class}) // 用于导入其他配置类,在引入其他配置类时,可以不用再写@Configuration 注解。当然,写上也没问题。
public class SpringConfiguration {
}
/* @PropertySource
 用于加载.properties 文件中的配置。例如我们配置数据源时,可以把连接数据库的信息写到
 properties 配置文件中,就可以使用此注解指定 properties 配置文件的位置。
 属性:
 value[]:用于指定 properties 文件位置。如果是在类路径下,需要写上 classpath:
*/
// @Configuration 写不写都行
@PropertySource("classpath:jdbc.properties")
public class JdbcConfig {
   // @Value 获取配置文件的值
   @Value("${jdbc.driver}")
   private String driver;
   @Value("${jdbc.url}")
   private String url;
   @Value("${jdbc.username}")
   private String username;
   @Value("${jdbc.password}")
   private String password;
   /* @Bean作用:
       该注解只能写在方法上,表明使用此方法创建一个对象,并且放入 spring 容器。
       属性:
       name:给当前@Bean 注解方法创建的对象指定一个名称(即 bean 的 id)
  */
   //创建一个数据源 并存入spring容器中
   @Bean(name = "dataSource")
   public DataSource createDataSource() {
       try {
           ComboPooledDataSource ds = new ComboPooledDataSource();
           ds.setUser(username);
           ds.setPassword(password);
           ds.setDriverClass(driver);
           ds.setJdbcUrl(url);
           return ds;
       } catch (PropertyVetoException e) {
           throw new RuntimeException();
       }
   }

   @Bean("runner")
   @Scope("prototype")
   public QueryRunner createQueryRunner(DataSource dataSource) {
       return new QueryRunner(dataSource);
   }
}
# jdbc.properties
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql:///chenqi
jdbc.username=root
jdbc.password=111
@Repository("accountDao")
public class AccountDaoImpl implements IAccountDao {
   @Autowired
   private QueryRunner qr;

   @Override
   public List<Account> select() throws SQLException {
       List<Account> list = null;
       try {
           String sql = "select * from account";
           list = qr.query(sql, new BeanListHandler<Account>(Account.class));
           for (Account account : list) {
               System.out.println(account);
           }
           return list;
       } catch (SQLException e) {
           e.printStackTrace();
       }
       return null;
   }
}
@Service("accountService")
public class AccountServiceImpl implements IAccountService {
   @Autowired
   IAccountDao dao;
 
   @Override
   public List<Account> select() throws SQLException {
       return dao.select();
   }
}

Spring 整合 Junit

在测试类中,每个测试方法都有以下两行代码:ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");IAccountService as = ac.getBean("accountService",IAccountService.class);这两行代码的作用是获取容器,如果不写的话,直接会提示空指针异常。所以又不能轻易删掉。

这时,我们需要依靠 spring 框架,因为它提供了一个运行器,可以读取配置文件(或注解)来创建容器。我 们只需要告诉它配置文件在哪就行了。

步骤:

导入坐标:

<!--此处需要注意的是, spring5 及以上版本要求 junit 的版本必须是 4.12 及以上,否则用不了。-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>

使用@RunWith 注解替换原有运行器

使用@ContextConfiguration 指定 spring 配置文件的位置

@RunWith(SpringJUnit4ClassRunner.class) // 使用@RunWith 注解替换原有运行器
@ContextConfiguration(classes = SpringConfiguration.class)
// 使用@ContextConfiguration 指定 spring 配置文件的位置
// classes 属性: 用于指定注解的类。 当不使用 xml 配置时,需要用此属性指定注解类的位置
// locations 属性: 用于指定配置文件的位置。如果是类路径下,需要用 classpath:表明
// @ContextConfiguration(locations= {"classpath:bean.xml"})
public class AccountTest {
   @Autowired
   private IAccountService service;

   @Test
   public void sel() throws SQLException {
       service.select();
   }
}

发表评论