在Web开发中,一般都分3层。
Controller/Action 控制层,
Service/Business 服务层/业务逻辑层,
Dao 数据访问层/数据持久层。

在学习和工作的实践过程中,我发现很多功能是比较通用的,我们可以把他们抽象成API接口。

下文通过一段较长的代码,Hibernate实现,来展示如何设计一些通用的API。

说明:代码只是起到一个示范(Demo)的作用,实际上完全可以做得更强大。
我最近已经在现在的基础上大大改进了,现在把比较基础的实现分享给大家。

package cn.fansunion.demo.db.dao; 

import java.lang.reflect.ParameterizedType;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set; import javax.annotation.Resource; import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.SessionFactory; import cn.fansunion.common.util.EmptyUtils; /**
* Dao的父类。采用泛型,实现了 一些通用的功能,大大减少了子类代码的重复。 * 目前只能适用于1个表或实体。
*
* @author leiwen@fansunion.cn
*/
public abstract class BaseDao<T> { private Class<T> modelClazz; @Resource
private SessionFactory sessionFactory; // ////////////////////////////////////////////////////////////// public BaseDao() {
this.modelClazz = (Class<T>) ((ParameterizedType) getClass()
.getGenericSuperclass()).getActualTypeArguments()[0];
} // //////////////////////////////////////////////////////
// //////////////////////泛型方法-CRUD/////////////////////////
// ///////////////////////////////////////////////////// /**
* 根据主键查询
*
* @param id
* 主键
* @return 实体对象
*/
public T get(Integer id) {
T entity = (T) getCurrentSession().get(modelClazz, id);
return entity;
} /**
* 增加
*
* @param entity
* 实体对象
*/
public void add(T entity) {
getCurrentSession().save(entity); } /**
* 物理删除
*
* @param entity
* 实体对象
*/
public void delete(T entity) {
getCurrentSession().delete(entity);
} /**
* 更新
*
* @param entity
* 持久态的实体对象
*/
public void update(T entity) {
getCurrentSession().update(entity);
} /**
* 逻辑删除
*
* @param id
* 主键
*/
public void remove(Integer id) { Session session = getCurrentSession();
String sql = "update " + modelClazz
+ " set isDeleted = 1 where id = :id";
Query query = session.createQuery(sql);
query.setParameter("id", id);
query.executeUpdate(); } // ///////////////////////////////////////////////
// ////////////获取记录总数/////////////////////////
// ///////////////////////////////////////////////
/**
* 获得1个整数
*
* @param hql
* hql语句
* @return 1个整数
*/
protected Integer getCount(String hql) {
Integer count = 0;
Query query = createQuery(hql);
count = getCount(query);
return count;
} protected Integer getCount(String hql, String key, Object value) {
Integer count = 0;
Query query = createQuery(hql, key, value);
count = getCount(query);
return count;
} // 带参数的hql语句
protected Integer getCount(String hql, Map<String, Object> params) {
Integer count = 0;
Query query = createQuery(hql, params);
count = getCount(query);
return count;
} private Integer getCount(Query query) {
Integer count = 0;
Object uniqueResult = query.uniqueResult();
if (uniqueResult != null) {
count = Integer.parseInt(uniqueResult.toString());
}
return count;
} // ///////////////////////////////////////////////
// ////////////获取一个对象/////////////////////////
// ///////////////////////////////////////////////
protected List findByProperty(String name, Object value) {
String hql = "from " + modelClazz.getSimpleName() + " where " + name
+ "=" + ":" + name;
return executeQueryList(hql, name, value);
} protected T executeQueryUnique(String hql) {
T result = null;
List<T> list = executeQueryList(hql);
if (list != null && list.size() >= 1) {
result = list.get(0);
}
return result;
} protected T executeQueryUnique(String hql, String key, Object value) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key, value);
return executeQueryUnique(hql, params);
} // 根据属性获得对象
protected T executeQueryUnique(String hql, String key1, Object value1,
String key2, Object value2) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key1, value1);
params.put(key2, value2);
return executeQueryUnique(hql, params);
} protected T executeQueryUnique(String hql, Map<String, Object> params) {
T result = null;
List<T> list = executeQueryList(hql, params);
if (EmptyUtils.isNotEmpty(list)) {
result = list.get(0);
}
return result;
} // ///////////////////////////////////////////////
// //////////////获取一个列表(不使用泛型 List<T>)/////////////////
// ///////////////////////////////////////////////
// 执行不带参数的hql查询,返回一个结果集List
protected List executeQueryList(String hql) {
return executeQueryList(hql, null);
} protected List executeQueryList(String hql, String key, Object value) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key, value);
return executeQueryList(hql, params);
} protected List executeQueryList(String hql, String key1, Object value1,
String key2, Object value2) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key1, value1);
params.put(key2, value2);
return executeQueryList(hql, params);
} protected List executeQueryList(String hql, Map<String, Object> params) {
return executeQueryList(hql, params, -1, -1);
} protected List executeQueryList(String hql, Integer firstResult,
Integer maxResults) {
return executeQueryList(hql, null, firstResult, maxResults);
} protected List executeQueryList(String hql, String key, Object value,
Integer firstResult, Integer maxResults) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key, value);
return executeQueryList(hql, params, firstResult, maxResults);
} // 执行带参数并且含有分页的hql查询
protected List executeQueryList(String hql, Map<String, Object> params,
Integer firstResult, Integer maxResults) {
Query query = createQuery(hql, params);
if (firstResult > 0) {
query.setFirstResult(firstResult);
} if (maxResults > 0) {
query.setMaxResults(maxResults);
} return query.list();
} // ///////////////////////////////////////////////
// ////////////更新操作/////////////////////////
// /////////////////////////////////////////////// protected int executeUpdate(String hql) {
return executeUpdate(hql, null);
} protected int executeUpdate(String hql, String key, Object value) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key, value);
return executeUpdate(hql, params);
} protected int executeUpdate(String hql, String key1, Object value1,
String key2, Object value2) {
Map<String, Object> params = new HashMap<String, Object>();
params.put(key1, value1);
params.put(key2, value2);
return executeUpdate(hql, params);
} // 执行带参数的hql更新语句
protected int executeUpdate(String hql, Map<String, Object> params) {
Query query = createQuery(hql, params);
return query.executeUpdate();
} // ///////////////////////////////////////////////
// ////////////创建Query对象/////////////////////////
// ///////////////////////////////////////////////
private Query createQuery(String hql, Map<String, Object> params) {
Query query = getCurrentSession().createQuery(hql);
if (params != null) {
Set<Entry<String, Object>> entrySet = params.entrySet();
for (Map.Entry<String, Object> entry : entrySet) {
Object value = entry.getValue();
String key = entry.getKey();
if (value instanceof Collection) {
query.setParameterList(key, (Collection) value);
} else if (value instanceof Object[]) {
query.setParameterList(key, (Object[]) value);
} else {
query.setParameter(key, value);
}
}
}
return query;
} private Query createQuery(String hql, String key, Object value) {
Query query = getCurrentSession().createQuery(hql);
if (key != null) {
query.setParameter(key, value);
}
return query;
} private Query createQuery(String hql) {
return getCurrentSession().createQuery(hql);
} /**
* 获取主数源
*/
protected Session getCurrentSession() {
return sessionFactory.getCurrentSession();
} }

