先来看一段常见的数据库操作代码:

```
protected User getDataFromDatabase(long id){
String sql = "select firstname from user where id=?";//1
Connection conn = DBConnectionFactory.getConnection();
PreparedStatement stat;
User user;//2
try {
stat = conn.prepareStatement(sql);
stat.setObject(1, id);
ResultSet rs = stat.executeQuery();
user.setFirstName(rs.getString("firstname"));//3
stat.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return user;
}
``` 在上面代码中存在三个可变部分,如果需要把数据库操作代码修改为任何对象都能使用的同一段代码,则需要提取出这三个可变部分。所以我们修改一下上面代码。
1、首先提取一下sql语句部分。如果我们可以动态生成该sql语句,数据库对象映射器应该就完成了一半。所以我们先来拆解sql语句,看看下面这段代码:
```
String selectList="firstname";
String whereClause="id=?";
String tableName = "user";
String sql = "select " + selectList +" from " + tableName +" where " + whereClause;
protected User getDataFromDatabase(String sql,long id){
//String sql = "select firstname from user where id=?";//1
Connection conn = DBConnectionFactory.getConnection();
PreparedStatement stat;
User user;//2
try {
stat = conn.prepareStatement(sql);
stat.setObject(1, id);
ResultSet rs = stat.executeQuery();
user.setFirstName(rs.getString("firstname"));//3
stat.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return user;
}
```
拆解了第一部分,现在看起来好像也感觉差别不大,仅仅是把sql当作参数传递进去。关于第二第三可变部分,我们再来看看下面这段代码: ```
String selectList="firstname";
String whereClause="id=?";
String tableName = "user";
String sql = "select " + selectList +" from " + tableName +" where " + whereClause;
Class klass = User.class;
String fieldName = "firstName";
String columnName = "firstname";
Field field = klass.getDeclaredField(fieldName);
protected Object getDataFromDatabase(String sql,Object id){
//String sql = "select id,firstname from user where id=?";//1
Connection conn = DBConnectionFactory.getConnection();
PreparedStatement stat;
Object user;//2
try {
stat = conn.prepareStatement(sql);
stat.setObject(1, id);
ResultSet rs = stat.executeQuery();
user = klass.newInstance();
field.set(user,rs.getObject(columnName));
//user.setFirstName(rs.getString("firstname"));//3
stat.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return user;
}
```
现在再重新看函数体内,与特定user对象的信息,好像没有了,当然这里为了简化问题,user对象只有一个firstname,后面会看到有多个变量时,仅仅是field.set()那里变成了一个for循环。然后我们再来优化一下代码,毕竟所有代码堆在一个地方乱七八糟,非常不容易理解。而且如果代码仅仅是这样的话,也没有任何的可重用性。我们先把函数体提取到BaseMapper.class。把其他可变的部分,提取到TableMap.class和OneToOneColumnMap.class。TableMap存储对象与数据库表映射信息,OneToOneColumnMap存储对象field与表列的映射信息。每一个对象都会有一个BaseMapper实例,存储各自的对象关系映射信息。代码如下:
```
class BassMapper{
protected TableMap<T> tableMap;
public Object getDataFromDatabase(String sql,Object... param){
Connection conn = DBConnectionFactory.getConnection();
PreparedStatement stat;
Object result = null;
try {
stat = conn.prepareStatement(sql);
for(int i = 0;i < param.length;i++){
stat.setObject(i + 1, param[i]);
}
ResultSet rs = stat.executeQuery();
if(rs.next()){
result = (T) tableMap.getKlass().newInstance();
for(Iterator<OneToOneColumnMap> it = tableMap.getOneToOneColumns();it.hasNext();){
OneToOneColumnMap columnMap = it.next();
Object columnValue = rs.getObject(columnMap.getColumnName());
columnMap.setField(result, columnValue);
}
}
stat.close();
} catch (SQLException | InstantiationException | IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return result;
}
}
``` ```
class OneToOneColumnMap{
private String columnName;
private String fieldName;
protected Field field;
private TableMap dataMap; public OneToOneColumnMap(String columnName,String fieldName,TableMap dataMap){
this.columnName = columnName;
this.fieldName = fieldName;
this.dataMap = dataMap;
initField();
} public String getColumnName(){
return this.columnName;
} public String getFieldName(){
return this.fieldName;
} public Object getValue(Object subject){
try {
return field.get(subject);
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null;
} public void setField(Object result,Object columnValue){
try {
field.set(result, columnValue);
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} protected void initField(){
try {
this.field = dataMap.getKlass().getDeclaredField(getFieldName());
field.setAccessible(true);
} catch (NoSuchFieldException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SecurityException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
``` ```
class TableMap{
private Class<T> domainClass;
private String tableName;
private OneToOneColumnMap primaryKeyColumn;
private List<OneToOneColumnMap> oneToOneColumnMaps = new ArrayList<OneToOneColumnMap>(); public TableMap(String tableName,Class<T> domainClass){
this.domainClass = domainClass;
this.tableName = tableName;
}
public void addOneToOneColumn(String columnName,String fieldName){
OneToOneColumnMap columnMap = new OneToOneColumnMap(columnName,fieldName,this);
if(!oneToOneColumnMaps.contains(columnMap)){
if(primaryKeyColumn == null){
primaryKeyColumn = columnMap;
}
oneToOneColumnMaps.add(columnMap);
}
} public void setPrimaryKeyColumn(String columnName, String fieldName){
this.primaryKeyColumn = new OneToOneColumnMap(columnName,fieldName,this);
if(!oneToOneColumnMaps.contains(primaryKeyColumn)){
oneToOneColumnMaps.add(primaryKeyColumn);
}
} public String primaryKeyWhereClause(){
return primaryKeyColumn.getColumnName() + " = ? ";
} public Object primaryKeyColumnName(){
return primaryKeyColumn.getColumnName();
} public Object primaryKeyValue(Object domainObject){
return primaryKeyColumn.getValue(domainObject);
} public String getTableName(){
return this.tableName;
} public String insertList(){
StringBuffer result = new StringBuffer("?");
for(int i = 0;i < oneToOneColumnMaps.size() - 1;i++){
result.append(",");
result.append("?");
}
return result.toString();
} public String columnList(){
StringBuffer result = new StringBuffer(" ");
for(Iterator<OneToOneColumnMap> it = getOneToOneColumns();it.hasNext();){
OneToOneColumnMap columnMap = it.next();
result.append(columnMap.getColumnName());
result.append(",");
}
result.setLength(result.length() - 1);
return result.toString();
} public String updateList(){
StringBuffer result = new StringBuffer(" SET ");
for(Iterator<OneToOneColumnMap> it = getOneToOneColumns();it.hasNext();){
OneToOneColumnMap column = it.next();
result.append(column.getColumnName());
result.append("=?,");
}
result.setLength(result.length() - 1);
return result.toString();
} public String getColumnForField(String fieldName){
for(Iterator<OneToOneColumnMap> it = getOneToOneColumns();it.hasNext();){
OneToOneColumnMap columnMap = it.next();
if(columnMap.getFieldName().equals(fieldName)){
return columnMap.getColumnName();
}
}
return null;
}
}
```
BaseMapper通过一个TableMap来初始化,依赖关系如下:
![这里写图片描述](http://img.blog.csdn.net/20170211104536444?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvaHV4dWVtaW4=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast)
项目代码:https://github.com/hu-xuemin/xBlog.git

一个基于注解的orm简单实现(二):实现思路的更多相关文章

  1. 基于注解的SpringMVC简单介绍

    SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是DispatcherServlet,DispatcherServlet负责转发每一个Request请 ...

  2. 【转】基于注解的SpirngMVC简单介绍

    转载地址:http://haohaoxuexi.iteye.com/blog/1343761 SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是 Di ...

  3. SpringMVC学习总结(四)——基于注解的SpringMVC简单介绍

    SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是 DispatcherServlet,DispatcherServlet负责转发每一个Request ...

  4. 【转载】基于注解的SpringMVC简单介绍

    SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是DispatcherServlet,DispatcherServlet负责转发每一个Request请 ...

  5. 实现一个基于tcc/tlink的简单的编译链接工具

    一.基础研究 在这里我们需要提供一套新的c语言开发工具cc,它支持的c程序不是从main开始运行而是从CMain开始运行. 书上已经对该工具程序进行了需求分析:(1)要在屏幕中间显示彩色的字符串:(2 ...

  6. LineCalc,一个基于Lex&Yacc的简单行计算工具

    LineCalc是基于Lex&Yacc的一个简单的行计算工具,支持常见的运算符和部分POSIX中定义于math.h中的数学函数:同时,LineCalc还提供了一个简单的错误处理模块,能检测公式 ...

  7. 一个基于EntityFramework Core的简单数据库访问层,适用于轻量级数据库业务

    这个访问层的代码实际上是园子里某个前辈的,本人只是觉得好使,记录了下来. 本访问层需要通过Nuget安装EntityFramework Core,不过个人认为EF 6同样可以使用. 搭配数据库,最好是 ...

  8. 一个基于tcp的socket简单对话小例子

    首先我们需要写连个py文件,一个server,一个client. import socket sk = socket.socket() # sk.bind(('ip',port)) sk.bind(( ...

  9. Mario是一个基于.NETCore的简单快速开发框架

    Mario .NET Core简单快速开发框架 Mario是一个基于.NET Core的简单快速开发框架 GitHub:https://github.com/deeround/Mario 技术特点 基 ...

随机推荐

  1. OmniGraffle-新手指南

    OmniGraffle是一款不错的可视化软件,通过它你可以把你想要展现的数据简介.直观的展现在图.表上,这是我在数据可视化工具这篇随笔中说过的功能.但是当我真正用它时,我发现它可以做的事情还有很多. ...

  2. bzoj3809

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3809 题目大意: Autumn和Bakser又在研究Gty的妹子序列了!但他们遇到了一个难题 ...

  3. mysql配置主从数据库

    1.目的 1.1 实现数据备份 1.2 项目访问时可以实现读写分离,提高访问和操作数据的速度<读写分离好处> 2.背景 这次的主从配置主要实现主库数据的改变可以实现同步到从库中: 此次试验 ...

  4. oracle语句总结(一)

    1,建外键 新建两个表来实现外键的关联. create table main_tab ( id number, name varchar2(30) ); create table sub_tab ( ...

  5. 推荐系统中的Graph Model

    转自:http://www.cnblogs.com/wentingtu/archive/2012/05/28/2521166.html 推荐中对graph model的研究主要有两个方面,一个是如何构 ...

  6. 判断div是否有滚动条

    判断div是否有滚动条 var obj=document.getElementById("showDiv"); if(obj.scrollHeight>obj.clientH ...

  7. 如何给js动态创建的dom添加事件

    delegate() 方法 实例 当点击鼠标时,隐藏或显示 p 元素: $("div").delegate("button","click" ...

  8. Linux安装配置VPN服务器

    一.实验简介 VPN ,中文翻译为虚拟专有网络,英文全称是 Virtual Private Network .现在 VPN 被普遍定义为通过 一个公用互联网络建立一个临时的.安全的连接,是一条穿过混乱 ...

  9. Spring集合配置

    <?xml version="1.0" encoding="UTF-8"?> <beans xmlns:xsi="http://ww ...

  10. Linux笔记(九) - 软件包管理

    (1)软件包简介1.源码包   脚本安装包2.二进制包(rpm,系统默认包) (2)rpm包管理安装-i 安装 -v 显示详细信息 -h 显示进度--nodeps 不检测依赖性-U 升级例:安装: r ...