spring 使用Spring表达式(Spring EL)

  Spring还提供了更灵活的注入方式,那就是Spring表达式,实际上Spring EL远比以上注入方式强大,我们需要学习它。Spring EL拥有很多功能。
  使用Bean的id来引用Bean。
  •调用指定对象的方法和访问对象的属性。
  •进行运算。
  •提供正则表达式进行匹配。
  •集合配置。
  这些都是Spring表达式的内容,使用Spring表达式可以获得比使用Properties文件更为强大的装配功能,只是有时候为了方便测试可以使用Spring EL定义的解析类进行测试,为此我们先来认识它们。

Spring EL相关的类

  简要介绍Spring EL的相关类,以便我们进行测试和理解。首先是ExpressionParser接口,它是一个表达式的解析接口,既然是一个接口,那么它就不具备任何具体的功能,显然Spring会提供更多的实现类

  

  代码清单:举例说明Spring EL的使用

//表达式解析器
ExpressionParser parser = new SpelExpressionParser();
// 设置表达式
Expression exp = parser.parseExpression("‘hello world‘");
String str = (String) exp.getValue();
System.out.println(str);
//通过EL访问普通方法
exp = parser.parseExpression("‘hello world‘.charAt(0)");
char ch = (Character) exp.getValue();
System.out.println(ch);
//通过EL访问的getter方法
exp = parser.parseExpression("‘hello world‘.bytes");
byte[] bytes = (byte[]) exp.getValue();
System.out.println(bytes);
//通过EL访问属性,相当于"hello world".getBytes().length
exp = parser.parseExpression("‘hello world‘.bytes.length");
int length = (Integer) exp.getValue();
System.out.println(length);
exp = parser.parseExpression("new String(‘abc‘)");
String abc = (String) exp.getValue();
System.out.println(abc);

//表达式解析器
ExpressionParser parser = new SpelExpressionParser();
//创建角色对象
Role2 role = new Role2(1L, "role_name", "note");
Expression exp = parser.parseExpression("note");
//相当于从role中获取备注信息
String note = (String) exp.getValue(role);
System.out.println(note);
//变量环境类,并且将角色对象role作为其根节点
EvaluationContext ctx = new StandardEvaluationContext(role);
//变量环境类操作根节点
parser.parseExpression("note").setValue(ctx, "new_note");
//获取备注,这里的String.class指明,我们希望返回的是一个字符串
note = parser.parseExpression("note").getValue(ctx, String.class);
System.out.println(note);
//调用getRoleName方法
String roleName = parser.parseExpression("getRoleName()").getValue(ctx, String.class);
System.out.println(roleName);
//新增环境变量
List<String> list = new ArrayList<String>();
list.add("value1");
list.add("value2");
//给变量环境增加变量
ctx.setVariable("list", list);
//通过表达式去读/写环境变量的值
parser.parseExpression("#list[1]").setValue(ctx, "update_value2");
System.out.println(parser.parseExpression("#list[1]").getValue(ctx));

  EvaluationContext使用了它的实现类StandardEvaluationContext,进行了实例化,在构造方法中将角色对象传递给它了,那么估值内容就会基于这个类进行解析。所以后面表达式的setValue和getValue方法都把这个估值内容传递进去,这样就能够读/写根节点的内容了,并且通过getRole()的例子,还可以知道它甚至能够支持方法的调用。为了更加灵活,估值内容还支持了其他变量的新增和操作,正如代码中创建了一个List,并且把List用估值内容的setVariable方法设置,其键为"list",这样就允许我们在表达式里面通过#list去引用它,而给出的下标1,则是代表引用List的第二个元素(list是以下标0标识第一个元素的)。
  上面介绍了Spring具有对表达式的解析功能,Spring EL最重要的功能就是对Bean属性进行注入,让我们以注解的方式为主去学习它们。

Bean的属性和方法

  使用注解的方式需要用到注解@Value,在属性文件的读取中使用的是“$”,而在Spring EL中则使用“#”。下面以角色类为例进行讨论,我们可以这样初始化它的属性,如代码清单所示。
  代码清单:使用Spring EL初始化角色类

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

@Component("role2")
public class Role2 {

//赋值long型
@Value("#{2}")
private Long id;
//字符串赋值
@Value("#{‘role_name_2‘}")
private String roleName;
//字符串赋值
@Value("#{‘note_2‘}")
private String note;
}

  代码清单:通过Spring EL引用role的属性,调用其方法

