手写spring+springmvc+mybatis框架篇

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 本系列文章抽丝剥茧,源码分析百度有很多,在这里只列举几个我收录的优秀的博客推荐给大家,本文不过多的分析源码,更注重将代码理解后的实现。

我们平日开发时所用的SSM框架,可是你真的了解它吗?技术革新,换代应接不暇,只有理解了源码才能保证在技术快速更迭的时代中真正站稳脚跟。

本系列文章抽丝剥茧,源码分析百度有很多,在这里只列举几个我收录的优秀的博客推荐给大家,本文不过多的分析源码,更注重将代码理解后的实现。

话不多说,开启正题。(由于本人是小白,理解的不深所以写出来的东西比较浅显易懂。如果有错误还望各位大佬指出来,帮助我提高)

项目介绍:此项目是在手写SSM框架的基础上开发了一个简单的登录功能。

首先是spring框架的实现,spring的依赖注入特性是集成其他所有框架的基础。在spring2.5版本之前,只支持配置文件注入。在2.5之后加入了@Autowired注解,实现了注解注入。我们的这个spring框架当然是都支持啦。首先是xml版本注入。目前支持的是构造器注入和set属性注入。

实现思路:
用dom4j解析xml文件。获取各个节点的属性和内容。
用枚举定义IOC的bean的规则,用BeanFacory的getBean方法读取配置信息,如果xml读取到的属性和IOCRULES的枚举内容匹配的话,用BeanDefinitionMap对象保存,然后用反射实例化一个对象。
读取在xml文件中读取到的扫包路径,扫描此路径下的有注解的类按照自下而上的顺序存储在componentList中。
将componentList对象按照顺序实例化出对象。

一些高大上的名词,其实就是一些对象或者数据结构。比如我们常说的spring容器其实就指的是一个map对象集合,在spring源码其实就是DefaultListableBeanFactory类中如下对象

private final Map<String, BeanDefinition> beanDefinitionMap 
= new ConcurrentHashMap<String, BeanDefinition>(256);

将这些对象实例化之后就是我们说的注入。再比如springmvc中的handlerMapping,其实就是map集合,key是方法名,value是@RequestMapping的路径。再比如...还有很多,在实战之前我先提出几个我自己想到的问题,希望大家看的时候带着问题提来看:

spring有xml配置,也有注解配置,那么先实例化的是哪种版本呢?

构造器注入,属性注入如何实现的?

在指定包的路径下有那么多要注入的bean,service层调用dao,所以dao层一定要在service之前注入,然后在service层属性注入的时候才能将dao层注入,那么如何保证实例化的顺序呢?这也是难点。

我们使用的时候只需要用@Autowired标记接口,即可使用。但是我们知道接口并不能实例化对象,那么spring是如何做到的呢?而且有的接口有实现类,在mybatis中连实现类都没有,如何实现的呢?

效果图

image

下面开始实战:
项目必须用jdk1.8,不然会报错.里面一些方法只有1.8才有。
用到了lombok这个jar包,主要是简化了代码量,一个注解就可以省略getset方法。
解析xml用的dom4j,在小型xml文件的解析和操作中dom4j是最佳选择
在最后的demo中,RegisterServiceImpl,UserServiceImpl,RegisterService,UserService等文件没有真正的使用。只是考虑多层注入的时候,为了测试注入的顺序是否是对的。在dao层真正用的是UserMapper这个接口。

先介绍一下工具类吧。省的代码中出现的时候大家看不懂

package spring.Utils;

/**
* Created by Xiao Liang on 2018/6/27.
* 注解工具类:判断注解是否为空
*/
public class  AnnotationUtils {
   public  static <T>  boolean isEmpty(T t){
       return  t == null ? true : false;
   }
}

package spring.Utils;

/**
* @ClassName ConvertUtis
* @Description 根据传入的属性和类名,将属性名强转为类名的属性
* @Data 2018/7/4
* @Author xiao liang
*/
public class ConvertUtis {

