最近很多小伙伴对ORM框架的实现很感兴趣,不少读者在冰河的微信上问:冰河,你知道ORM框架是如何实现的吗?比如像MyBatis和Hibernte这种ORM框架,它们是如何实现的呢?

为了能够让小伙伴们更加深刻并且清晰的理解ORM框架的实现原理,冰河决定自己手撸一个极简版的ORM框架,让小伙伴们一看就能够明白什么是ORM框架?ORM框架到底是如何运行的?ORM框架是如何将程序对象与数据库中的数据进行映射的?不过,在正式开始手撸ORM框架之前,我们要先来搞清楚什么是ORM框架。

什么是ORM框架?

ORM全称为:Object Relational Mapping,翻译成中文就是:对象关系映射。也就是说ORM框架就是对象关系映射框架,它通过元数据描述对象与关系映射的细节,ORM框架在运行的时候,可以根据对应与映射之间的关系将数据持久化到数据库中。

其实,从本质上讲,ORM框架主要实现的是程序对象到关系数据库数据的映射。说的直白点:ORM框架就是将实体和实体与实体之间的关系,转化为对应的SQL语句,通过SQL语句操作数据库,将数据持久化到数据库中,并且对数据进行相应的增删改查操作。

最常用的几种ORM框架为:MyBatis、Hibernate和JFinal。

手撸ORM框架

这里,我们模拟的是手撸Hibernate框架实现ORM,小伙伴们也可以模拟其他的ORM框架实现,核心原理都是相通的。如果大家在模拟其他框架手撸实现ORM时,遇到问题的话,都可以私聊我沟通,我看到的话,会第一时间回复大家。

好了,说干就干,我们开始吧。

@Table注解的实现

首先,我们创建一个io.mykit.annotation.jdk.db.provider Java包,在这个Java包创建一个@Table注解,@Table注解标注在Java类上,表示当前类会被映射到数据库中的哪张数据表上,如下所示。

package io.mykit.annotation.jdk.db.provider;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/*** 自定义Table注解* @author binghe**/
@Inherited
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Table {String value() default "";
}

@Column注解的实现

同样的,在io.mykit.annotation.jdk.db.provider包下创建一个@Column注解,@Column注解标注在类中的字段上,表示当前类中的字段映射到数据表中的哪个字段上,如下所示。

package io.mykit.annotation.jdk.db.provider;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;/*** 自定义Column注解* @author binghe**/
@Inherited
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Column {String value() default "";
}

看到这里,不管是使用过MyBatis的小伙伴还是使用过Hibernate的小伙伴,应该都会有所体会吧?没错,@Table注解和@Column注解,不管是在MyBatis框架还是Hibernate框架中,都会被使用到。这里,我们在收录极简版ORM框架时,也使用了这两个经典的注解。

创建实体类

io.mykit.annotation.jdk.db.provider.entity包下创建实体类User,并且@Table注解和@Column注解会被分别标注在User类上和User类中的字段上,将其映射到数据库中的数据表和数据表中的字段上,如下所示。

package io.mykit.annotation.jdk.db.provider.entity;import io.mykit.annotation.jdk.db.provider.Column;
import io.mykit.annotation.jdk.db.provider.Table;/*** 自定义使用注解的实体* @author binghe**/
@Table("t_user")
public class User implements Serializable{@Column("id")private String id;@Column("name")private String name;public User() {super();}public User(String id, String name) {super();this.id = id;this.name = name;}public String getId() {return id;}public void setId(String id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "User [id=" + id + ", name=" + name + "]";}}

注解解析类的实现

io.mykit.annotation.jdk.db.provider.parser包中创建一个AnnotationParser类,AnnotationParser 类是整个框架的核心,它负责解析标注在实体类上的注解,并且将对应的实体类及其字段信息映射到对应的数据表和字段上,如下所示。

