10分钟手撸极简版ORM框架!

开发 架构
ORM框架到底是如何运行的?ORM框架是如何将程序对象与数据库中的数据进行映射的?不过,在正式开始手撸ORM框架之前,我们要先来搞清楚什么是ORM框架。

[[402491]]

大家好,我是冰河~~

最近很多小伙伴对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类上,表示当前类会被映射到数据库中的哪张数据表上,如下所示。

  1. package io.mykit.annotation.jdk.db.provider; 
  2. import java.lang.annotation.Documented; 
  3. import java.lang.annotation.ElementType; 
  4. import java.lang.annotation.Inherited; 
  5. import java.lang.annotation.Retention; 
  6. import java.lang.annotation.RetentionPolicy; 
  7. import java.lang.annotation.Target; 
  8. /** 
  9.  * 自定义Table注解 
  10.  * @author binghe 
  11.  * 
  12.  */ 
  13. @Inherited 
  14. @Target({ElementType.TYPE}) 
  15. @Retention(RetentionPolicy.RUNTIME) 
  16. @Documented 
  17. public @interface Table { 
  18.  String value() default ""

@Column注解的实现

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

  1. package io.mykit.annotation.jdk.db.provider; 
  2.   
  3. import java.lang.annotation.Documented; 
  4. import java.lang.annotation.ElementType; 
  5. import java.lang.annotation.Inherited; 
  6. import java.lang.annotation.Retention; 
  7. import java.lang.annotation.RetentionPolicy; 
  8. import java.lang.annotation.Target; 
  9.   
  10. /** 
  11.  * 自定义Column注解 
  12.  * @author binghe 
  13.  * 
  14.  */ 
  15. @Inherited 
  16. @Target({ElementType.FIELD}) 
  17. @Retention(RetentionPolicy.RUNTIME) 
  18. @Documented 
  19. public @interface Column { 
  20.  String value() default ""

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

创建实体类

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

  1. package io.mykit.annotation.jdk.db.provider.entity; 
  2.   
  3. import io.mykit.annotation.jdk.db.provider.Column
  4. import io.mykit.annotation.jdk.db.provider.Table
  5.   
  6. /** 
  7.  * 自定义使用注解的实体 
  8.  * @author binghe 
  9.  * 
  10.  */ 
  11. @Table("t_user"
  12. public class User implements Serializable
  13.   
  14.  @Column("id"
  15.  private String id; 
  16.   
  17.  @Column("name"
  18.  private String name
  19.   
  20.  public User() { 
  21.   super(); 
  22.  } 
  23.   
  24.  public User(String id, String name) { 
  25.   super(); 
  26.   this.id = id; 
  27.   this.name = name
  28.  } 
  29.   
  30.  public String getId() { 
  31.   return id; 
  32.  } 
  33.   
  34.  public void setId(String id) { 
  35.   this.id = id; 
  36.  } 
  37.   
  38.  public String getName() { 
  39.   return name
  40.  } 
  41.   
  42.  public void setName(String name) { 
  43.   this.name = name
  44.  } 
  45.   
  46.  @Override 
  47.  public String toString() { 
  48.   return "User [id=" + id + ", name=" + name + "]"
  49.  } 
  50.   

注解解析类的实现

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

  1. package io.mykit.annotation.jdk.db.provider.parser; 
  2.   
  3. import java.lang.reflect.Field; 
  4. import java.lang.reflect.Method; 
  5.   
  6. import io.mykit.annotation.jdk.db.provider.Column
  7. import io.mykit.annotation.jdk.db.provider.Table
  8.   
  9. /** 
  10.  * 解析自定义注解 
  11.  * @author binghe 
  12.  * 
  13.  */ 
  14. public class AnnotationParser { 
  15.  /**  
  16.      * 通过注解来组装查询条件,生成查询语句  
  17.      * @param obj  
  18.      * @return  
  19.      */   
  20.     public static String assembleSqlFromObj(Object obj) {   
  21.         Table table = obj.getClass().getAnnotation(Table.class);   
  22.         StringBuffer sbSql = new StringBuffer();   
  23.         String tableName = table.value();   
  24.         sbSql.append("select * from " + tableName + " where 1=1 ");   
  25.         Field[] fileds = obj.getClass().getDeclaredFields();   
  26.         for (Field f : fileds) {   
  27.             String fieldName = f.getName();   
  28.             String methodName = "get" + fieldName.substring(0, 1).toUpperCase()   
  29.                     + fieldName.substring(1);   
  30.             try {   
  31.                 Column column = f.getAnnotation(Column.class);   
  32.                 if (column != null) {   
  33.                     Method method = obj.getClass().getMethod(methodName);   
  34.                     Object v = method.invoke(obj);   
  35.                     if (v != null) {   
  36.                         if (v instanceof String) {   
  37.                          String value = v.toString().trim(); 
  38.                             // 判断参数是不是 in 类型参数 1,2,3   
  39.                             if (value.contains(",")) {   
  40.                              //去掉value中的, 
  41.                              String sqlParams = value.replace(",""").trim(); 
  42.                              //value中都是纯数字 
  43.                              if(isNum(sqlParams)){ 
  44.                               sbSql.append(" and " + column.value() + " in (" + value + ") ");   
  45.                              }else
  46.                               String[] split = value.split(","); 
  47.                               //将value重置为空 
  48.                               value = ""
  49.                               for(int i = 0; i < split.length - 1; i++){ 
  50.                                value += "'"+split[i]+"',"
  51.                               } 
  52.                               value += "'"+split[split.length - 1]+"'"
  53.                               sbSql.append(" and " + column.value() + " in (" + value + ") ");   
  54.                              } 
  55.                             } else {   
  56.                              if(value != null && value.length() > 0){ 
  57.                               sbSql.append(" and " + column.value() + " like '%" + value + "%' ");   
  58.                              } 
  59.                             }   
  60.                         } else { 
  61.                             sbSql.append(" and " + column.value() + "=" + v.toString() + " ");   
  62.                         }   
  63.                     }   
  64.                 }   
  65.             } catch (Exception e) {   
  66.                 e.printStackTrace();   
  67.             }   
  68.         }   
  69.         return sbSql.toString();   
  70.     }   
  71.    
  72.     /**  
  73.      * 检查给定的值是不是 id 类型 1.检查字段名称 2.检查字段值  
  74.      *   
  75.      * @param target  
  76.      * @return  
  77.      */   
  78.     public static boolean isNum(String target) {   
  79.         boolean isNum = false;   
  80.         if (target.toLowerCase().contains("id")) {   
  81.             isNum = true;   
  82.         }   
  83.         if (target.matches("\\d+")) {   
  84.             isNum = true;   
  85.         }   
  86.         return isNum;   
  87.     }   

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

测试类的实现

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

  1. package io.mykit.annotation.jdk.provider; 
  2.   
  3. import org.junit.Test; 
  4.   
  5. import io.mykit.annotation.jdk.db.provider.entity.User
  6. import io.mykit.annotation.jdk.db.provider.parser.AnnotationParser; 
  7. import io.mykit.annotation.jdk.provider.parser.AnnotationProcessor; 
  8.   
  9. /** 
  10.  * 测试自定义注解 
  11.  * @author binghe 
  12.  * 
  13.  */ 
  14. public class AnnotationTest { 
  15.   
  16.  @Test 
  17.  public void testDBAnnotation(){ 
  18.   User testDto = new User("123""34");   
  19.   User testDto1 = new User("123""test1");   
  20.   User testDto2 = new User("""test1,test2,test3,test4");   
  21.         String sql = AnnotationParser.assembleSqlFromObj(testDto);   
  22.         String sql1 = AnnotationParser.assembleSqlFromObj(testDto1);   
  23.         String sql2 = AnnotationParser.assembleSqlFromObj(testDto2);   
  24.         System.out.println(sql);   
  25.         System.out.println(sql1);   
  26.         System.out.println(sql2);   
  27.  } 

运行测试

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

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

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

本文转载自微信公众号「冰河技术」,可以通过以下二维码关注。转载本文请联系冰河技术公众号。

 

责任编辑:武晓燕 来源: 冰河技术
相关推荐

2021-10-12 09:24:02

Java线程池源码

2020-08-17 10:50:29

Python代码get

2019-08-15 07:13:54

负载平衡服务器迁移IIS

2020-02-17 13:45:27

抓取代码工具

2014-03-25 09:50:00

解释器编程语言

2021-07-15 06:43:11

Bash调试脚本

2013-09-13 14:08:01

2021-04-23 09:50:41

topLinux命令

2014-08-08 09:30:04

android scrollview

2020-10-13 18:22:58

DevOps工具开发

2021-08-01 21:38:07

网页点灯网关

2018-11-28 11:20:53

Python函数式编程编程语言

2022-08-29 15:02:53

远程桌面

2020-10-29 08:28:42

Java NIO异步非阻塞

2020-12-07 11:23:32

Scrapy爬虫Python

2023-06-03 21:14:47

Linux

2021-05-28 07:38:20

内存溢出场景

2021-05-17 20:13:50

数仓操作型数据库

2019-07-18 16:32:06

Python函数数据

2015-09-06 09:22:24

框架搭建快速高效app
点赞
收藏

51CTO技术栈公众号