前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java中Vo类的理解

Java中Vo类的理解

作者头像
Tom2Code
发布2022-11-21 12:12:04
2.8K0
发布2022-11-21 12:12:04
举报
文章被收录于专栏:TomTom

先来解释两个概念

实体类(持久对象)=PO

值对象(新的对象)=VO

代码语言:javascript
复制
一、PO:persistant?object?持久对象
可以看成是与数据库中的表相映射的java对象。使用Hibernate来生成PO是不错的选择。

二、VO:value object值对象。
通常用于业务层之间的数据传递,和PO一样也是仅仅包含数据而已。但应是抽象出的业务对象
可以和表对应,也可以不,这根据业务的需要.

有一种观点就是:PO只能用在数据层,VO用在商业逻辑层和表示层。各层操作属于该层自己的数据对象
这样就可以降低各层之间的耦合,便于以后系统的维护和扩展。
如果将PO用在各个层中就相当于我们使用全局变量,我们知道在OO设计非常不赞成使用全局变量。
但是每次都得进行VO-PO的转换,也确实很烦。我觉得有时候也可以在某个商业逻辑或者表示层使用PO
此时在这个商业逻辑的过程中PO的状态是不发生变化的,比如显示一条商品详细信息的商业逻辑。
在开发过的项目中,规模都很小,我一直都把PO当VO用,因为PO确实很方便,结合Hibernate的DAO
我使用JAVA的集合对象作为值传递的载体,当然Struts也是我的不二之选。
我认为:在一些直观的,简单的,不易发生变化的,不需要涉及多个PO时,传递值还是使用PO好
这样可以减少大量的工作量(也就意味着减少bug,减少风险),也不需要担心未来的维护工作!

vo:value object,值对象
一般在java中用的多的是pojo:plain oriented java object
原始java对象,pojo一般和数据库中的表是一一对应的。
vo一般是来做值的存储与传递。

既然有了实体类与数据库中的字段一一对应了 那为什么还要VO呢

答案是因为在复杂的业务逻辑中,往往单一实体类无法满足我们的需求,就举个简单的例子,一个课程系统中有一级分类和二级分类,那么一个一级分类应该会对应多个二级分类,如果我们使用二级菜单的话,就应该可以实现这种联动的效果,一旦一级菜单确定下来了,那么二级菜单的下拉项中的选项只能是一级菜单下的二级菜单,也许有点绕 发证大意就是需要做到 一对多

一个一级菜单对应多个二级菜单

下面就来一个例子吧,首先有一个在线课堂的项目中有一个分类表

我们先看一下表结构

id就是主键

titie是名称

那么parentId就是上级分类的ID 如果值为0则为一级分类 如果为其他

Springboot中的实体类PO:

代码语言:javascript
复制
package cn.tompro.eduservice.entity;

import com.baomidou.mybatisplus.annotation.FieldFill;
import com.baomidou.mybatisplus.annotation.IdType;
import java.util.Date;

import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import java.io.Serializable;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;

/**
 * <p>
 * 课程科目
 * </p>
 *
 * @author testjava
 * @since 2022-07-12
 */
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@ApiModel(value="EduSubject对象", description="课程科目")
public class EduSubject implements Serializable {

    private static final long serialVersionUID = 1L;

    @ApiModelProperty(value = "课程类别ID")
    @TableId(value = "id", type = IdType.ID_WORKER_STR)
    private String id;

    @ApiModelProperty(value = "类别名称")
    private String title;

    @ApiModelProperty(value = "父ID")
    private String parentId;

    @ApiModelProperty(value = "排序字段")
    private Integer sort;

    @ApiModelProperty(value = "创建时间")
    @TableField(fill = FieldFill.INSERT)
    private Date gmtCreate;

    @ApiModelProperty(value = "更新时间")
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private Date gmtModified;


}

但是 正如我在前面说到过的 这种单一实体类 是满足不了我们的需求的

因为例如下图中的需求

我们选中一个一级分类之后 那么下一个下拉列表中的选项只能是 已选中的一节分类下的对应的二级分类

是会有一个联动的反应,所以我们的单一实体类就不能做到了。

这时候,引入我们的Vo类

首先新建一级分类Vo类

代码语言:javascript
复制
package cn.tompro.eduservice.entity.subject;

import lombok.Data;

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

/**
 * 2 * @Author: AkaTom
 * 3 * @Date: 2022/7/12 21:43
 * 4 一级分类
 */
