大家好,我是烤鸭:

    今天分享一下关于mybatis的级联查询。

    环境:

        mybatis   3.2.8

        spring      4.1.9

 

1.   业务场景

    在一个人申请某些账号或者权限的时候,比如微信的认证流程。

   会让你一步一步按要求输入,比如第一步:点击微信认证,第二步:认证联系人信息填写;最后一步:填写公众号名称、功能介绍、选择运营地区就。

   当你在填写完第二步退出之后,再重新登录会发现第一步的信息已经记录了。

   这里至少要设计四张表,流程表和每一步的表,都是主键一一关联,这是一对一。

   又或者如其他平台的企业版的认证,可以添加下级的机构,这就是一对多了。

2.    mybatis 构建

     get,set方法是必须要的,我这里没粘贴出来。

pojo:

public class ApplyCompanyBackResp{
    private static final long serialVersionUID = 1L;

    private ApplyCompanyBusinessA businessA1;
    private ApplyCompanyBusinessAl businessAl;
    private ApplyCompanyBusinessCt businessC1;
    private ApplyCompanyBusinessCr businessC2;
    private ApplyCompanyBusinessJ businessAliantJ1;
    private ApplyCompanyBusinessM businessM1;
    private ApplyCompanyBusinessMo businessAliantM2;
    private List<ApplyCompanyBusinessJ1Store> businessJ1StoreList;
   }

mapper:

<!--根据当前页面获取该企业开通的权限-->
	<select id="applyCompanyAuthorityBusiness" parameterType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyAuthority"
			resultMap="applyCompanyResp">
		SELECT
			acbc.`status` AS 'c1Status',
			'c1' AS 'c1Remarks',
			acbcc.`status` AS 'c2Status',
			'c2' AS 'c2Remarks',
			acba.`status` AS 'a1Status',
			'a1' AS 'a1Remarks',
			acbac.`status` AS 'a2Status',
			'a2' AS 'a2Remarks',
			acbj.status AS 'J1Status',
			'j1' AS 'j1Remarks',
			acbmr.`status` AS 'm1Status',
			'm1' AS 'm1Remarks',
			acbmr2.status AS 'm2Status',
			'm2' AS 'm2Remarks',
			acbjs.jx_name AS 'j1Name',
			acbjs.status AS 'j1StoreStatus'
		FROM apply_company_process acp
		LEFT JOIN apply_company_business_c1 acbc ON acbc.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_c2 acbcc ON acbcc.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_a1 acba ON acba.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_a2 acbac ON acbac.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_j1 acbj ON acbj.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_m1 acbmr ON acbmr.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_m2 acbmr2 ON acbmr2.apply_id = acp.apply_id
		LEFT JOIN apply_company_business_j1_store acbjs ON acbjs.apply_id = acbj.apply_id
		<where>
			1 = 1 AND 'needOptimization' = 'needOptimization' 
			AND acp.apply_id = #{applyId}
		</where>
	</select>
	<resultMap type="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBackResp" id="applyCompanyResp">
		<!-- 配置xx相关的映射信息 -->
		<association property="businessA1" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessA1">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="a1Remarks"/>
			<result property="status" column="a1Status"/>
		</association>
		<!-- 配置xx的映射信息 -->
		<association property="businessA2" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessA2">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="a2Remarks"/>
			<result property="status" column="a2Status"/>
		</association>
		<!-- 配置xx相关的映射信息 -->
		<association property="businessC1" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessC1">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="c1Remarks"/>
			<result property="status" column="c1Status"/>
		</association>
		<!-- 配置xx相关的映射信息 -->
		<association property="businessC2" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessC2">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="c2Remarks"/>
			<result property="status" column="c2Status"/>
		</association>
		<!-- 配置xx相关的映射信息 -->
		<association property="businessAliantJ1" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessJ1">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="j1Remarks"/>
			<result property="status" column="j1Status"/>
		</association>
		<!-- 配置xx相关的映射信息 -->
		<association property="businessM1" column="businessMobileRecycling.aliStatus" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessM1">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="m1Remarks"/>
			<result property="status" column="m1Status"/>
		</association>
		<!-- 配置xx相关的映射信息 -->
		<association property="businessAliantM2" column="apply_id" javaType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessM2">
			<id  property="applyId" column="applyId" />
			<result property="remarks" column="m2Remarks"/>
			<result property="status" column="m2Status"/>
		</association>
		<!-- 配置xx相关的信息 -->
		<!--collection用于指定集合类型的映射.property指定对应的集合名称.ofType指定集合中存放的数据的类型  -->
		<collection property="businessJ1StoreList" ofType="com.xxx.xxxx.xxx.xxx.xxx.entity.ApplyCompanyBusinessJ1Store">
			<!-- id和result指定了需要配置的列和属性名和映射 -->
			<result property="applyId" column="applyId" />
			<result property="jxName" column="j1Name"/>
			<result property="status" column="j1StoreStatus"/>
		</collection>
	</resultMap>

从sql我们可以看出来,有一张process主表,其他的表都是以applyId作为主键关联。其中store表是一对多的关系,其他都是一对一。如果我们想把这个企业的所有权限信息展示,就需要关联这些所有的表。

需求是查询这些权限的状态,所以不需要太多其他的字段。

association  一对一。

property :

association  中的 property 对应的是 resultMap实体中的属性名称,在上面就是这个ApplyCompanyBackResp。

ApplyCompanyBackResp 这个pojo中的属性名称,比如 businessA1 对应的 property 就是 businessA1。

column:

column 是一定要写的 ,对应的是数据库的关联字段名称,比如上边的是apply_id 。

javaType:

返回的实体类型。

id:

数据库表主键,我这边是主键关联,所以id就是apply_id。property 是 实体属性 ,column是 查询的结果字段(别名)。比如上边的各种status,是起的别名,这里要和下边的 column 对应上。

针对于我这边不需要查询主键,可以不查。

result:

和id是一样的, 是非主键的字段映射。property 是 实体属性 ,column是 查询的结果字段(别名)。

 

collection 一对多:

property :

association  中的 property 对应的是 resultMap实体中的属性名称,在上面就是这个ApplyCompanyBackResp。

ApplyCompanyBackResp 这个pojo中的属性名称,比如 businessJ1StoreList对应的 property 就是 businessJ1StoreList。

ofType:

返回的实体类型。

id:

数据库表主键,我这边是主键关联,所以id就是apply_id。property 是 实体属性 ,column是 查询的结果字段(别名)。比如上边的各种status,是起的别名,这里要和下边的 column 对应上。

针对于我这边不需要查询主键,可以不查。

result:

和id是一样的, 是非主键的字段映射。property 是 实体属性 ,column是 查询的结果字段(别名)。

 

3.    总结

association标签内加 column属性。这样查询出来的才能封装到 结果对象 中。

    如果是一对多,mysql查询出来的数据结果就会是多条。比如上面的情况,查询出来的结果如果是2条,这两条数据collection前面的结果是一样的,只是针对于collection中的属性名称的 结果 不一样。类似下图,应该j1Name,j1StoreStatus 这列是从别的表查询出来的,其他列数据是相同的。

JAVA mybaties 级联查询_级联查询

    如果你想清晰一点你的话,也可以在 association 或者 collection 写 select标签,值是 select 语句。类似下图

    

JAVA mybaties 级联查询_级联查询_02