• 一步步学Mybatis-实现多表联合查询(4)


      上一章节中我们已经完成了对单表的CRUD操作,接下来今天这一讲讲述的是关于Mybatis在多表查询时候的应用,毕竟实际业务中也是多表的联合查询比较多嘛~

    还记得最一开始我们新建过一张Website表吗,在那张表里有个VisitorId字段,表示访问者访问过哪些网站,现在我们先按照上一张中的要求把关于Website的JavaBean实体先建立出来。

    还是在david.mybatis.model包下面新建一个Website类,用来持久化数据之用,重写下相应toString()方法,方便测试程序之用。

    package david.mybatis.model;
    
    import java.text.SimpleDateFormat;
    import java.util.Date;
    
    public class Website {
        private int id;
        private String name;
        private int visitorId;
        private int status;
        private Date createTime;
        private Visitor visitor;
    
        public Website() {
            // TODO Auto-generated constructor stub
            createTime = new Date();
            visitor = new Visitor();
        }
    
        public Website(String name, int visitorId) {
            this.name = name;
            this.visitorId = visitorId;
            visitor = new Visitor();
            status = 1;
            createTime = new Date();
        }
    
        public int getId() {
            return id;
        }
    
        public void setId(int id) {
            this.id = id;
        }
    
        public Visitor getVisitor() {
            return visitor;
        }
    
        public void setVisitor(Visitor visitor) {
            this.visitor = visitor;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public int getStatus() {
            return status;
        }
    
        public void setStatus(int status) {
            this.status = status;
        }
    
        public Date getCreateTime() {
            return createTime;
        }
    
        public void setCreateTime(Date createTime) {
            this.createTime = createTime;
        }
    
        public int getVisitorId() {
            int id = 0;
            if (visitor == null)
                id = visitorId;
            else
                id = visitor.getId();
            return id;
        }
    
        public void setVisitorId(int visitorId) {
            this.visitorId = visitorId;
        }
    
        @Override
        public String toString() {
            StringBuilder sb = new StringBuilder(String.format("Website=> {Id:%d, Name:%s, CreateTime:%s}
    ", id, name,
                    new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(createTime)));
            if (visitor != null)
                sb.append(String.format("Visitor=> %s", visitor.toString()));
            return sb.toString();
        }
    }
    Website实体类

    在david.mybatis.demo下面分别新建相应的操作接口:

    package david.mybatis.demo;
    
    import java.util.List;
    import david.mybatis.model.Website;
    
    public interface IWebsiteOperation {
        
        public int add(Website website);
        
        public int delete(int id);
        
        public int update(Website website);
        
        public Website query(int id);
        
        public List<Website> getList();
        
    }
    Website操作接口

    在mapper文件夹下新建WebsiteMapper.xml映射文件,分别参照上一张所说的把增删改查的单表操作配置分别放进去,这样你可以建造一点测试数据。如下

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE mapper
      PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
      "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
    <mapper namespace="david.mybatis.demo.IWebsiteOperation">
        <sql id="getListSql">
            select id,
            name, VisitorId, status, createTime from Website
            where status>0
        </sql>
        <insert id="add" parameterType="Website" useGeneratedKeys="true"
            keyProperty="Id">
            insert into Website (Name, VisitorId, Status, CreateTime)
            values (#{name}, #{visitorId}, #{status}, #{createTime})
        </insert>
        <delete id="delete" parameterType="int">
            delete from website where
            status>0 and id = #{id}
        </delete>
        <update id="update" parameterType="Website">
            update website set
            name=#{name} where status>0 and id=#{id}
        </update>
        <select id="query" parameterType="int" resultMap="websiteRs">
            select
            Website.id siteId, Website.name siteName, Visitor.Id visitorId,
            Visitor.name visitorName,
            Website.status siteStatus, Website.createtime
            siteCreateTime from Website
            inner join Visitor on Website.visitorid =
            Visitor.id where Website.status>0 and
            Website.id=#{id}
        </select>
        <resultMap type="Website" id="websiteRs">
            <id column="siteId" property="id" />
            <result column="siteName" property="name" />
            <result column="siteStatus" property="status" />
            <result column="siteCreateTime" property="createTime" />
            <association property="visitor" javaType="Visitor" resultMap="visitorRs" />
        </resultMap>
        <resultMap type="Visitor" id="visitorRs">
            <id column="visitorId" property="id" />
            <result column="visitorName" property="name" />
        </resultMap>
        <select id="getList" resultMap="websiteByVisitorIdRs">
        <include refid="getListSql" />
        </select>    
    </mapper>
    WebsiteMapper配置

    这里今天主要说的就是那个查,现在我们想要查询网站的同时分别把相应的访问者信息一起拿出来,怎么做呢,大家可以参照配置中的query,写下联表查询的SQL,

    这里主要要注意的是,Website实体与Visit的实体里面Id与Name这2个属性都是一样的,所以为了避免映射出现出错现象,把相应的查询结果列起上不一样的别名,这样绑定的时候就可以避免

    假如我像下面一样配置会得到什么呢?

    <select id="query" parameterType="int" resultMap="websiteRs">
        select
        Website.id, Website.name siteName, Visitor.Id,
        Visitor.name visitorName,
        Website.status siteStatus, Website.createtime
        siteCreateTime from Website
        inner join Visitor on Website.visitorid =
        Visitor.id where Website.status>0 and
        Website.id=#{id}
    </select>
    <resultMap type="Website" id="websiteRs">
        <id column="id" property="id" />
        <result column="siteName" property="name" />
        <result column="siteStatus" property="status" />
        <result column="siteCreateTime" property="createTime" />
        <association property="visitor" javaType="Visitor"
            resultMap="visitorRs" />
    </resultMap>
    <resultMap type="Visitor" id="visitorRs">
        <id column="id" property="id" />
        <result column="visitorName" property="name" />
    </resultMap>
    可能引起错误的配置

    有木有发觉,Visitor的Id也变成2了,这个其实它默认映射了Website的ID,因为SQL语句查询出来的结果2个ID都是变成2了,有人会问为什么不是4呢,因为他默认匹配第一个如果你把Website.Id与Visit.Id的位置,相互换下就会发现结果又神奇的变了

    所以需要起个别名避免这种情况,这样你就会发现真相其实只有一个就是下面的:

    大家可以看到其实多表处理resultMap的方式和单表是一致的,也无非是吧列明与Javabean属性名成对应上去,可以看到在Website的<resultMap>节点里面前台另外一个resultMap,他就是代表Visit实体所需要映射的实体,可以使用以下方式进行关联

    <association property="visitor" javaType="Visitor" resultMap="visitorRs" />

    其中的visitor就是Website实体中的visit字段名,必须保证名称一致,否则就会抛出There is no getter for property named 'XXX' in 'class david.mybatis.model.Website'的异常,这在上几章已经讲述了,当然如果你觉得不用嵌套resultMap也行,嵌套也是出于其他地方可以还要用到这个配置那就提炼出来的过程,也是抽象出来的一种思想。具体使用<resultMap>中的ID与Result可以从官网查找相应区别说明:http://mybatis.github.io/mybatis-3/sqlmap-xml.html#Result_Maps

    这样,一个简单的多表联合查询就出来啦~,如果还有更加复杂的查询业务费是在这个基础上些许的变通修改。

    这章就到此为止啦,下一章会继续跟讲下,如何弄一个简单的Mybatis下的分页效果~^0^

  • 相关阅读:
    类间关系总结
    Android数据持久化技术
    广播
    活动
    Clean Code
    理解async特性
    async和await构成的异步方法
    ubuntu开启ssh服务
    lumen可以使用laravel-ide-helper
    laravel excel迁移到lumen
  • 原文地址:https://www.cnblogs.com/daviddai/p/3485590.html
Copyright © 2020-2023  润新知