@Data
public class OneSubject {

    private String id;
    private String title;

    //一个一级分类有多个二级分类
    private List<twoSubject> children=new ArrayList<>();

}

然后是二级分类Vo类

代码语言:javascript
复制
package cn.tompro.eduservice.entity.subject;

import lombok.Data;

/**
 * 2 * @Author: AkaTom
 * 3 * @Date: 2022/7/12 21:43
 * 4 二级分类
 */
@Data
public class twoSubject {
    private String id;
    private String title;
}

然后在前端的时候我们则需要获取我们的vo类

vue中的methods:

代码语言:javascript
复制
    //点击某个一级分类,触发change,显示对应二级分类
    subjectLevelOneChanged(value) {
      //value就是一级分类id值
      //遍历所有的分类,包含一级和二级
      for(var i=0;i<this.subjectOneList.length;i++) {
        //每个一级分类
        var oneSubject = this.subjectOneList[i]
        //判断:所有一级分类id 和 点击一级分类id是否一样
        if(value === oneSubject.id) {
          //从一级分类获取里面所有的二级分类
          this.subjectTwoList = oneSubject.children
          //把二级分类id值清空
          this.courseInfo.subjectId = ''
        }
      }
    },
    //查询所有的一级分类
    getOneSubject() {
      subject.getSubjectList()
        .then(response => {
          this.subjectOneList = response.data.list
        })
    },

data:

代码语言:javascript
复制
subjectOneList:[],//一级分类
subjectTwoList:[]//二级分类

页面:

代码语言:javascript
复制
<!-- 所属分类 TODO -->
      <el-form-item label="课程分类">
        <el-select
          v-model="courseInfo.subjectParentId"
          placeholder="一级分类" @change="subjectLevelOneChanged">

          <el-option
            v-for="subject in subjectOneList"
            :key="subject.id"
            :label="subject.title"
            :value="subject.id"/>

        </el-select>

        <!-- 二级分类 -->
        <el-select v-model="courseInfo.subjectId" placeholder="二级分类">
          <el-option
            v-for="subject in subjectTwoList"
            :key="subject.id"
            :label="subject.title"
            :value="subject.id"/>
        </el-select>
      </el-form-item>

后端service层中的主要方法:

代码语言:javascript
复制
public List<OneSubject> getAllOneTwoSubject() {
        //1 查询所有1级分类
        QueryWrapper<EduSubject> wrapperOne=new QueryWrapper<>();
        wrapperOne.eq("parent_id","0");
        List<EduSubject> oneSubjectList = baseMapper.selectList(wrapperOne);

        //2 查询所有二级分类
        QueryWrapper<EduSubject> wrapperTwo=new QueryWrapper<>();
        wrapperTwo.ne("parent_id","0");
        List<EduSubject> twoSubjectList = baseMapper.selectList(wrapperTwo);
        //创建list集合 存储最终封装数据
        List<OneSubject> finalSubjectList=new ArrayList<>();
        //封装一级分类
        //查询出来所有的一级分类list集合遍历,得到每个一级分类对象,获取每个以及分类对象值
        //封装到要求的list集合里面
        for (int i = 0; i < oneSubjectList.size(); i++) {
            //得到oneSubject每个对象
            EduSubject eduSubject = oneSubjectList.get(i);
            //把edusubject的值获取出来 放到onesubject中
            OneSubject oneSubject=new OneSubject();
//            oneSubject.setId(eduSubject.getId());
//            oneSubject.setTitle(eduSubject.getTitle());
            BeanUtils.copyProperties(eduSubject,oneSubject);
            finalSubjectList.add(oneSubject);
            // 在一级分类循环遍历查询所有的二级分类
            // 创建list集合封装每一个一级分类的二级分类
            List<twoSubject> twoFinalSubjectList=new ArrayList<>();
            for (int m = 0; m < twoSubjectList.size(); m++) {
                EduSubject eduSubject1=twoSubjectList.get(m);
                if (eduSubject1.getParentId().equals(oneSubject.getId())){
                    //把eduSubject1复制到twosubject中 放到twofinalsubjectlist中
                    twoSubject twosubject=new twoSubject();
                    BeanUtils.copyProperties(eduSubject1,twosubject);
                    twoFinalSubjectList.add(twosubject);
                }
            }
            oneSubject.setChildren(twoFinalSubjectList);
        }
        //封装二级分类
        return finalSubjectList;
    }

END

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-07-22,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Tom的小院 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档