Struts2之OGNL表达式

  OGNL(Object-Graph Navigation Language的简称),对象图导航语言,它是一门表达式语言,除了用来设置和获取Java对象的属性之外,另外提供诸如集合的投影和过滤以及lambda表达式等。在Struts2中有大量的使用,本篇我们一起来研究一下OGNL表达式在Struts2中的使用。

  首先是获取值栈中的普通属性,问题来了,哪些属性会被封装到值栈中呢?首先是我们在Action中设置的默认参数,其次是我们通过url地址传递给Action的参数,好了接下来我们做一下测试:首先是我们看一下我们的Action文件:

public class OJNL extends ActionSupport{
    
    private String Name;//普通属性不设置默认值
    private Integer age = 18;//为该属性添加默认值

    public String getName() {
        return Name;
    }

    public void setName(String name) {
        Name = name;
    }

    public Integer getAge() {
        return age;
    }

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

    @Override
    public String execute() throws Exception {
        return SUCCESS;
    }
    
}

  按照我们刚刚的介绍,name、age两个属性将会被添加到值栈中,我们接下来就看一下我们如何在jsp页面获得这两个属性值:

<ol>
  <li>访问值栈中的普通属性:name:<s:property value="name" /></li>
  <li>访问值栈中的普通属性:age:<s:property value="age" /></li>
  <li>访问值栈中的普通属性:password:<s:property value="password" /></li>
  <li><s:debug></s:debug></li>
</ol>

  url请求:<a href="http://localhost:8080/Struts/ognl?name=hpugs&password=123456">OGNL表达式</a>

  请求结果:

  

  值栈中的数据内容:

  


   看完了如何从值栈中获取普通属性,下面我们看一下如何从值栈中获取对象属性。通过获取对象属性我们一起回顾一下,如何给对象类型传递参数:1、url中对象名.属性名传参;2、实现ModelDriven<T>接口;这些不是今天的重点,如果你有不懂的地方,请查阅之前的分享,接下来回归正题,当我们通过上面的形式传递参数到对象后,值栈中是不是就会出现这个对象?我们是不是就可以通过类名.属性名来获去参数?如果你和我一样有疑惑,下面我们就一起通过代码实例,简单测试一下。下面我们分几种情况来看一下我们的值栈中对象的创建情况:

第一种:

  访问值栈中的对象的普通属性(地址栏传参),首先创建一个User对象类:

public class User {
    private int age;

    public User(){
        
    }
    
    public User(int age){
        this.age = age;
    }

    @Override
    public String toString() {
        return "User [age=" + age + "]";
    }
    
    public int getAge() {
        return age;
    }

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

  这里为什么要添加构造方法,在接下来的测试中我们会重点提这一点,大家可以先按上面的内容创建一下类对象。有了类对象,接下来就是我们Action对象了:

public class OGNL extends ActionSupport {
    
    private User user;//声明一个User对象
    
    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    @Override
    public String execute() throws Exception {
        // TODO Auto-generated method stub
        return SUCCESS;
    }
    
}

  写好了上面的内容,我们知道要为Action添加参数,我们需要在地址栏通过类名.属性名来传递,我们的url请求:

<a href="http://localhost:8080/Struts/ognl?user.age=24">OGNL表达式</a>

  我们的jsp页面获取OGNL代码:

<ol>
    <li>访问值栈中的对象的普通属性(地址栏传参):user.age:<s:property value="user.age" /></li>
    <li><s:debug></s:debug></li>
</ol>

  访问结果:

  

第二种:

  访问值栈中的对象的普通属性(对象中设置默认值,地址栏不传参),与上面不同的地方是,我们不再通过地址栏传递参数到Action,我们直接在对象中为age添加默认值,我们的User类:

public class User {
    private int age = 18;

    public User(){
        
    }
    
    public User(int age){
        this.age = age;
    }

    @Override
    public String toString() {
        return "User [age=" + age + "]";
    }
    