   public static Object  convert(String className,String parameter){
       if (className.equals("String")){
           return parameter;
       }
       else if (className.equals("Integer")){
           return Integer.valueOf(parameter);
       }
       else if (className.equals("int")){
           return Integer.valueOf(parameter);
       }
       else if (className.equals("Float")){
           return Float.valueOf(parameter);
       }
       else if (className.equals("Double")){
           return Integer.valueOf(parameter);
       }
       else if (className.equals("Long")){
           return Long.valueOf(parameter);
       }
       else if (className.equals("Short")){
           return Short.valueOf(parameter);
       }
       else if (className.equals("Byte")){
           return Byte.valueOf(parameter);
       }
       else if (className.equals("Boolean")){
           return Boolean.valueOf(parameter);
       }
       return null;

   }
}

package spring.Utils;

/**
* @ClassName GetMethodName
* @Description 根据属性名拼接set方法
* @Data 2018/7/4
* @Author xiao liang
*/
public class GetMethodName {
   public   static  String getSetMethodNameByField(String propertyName) {
       String methodName = "set" + propertyName.substring(0, 1).toUpperCase() + propertyName.substring(1);
       return methodName;
   }
}

package spring.Utils;

/**
* @ClassName isBasicType
* @Description 判断是不是基本数据类型
* @Data 2018/7/4
* @Author xiao liang
*/
public class isBasicTypeUtils {
   public static boolean  isBasicType(String typeName){
       if (typeName.equals("String")){
           return true;
       }
       else if(typeName.equals("Integer")){
           return true;
       }
       else if(typeName.equals("int")){
           return true;
       }
       else if(typeName.equals("Long")){
           return true;
       }
       else if(typeName.equals("Short")){
           return true;
       }
       else if(typeName.equals("Float")){
           return true;
       }
       else if(typeName.equals("Double")){
           return true;
       }
       else if(typeName.equals("Byte")){
           return true;
       }
       return false;
   }
}

package spring.Utils;

import java.util.HashSet;
import java.util.List;
import java.util.Set;

/**
*  @Author xiao liang
*  @Desprition 在链表中添加数据,添加时保证只有一个相同的实例
*/
public class ListAddUtils {
   public static <T> void add(List<T> list ,T t) {
       Set<T> set1 = new HashSet<>(list);
       if (set1.add(t)){
           list.add(t);
       }
   }
}

package spring.Utils;
/**
*  @Author xiao liang
*  判断字符串是否为空
*/
public class StringUtils {
   public static boolean isEmpty(String string) {
       if ((string == null) || "".equals(string)) {
           return true;
       }
       return false;
   }
}

还有一个常量类。

package spring.constants;

import spring.xml.FileSystemXmlApplicationContext;
/**
* Created by Xiao Liang on 2018/6/27.
* @Description :保存的是各个配置文件的路径
*/
public interface Constants {
   String PATH = FileSystemXmlApplicationContext.class.getResource("/").getPath();
   String contextConfigLocation = "application.xml";
   String springmvcConfigLocation = "spring-mvc.xml";
   String mybatisConfigLocation = "MyUserMapper.xml";
}

还有三个xml文件:
application.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans>
   <bean id="hostess" class="spring.Person">
       <property name ="userName" value = "admin"></property>
       <property name ="passWord" value = "admin"></property>
   </bean>
   <bean class="spring.mybatis.JDBCUtils" id="dataSource">
       <property name="user" value="root"></property>
       <property name="password" value="qinliang"></property>
       <property name="driver" value="com.mysql.jdbc.Driver"></property>
       <property name="url" value="jdbc:mysql://localhost:3306/test"></property>
   </bean>

   <component-scan base-package="spring.demo" >
   </component-scan>

</beans>

springmvc.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans>
   <bean id="viewResolver" class="spring.springmvc.ViewResolver">
       <property name="prefix" value="/WEB-INF/views/"></property>
       <property name="suffix" value=".jsp"></property>
   </bean>
</beans>

MyUserMapper.xml

<?xml version="1.0" encoding="UTF-8"?>
<mapper namespace="spring.demo.repository.UserMapper">

   <select id="queryUser" resultType="spring.dataObject.User">
       select * from login_user where username = #{username} AND password = #{password}
   </select>

</mapper>

