MyBatis中如何通过继承SqlSessionDaoSupport来编写DAO(二)

(本文示例工程源代码下载地址:http://down.51cto.com/data/1975295

在上一篇博文的最后,介绍了使用@PostConstruct注解标注StudentDao的init方法,这样在Spring完成依赖注入后此方法即会被Spring调用,从而也就完成了studentMapper的初始化工作。

如果只有StudentDao一个DAO类,这样做当然没有问题。不过在实际应用中,必定存在多个DAO类。每个DAO类的初始化方法,除了传入的映射器接口类型(如StudentMapper接口)不同外,代码都是一样的,也就是说,同样的代码会重复多遍。显然,这种情况是需要避免的。那么更好的做法,是只写一个初始化方法,就能初始化所有的DAO对象。本文就来探讨如何实现这个目标。

初始化方法只想写一次,就能初始化所有的DAO对象,那么这个初始化方法就只能写在父类中,在本文的例子中,也就是BaseDao了(当然,@PostConstruct注解是必需的)。在初始化方法中对子类的映射器属性(如StudentDao的StudentMapper类型的studentMapper属性)进行初始化,显然是不现实的,因为父类的方法不能访问子类的属性。那么,子类就不能定义自己的映射器属性,只能是在父类中定义,子类继承。不过这又遇到一个问题,父类不知道映射器属性具体的类型——对于StudentDao来说是StudentMapper类型,对于TeacherDao来说就是TeacherMapper类型了。这个问题如何解决呢?也许你已经想到了,我们可以用泛型。把BaseDao定义为泛型类,用泛型来定义映射器变量,子类在继承BaseDao时,再把泛型指定为自己需要的具体的映射器类型。所以,目前我们的BaseDao代码如下:

package com.abc.dao.base;
import java.lang.reflect.ParameterizedType;
import javax.annotation.PostConstruct;
import org.mybatis.spring.SqlSessionTemplate;
import org.mybatis.spring.support.SqlSessionDaoSupport;
import org.springframework.beans.factory.annotation.Autowired;
public abstract class BaseDao<T> extends SqlSessionDaoSupport {
 //保护类型,子类可直接访问
 protected T mapper;
 
 @Autowired
 public void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate)
    {
     super.setSqlSessionTemplate(sqlSessionTemplate);
    }
   
}

相应地,在StudentDao中,应指定泛型T为StudentMapper类型,删除init方法和studentMapper属性,并使用继承过来的mapper属性代替studentMapper属性。修改后的StudentDao代码如下:

package com.abc.dao;
import org.springframework.stereotype.Repository;
import com.abc.dao.base.BaseDao;
import com.abc.domain.Student;
import com.abc.mapper.StudentMapper;
@Repository
public class StudentDao extends BaseDao<StudentMapper>{
 
 public Student getById(int id)
 {
  return this.mapper.getById(id);
 }
 
 public void deleteById(int id)
 {
  int count = this.mapper.delete(id);
  System.out.println("删除了" + count + "行数据。");
 }
 
 public void update(Student student)
 {
  int count = this.mapper.update(student);
  System.out.println("修改了" + count + "行数据。");
 }
 public void add(Student student) {
  // TODO Auto-generated method stub
  int count = this.mapper.add(student);
  System.out.println("添加了" + count + "行数据。");
 }
}

最后的关键点,同时也是难点,是如何在BaseDao中编写这个init方法。其实本质上关键的地方就是如何获取子类的映射器类型,有了这个类型,获取映射器对象就很容易了。这里需要用到反射中有关泛型的知识,init方法的代码如下:

 @PostConstruct
 public void init()
 {
  //在init方法被通过子类(如StudentDao)的对象被调用时,this
  //指代的是子类的对象,this.getClass()返回代表子类的Class对象,
  //再接着调用getGenericSuperclass()方法,可以返回代表子类的直接
  //超类(也就是BaseDao类)的Type对象。因为它是泛型,因此可强制类型
  //转换为ParameterizedType类型,再调用getActualTypeArguments()
  //方法,可获得子类给泛型指定的实际类型的数组。因为这里只有一个泛型
  //参数,所以取数组的第0个元素,即为子类的映射器类型。变量mapperType
  //代表子类的映射器类型。
  @SuppressWarnings("unchecked")
  Class<T> mapperType = (Class<T>)((ParameterizedType)this.getClass().getGenericSuperclass()).getActualTypeArguments()[0];
  System.out.println("初始化..." + "   " + mapperType.getName());
  //初始化映射器属性
  this.mapper = this.getSqlSession().getMapper(mapperType);
 }