    public int getAge() {
        return age;
    }

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

  我们的Action与上面的一致,这里就不在增加篇幅了,既然我们已经在User对象里面为age属性添加了默认值,那么我们就不在通过url传递参数,我们来测试一下,前台jsp是否可以从值栈中获取到age属性:

  我们的url请求:

<a href="http://localhost:8080/Struts/ognl">OGNL表达式</a>

  我们的jsp页面处理:

<ol>
    <li>访问值栈中的对象的普通属性(对象中设置默认值,地址栏不传参):user.age:<s:property value="user.age" /></li>
    <li><s:debug></s:debug></li>
</ol>

  测试结果:

  

  小伙伴们是不是不淡定了,我们明明设置了默认值,为什么值栈中是NULL呢?大家想一想第一种,当我们通过Url将user.age传递到Action后,Action会为我创建一个User对象,并将age属性传递过去,而现在我们不再通过url传递user.age参数了,所以Action也就不再为我们创建User对象,所以值栈中user == null

第三种:

  对于第二种情况,你是不是和我一样有疑惑,难得我们必须通过Url才能使Action为我们创建user对象吗?答案是肯定的。如何实现呢?既然是Action没有为我们实例化user对象,我们是不是可以手动在Action中实例化user对象,下面我们做一下测试,修改一下我们的Action类:

public class OGNL extends ActionSupport{
    
    private User user = new User();
    
    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    @Override
    public String execute() throws Exception {
        // TODO Auto-generated method stub
        return SUCCESS;
    }
    
}

  接下来我们再次运行我们的项目,看一下值栈中的数据:

  

  这样值栈中就有了user对象。

第四种:

   看到第三种你是不是和我一样有一个小疑问,Action是如何为我们实例化user对象的,其实明显了,就是通过调用User类的无参构造方法,当然我们这里也简单做下测试。这里我们一起回忆一下关于java中类的构造方法的知识,首先构造方法,方法名要与类名一致,其次方法没有返回值。我们还知道,当我们没有为类添加构造方法时,系统默认为我们实现一个无参构造方法,当我们添加了有参构造方法后,系统将不再为我们添加默认构造方法。这里我们就将User类的构造方法去掉,保留有参构造方法:

public class User {
    private int age = 12;
    
    public User(int age){
        this.age = age;
    }
    
    public int getAge() {
        return age;
    }

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

  我们的Action配置与第一种一摸一样,这里就不再添加篇幅,下面直接看结果:

  

  关于值栈中对象的普通属性的获取就和大家先聊到这里


  上面我们介绍了单个对象属性获取,下面我们介绍一下通过一个对象去获取另一个对象的属性,比如有一只猫(加菲),有一个好朋友(欧第),我们通过加菲拿到欧弟的属性:

  首先我们先创建两个对象,Cat.class、Dog.class:

public class Cat {

    private Dog friend;

    public Dog getFriend() {
        return friend;
    }

    public void setFriend(Dog friend) {
        this.friend = friend;
    }
    
}
public class Dog {
    private String name;

    public Dog() {
    }

    public Dog(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Dog [name=" + name + "]";
    }
    
}

  下面就是我们的Action了,我们在Action中声明一个Cat对象:

public class OGNL extends ActionSupport {

    private Cat cat;
    
    public Cat getCat() {
        return cat;
    }

    public void setCat(Cat cat) {
        this.cat = cat;
    }

    @Override
    public String execute() throws Exception {
        return SUCCESS;
    }
}

  接下来我们就通过Url为我们的Dog对象起一个名字:欧弟

<a href="http://localhost:8080/Struts/ognl?cat.friend.name=欧弟">OGNL</a>

  我们的jsp参数接收方式;

<ol>
    <li>访问值栈中的对象的普通属性:cat.friend.name:<s:property value="cat.friend.name" /></li>
    <li><s:debug></s:debug></li>
</ol>

