Spring的Java配置方式

1、@Configuration 和 @Bean

Java配置是Spring4.x推荐的配置方式,可以完全替代xml配置。Spring的Java配置方式是通过@Configuration 和 @Bean 这两个注解实现的:
1、@Configuration 作用于类上,相当于一个xml配置文件;
2、@Bean 作用于方法上,相当于xml配置中的;

2、示例

该示例演示了通过Java配置的方式进行配置Spring,并且实现了Spring IOC功能。

2.1 创建工程以及导入以来(Maven)
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>cn.qiaoliqiang</groupId>
    <artifactId>spring-boot</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>war</packaging>

    <dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>4.3.7.RELEASE</version>
        </dependency>
        <!-- 连接池 -->
        <dependency>
            <groupId>com.jolbox</groupId>
            <artifactId>bonecp-spring</artifactId>
            <version>0.8.0.RELEASE</version>
        </dependency>
    </dependencies>
    <build>
        <finalName>${project.artifactId}</finalName>
        <plugins>
            <!-- 资源文件拷贝插件 -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <configuration>
                    <encoding>UTF-8</encoding>
                </configuration>
            </plugin>
            <!-- java编译插件 -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.7</source>
                    <target>1.7</target>
                    <encoding>UTF-8</encoding>
                </configuration>
            </plugin>
        </plugins>
        <pluginManagement>
            <plugins>
                <!-- 配置Tomcat插件 -->
                <plugin>
                    <groupId>org.apache.tomcat.maven</groupId>
                    <artifactId>tomcat7-maven-plugin</artifactId>
                    <version>2.2</version>
                </plugin>
            </plugins>
        </pluginManagement>
    </build>
</project>
2.2、编写User对象
package cn.qlq;

public class User {
	
	private String userName;
	
	private String passWord;
	
	private int age;

	public String getUserName() {
		return userName;
	}

	public void setUserName(String userName) {
		this.userName = userName;
	}

	public String getPassWord() {
		return passWord;
	}

	public void setPassWord(String passWord) {
		this.passWord = passWord;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}
	
}

2.3 编写UserDao用于模拟与数据库的交互(注意此Dao没有打注解)
package cn.qlq;

import java.util.ArrayList;
import java.util.List;

/**
 * 模拟UserDao查询数据库
 * 
 * @date 2019年12月19日上午10:04:55
 */
public class UserDao {
	
	/**
	 * 模拟查到10个用户
	 * 
	 * @return
	 */
	public List<User> queryUserList() {
		List<User> result = new ArrayList<User>();
		// 模拟数据库的查询
		for (int i = 0; i < 10; i++) {
			User user = new User();
			user.setUserName("username_" + i);
			user.setPassWord("password_" + i);
			user.setAge(i + 1);
			result.add(user);

		}
		return result;
	}

}

2.4 编写UserService用于实现User数据操作业务逻辑(申明service注解,且自动注入dao对象)
package cn.qlq;

import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

//声明是service层对象
@Service
public class UserService {
	
	//自动注入Spring容器中的dao层对象(byType注入)
	@Autowired
	private UserDao UserDao;
	
	public List<User> queryUserList(){
		
		//调用UserDao中的方法进行查询
		return this.UserDao.queryUserList();
	}

}

2.5 编写SpringConfig 用于实例化Spring容器

打上Configuration注解,同时打上@ComponentScan配置扫描的包。
@Bean用于向容器中放入对象,如果在UserDao类前面打上@Repository注解就不用@Bean方式

package cn.qlq;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

// 通过该注解来表明该类是一个Spring的配置,相当于一个xml文件
@Configuration
// 配置扫描的包
@ComponentScan(basePackages = "cn.qlq")
public class SpringConfig {

	// 通过该注解来表明是一个Bean对象,相当于xml中的<bean>
	@Bean
	public UserDao getUserDao() {
		return new UserDao(); // 直接new对象做演示
	}

}

注意:方法名是作为返回对象的名字的,因此一般不带get,也就是上述放入spring容器的bean 的name为getUserDao

2.6 编写测试方法,用于起订Spring容器
package cn.qlq;

import java.util.List;

import org.springframework.context.annotation.AnnotationConfigApplicationContext;

/**
 * 测试类
 * 
 */
public class Test {

	public static void main(String[] args) {

		// 通过Java配置来实例化Spring容器
		AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);

