前言
本篇文章主要讲诉多环境注入适应不同场景问题和Spring EL。
一、@Profile
- 在企业开发的过程中,项目往往要面临开发环境、测试环境、准生产环境(用于模拟真实生产环境部署所用〉和生产环境等的切换,所以在企业开发中往往需要有多套环境,而每一套环境的上下文是不一样的。例如,它们会有各自的数据库资源,这样就要求我们在不同的数据库之间进行切换。为了方便, Spring还提供了 Profile机制, 使我们可以很方便地实现各个环境之间的切换。在使用DI来依赖注入的时候,能够根据@profile标明的环境,将注入符合当前运行环境的相应的bean。
- 使用要求:
- @Component或@Configuration注解的类可以使用@profile
- @Profile中需要指定一个字符串,约定生效的环境
注解使用
(1) @Prifile修饰类
@Configuration @Profile("dev") public class MyConfig{ @Bean(destroyMethod="destory") public DataSource getDevDataSource () { Properties props = new Properties(); props.setProperty("driver","com.mysql.jdbc.Driver"); props.setProperty("url","jdbc:mysql://localhost:3306/dev_spring_boot") ; props.setProperty("sername","root"); props.setProperty ("password","root") ; DataSource dataSource = null; try { dataSource = BasicDataSourceFactory.createDataSource(props) ; } catch (Exception e) { e.printStackTrace() ; } return dataSource; } }
(2)@Profile修饰方法
现在有两个数据库,使用注解@Profile 定义两个Bean
@Bean(name = "dataSource", destroyMethod = "close" ) @Profile("dev") public DataSource getDevDataSource () { Properties props = new Properties(); props.setProperty("driver","com.mysql.jdbc.Driver"); props.setProperty("url","jdbc:mysql://localhost:3306/dev_spring_boot") ; props.setProperty("sername","root"); props.setProperty ("password","root") ; DataSource dataSource = null; try { dataSource = BasicDataSourceFactory.createDataSource(props) ; } catch (Exception e) { e.printStackTrace() ; } return dataSource; } @Bean(name = "dataSource",destroyMethod = "close") @Profile("test") public DataSource getTestDataSource () { Properties props= new Properties() ; props.setProperty("driver", "com.mysql.jdbc.Driver"); props.setProperty("url","jdbc:mysql://localhost:3306/test_spring_boot"); props.setProperty("username","root") ; props.setProperty("password", "root") ; DataSource dataSource = null; try { dataSource = BasicDataSourceFactory.createDataSource(props); }catch (Exception e) { e.printStackTrace(); } return dataSource; }
(3)@Profile修饰注解
了一个注解实现自定义场景,该注解可以使注入的bean使用dev的这个场景,后续就不再需要使用@Profile(“dev”)的方式,只需要在使用@Dev。
@Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Profile("dev") public @interface Dev { }
(4)启动激活
在 Spring 中存在两个参数可以提供给我们配置,以修改启动 Profile 机制, 一个是spring.profiles.active, 另一个是 spring.profiles.default。在这两个属性都没有配置的情况下, Spring 将不会启动Profile 机制,这就意味着被@Profile 标注的 Bean 将不会被 Spring装配到 roe 容器中。Spring是先判定是否存在spring.profiles.active 配置后 , 再去查找 spring.profiles.default 配置的,所以spring.profiles.active 的优先级要大于 spring.profiles.default。
在Java 启动项目中,使用dev环境,我们只需要如下配置就能够启动Profile机制(idea和eclipse的配置):
也可以打包后运行:
java -jar spring-boot-02-config-0.0.1-SNAPSHOT.jar --spring.profiles.active=dev;
资源配置文件
- springboot的环境隔离还可以使用多资源文件的方式,进行一些参数的配置。
- Springboot的资源配置文件除了application.properties之外,还可以有对应的资源文件application-{profile}.properties。假设,一个应用的工作环境有:dev、test、prod那么,我们可以添加 4 个配置文件:
- applcation.properties : 公共配置
- application-dev.properties : 开发环境配置
- application-test.properties : 测试环境配置
- application-prod.properties : 生产环境配置
- 不同的properties配置文件也可以是在 applcation.properties 文件中来激活profile:
spring.profiles.active = dev
- 这个时候请注意,按照 Spring Boot 的规则,假设把选项-Dspring.profiles.active 配置的值记为{profile},则它会用 application-{profile}.properties 文件去代替原来默认的 application.properties 文件,然后启动Spring Boot 的程序。
例如:
application.properties:
spring.profiles.active=prod
application-prod.properties:
server.port=2222 #定义一些自己使用的属性,然后通过@Value("${属性名}}")注解来加载对应的配置属性 com.name=Prod com.location=Hubei
application-dev.properties:
# 服务端口 server.port=1111 #定义一些自己使用的属性,然后通过@Value("${属性名}}")注解来加载对应的配置属性 com.name=DEV com.location=HeNan
控制层:
@Controller @RequestMapping("/test") public class test { @Value("${com.name}") private String name; @Value("${com.location}") private String location; @RequestMapping("hello") @ResponseBody public String test1(){ System.out.println(name + "hello Spring Boot, " +location); return name + ",hello Spring Boot! " +location; } }
启动Springboot后,访问http://localhost:2222/test/hello如下:
访问http://localhost:1111/test/hello是报错的,因为此时激活的是prod环境,使用的是application-dev.properties配置文件。
二、Spring EL
为了更加灵活, Spring还提供了表达式语言Spring EL。通过 Spring EL 可以拥有更为强大的运算规则来更好地装配Bean。
最常用的当然是读取属性文件的值, 例如
@Value("${database.driverName}") String driver
这里的@Value 中的${......}代表占位符,它会读取上下文的属性值装配到属性中,这便是一个最简单的Spring 表达式。 除此之外,它还能够调用方法,例如,我们记录一个Bean 的初始化时间:
@Value ("#{T(System).currentTimeMillis()}") private Long initTime = null;
- 这里采用 #{.....}代表启用 Spring 表达式,它将具有运算的功能;
- T(…)代表的是引入类:System 是java.lang.*包的类, 这是 Java 默认加载的包,因此可以不必写全限定名,如果是其他的包,则需要写出全限定名才能引用类: currentTimeMillis 是它的静态(static)方法,也就是我们调用一次System.currentTimeMillis()方法来为这个属性赋值。
此外还可以给属性直接进行赋值:
//赋值字符串 @Value( "#{'使用 Spring EL 赋值字符串'}") private String str = null; //科学计数法赋值 @Value("#(9.3E3}") private double d; //赋值浮点数 @Value ("#(3.14 }") private float pi;
还可以获取其他Spring Bean 的属性来给当前的Bean属性赋值:
@Value ("#{beanName.str}") private String otherBeanProp = null;
- beanName 是 Spring IoC 容器 Bean 的名称。
- str 是其属性,代表引用对应的 Bean的属性给当前属性赋值。
有时候, 我们还希望这个属性的宇母全部变为大写,这个时候就可以写成:
@Value("#{beanName.str?.toUpperCase()}") private String otherBeanProp = null;
再次注意这里的Spring EL。这里引用由属性后跟着是一个?,这个符号的含义是判断这个属性是否为空。如果不为空才会去执行toUppercase 方法,进而把引用到的属性转换为大写,赋予当前属性。
还可以使用 Spring EL进行一定的运算:
# 数学运算 @Value ("#{1+2}") private int run; # 浮点数比较运算 @Value ("#{beanName.pi == 3.14f}") private boolean piFlag; # 字符串比较运算 @Value ("#{beanName.str eq 'Spring Boot'}") pri. vate boolean strFlag; # 字符串连接 @Value ("#{beanName.str + ' 连接字符串 '}") private String strApp = null; # 三元运算 @Value ("#{beanName.d > 1000 ? '大子' : '小子'}") private String resultDesc = null;
从上面的代码可以看出, SpringEL 能够支持的运算还有很多,其中等值比较如果是数字型的可以使用=比较符,如果是字符串型的可以使用 eq 比较符。
总结
以上就是多环境注入和Spring EL的全部讲解。