什么是ORM框架?

ORM全称为:Object Relational Mapping,翻译成中文就是:对象关系映射。也就是说ORM框架就是对象关系映射框架,它通过元数据描述对象与关系映射的细节,ORM框架在运行的时候,可以根据对应与映射之间的关系将数据持久化到数据库中。
其实,从本质上讲,ORM框架主要实现的是程序对象到关系数据库数据的映射。说的直白点:ORM框架就是将实体和实体与实体之间的关系,转化为对应的SQL语句,通过SQL语句操作数据库,将数据持久化到数据库中,并且对数据进行相应的增删改查操作。
最常用的几种ORM框架为:MyBatis、Hibernate和JFinal。

手撸ORM框架

@Table注解的实现

首先,我们创建一个io.mykit.annotation.jdk.db.providerJava包,在这个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 "";
  19. }

@Column注解的实现

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

  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. * 自定义Column注解
  10. * @author binghe
  11. *
  12. */
  13. @Inherited
  14. @Target({ElementType.FIELD})
  15. @Retention(RetentionPolicy.RUNTIME)
  16. @Documented
  17. public @interface Column {
  18. String value() default "";
  19. }

看到这里,不管是使用过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. import io.mykit.annotation.jdk.db.provider.Column;
  3. import io.mykit.annotation.jdk.db.provider.Table;
  4. /**
  5. * 自定义使用注解的实体
  6. * @author binghe
  7. *
  8. */
  9. @Table("t_user")
  10. public class User implements Serializable{
  11. @Column("id")
  12. private String id;
  13. @Column("name")
  14. private String name;
  15. public User() {
  16. super();
  17. }
  18. public User(String id, String name) {
  19. super();
  20. this.id = id;
  21. this.name = name;
  22. }
  23. public String getId() {
  24. return id;
  25. }
  26. public void setId(String id) {
  27. this.id = id;
  28. }
  29. public String getName() {
  30. return name;
  31. }
  32. public void setName(String name) {
  33. this.name = name;
  34. }
  35. @Override
  36. public String toString() {
  37. return "User [id=" + id + ", name=" + name + "]";
  38. }
  39. }

注解解析类的实现

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

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

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

测试类的实现

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

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

我们运行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语句返回。