以上代码中提到的Type,JAVA API文档的解释是“Type 是 Java 编程语言中所有类型的公共高级接口。它们包括原始类型、参数化类型、数组类型、类型变量和基本类型”。Class类是它的实现类,ParameterizedType是它的子接口,表示参数化类型,其实也就是泛型。

仿照StudentDao,我们可以写出TeacherDao如下(简单起见,只有一个方法):

package com.abc.dao;
import org.springframework.stereotype.Repository;
import com.abc.dao.base.BaseDao;
import com.abc.domain.Student;
import com.abc.domain.Teacher;
import com.abc.mapper.StudentMapper;
import com.abc.mapper.TeacherMapper;
@Repository
public class TeacherDao extends BaseDao<TeacherMapper>{
 
 public Teacher getById(int id)
 {
  return this.mapper.getById(id);
 }
 
}

Spring会把这两个DAO类的对象注入到相应的Service组件中(具体请参见源代码中Spring的主配置文件applicationContext.xml中context:component-scan元素的配置,以及相应Service类和DAO类上的注解。本文示例工程源代码下载地址:http://down.51cto.com/data/1975295),而测试、执行的类(TestGenericDaoSupport)的代码如下:

package com.demo;
import org.springframework.context.ApplicationContext;
import com.abc.service.StudentService;
import com.abc.service.TeacherService;
import com.abc.domain.Student;
import com.abc.domain.Teacher;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class TestGenericDaoSupport {
 private static ApplicationContext ctx;
 static {
  // 在类路径下寻找spring主配置文件,启动spring容器
  ctx = new ClassPathXmlApplicationContext(
    "classpath:/applicationContext.xml");
 }
 public static void main(String[] args) {
  // 从Spring容器中请求服务组件
  StudentService studentService = (StudentService) ctx
    .getBean("studentService");
  TeacherService teacherService = (TeacherService) ctx
    .getBean("teacherService");
  studentService.deleteById(11);
  Teacher teacher = teacherService.getById(1);
  System.out.println("查询到的教师的姓名:" + teacher.getName());
 }
}

执行结果如下,注意红框内init方法被调用时打印的信息。

(本文示例工程源代码下载地址:http://down.51cto.com/data/1975295

MyBatis技术交流群:188972810,或扫描二维码:



点击此处观看作者的MyBatis 3从入门到提高视频教程

【MyBatis学习笔记】系列之预备篇一:ant的下载与安装

【MyBatis学习笔记】系列之预备篇二:ant入门示例

【MyBatis学习笔记】系列之一:MyBatis入门示例

【MyBatis学习笔记】系列之二:MyBatis增删改示例

【MyBatis学习笔记】系列之三:MyBatis的association示例

【MyBatis学习笔记】系列之四:MyBatis association的两种形式

【MyBatis学习笔记】系列之五:MyBatis与Spring集成示例

【MyBatis学习笔记】系列之六:MyBatis与Spring集成示例续

【MyBatis学习笔记】系列之七:MyBatis一对多双向关联

【MyBatis学习笔记】系列之八:MyBatis MapperScannerConfigurer配置

【MyBatis学习笔记】系列之九:MyBatis collection的两种形式

【MyBatis学习笔记】系列之十:MyBatis日志之Log4j示例

【MyBatis学习笔记】系列之十一:MyBatis多参数传递之注解方式示例

【MyBatis学习笔记】系列之十二:MyBatis多参数传递之默认命名方式示例

【MyBatis学习笔记】系列之十三:MyBatis多参数传递之Map方式示例

【MyBatis学习笔记】系列之十四:MyBatis中的N+1问题

【MyBatis学习笔记】系列之十五:MyBatis多参数传递之混合方式

【MyBatis学习笔记】系列之十六:Spring声明式事务管理示例

【MyBatis学习笔记】系列之十七:MyBatis多对多保存示例

【MyBatis学习笔记】系列之十八:MyBatis多对多关联查询示例

【MyBatis学习笔记】系列之十九:如何在MyBatis-3.2.7中使用Log4j2 rc2

MyBatis中如何通过继承SqlSessionDaoSupport来编写DAO(一)

MyBatis中如何通过继承SqlSessionDaoSupport来编写DAO(二)

时间: 2024-10-05 04:14:57

MyBatis中如何通过继承SqlSessionDaoSupport来编写DAO(二)的相关文章

MyBatis中如何通过继承SqlSessionDaoSupport来编写DAO(一)

在MyBatis中,当我们编写好访问数据库的映射器接口后,MapperScannerConfigurer就能自动成批地帮助我们根据这些接口生成DAO对象(),然后我们再使用Spring把这些DAO对象注入到业务逻辑层的对象(Service类的对象).因此,在这种情况下的DAO层,我们几乎不用编写代码,而且也没有地方编写,因为只有接口.这固然方便,不过如果我们需要在DAO层写一些代码的话,这种方式就无能为力了.此时,MyBatis-Spring提供给我们的SqlSessionDaoSupport类

关于INTEGER 类型在Mybatis中使用if的注意

Mybatis中是使用<if>用来编写动态语句是很方便的一个操作,不过在最近的项目开发中当使用的一个变量为inteter类型的时候,用<if>来编写发现有点问题,源码如下面: <update id="updateWeightintervalInfo" parameterType="com.exiao.platform.core.logistics.data.WeightInterval"> update tm_weightinte

详解Java的MyBatis框架中SQL语句映射部分的编写

这篇文章主要介绍了Java的MyBatis框架中SQL语句映射部分的编写,文中分为resultMap和增删查改实现两个部分来讲解,需要的朋友可以参考下 1.resultMap SQL 映射XML 文件是所有sql语句放置的地方.需要定义一个workspace,一般定义为对应的接口类的路径.写好SQL语句映射文件后,需要在MyBAtis配置文件mappers标签中引用,例如: ? 1 2 3 4 5 6 <mappers>   <mapper resource="com/limi

mybatis中的resultMap 的高级映射

引用:https://www.cnblogs.com/yansum/p/5774873.html Mybatis 高级结果映射 ResultMap Association Collection 在阅读本文章时,先说几个mybatis中容易混淆的地方: 1. mybatis中的列不是数据库里的列而是查询里的列,可以是别名(如 select user_name as userName,这时column='userName' property="userName") 2.Mysql中使用自增

MyBatis中的OGNL教程

MyBatis中的OGNL教程 有些人可能不知道MyBatis中使用了OGNL,有些人知道用到了OGNL却不知道在MyBatis中如何使用,本文就是讲如何在MyBatis中使用OGNL. 如果我们搜索OGNL相关的内容,通常的结果都是和Struts有关的,你肯定搜不到和MyBatis有关的,虽然和Struts中的用法类似但是换种方式理解起来就有难度. MyBatis常用OGNL表达式 e1 or e2 e1 and e2 e1 == e2,e1 eq e2 e1 != e2,e1 neq e2

mybatis 中mapper 的namespace有什么用

原文:http://zhidao.baidu.com/link?url=ovFuTn7-02s7Qd40BOnwHImuPxNg8tXJF3nrx1SSngNY5e0CaSP1E4C9E5J6Xv5fI9P_dTMqHeBRGOID9bk9IcY1o9h6O21l6rHRAwj_Km3 ----------------------------------------------------------------------------------------------------------

Django中的Model继承

Django 中的 model 继承和 Python 中的类继承非常相似,只不过你要选择具体的实现方式:让父 model 拥有独立的数据库:还是让父 model 只包含基本的公共信息,而这些信息只能由子 model 呈现. Django中有三种继承关系: 1.通常,你只是想用父 model 来保存那些你不想在子 model 中重复录入的信息.父类是不使用的也就是不生成单独的数据表,这种情况下使用抽象基类继承 Abstract base classes. 2.如果你想从现有的Model继承并让每个

mybatis中#{}与${}的差别(如何防止sql注入)

默认情况下,使用#{}语法,MyBatis会产生PreparedStatement语句中,并且安全的设置PreparedStatement参数,这个过程中MyBatis会进行必要的安全检查和转义. #相当于对数据 加上 双引号,$相当于直接显示数据 示例1:执行SQL:select * from emp where name = #{employeeName}参数:employeeName=>Smith解析后执行的SQL:select * from emp where name = ? 示例2:

谈一下思考,关于mybatis中&lt;foreach collection=&quot;list&quot;&gt;中list得来的原因 没看到官方说明

<foreach> 是在sql语句中进行多个id查询 时用到的,因为mybatis代替jdbc和hibernate, 使用 在xml文件中编写sql语句,这是一个标签文件.然后在 dao层?,用sqlSession.来调用这个写好的sql语句. 有时候我想会不会jdbc更简单一些?? 然后,在xml中写sql语句,会碰到根据条件或者模糊查询,这个时候需要接收来自dao层的 查询条件参数.此时需要在xml文件中的sql语句中接收,并且应该没有接收,只是确定有几个占位符,等到预编译之后,再在dao