这是一个maven项目,pom.xml文件内容如下

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
   <modelVersion>4.0.0</modelVersion>
   <groupId>com.myspring</groupId>
   <artifactId>start</artifactId>
   <version>1.0-SNAPSHOT</version>
   <packaging>war</packaging>
   <name>start Maven Webapp</name>
   <!-- FIXME change it to the project's website -->
   <url>http://www.example.com</url>

   <properties>
       <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
       <maven.compiler.source>1.8</maven.compiler.source>
       <maven.compiler.target>1.8</maven.compiler.target>
   </properties>

   <dependencies>
       <dependency>
           <groupId>junit</groupId>
           <artifactId>junit</artifactId>
           <version>4.11</version>
           <scope>test</scope>
       </dependency>

       <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
       <dependency>
           <groupId>org.projectlombok</groupId>
           <artifactId>lombok</artifactId>
           <version>1.18.0</version>
           <scope>provided</scope>
       </dependency>
       <!-- https://mvnrepository.com/artifact/org.slf4j/slf4j-api -->
       <dependency>
           <groupId>org.slf4j</groupId>
           <artifactId>slf4j-api</artifactId>
           <version>1.7.25</version>
       </dependency>

       <dependency>
           <groupId>org.slf4j</groupId>
           <artifactId>slf4j-simple</artifactId>
           <version>1.7.25</version>
       </dependency>


       <dependency>
           <groupId>javax.servlet</groupId>
           <artifactId>servlet-api</artifactId>
           <version>2.5</version>
       </dependency>

       <dependency>
           <groupId>javax.servlet.jsp</groupId>
           <artifactId>jsp-api</artifactId>
           <version>2.1.3-b06</version>
       </dependency>

       <dependency>
           <groupId>dom4j</groupId>
           <artifactId>dom4j</artifactId>
           <version>1.6.1</version>
       </dependency>

       <!-- https://mvnrepository.com/artifact/jstl/jstl -->
       <dependency>
           <groupId>jstl</groupId>
           <artifactId>jstl</artifactId>
           <version>1.2</version>
       </dependency>
       <!--数据库连接池和数据库连接-->
       <dependency>
           <groupId>mysql</groupId>
           <artifactId>mysql-connector-java</artifactId>
           <version>5.1.8</version>
       </dependency>


   </dependencies>

   <build>
       <finalName>start</finalName>
       <pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) -->
           <plugins>
               <plugin>
                   <artifactId>maven-clean-plugin</artifactId>
                   <version>3.0.0</version>
               </plugin>
               <!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging -->
               <plugin>
                   <artifactId>maven-resources-plugin</artifactId>
                   <version>3.0.2</version>
               </plugin>
               <plugin>
                   <artifactId>maven-compiler-plugin</artifactId>
                   <version>3.7.0</version>
               </plugin>
               <plugin>
                   <artifactId>maven-surefire-plugin</artifactId>
                   <version>2.20.1</version>
               </plugin>
               <plugin>
                   <artifactId>maven-war-plugin</artifactId>
                   <version>3.2.0</version>
               </plugin>
               <plugin>
                   <artifactId>maven-install-plugin</artifactId>
                   <version>2.5.2</version>
               </plugin>
               <plugin>
                   <artifactId>maven-deploy-plugin</artifactId>
                   <version>2.8.2</version>
               </plugin>
           </plugins>
       </pluginManagement>
   </build>
</project>

下面是项目的业务逻辑代码,可以过一下,就是最简单的三层架构。业务逻辑代码中中有一些为了测试用,并没有实际的意义

package spring.demo.controller;

import spring.annotation.MyAutowired;
import spring.annotation.MyController;
import spring.annotation.MyModelAttribute;
import spring.annotation.MyRequestMapping;
import spring.dataObject.User;
import spring.demo.service.UserService;
import spring.springmvc.MyModelAndView;
import spring.springmvc.MyModelMap;
import spring.xmlRules.RequestMethod;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyController
public class LoginController {