在工作和学习的实践中,我发现Web开发有很大程度上的通用性。

我希望,也在努力地总结这些规律,争取早点弄出一套可以大大提高生产力的方法和代码框架。

技术不能改变程序员的命运,而生产力可以。

提高生产力,是我目前迫切的追求。

过去,现在和未来,我都将为之而努力。

我的博客网站:http://FansUnion.cn

原文参见:http://fansunion.cn/articles/2264

数据持久层(DAO)通用API的实现的更多相关文章

  1. [置顶] 数据持久层(DAO)常用功能–通用API的实现

    在Web开发中,一般都分3层. Controller/Action 控制层, Service/Business 服务层/业务逻辑层, Dao 数据访问层/数据持久层. 在学习和工作的实践过程中,我发现 ...

  2. Java数据持久层

    一.前言 1.持久层 Java数据持久层,其本身是为了实现与数据源进行数据交互的存在,其目的是通过分层架构风格,进行应用&数据的解耦. 我从整体角度,依次阐述JDBC.Mybatis.Myba ...

  3. Restful.Data v1.0 - 轻量级数据持久层组件, 正式开源发布了

    经过几个星期的优化调整,今天 Restful.Data 正式开源发布. 源码地址:https://github.com/linli8/Restful 今天不写那么多废话了,还是重新介绍一下 Restf ...

  4. Java数据持久层框架 MyBatis之背景知识一

    对于MyBatis的学习而言,最好去MyBatis的官方文档:http://www.mybatis.org/mybatis-3/zh/index.html 对于语言的学习而言,马上上手去编程,多多练习 ...

  5. .NET平台下,关于数据持久层框架

    在.NET平台下,关于数据持久层框架非常多,本文主要对如下几种做简要的介绍并推荐一些学习的资源: 1.NHibernate 2.NBear 3.Castle ActiveRecord 4.iBATIS ...

  6. .NET开源项目介绍及资源推荐:数据持久层

    在.NET平台下,关于数据持久层框架非常多,本文主要对如下几种做简要的介绍并推荐一些学习的资源: 1.NHibernate 2.NBear 3.Castle ActiveRecord 4.iBATIS ...

  7. c++ 数据持久层研究(一)

    C++ORM框架自动生成代码数据库  用过Java的都知道SSH框架,特别对于数据库开发,Java领域有无数的ORM框架,供数据持久层调用,如Hibernate,iBatis(现在改名叫MyBatis ...

  8. UWP开发之ORM实践:如何使用Entity Framework Core做SQLite数据持久层?

    选择SQLite的理由 在做UWP开发的时候我们首选的本地数据库一般都是Sqlite,我以前也不知道为啥?后来仔细研究了一下也是有原因的: 1,微软做的UWP应用大部分也是用Sqlite.或者说是微软 ...

  9. .NET平台数据持久层框架

    在.NET平台下的几个数据持久层框架: 1.NHibernate 2.NBear 3.Castle ActiveRecord 4.iBATIS.NET 5.DAAB 6.DLinq

随机推荐

  1. 开源 java CMS - FreeCMS2.2 工作流管理

    项目地址:http://www.freeteam.cn/ 工作流管理 从FreeCMS 2.2開始支持 管理系统中使用到的工作流,如信息审核工作流. 1. 工作流组管理 从左側管理菜单点击工作流组管理 ...

  2. 飘逸的python - 极简的二叉树前中后序通杀函数

    对于任一结点.能够按某种次序运行三个操作: 訪问结点本身(N) 遍历该结点的左子树(L) 遍历该结点的右子树(R) 用来表示顺序,即,前序NLR/中序LNR/后序LRN. 以下我们用namedtupl ...

  3. Codeforces Round #276 (Div. 1) A. Bits 贪心

    A. Bits   Let's denote as  the number of bits set ('1' bits) in the binary representation of the non ...

  4. oc65--协议应用1,接口.做数据类型限定

    // WifeCondition.h #import <Foundation/Foundation.h> @protocol WifeCondition <NSObject> ...

  5. [整理] C#调用SQLDMO.DLL时间数据库备份 / 还原。 (香神无涯) // C#实现SQLSERVER2000数据库备份还原的两种方法 (带进度条)

    /// <summary>/// 通过调用MSSQL的SQLDMO.DLL文件来实现备份数据库/// 1.首先在在项目中引用SQLDMO.DLL文件./// 2.在引用中的SQLDMO.D ...

  6. PCB MS CLR 聚合函数 joinString加排序实现

    准备着手,动态列SQL查询,要实现动态列SQL,会运用到聚合函数,而SQL本身聚合函数有限, 无满足此用户任意字段组合变化,再加上工艺流程重复相同工序,如;沉铜1,沉铜2对应工序代码都是相同的 而通常 ...

  7. Django day08 多表操作 (一) 多表模型创建

    多表模型创建分析:1)作者表:一个作者有姓名和年龄2)作者信息表: 有作者就有信息,点击作者的名字可以查询他的电话和地址, 作者表对应作者信息表,所以他们之间是一对一对的关系3)出版社表: 出版社有对 ...

  8. java input 实现调用手机相机和本地照片上传图片到服务器然后压缩

    在微信公众号里面需要上传头像,时间比较紧,调用学习jssdk并使用 来不及  就用了input 使用input:file标签, 去调用系统默认相机,摄像,录音功能,其实是有个capture属性,直接说 ...

  9. 另一种压缩图片的方法---Machine learning 之 PCA(Principle Component Analysis)

    PCA最主要的用途是用来减少特征向量的数目,N个特征向量 减小到 K个特征向量.如果为了可视化,k可以使3 或者 2.这样可以加速算法的学习速度. PCA用来压缩图像同一有效. 具体方式以及原理在gi ...

  10. java 实现yaml 数据转json与map

    首先引入snakeyaml-1.16.jar的包. 直接上代码: package com.ming.yaml; import java.util.Map; import org.yaml.snakey ...