package io.mykit.annotation.jdk.db.provider.parser;import java.lang.reflect.Field;
import java.lang.reflect.Method;import io.mykit.annotation.jdk.db.provider.Column;
import io.mykit.annotation.jdk.db.provider.Table;/*** 解析自定义注解* @author binghe**/
public class AnnotationParser {/** * 通过注解来组装查询条件,生成查询语句 * @param obj * @return */  public static String assembleSqlFromObj(Object obj) {  Table table = obj.getClass().getAnnotation(Table.class);  StringBuffer sbSql = new StringBuffer();  String tableName = table.value();  sbSql.append("select * from " + tableName + " where 1=1 ");  Field[] fileds = obj.getClass().getDeclaredFields();  for (Field f : fileds) {  String fieldName = f.getName();  String methodName = "get" + fieldName.substring(0, 1).toUpperCase()  + fieldName.substring(1);  try {  Column column = f.getAnnotation(Column.class);  if (column != null) {  Method method = obj.getClass().getMethod(methodName);  Object v = method.invoke(obj);  if (v != null) {  if (v instanceof String) {  String value = v.toString().trim();// 判断参数是不是 in 类型参数 1,2,3  if (value.contains(",")) {  //去掉value中的,String sqlParams = value.replace(",", "").trim();//value中都是纯数字if(isNum(sqlParams)){sbSql.append(" and " + column.value() + " in (" + value + ") ");  }else{String[] split = value.split(",");//将value重置为空value = "";for(int i = 0; i < split.length - 1; i++){value += "'"+split[i]+"',";}value += "'"+split[split.length - 1]+"'";sbSql.append(" and " + column.value() + " in (" + value + ") ");  }} else {  if(value != null && value.length() > 0){sbSql.append(" and " + column.value() + " like '%" + value + "%' ");  }}  } else {sbSql.append(" and " + column.value() + "=" + v.toString() + " ");  }  }  }  } catch (Exception e) {  e.printStackTrace();  }  }  return sbSql.toString();  }  /** * 检查给定的值是不是 id 类型 1.检查字段名称 2.检查字段值 *  * @param target * @return */  public static boolean isNum(String target) {  boolean isNum = false;  if (target.toLowerCase().contains("id")) {  isNum = true;  }  if (target.matches("\\d+")) {  isNum = true;  }  return isNum;  }
}

至此,我们的极简版ORM框架就实现好了,不过实现完还不行,我们还要对其进行测试验证。

测试类的实现

io.mykit.annotation.jdk.provider包下创建AnnotationTest 类,用以测试我们实现的极简ORM框架的效果,具体如下所示。

package io.mykit.annotation.jdk.provider;import org.junit.Test;import io.mykit.annotation.jdk.db.provider.entity.User;
import io.mykit.annotation.jdk.db.provider.parser.AnnotationParser;
import io.mykit.annotation.jdk.provider.parser.AnnotationProcessor;/*** 测试自定义注解* @author binghe**/
public class AnnotationTest {@Testpublic void testDBAnnotation(){User testDto = new User("123", "34");  User testDto1 = new User("123", "test1");  User testDto2 = new User("", "test1,test2,test3,test4");  String sql = AnnotationParser.assembleSqlFromObj(testDto);  String sql1 = AnnotationParser.assembleSqlFromObj(testDto1);  String sql2 = AnnotationParser.assembleSqlFromObj(testDto2);  System.out.println(sql);  System.out.println(sql1);  System.out.println(sql2);  }
}

运行测试

我们运行AnnotationTest#testDBAnnotation()方法,命令行会输出如下信息。

select * from t_user where 1=1  and id like '%123%'  and name like '%34%'
select * from t_user where 1=1  and id like '%123%'  and name like '%test1%'
select * from t_user where 1=1  and name in ('test1','test2','test3','test4')

可以看到,我们在测试程序中,并没有在测试类中传入或者执行任何SQL语句,而是直接创建User类的对象,并调用AnnotationParser#assembleSqlFromObj()进行解析,并且将对应的实体类对象转换为SQL语句返回。

其实,MyBatis和Hibernate的底层核心原理都是这样的,大家学会了吗?有不懂的地方欢迎私聊我沟通。赶紧打开你的开发环境,手撸个极简版ORM框架吧!!

10分钟手撸极简版ORM框架!相关推荐

  1. 很多小伙伴不太了解ORM框架的底层原理,这不,冰河带你10分钟手撸一个极简版ORM框架(赶快收藏吧)

    大家好,我是冰河~~ 最近很多小伙伴对ORM框架的实现很感兴趣,不少读者在冰河的微信上问:冰河,你知道ORM框架是如何实现的吗?比如像MyBatis和Hibernate这种ORM框架,它们是如何实现的 ...

  2. 来,一起手撸一个简版 Redis(附源码)

    点击上方 视学算法,选择 设为星标 优质文章,及时送达 作者 | 凯京技术团队 来自 | my.oschina.net/keking/blog/3037372 今天主要介绍两个开源项目,然后创建应用最 ...