   @MyAutowired
   private UserService userService;
   //测试用的@MyRequstParam(value = "userName") String userName,  @MyRequstParam(value = "passWord") Integer passWord
   //返回值只支持MyModelAndView,数据模型和视图模型相结合
   @MyRequestMapping(value = "/hello", method = RequestMethod.POST)
   public MyModelAndView login(@MyModelAttribute("User") User user) {
       MyModelAndView myModelAndView = new MyModelAndView("success");
       MyModelMap myModel = new MyModelMap();
       User user1 = userService.queryUser("admin", "admin");
       myModel.addAttribute("test", user1.getUserName());
       myModelAndView.setModelMap(myModel);
       return myModelAndView;
   }
   @MyRequestMapping("/hello22")
   public String test() {
       return "success";
   }
}

package spring.demo.controller;

import spring.annotation.MyAutowired;
import spring.annotation.MyController;
import spring.annotation.MyRequestMapping;
import spring.demo.service.RegisterService;
import spring.demo.service.UserService;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyController
public class RegisterController {
   @MyAutowired
   private UserService userService;
   @MyAutowired
   private RegisterService registerService;
   @MyRequestMapping("/register")
   public void regeister(){
       userService.queryUser("","");
       registerService.register();
   }
}

package spring.demo.service;

import spring.dataObject.User;
/**
* Created by Xiao Liang on 2018/6/27.
*/
public interface UserService {
    User queryUser(String userName, String passWord);
}

package spring.demo.service;

/**
* Created by Xiao Liang on 2018/6/27.
*/
public interface RegisterService {
   void register();
}

package spring.demo.service.impl;

import spring.annotation.MyAutowired;
import spring.annotation.MyService;
import spring.dataObject.User;
import spring.demo.repository.RegisterDao;
import spring.demo.repository.UserDao;
import spring.demo.repository.UserMapper;
import spring.demo.service.UserService;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyService
public class UserServiceImpl implements UserService {
   @MyAutowired
   private UserDao userDao;
   @MyAutowired
   private RegisterDao registerDao;
   @MyAutowired
   private UserMapper userMapper;
   @Override
   public User queryUser(String userName, String passWord) {
       return userMapper.queryUser(userName,passWord);
   }
}

package spring.demo.service.impl;

import spring.annotation.MyAutowired;
import spring.annotation.MyService;
import spring.demo.repository.RegisterDao;
import spring.demo.service.RegisterService;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyService
public class RegisterServiceImpl implements RegisterService{
   @MyAutowired
   private RegisterDao registerDao;
   @Override
   public void register() {
       registerDao.register();
   }
}

package spring.demo.repository;

/**
* Created by Xiao Liang on 2018/6/27.
*/
public interface UserDao {
     void test();
}

package spring.demo.repository;

/**
* Created by Xiao Liang on 2018/6/27.
*/
public interface RegisterDao {
    void register();
}

package spring.demo.repository.impl;

import spring.annotation.MyRepository;
import spring.demo.repository.UserDao;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyRepository
public class UserDaoImpl implements UserDao {
   @Override
   public void test() {
       System.out.println("我是UserDao");
   }
}

package spring.demo.repository.impl;

import spring.annotation.MyRepository;
import spring.demo.repository.RegisterDao;
/**
* Created by Xiao Liang on 2018/6/27.
*/
@MyRepository
public class RegisterDaoImpl implements RegisterDao{
   @Override
   public void register() {
       System.out.println("我是RegisterDao");
   }
}

package spring.demo.repository;

import spring.dataObject.User;
/**
* @ClassName UserMapper
* @Description
* @Data 2018/7/7
* @Author xiao liang
*/
public interface UserMapper {
   User queryUser(String userName,String passWord);
}

