spring_02

1 IOC注解

1.1 XML和注解的区别

1.1.1 XML配置

优点有:

  1. XML配置方式进一步降低了耦合,使得应用更加容易扩展,即使对配置文件进一步修改也不需要工程进行修改和重新编译。

  2. 在处理大的业务量的时候,用XML配置应该更加好一些。因为XML更加清晰的表明了各个对象之间的关系,各个业务类之间的调用。同时spring的相关配置也能一目了然。  

缺点有:

配置文件读取和解析需要花费一定的时间,配置文件过多的时候难以管理,无法对配置的正确性进行校验,增加了测试难度。

1.1.2 annotation配置

优点有:

  1. 在class文件中,可以降低维护成本,annotation的配置机制很明显简单  

  2. 不需要第三方的解析工具,利用java反射技术就可以完成任务  

  3. 编辑期可以验证正确性,查错变得容易  

  4. 提高开发效率  

缺点有:

  1. 如果需要对于annotation进行修改,那么要重新编译整个工程  

  2. 业务类之间的关系不如XML配置那样容易把握。  

  3. 如果在程序中annotation比较多,直接影响代码质量,对于代码的简洁度有一定的影响

Spring 的IOC 的XML方式我们已经掌握的差不多了,下面来学习学习注解的方式

1.2 注解注入使用步骤

1.2.1 Autowired

1.2.1.1 创建项目并导包

<dependency>

  <groupId>org.springframework</groupId>

  <artifactId>spring-aop</artifactId>

    <version>${spring.version}</version>

</dependency>

1.2.1.2 开启注解的支持

<?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">

    <!-- 引入注解约束 -->

    <!-- 使用注解形式自动装配 -->

    <context:annotation-config />

    <context:component-scan base-package="需要扫描的包名"/>

</beans>

1.2.1.3 业务类

业务类内容不变,只是要多加几个注解

//类名上添加Component注解

@Component

public class UserService {

    private IUserDao userDao;

//需要注入的变量上添加@Autowired

    @Autowired

    public void setUserDao(IUserDao userDao) {

        this.userDao = userDao;

}

}

//获取bean时需要类名小驼峰

application.getBean("addressServiceImpl")

1.2.1.4 知识点

@Autowired(自动封装)

该注解可以加在set方法上或者直接加载属性上,如果写在setter方法上,就会通过setter方法进行注入,如果写在变量上,就直接通过反射设置变量的值,不经过setter方法。

注入时,会从spring容器中,找到一个和这个属性数据类型匹配的实例化对象注入进来,默认使用byType,根据类型匹配。

如果只能找到一个这个数据类型的对象的时候,就直接注入该对象。

如果找到了多个同一个类型的对象的时候,就会自动更改为byName来进行匹配,根据set方法对应的参数列表的局部变量名来匹配。

private IUserDao userDao;

@Autowired

public void setUserDao(IUserDao userDao){};

会先找符合IUserDao类型的对象有多少,一个的话就直接拿过来

多个的话,就按照setUserDao方法的参数列表的局部变量名来找

找不到就报错

@Autowired(required=false) 就说明 这个值可以为null,如果Spring容器中没有对应的对象,不会报错

默认为true,比如beans.xml中没有创建dao对象,就会报错,加上required=false就不会报错

@Qualifier :  

以指定名字进行匹配  

private IUserDao userDao;

@Autowired

public void setUserDao(@Qualifier(“userDao2”)IUserDao userDao){};

这时候就不会按照userDao来进行匹配了,而是强制使用userDao2来进行比配,也就不会按照类型匹配了

1.2.1.5 配置文件

注意增加xml解析器内容,即头部beans内容

<?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">

    <!-- 引入注解约束 -->

    <!-- 使用注解形式自动装配 -->

    <context:annotation-config />

    <bean name="userDao" class="com.tledu.zrz.dao.impl.UserDaoImpl">

  <property name="daoId" value="1"></property>

    </bean>

  <bean name="userDao2" class="com.tledu.zrz.dao.impl.UserDaoImpl">

  <property name="daoId" value="2"></property>

    </bean>

    <bean id="userService" class="com.tledu.zrz.service.UserService">

    </bean>

</beans>

1.2.1.6 测试

@Test

public void testAdd() {

  ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext(

    "beans.xml");

  // userService是bean的id或name

   UserService userService = (UserService) applicationContext

   .getBean("userService");

   System.out.println(userService.getUserDao());

}

     因为创建了两个对象,所以Dao构造方法执行两次

由于代码中定义了根据name匹配,所以最终找到的是daoId=2的对象

1.2.2 Resource

Resource这个注解是javaEE的,在javax包下,所以不需要导入其他jar包

@Resource默认使用byName的方式,按照名字匹配,可以写在setter方法上也可以写在变量上

先匹配set方法的名字,匹配不上再匹配方法参数列表的名字

如果还是匹配不上就会转换为byType,根据类型匹配

当然我们也可以指定名字

@Resource(name=”userDao”)  

就相当于 Autowired和Qualifier 一起使用

相关的还有一个 @Inject 根据type匹配,通过named指定名字,自行学习  

导包javax.inject.Inject

1.2.2.1 业务类