  访问结果:

  


  下面我们接上面的内容,一起来探讨一下,如何访问值栈中的对象的普通方法,一种:值栈中基本方法(length()这类方法);一种Action中对象的普通方法。

  这里我们在Cat类里面添加一个Eat()方法,用于jsp页面进行调用:

public class Cat {

    private Dog friend;

    public Dog getFriend() {
        return friend;
    }

    public void setFriend(Dog friend) {
        this.friend = friend;
    }
    
    public String Eat(){
        return "猫爱吃鱼";
    }
}

  下面我们看一下我们的jsp中的参数处理:

<ol>
    <li>访问值栈中的对象的普通属性:cat.friend.name:<s:property value="cat.friend.name" /></li>
    <li>访问值栈中的对象的普通方法:cat.friend.name.length():<s:property value="cat.friend.name.length()" /></li>
    <li>访问值栈中的对象的普通方法:cat.Eat():<s:property value="cat.Eat()" /></li>
    <li><s:debug></s:debug></li>
</ol>

  下面看一下处理结果:

  


   接下来我们一起来探讨一下如何访问Action中的静态方法和静态属性,这次我先来看一下jsp参数的处理模块:

<ol>              
    <li>访问静态属性:YEAR:<s:property value="@com.edu.action.OGNL@YEAR" /></li>
    <li>访问静态方法:GetDate():<s:property value="@com.edu.action.OGNL@GetDate()" /></li>
    <li>访问Max的静态方法:max(1,1):<s:property value="@@max(1,1)" /></li>
              
    <li>访问普通对象的构造方法:new User(8):<s:property value="new com.edu.model.User(8)" /></li>
    <li><s:debug></s:debug></li>
</ol>

  由此来设计我们的User类:

public class User {
    private int age;
    
    public User(int age){
        this.age = age;
    }

    @Override
    public String toString() {
        return "User [age=" + age + "]";
    }
    
    public int getAge() {
        return age;
    }

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

  然后是我们的Action类:

public class OGNL extends ActionSupport{
    
    //静态属性
    public static final Date YEAR = new Date();
    
    //静态方法
    public static String GetDate(){
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy年MM月dd日  HH:mm:ss");
        return simpleDateFormat.format(new Date());
    }
    
    @Override
    public String execute() throws Exception {
        return SUCCESS;
    }
    
}

  URL请求地址:

<a href="http://localhost:8080/Struts/ognl">OGNL表达式</a>

  最后的处理结果:

  


   最后是关于OGNL操作List、Set、Map的知识,这里就不再一一展开详述。这路用到两个类:User和Dog,这两个类的没有什么特别需要说明的内容,只是两个基本的对象类,这里不再展示,我直接从Action类开始:

public class OGNL extends ActionSupport {

    private List<User> users = new ArrayList<User>();
    
    private Set<Dog> dogs = new HashSet<Dog>();
    
    private Map<String, Dog> dogMap = new HashMap<String, Dog>();
    
    //每次访问Action都会执行,放置一些类的初始化操作
    public OGNL(){
        users.add(new User(1));
        users.add(new User(2));
        users.add(new User(3));
        
        dogs.add(new Dog("dog1"));
        dogs.add(new Dog("dog2"));
        dogs.add(new Dog("dog3"));
        
        dogMap.put("dog11", new Dog("dog11"));
        dogMap.put("dog22", new Dog("dog22"));
        dogMap.put("dog33", new Dog("dog33"));
    }

    public List<User> getUsers() {
        return users;
    }

    public void setUsers(List<User> users) {
        this.users = users;
    }

    public Set<Dog> getDogs() {
        return dogs;
    }

    public void setDogs(Set<Dog> dogs) {
        this.dogs = dogs;
    }

    public Map<String, Dog> getDogMap() {
        return dogMap;
    }

    public void setDogMap(Map<String, Dog> dogMap) {
        this.dogMap = dogMap;
    }
    
