zoukankan      html  css  js  c++  java
  • 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属性组成一个数组,最后判断数组是否为空


  • 相关阅读:
    字符数组,字符指针,字符串常量以及其sizeof的一些总结
    Linux——makefile
    动态定义多维数组
    二叉树的前序、中序、后序遍历(非递归)
    构造函数、析构函数抛出异常的问题
    倒排索引
    宏定义
    sizeof && strlen
    搜索引擎技术原理
    最小生成树
  • 原文地址:https://www.cnblogs.com/AndroidJotting/p/6596604.html
Copyright © 2011-2022 走看看