原文发布时间为:2018-07-17
本文作者:小亮
本文来自云栖社区合作伙伴“Java知音”,了解相关信息可以关注“Java知音

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
13天前
|
Java API 数据库
构建RESTful API已经成为现代Web开发的标准做法之一。Spring Boot框架因其简洁的配置、快速的启动特性及丰富的功能集而备受开发者青睐。
【10月更文挑战第11天】本文介绍如何使用Spring Boot构建在线图书管理系统的RESTful API。通过创建Spring Boot项目,定义`Book`实体类、`BookRepository`接口和`BookService`服务类,最后实现`BookController`控制器来处理HTTP请求,展示了从基础环境搭建到API测试的完整过程。
31 4
|
10天前
|
Java API 数据库
Spring Boot框架因其简洁的配置、快速的启动特性及丰富的功能集而备受开发者青睐
本文通过在线图书管理系统案例,详细介绍如何使用Spring Boot构建RESTful API。从项目基础环境搭建、实体类与数据访问层定义,到业务逻辑实现和控制器编写,逐步展示了Spring Boot的简洁配置和强大功能。最后,通过Postman测试API,并介绍了如何添加安全性和异常处理,确保API的稳定性和安全性。
25 0
|
5天前
|
前端开发 Java 数据库连接
Spring 框架:Java 开发者的春天
Spring 框架是一个功能强大的开源框架,主要用于简化 Java 企业级应用的开发,由被称为“Spring 之父”的 Rod Johnson 于 2002 年提出并创立,并由Pivotal团队维护。
21 1
Spring 框架:Java 开发者的春天
|
1天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
5天前
|
Java 数据库连接 开发者
Spring 框架:Java 开发者的春天
【10月更文挑战第27天】Spring 框架由 Rod Johnson 在 2002 年创建,旨在解决 Java 企业级开发中的复杂性问题。它通过控制反转(IOC)和面向切面的编程(AOP)等核心机制,提供了轻量级的容器和丰富的功能,支持 Web 开发、数据访问等领域,显著提高了开发效率和应用的可维护性。Spring 拥有强大的社区支持和丰富的生态系统,是 Java 开发不可或缺的工具。
|
11天前
|
人工智能 开发框架 Java
总计 30 万奖金,Spring AI Alibaba 应用框架挑战赛开赛
Spring AI Alibaba 应用框架挑战赛邀请广大开发者参与开源项目的共建,助力项目快速发展,掌握 AI 应用开发模式。大赛分为《支持 Spring AI Alibaba 应用可视化调试与追踪本地工具》和《基于 Flow 的 AI 编排机制设计与实现》两个赛道,总计 30 万奖金。
|
12天前
|
人工智能 Java API
阿里云开源 AI 应用开发框架:Spring AI Alibaba
近期,阿里云重磅发布了首款面向 Java 开发者的开源 AI 应用开发框架:Spring AI Alibaba(项目 Github 仓库地址:alibaba/spring-ai-alibaba),Spring AI Alibaba 项目基于 Spring AI 构建,是阿里云通义系列模型及服务在 Java AI 应用开发领域的最佳实践,提供高层次的 AI API 抽象与云原生基础设施集成方案,帮助开发者快速构建 AI 应用。本文将详细介绍 Spring AI Alibaba 的核心特性,并通过「智能机票助手」的示例直观的展示 Spring AI Alibaba 开发 AI 应用的便利性。示例源
|
7天前
|
缓存 Cloud Native 安全
探索阿里巴巴新型ORM框架:超越MybatisPlus?
【10月更文挑战第9天】在Java开发领域,Mybatis及其增强工具MybatisPlus长期占据着ORM(对象关系映射)技术的主导地位。然而,随着技术的发展,阿里巴巴集团推出了一种新型ORM框架,旨在提供更高效、更简洁的开发体验。本文将对这一新型ORM框架进行探索,分析其特性,并与MybatisPlus进行比较。
15 0
|
6月前
|
设计模式 前端开发 Java
了解 Spring MVC 架构、Dispatcher Servlet 和 JSP 文件的关键作用
Spring MVC 是 Spring 框架的一部分,是一个 Web 应用程序框架。它旨在使用 Model-View-Controller(MVC) 设计模式轻松构建Web应用程序。
99 0
|
25天前
|
存储 设计模式 前端开发
什么是SpringMVC?简单好理解!什么是应用分层?SpringMVC与应用分层的关系? 什么是三层架构?SpringMVC与三层架构的关系?
文章解释了SpringMVC的概念和各部分功能,探讨了应用分层的原因和具体实施的三层架构,以及SpringMVC与三层架构之间的关系和联系。
20 1
什么是SpringMVC?简单好理解!什么是应用分层?SpringMVC与应用分层的关系? 什么是三层架构?SpringMVC与三层架构的关系?