    @Override
    public String execute() throws Exception {
        return SUCCESS;
    }
    
}

  通过Action的无参构造方法,初始化我们ListSetMap,接下来我们一起看一下jsp中参数的处理:

第一种:List类型

<ol>
    <li>访问List:<s:property value="users" /></li>
    <li>访问List中某个元素users[1]:<s:property value="users[1]" /></li>
    <li>访问List中某个属性的集合users.{age}:<s:property value="users.{age}" /></li>
    <li>访问List中某个属性集合中的特定值users.{age}[2]:<s:property value="users.{age}[2]" /></li>
    <li><s:debug></s:debug></li>
</ol>

  结果页面:

  

  通过值栈的信息结合jsp页面的处理内容,List的结果处理没有太多难处。

第二种:Set类型

<ol>
    <li>访问Set:<s:property value="dogs" /></li>
    <li>访问Set中的某个元素:<s:property value="dogs[1]" /> | <span style="color:red">set中的数据是无序的,所以没办法获取set中的具体元素</span></li>
</ol>

  结果页面:

  

  和List的区别在于,List中可以根据索引找到集合中的数据,而Set由于元素无序且不能重复,故不能通过索引取得值。

第三种:Map类型

<ol>
    <li>访问Map:<s:property value="dogMap" /></li>
    <li>访问Map某个元素:<s:property value="dogMap.dog11" />|<s:property value="dogMap.dog22" />|<s:property value="dogMap.dog33" /></li>
    <li>访问Map所有Key:<s:property value="dogMap.keys" /></li>
    <li>访问Map所有Value:<s:property value="dogMap.values" /></li>
    <li>访问Map容器大小:dogMap.size<s:property value="dogMap.size" />|users.size<s:property value="users.size" />|dogs.size<s:property value="dogs.size" /></li>
    <li>访问Map容器大小:dogMap.size()<s:property value="dogMap.size()" />|users.size()<s:property value="users.size()" />|dogs.size()<s:property value="dogs.size()" /></li>
</ol>

  结果页面:

  

  Map中数据都是以key-value的形式保存的,当需要通过key得到指定的value时,我们只需要通过Map对象.key即可拿到值。对于获得List、Set、Map中的数据总数可以通过.size或.size()来获取。


  下面我们来一起看一下关于集合投影和过滤的内容,这里先要解释三个符号:'?':过滤;'^':开头;'$':结尾,下面我们来看一下在OGNL中的具体使用:

<ol>
    <li>投影(过滤):<s:property value="users.{?#this.age == 1}.{age}" />-----?:过滤</li>
    <li>投影(过滤):<s:property value="users.{?#this.age == 1}.{age}[0]" />-----?:过滤</li>
    <li>投影:<s:property value="users.{^#this.age > 1}.{age}" />-----^:开头</li>
    <li>投影:<s:property value="users.{$#this.age > 1}.{age}" />-----#:结尾</li>
    <li>投影:<s:property value="users.{$#this.age > 1}.{age} == null" />------判断集合结果是否为空</li>
</ol>

  结果页面:

  

  1、解释一下users.{?#this.age == 1}.{age}:过滤users中age==1的对象,将其age属性组成一个数组

  2、users.{?#this.age == 1}.{age}[0]:过滤users中age==1的对象,将其age属性组成一个数组,取数组中的第一个age元素

  3、users.{^#this.age > 1}.{age}:过滤users中age>1的对象,将其age属性组成一个数组,取出数组中的第一个age元素

  4、users.{$#this.age > 1}.{age}:过滤users中age>1的对象,将其age属性组成一个数组,取出数组中的最后一个age元素

  5、users.{$#this.age > 1}.{age} == null:过滤users中age>1的对象,将其age属性组成一个数组,最后判断数组是否为空


原文地址:https://www.cnblogs.com/AndroidJotting/p/6596604.html