@Component("elBean")
public class ElBean {

//通过beanName获取bean,然后注入
@Value("#{role2}")
private Role2 role2;

//获取bean的属性id
@Value("#{role2.id}")
private Long id;

//调用bean的getNote方法,获取角色名称
// @Value("#{role.getNote().toString()}")
@Value("#{role2.getNote()?.toString()}")
private String note;

@Value("#{T(Math).PI}")
private double pi;

@Value("#{T(Math).random()}")
private double random;

@Value("#{role.id+1}")
private int num;
}
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ApplicationConfig3.class);
Role2 role2 = context.getBean(Role2.class);
System.out.println(role2.toString());
ElBean elBean = context.getBean(ElBean.class);
System.out.println(elBean.toString());

使用类的静态常量和方法  

  有时候我们可能希望使用一些静态方法和常量,比如圆周率π,而在Java中就是Math类的PI常量了,需要注入它十分简单,在ElBean中如同下面一样操作就可以了:
  @Value("#{T(Math).PI}")
  private double pi;
  这里的Math代表的是java.lang.*包下的Math类。当在Java代码中使用该包是不需要先使用import关键字引入的,对于Spring EL也是如此。如果在Spring中使用一个非该包的内容,那么要给出该类的全限定名,需要写成类似这样:
  @Value("#{T(java.lang.Math).PI}")
  private double pi;
  同样,有时候使用Math类的静态方法去生产随机数(0到1之间的随机双精度数字),这个时候就需要使用它的random方法了,比如:
  @Value("#{T(Math).random()}")
  private double random;
  这样就可以通过调用类的静态方法加载对应的数据了。

Spring EL运算

  上面讨论了如何获取值,除此之外Spring EL还可以进行运算,比如在ElBean上增加一个数字num,其值默认为要求是角色编号(id)+1,那么我们就可以写成:

  @Value("#{role.id+1}")
  private int num;
  有时候“+”运算符也可以运用在字符串的连接上,比如下面的这个字段,把角色对象中的属性roleName和note相连:
  @Value("#{role.roleName + role.note}")
  private String str;
  这样就能够得到一个角色名称和备注相连接的字符串。比较两个值是否相等,比如角色id是否为1,角色名称是否为"role_name_001"。数字和字符串都可以使用“eq”或者“==”进行相等比较。除此之外,还有大于、小于等数学运算,比如:
  @Value("#{role.id == 1}")
  private boolean equalNum;
  @Value("#{role.note eq ‘note_1‘}")
  private boolean eqaulString;
  @Value("#{role.id > 2}")
  private boolean greater;
  @Value("#{role.id < 2}")
  private boolean less;
  在Java中,也许你会怀念三目运算,比如,如果角色编号大于1,那么取值5,否则取值1,那么在Java中可以写成:
  int max = (role.getId()>1? 5:1);
  如果角色的备注为空,我们给它一个默认的初始值“note”,使用Java则写成:  
  String defaultString = (role.getNote() == null? "hello" : role.getNote());
  下面让我们通过String EL去实现上述的功能。
  @Value("#{role.id > 1 ? 5 : 1}")
  private int max;
  @Value("#{role.note?: ‘hello‘}")
  private String defaultString;
  实际上Spring EL的功能远不止这些,上面只介绍了一些最基础、最常用的功能,熟练运用它还需要读者们多动手实践。

文章来源:ssm10.10

原文地址:https://www.cnblogs.com/ooo0/p/10987630.html

时间: 2024-08-05 23:18:21

spring 使用Spring表达式(Spring EL)的相关文章

Spring基础系列6 -- Spring表达式语言(Spring EL)

Spring基础系列6 -- Spring表达式语言(Spring EL) 转载:http://www.cnblogs.com/leiOOlei/p/3543222.html 本篇讲述了Spring Expression Language —— 即Spring3中功能丰富强大的表达式语言,简称SpEL.SpEL是类似于OGNL和JSF EL的表达式语言,能够在运行时构建复杂表达式,存取对象属性.对象方法调用等.所有的SpEL都支持XML和Annotation两种方式,格式:#{ SpEL exp

Spring Security——基于表达式的权限控制

