一个基于注解的orm简单实现(二):实现思路
先来看一段常见的数据库操作代码: ```
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来初始化,依赖关系如下:

项目代码:https://github.com/hu-xuemin/xBlog.git
一个基于注解的orm简单实现(二):实现思路的更多相关文章
- 基于注解的SpringMVC简单介绍
SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是DispatcherServlet,DispatcherServlet负责转发每一个Request请 ...
- 【转】基于注解的SpirngMVC简单介绍
转载地址:http://haohaoxuexi.iteye.com/blog/1343761 SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是 Di ...
- SpringMVC学习总结(四)——基于注解的SpringMVC简单介绍
SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是 DispatcherServlet,DispatcherServlet负责转发每一个Request ...
- 【转载】基于注解的SpringMVC简单介绍
SpringMVC是一个基于DispatcherServlet的MVC框架,每一个请求最先访问的都是DispatcherServlet,DispatcherServlet负责转发每一个Request请 ...
- 实现一个基于tcc/tlink的简单的编译链接工具
一.基础研究 在这里我们需要提供一套新的c语言开发工具cc,它支持的c程序不是从main开始运行而是从CMain开始运行. 书上已经对该工具程序进行了需求分析:(1)要在屏幕中间显示彩色的字符串:(2 ...
- LineCalc,一个基于Lex&Yacc的简单行计算工具
LineCalc是基于Lex&Yacc的一个简单的行计算工具,支持常见的运算符和部分POSIX中定义于math.h中的数学函数:同时,LineCalc还提供了一个简单的错误处理模块,能检测公式 ...
- 一个基于EntityFramework Core的简单数据库访问层,适用于轻量级数据库业务
这个访问层的代码实际上是园子里某个前辈的,本人只是觉得好使,记录了下来. 本访问层需要通过Nuget安装EntityFramework Core,不过个人认为EF 6同样可以使用. 搭配数据库,最好是 ...
- 一个基于tcp的socket简单对话小例子
首先我们需要写连个py文件,一个server,一个client. import socket sk = socket.socket() # sk.bind(('ip',port)) sk.bind(( ...
- Mario是一个基于.NETCore的简单快速开发框架
Mario .NET Core简单快速开发框架 Mario是一个基于.NET Core的简单快速开发框架 GitHub:https://github.com/deeround/Mario 技术特点 基 ...
随机推荐
- android 菜单的总结
安卓菜单有三种菜单. 选项菜单: 点击系统菜单按钮会触发 上下文菜单:长按屏幕触发 子菜单:某一个菜单的下一级菜单 具体的描叙:http://blog.csdn.net/zqiang_55/artic ...
- 将MPLS编译进linux内核中
系统环境:linux kernel 2.6.35.(此环境是上一篇文章中将ubuntu内核替换后的环境) 编译过程如下: 1)首先需要下载patch文件:linux-kernel-v2.6.35-mp ...
- jQuery图片轮播的具体实现
先看一看html代码,以及对应的css代码: <div id="scrollPics"> <ul class="slider" > ...
- AC日记——统计难题 hdu 1251
统计难题 Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131070/65535 K (Java/Others)Total Submi ...
- Delphi 中的常用事件
OnActive 焦点称到窗体或控件时发生 OnClick 鼠标单击事件 OnDbClick 鼠标双击事件 OnClose和OnCloseQuery 当关闭一个窗体时就会响应OnClose和OnClo ...
- fork() && fork() || fork()
http://blog.csdn.net/hs794502825/article/details/10242091 #include <unistd.h> #include <std ...
- JS前端的分享功能
给网页加上分享代码,借助网友的力量推广网站,目前已经很流行了 以下是网页代码 QQ空间分享代码如下: <a href="javascript:void(0);" onclic ...
- MyBatis 使用foreach与其他方式的时候参数传递方式
Mapper文件: <select id="selectPersonByIds" parameterType="map" resultMap=" ...
- Flash神奇的视频利器StageVideo
在过去的几年里,视频已经成为web网页上最主流的趋势之一,这主要是由Adobe Flash Player来推动的.2007年Flash Player 9中引入了H.264和全屏支持技术,通过在web页 ...
- easyUI parser的使用
easyUI parser的使用: <!DOCTYPE html> <html lang="en"> <head> <meta chars ...