		// 在Spring容器中获取Bean对象
		UserService userService = context.getBean(UserService.class);

		// 调用对象中的方法
		List<User> list = userService.queryUserList();
		for (User user : list) {
			System.out.println(user.getUserName() + "," + user.getPassWord() + "," + user.getAge());
		}

		// 销毁该容器
		context.destroy();
	}

}

结果:

username_0,password_0,1
username_1,password_1,2
username_2,password_2,3
username_3,password_3,4
username_4,password_4,5
username_5,password_5,6
username_6,password_6,7
username_7,password_7,8
username_8,password_8,9
username_9,password_9,10
3、总结

从以上的示例可以可以看出,使用Java代码就完美的替代xml配置文件,并且结构更加的清晰。

使用方法:

Spring 对Java 配置的支持是由@Configuration注解和@Bean注解来实现的。由@Bean直接的方法将会实例化、配置和初始化一个新对象,这个对象将由Spring的IOC容器来管理。@Bean声明所起到的作用与元素类似。被@Configuration所注解的类则表示这个类的主要目的是作为bean定义的资源。被@Configuration声明的类可以通过在同一个类的内部调用@bean方法来设置嵌入bean 的依赖关系。

最简单的@Configuration声明类请参考下面代码:(放入spring的name为方法名字,因此一般不加get)

@Configuration
public class AppConfig {
	
	@Bean
	public MyService myService(){
		return new MyServiceImpl;
	}

}

对于上面的@Bean配置文件相同的XML配置文件如下:

<beans>
	<bean id="myService"  class="com.somnus.services.MyServiceImpl" />
</beans>

上述配置方式的实例化方式如下:

利用AnnotationConfigApplicationContext 类进行实例化

public static void main(String[] args) {
		ApplicationContext ctx = new AnnotationConfigApplicationContext(AppConfig.class);
		MyService myService = ctx.getBean(MyService.class);
		myService.doStuff();
}

要使用组件组建扫描,仅用@Configuration进行注解即可:

@Configuration
@ComponentScan(basePackages="com.somnus")
public class AppConfig {
	...
}

在上面的例子中,com.somnus包首先会被扫到,然后在容器内查找被@Component声明的类,找到后将这些类按照Spring bean定义进行注册。

补充:关于@Bean的参数注入问题:

@Bean
    public User user2(User user) { 
        System.out.println("user is -> "+ user);
        User user2 = new User();
        user2.setUsername("user2");
        return user2;
    }

如上,有参数user,若spring容器中只有一个User类型的bean,则不论参数取名为何都是按类型取bean user为参数,若有多个则参数取名必须为多个bean中的一个,否则报错。

例如:下面会报错

@Bean
    public User user1() {
        User user = new User();
        user.setUsername("user1");
        return user;
    }

    @Bean
    public User user2(User user) { 
        System.out.println("user is -> "+ user);
        User user2 = new User();
        user2.setUsername("user2");
        return user2;
    }
    
    @Bean
    public User user3(User user) { 
        System.out.println("user is -> "+ user);
        User user2 = new User();
        user2.setUsername("user2");
        return user2;
    }
Description:

Parameter 0 of method user2 in cn.qlq.config.RedisCacheConfig required a single bean, but 2 were found:
- user1: defined by method 'user1' in class path resource [cn/qlq/config/RedisCacheConfig.class]
- user3: defined by method 'user3' in class path resource [cn/qlq/config/RedisCacheConfig.class]
Action:
Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed

修改为以下即可成功:

@Bean
    public User user1() {
        User user = new User();
        user.setUsername("user1");
        return user;
    }

    @Bean
    public User user2(User user1) { 
        System.out.println("user1 is -> "+ user1);
        User user2 = new User();
        user2.setUsername("user2");
        return user2;
    }
    
    @Bean
    public User user3(User user2) { 
        System.out.println("user2 is -> "+ user2);
        User user23 = new User();
        user23.setUsername("user3");
        return user23;
    }

启动查看 注入的信息:(多个类型按name进行注入)

user1 is -> User [id=null, username=user1, password=null, userfullname=null, createtime=null, isdeleted=null, sex=null, address=null]
user2 is -> User [id=null, username=user2, password=null, userfullname=null, createtime=null, isdeleted=null, sex=null, address=null]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值