@Resource(name="userDao")

  public void setUserDao(UserDao userDao) {

      System.out.println("--------------");

      this.userDao2 = userDao;

    }

1.2.2.2 测试类

@Test

    public void testAdd() {

  ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext(

    "beans.xml");

   UserService userService = (UserService) applicationContext

   .getBean("userService");

   System.out.println(userService.getUserDao());

    }

1.3 注解实例化使用步骤

1.3.1 配置文件

创建项目和导包和上面的一样,业务类也一致,可以复制过来进行更改

在配置文件中设置使用注解形式实例化对象 只需要加入 <context:component-scan base-package="com.tledu" />

----------------------------------------------------------------------------------------------------------------------

<?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">

    <!-- 使用注解形式自动装配 -->

    <context:annotation-config />

    <!-- 使用注解形式实例化对象 -->

    <context:component-scan base-package="com.tledu" />

</beans>

1.3.2 业务类

所有需要实例化对象的类上面都加上@Component  

默认是以类名首字母小写作为名字进行存储

可以使用@Component(“xxx”) 或者@Component(value=”xxx”)来设置名字

@Component(value="userDao")

public class UserDaoImpl implements UserDao {

@Component

public class User {

@Component("userService")

public class UserService {

1.3.3 测试

@Test

    public void testAdd() {

  ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext(

    "beans.xml");

  // userService是value设置的值或者是类名首字母小写

   UserService userService = (UserService) applicationContext

   .getBean("userService");

   System.out.println(userService.getUserDao());

}

1.3.4 注解分类

像上面我们写的代码中,所有实例化对象都是要的是@Component 这样不是很好,官方给出了几个分类

@Controller :WEB 层 ,就是和页面交互的类

@Service :业务层 ,主要处理逻辑

@Repository :持久层 ,就是Dao操作数据库

这三个注解是为了让标注类本身的用途清晰,Spring 在后续版本会对其增强  

@Component: 最普通的组件,可以被注入到spring容器进行管理

@Value :用于注入普通类型. 可以写在变量上和setter方法上

@Autowired :自动装配,上面描述比较详细,可以参照上面

@Qualifier:强制使用名称注入.  

@Resource 相当于: @Autowired 和@Qualifier 一起使用

@Scope: 设置对象在spring容器中的生命周期

取值 :  

singleton:单例  

prototype:多例

@PostConstruct :相当于 init-method  

@PreDestroy :相当于 destroy-method

1.4 新注解

1.4.1 Configuration

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

属性: value:用于指定配置类的字节码

package com.tledu.zrz.config;

import org.springframework.context.annotation.Configuration;

@Configuration

public class SpringConfiguration {

}

1.4.2 ComponentScan

作用: 用于指定 spring 在初始化容器时要扫描的包。

作用和在 spring 的 xml 配置文件中的: <context:component-scan base-package="com.tledu.zrz.spring"/>是一样的。

属性: basePackages:用于指定要扫描的包。和该注解中的 value 属性作用一样。

@Configuration

@ComponentScan("com.tledu.zrz.spring")

public class SpringConfiguration {

}

1.4.3 Bean

作用: 该注解只能写在方法上,表明使用此方法创建一个对象,并且放入 spring 容器。

属性: name:给当前@Bean 注解方法创建的对象指定一个名称(即 bean 的 id)。

1.4.4 PropertySource

作用: 用于加载.properties 文件中的配置。例如我们配置数据源时,可以把连接数据库的信息写到properties 配置文件中,就可以使用此注解指定 properties 配置文件的位置。

属性: value[]:用于指定 properties 文件位置。如果是在类路径下,需要写上 classpath

package com.tledu.zrz.config;

import org.apache.commons.dbcp2.BasicDataSource;

import org.springframework.beans.factory.annotation.Value;

import org.springframework.context.annotation.Bean;

public class JdbcConfig {

    @Value("${jdbc.driver}")

    private String driver;

    @Value("${jdbc.url}")

    private String url;

    @Value("${jdbc.username}")

    private String username;

    @Value("${jdbc.password}")

    private String password;

}

jdbc.properties文件

jdbc.driver=com.mysql.jdbc.Driver

jdbc.url=jdbc:mysql://localhost:3306/test

jdbc.username=root

jdbc.password=root

@Value 在Spring5里支持${}取配置文件中的值

Spring会有一些内置变量,我们在进行变量命名的时候要加上前缀

1.4.5 Import

作用: 用于导入其他配置类,在引入其他配置类时,可以不用再写@Configuration 注解。当然,写上也没问题。

属性: value[]:用于指定其他配置类的字节码。

@Configuration

@ComponentScan("com.tledu.zrz.spring")

@Import({ JdbcConfig.class})

public class SpringConfiguration {

}

// jdbc的配置

@Configuration

@PropertySource("classpath:jdbc.properties")

public class JdbcConfig {}

1.4.6 通过注解获取容器

ApplicationContext applicationContext = new AnnotationConfigApplicationContext(SpringConfiguration.class);

1.4.7 测试类

@Test

public void testAdd() {

  ApplicationContext applicationContext = new AnnotationConfigApplicationContext(

    SpringConfiguration.class);

  UserService userService = (UserService) applicationContext

    .getBean("userService");

  userService.add(null);

}

  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值