  3. 从JDBC到手撸极简版Mybaties(3)JDBC自动解析配置文件

    简介 开门见山的说,java中有一个类叫Properties.该类主要用于读取Java的配置文件,不同的编程语言有自己所支持的配置文件,配置文件中很多变量是经常改变的,为了方便用户的配置,能让用户够脱 ...

  4. 打造手淘极简包的轻量化框架

    作者:周松(北傲) 如今app冗余与包大小一直都是很多业务挥之不去的优化项.行业内很多极简包如雨后春笋般冒出来.又或者是很多业务对极简包的探索也有积极的尝试. 极简包带来的价值,不仅是包大小上的质变, ...

  5. 《重大人生启示录》极简版

    <重大人生启示录>极简版 献给所有活着和将要死去的人们,献给所有历经悲伤的人们 本文摘录了启示录最重要的内容,不必非要看全本,不必购买书 序言 这是极为特殊的历史转折期,物质文明发展到这一 ...

  6. 7句话让Codex给我做了个小游戏,还是极简版塞尔达,一玩简直停不下来

    点击上方"视学算法",选择加"星标"或"置顶" 重磅干货,第一时间送达 梦晨 萧箫 发自 凹非寺 量子位 | 公众号 QbitAI 什么,7 ...

  7. openGauss 极简版安装

    openGauss 官网   openGauss 下载地址 支持的操作系统 ● ARM:   ● openEuler 20.03LTS(推荐采用此操作系统)   ● 麒麟V10   ● Asianux ...

  8. Android——教你10分钟手敲 Butter Knife(小刀注解)

    教你10分钟手敲 Butter Knife(小刀注解) 在用 AndroidStudio 集成 Butter Knife(小刀注解)的时候感觉像极了J2EE的Spring IOC容器 自己研究了一下, ...

  9. python入门笔记——飞机大战(极简版、未进行继承优化)

    python入门笔记--飞机大战(极简版.未进行继承优化) import random import pygame# 引用pygame里的模块 from pygame.locals import *# ...

最新文章

  1. 阿里规定超过三张表禁止JOIN,为啥呢?
  2. hadoop程序运行
  3. CentOS使用yum安装Docker
  4. 校长回应8名考生放弃清华北大:不喜欢学医才去其他
  5. 珠心算测验(洛谷-P2141)
  6. 使用Event Message 对 Package 进行Troubleshoot
  7. softmax函数_反向传播之一:softmax函数
  8. 华为鸿蒙mate,华为MatePad Pro发布亮相!华为首款鸿蒙平板全新体验!
  9. (转帖)無號數及有號數的乘加運算電路設計(Verilog)
  10. 毕设题目:Matlab语音识别
  11. mx350显卡天梯图_CPU天梯图与显卡天梯图2020年最新版
  12. matlab中plot3,mesh,grid三者画图的区别
  13. 操作必须使用一个可更新的查询
  14. Laravel核心概念:服务容器(ServiceContainer),服务提供者(Service Provider),门面(Facade),契约(Contracts)
  15. 银行账户管理(Bank Account Management)
  16. 解读《大话西游之大圣娶亲》
  17. FPGA控制TDC-GPX2时间间隔测量(三)
  18. 华为荣耀鸿蒙3.0,2.0解决此设备未获得play保护机制认证框架,安装谷歌Play商店服务框架
  19. 自行車基本知識 (zz)
  20. [转]OpenSynergy的COQOS Hypervisor SDK-实现仪表和车载信息娱乐系统共享显示

热门文章

  1. php更新点击数,始终如一的坚守,PHP对数据更新的更优解
  2. 中date转为string_股票数据获取篇(持续更新中...)
  3. 安装旧版本插件_iOS 应用降级插件,支持任意版本升降
  4. 迁移学习之DenseNet121(121层),DenseNet169(169层),DenseNet201(201层)(图像识别)
  5. 2020牛客多校第3场:[Points Construction Problem + 思维题+构造]
  6. Java切面理解_Spring AOP面向切面编程:理解篇
  7. SDOI2015 约数个数和(莫比乌斯反演经典、双上限整除分块)超详细笔记
  8. 【经验分享】如何优雅的在 Microsoft word中插入代码
  9. java nonewithrsa,如何使“MessageDigest SHA-1和Signature NONEwithRSA”等同于“Signature SHA1withRSA”...
  10. android xposed 编译,Xposed修改特征编译