Spring Security允许我们在定义URL访问或方法访问所应有的权限时使用Spring EL表达式,在定义所需的访问权限时如果对应的表达式返回结果为true则表示拥有对应的权限,反之则无.Spring Security可用表达式对象的基类是SecurityExpressionRoot,其为我们提供了如下在使用Spring EL表达式对URL或方法进行权限控制时通用的内置表达式. 表达式 描述 hasRole([role]) 当前用户是否拥有指定角色. hasAnyRole([role1,

Spring AOP切点表达式用法总结

1. 简介 面向对象编程,也称为OOP(即Object Oriented Programming)最大的优点在于能够将业务模块进行封装,从而达到功能复用的目的.通过面向对象编程,不同的模板可以相互组装,从而实现更为复杂的业务模块,其结构形式可用下图表示: 面向对象编程解决了业务模块的封装复用的问题,但是对于某些模块,其本身并不独属于摸个业务模块,而是根据不同的情况,贯穿于某几个或全部的模块之间的.例如登录验证,其只开放几个可以不用登录的接口给用户使用(一般登录使用拦截器实现,但是其切面思想是一致

必备技能:spring aop 切入点表达式,你都会么?

本文主要介绍spring aop中9种切入点表达式的写法 execute within this target args @target @within @annotation @args 1.execute表达式 拦截任意公共方法 execution(public * *(..)) 拦截以set开头的任意方法 execution(* set*(..)) 拦截类或者接口中的方法 execution(* com.xyz.service.AccountService.*(..)) 拦截Account

JAVAWEB开发之Spring详解之——Spring的入门以及IOC容器装配Bean(xml和注解的方式)、Spring整合web开发、整合Junit4测试

Spring框架学习路线 Spring的IOC Spring的AOP,AspectJ Spring的事务管理,三大框架的整合 Spring框架概述 什么是Spring? Spring是分层的JavaSE/EE full-stack(一站式)轻量级开源框架. 所谓分层: SUN提供的EE的三层结构:web层.业务层.数据访问层(也称持久层,集成层). Struts2是web层基于MVC设计模式框架. Hibernate是持久的一个ORM的框架. 所谓一站式:Spring框架有对三层的每层解决方案.

Spring学习1-初识Spring

一.简介   1.Spring是一个开源的控制反转(Inversion of Control ,IoC)和面向切面(AOP)的容器框架.它的主要目得是简化企业开发.  2.为何要使用Spring?    i:降低组件之间的耦合度,实现软件各层之间的解耦.    ii:可以使用容器提供的众多服务,如:事务管理服务.消息服务等等.当我们使用容器管理事务时,开发人员就不再需要手工控制事务.也不需处理复杂的事务传播. i3:容器提供单例模式支持,开发人员不再需要自己编写实现代码. i4:容器提供了AOP

Spring基础系列12 -- Spring AOP AspectJ

Spring基础系列12 -- Spring AOP AspectJ 转载:http://www.cnblogs.com/leiOOlei/p/3613352.html 本文讲述使用AspectJ框架实现Spring AOP. 再重复一下Spring AOP中的三个概念, Advice:向程序内部注入的代码. Pointcut:注入Advice的位置,切入点,一般为某方法. Advisor:Advice和Pointcut的结合单元,以便将Advice和Pointcut分开实现灵活配置. Aspe

1.1 Spring的整体架构--Spring源码深度解析

前言: Spring 始于2003年,轻量级 Java 开源框架. Spring 是为了解决企业应用开发的复杂性而创建的,它使用基本的 JavaBean 来完成以前只可能由 EJB 完成的事情. Spring 的用途不仅限于服务器端的开发,从简单性.可测试性和松耦合的角度而言,任何 Java 应用都可以从 Spring 中受益. Spring 的整体架构: Spring 框架是一个分层架构,被分为大约20个模块. (1)Core Container Core Container (核心容器)包含

攻城狮在路上(贰) Spring(三)--- Spring 资源访问利器Resource接口

Spring为了更好的满足各种底层资源的访问需求.设计了一个Resource接口,提供了更强的访问底层资源的能力.Spring框架使用Resource装载各种资源,包括配置文件资源.国际化属性文件资源等.一.Resource接口的主要方法有: boolean exists():资源是否存在. boolean isOpen():资源是否打开. URL getURL():如果底层资源可以表示为URL,该方法返回对应的URL对象. File getFile():如果底层资源对应一个文件,该方法返回对应

&lt;Spring Data JPA&gt;二 Spring Data Jpa

1.pom依赖 <?xml version="1.0" encoding="UTF-8"?> <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