面试官让我现场手写MyBatis框架,我10分钟就搞定了!!(全程实战,建议收藏)

£神魔★判官ぃ 2021-09-08 01:24 449阅读 0赞

大家好,我是冰河~~

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

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

什么是ORM框架?

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

其实,从本质上讲,ORM框架主要实现的是程序对象到关系数据库数据的映射。

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

手撸ORM框架

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

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

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2wxMDI4Mzg2ODA0_size_16_color_FFFFFF_t_70_pic_center

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

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

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

注解解析类的实现

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

至此,我们的极简版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. /** * 测试自定义注解 * @author binghe * */
  7. public class AnnotationTest {
  8. @Test
  9. public void testDBAnnotation(){
  10. User testDto = new User("123", "34");
  11. User testDto1 = new User("123", "test1");
  12. User testDto2 = new User("", "test1,test2,test3,test4");
  13. String sql = AnnotationParser.assembleSqlFromObj(testDto);
  14. String sql1 = AnnotationParser.assembleSqlFromObj(testDto1);
  15. String sql2 = AnnotationParser.assembleSqlFromObj(testDto2);
  16. System.out.println(sql);
  17. System.out.println(sql1);
  18. System.out.println(sql2);
  19. }
  20. }

运行测试

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

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

写在最后

如果你想进大厂,想升职加薪,或者对自己现有的工作比较迷茫,都可以私信我交流,希望我的一些经历能够帮助到大家~~

推荐阅读:

  • 《奉劝那些刚参加工作的学弟学妹们:要想进大厂,这些核心技能是你必须要掌握的!完整学习路线!!(建议收藏)》
  • 《奉劝那些刚参加工作的学弟学妹们:这些计算机与操作系统基础知识越早知道越好!万字长文太顶了!!(建议收藏)》
  • 《我用三天时间开发了一款老少皆宜的国民级游戏,支持播放音乐,现开放完整源代码和注释(建议收藏)!!》
  • 《我是全网最硬核的高并发编程作者,CSDN最值得关注的博主,大家同意吗?(建议收藏)》
  • 《毕业五年,从月薪3000到年薪百万,我掌握了哪些核心技能?(建议收藏)》
  • 《我入侵了隔壁妹子的Wifi,发现。。。(全程实战干货,建议收藏)》
  • 《千万不要轻易尝试“熊猫烧香”,这不,我后悔了!》
  • 《清明节偷偷训练“熊猫烧香”,结果我的电脑为熊猫“献身了”!》
  • 《7.3万字肝爆Java8新特性,我不信你能看完!(建议收藏)》
  • 《在业务高峰期拔掉服务器电源是一种怎样的体验?》

好了,今天就到这儿吧,小伙伴们点赞、收藏、评论,一键三连走起呀,我是冰河,我们下期见~~

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2wxMDI4Mzg2ODA0_size_16_color_FFFFFF_t_70_pic_center 1

发表评论

表情:
评论列表 (有 0 条评论,449人围观)

还没有评论,来说两句吧...

相关阅读