Springboot整合Mybatis

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: Springboot整合Mybatis

MyBatis简单介绍


MyBatis是一个Java持久化框架,它通过XML描述符或注解把对象与存储过程或SQL语句关联起来。大部分都是比较灵活的,我们可以进行多表写比较复杂的查询sql,比较灵活。hibernate相对于来说比较固定,实体和表进行一一对应,比较固定。对于业务比较复杂的,选择MyBatis框架比较合适。

建表脚本

我们新建一个订单表来测试。

CREATE TABLE test.`order` (

order_id INT auto_increment NOT NULL primary key COMMENT '订单ID',

order_no INT NULL COMMENT '订单号',

product_id int NULL COMMENT '产品id',

user_id INT NULL COMMENT '用户ID',

   order_num int comment '订单产品数量',

   order_amt double comment '订单金额',

order_status varchar(10) NULL COMMENT '订单状态',

pay_status varchar(10) NULL COMMENT '支付状态',

create_time DATETIME NULL COMMENT '创建时间'  default current_timestamp,

update_time DATETIME NULL COMMENT '更新时间' default current_timestamp,

create_user varchar(10) NULL COMMENT '创建人',

update_user varchar(10) NULL COMMENT '更新人'

)

ENGINE=InnoDB

DEFAULT CHARSET=utf8mb4

COLLATE=utf8mb4_0900_ai_ci

COMMENT='订单表';

需要的话,可以自己插入部分数据。

INSERT INTO test.`order`

(order_id, order_no, product_id, user_id, order_num, order_amt, order_status, pay_status, create_time, update_time, create_user, update_user)

VALUES(1, 1, 1, 1, 2, 0.00, '已发货', '支付完成', '2022-08-21 08:32:40', '2022-08-21 10:48:44', 'user1', 'user2');

INSERT INTO test.`order`

(order_id, order_no, product_id, user_id, order_num, order_amt, order_status, pay_status, create_time, update_time, create_user, update_user)

VALUES(3, 2, 2, 1, 2, 20.00, '取消下单', '未支付', '2022-08-21 11:20:29', '2022-08-21 11:20:29', 'annotation', 'annotation');

INSERT INTO test.`order`

(order_id, order_no, product_id, user_id, order_num, order_amt, order_status, pay_status, create_time, update_time, create_user, update_user)

VALUES(4, 4, 1, 1, 2, 20.00, '下单', '支付', '2022-08-21 11:25:09', '2022-08-21 11:25:09', 'annotation', 'annotation');

引入依赖

主要引入mybatis,mysql驱动以及druid数据源。

 <!--mybatis-->

   <dependency>

     <groupId>org.mybatis.spring.boot</groupId>

     <artifactId>mybatis-spring-boot-starter</artifactId>

     <version>2.1.4</version>

   </dependency>

   <!--mysql驱动-->

   <dependency>

     <groupId>mysql</groupId>

     <artifactId>mysql-connector-java</artifactId>

     <version>5.1.47</version>

   </dependency>

<!--druid数据源-->

   <dependency>

     <groupId>com.alibaba</groupId>

     <artifactId>druid</artifactId>

     <version>1.2.8</version>

   </dependency>

druid数据源

druid阿里的,可以方便我们进行监控。

配置类:

package com.elite.springboot.config;

import com.alibaba.druid.pool.DruidDataSource;

import com.alibaba.druid.support.http.StatViewServlet;

import com.alibaba.druid.support.http.WebStatFilter;

import org.springframework.boot.context.properties.ConfigurationProperties;

import org.springframework.boot.web.servlet.FilterRegistrationBean;

import org.springframework.boot.web.servlet.ServletRegistrationBean;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

import javax.sql.DataSource;

import java.util.Arrays;

import java.util.HashMap;

import java.util.Map;

//导入druid数据源

@Configuration

public class DruidConfig {

   @ConfigurationProperties(prefix = "spring.datasource")

   @Bean

   public DataSource druid(){

       return new DruidDataSource();

   }

   //配置Druid的监控

   //1、配置一个管理后台的Servlet

   @Bean

   public ServletRegistrationBean statViewServlet(){

       ServletRegistrationBean bean = new ServletRegistrationBean(new StatViewServlet(),

               "/druid/*");

       Map<String,String> initParams = new HashMap<>();

       initParams.put("loginUsername","admin");

       initParams.put("loginPassword","123456");

       initParams.put("allow","");//默认就是允许所有访问

       //initParams.put("deny","192.168.15.21");

       bean.setInitParameters(initParams);

       return bean;

   }

   //2、配置一个web监控的filter

   @Bean

   public FilterRegistrationBean webStatFilter(){

       FilterRegistrationBean bean = new FilterRegistrationBean();

       bean.setFilter(new WebStatFilter());

       Map<String,String> initParams = new HashMap<>();

       initParams.put("exclusions","*.js,*.css,/druid/*");

       bean.setInitParameters(initParams);

       bean.setUrlPatterns(Arrays.asList("/*"));

       return bean;

   }

}

yml配置

server:

 port: 8013

spring:

 application:

   name: SpringbootMybatis

   #配置数据库

 datasource:

   type: com.alibaba.druid.pool.DruidDataSource

   url: jdbc:mysql://数据库ip:3306/test

   driver-class-name: com.mysql.jdbc.Driver

   username: root

   password: 123456

   druid:

     # 初始化大小,最小,最大

     initialSize: 5

     minIdle: 5

     maxActive: 20

     # 配置获取连接等待超时的时间(毫秒)

     maxWait: 60000

     # 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒

     timeBetweenEvictionRunsMillis: 60000

     # 配置有一个连接在连接池中的最小生存时间,单位是毫秒

     minEvictableIdleTimeMillis: 300000

     validationQuery: SELECT 1 FROM DUAL

     testWhileIdle: true

     testOnBorrow: false

     testOnReturn: false

     # 打开PSCache,指定每个连接上PSCache的大小

     poolPreparedStatements: true

     maxPoolPreparedStatementPerConnectionSize: 20

     # 配置监控统计拦截的filters,去掉后监控界面sql将无法统计,'wall'用于防火墙

     filters: stat, wall, log4j

     # 通过connectProperties属性来打开mergeSql功能,慢SQL记录

     connectionProperties: druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000

#mybatis配置

mybatis:

 #config‐location: classpath:mybatis‐config.xml #指定全局配置文件的位置

 mapper‐locations: classpath:mapper/*.xml  #指定sql映射文件的位置

实体类Order

package com.elite.springboot.entity;

import lombok.Data;

import lombok.Getter;

import lombok.Setter;

import java.math.BigDecimal;

import java.sql.Date;

/**

* 订单表

*/

@Data

@Getter

@Setter

public class Order {

   //订单ID

   private Integer order_id;

   //订单号

   private Integer order_no;

   //商品编号

   private Integer product_id;

   //用户

   private Integer user_id;

   //'订单产品数量'

   private Integer order_num ;

   //'订单金额'

   private BigDecimal order_amt ;

   //订单状态 0待确认,1已确认,2已收货,3已取消,4已完成,5已作废

   private String order_status;

   //支付状态 0待支付,1已支付,2部分支付,3已退款,4拒绝退款

   private String pay_status;

   //创建时间

   private Date create_time;

   //更新时间

   private Date update_time;

   //创建用户

   private String create_user;

   //更新用户

   private String update_user;

}

MyBatis 注解和xml方式

XML配置方式

OrderMapper.xml

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

<!DOCTYPE mapper

       PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"

       "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="com.elite.springboot.mapper.OrderMapper">

   <!--查询订单-->

   <select id="getOneOrder" parameterType="Integer" resultType="com.elite.springboot.entity.Order">

       SELECT order_id,

              order_no,

              product_id,

              user_id,

              order_num,

              order_amt,

              order_status,

              pay_status,

              create_time,

              update_time,

              create_user,

              update_user

       from `order` where order_id=#{order_id};

   </select>

   <!--查询订单列表-->

   <select id="getAllOrder" resultType="com.elite.springboot.entity.Order" >

       SELECT order_id,

              order_no,

              product_id,

              user_id,

              order_num,

              order_amt,

              order_status,

              pay_status,

              create_time,

              update_time,

              create_user,

              update_user

       from `order`

   </select>

   <!--更新订单-->

   <update id="updateOrder" parameterType="com.elite.springboot.entity.Order">

      update `order`

       <set>

           <if test="order_no != null and order_no != ''">order_no = #{order_no},</if>

           <if test="product_id != null and product_id != ''">product_id = #{product_id},</if>

           <if test="user_id != null and user_id != ''">user_id = #{user_id},</if>

           <if test="order_num != null and order_num != ''">order_num = #{order_num},</if>

           <if test="order_amt != null and order_amt != ''">order_amt = #{order_amt},</if>

           <if test="order_status != null">order_status = #{order_status},</if>

           <if test="pay_status != null">pay_status = #{pay_status},</if>

           <if test="create_user != null and create_user != ''">create_user = #{create_user},</if>

           <if test="update_user != null and update_user != ''">update_user = #{update_user},</if>

           update_time =CURRENT_TIMESTAMP

       </set>

       where order_id = #{order_id}

   </update>

   <!--删除订单-->

   <delete id="delOrder" parameterType="Integer">

       DELETE FROM `order`

       WHERE order_id= #{order_id};

   </delete>

   <!--保存订单数据-->

   <insert id="saveOrder" parameterType="com.elite.springboot.entity.Order">

       INSERT INTO `order`(`order_no`,

        `product_id`,

        `user_id`,

        `order_num`,

        `order_amt`,

        `order_status`,

        `pay_status`,

        `create_user`,

        `update_user`)

       VALUES( #{order_no},

               #{product_id},

               #{user_id},

               #{order_num},

               #{order_amt},

               #{order_status},

               #{pay_status},

               #{create_user},

               #{update_user})

   </insert>

</mapper>

OrderMapper.java接口类

package com.elite.springboot.mapper;

import com.elite.springboot.entity.Order;

import org.apache.ibatis.annotations.Mapper;

import java.util.List;

/**

* 科室映射类

*/

//@Mapper

public interface OrderMapper {

   //保存数据

   void saveOrder(Order order);

   //更新订单

   void updateOrder(Order order);

   //删除订单

   void delOrder(Integer order_id);

   //获取订单

   Order getOneOrder(Integer order_id);

   //获取所有的订单

   List<Order> getAllOrder();

}

xml测试

package com.elite.springboot;

import com.elite.springboot.entity.Order;

import com.elite.springboot.mapper.OrderMapper;

import lombok.extern.slf4j.Slf4j;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.test.context.junit4.SpringRunner;

import javax.annotation.Resource;

import java.math.BigDecimal;

import java.util.List;

/**

* Unit test for simple App.

*/

@Slf4j

@RunWith(SpringRunner.class)

@SpringBootTest

public class AppTest {

   @Resource

   OrderMapper orderMapper;

   //保存数据

   @Test

   public void saveOrder(){

       Order order = new Order();

       order.setOrder_no(1);

       order.setProduct_id(1);

       order.setUser_id(1);

       order.setOrder_num(2);

       order.setOrder_amt(BigDecimal.valueOf(0));

       order.setOrder_status("下单");

       order.setPay_status("未支付");

       order.setCreate_user("user1");

       order.setUpdate_user("user2");

       orderMapper.saveOrder(order);

   }

   //更新订单

   @Test

   public void updateOrder(){

       Order order = new Order();

       order.setOrder_id(1);

       order.setOrder_status("已发货");

       order.setPay_status("支付完成");

       orderMapper.updateOrder(order);

   }

   //删除订单

   @Test

   public void delOrder(){

       orderMapper.delOrder(2);

   }

   //获取订单

   @Test

   public void getOneOrder(){

       Order oneOrder = orderMapper.getOneOrder(1);

       System.out.println(oneOrder);

       //Order(order_id=1, order_no=1, product_id=1, user_id=1, order_num=2,

       // order_amt=0.00, order_status=已发货, pay_status=支付完成, create_time=2022-08-21, update_time=2022-08-21, create_user=user1, update_user=user2)

   }

   //获取所有的订单

   @Test

   public void getAllOrder(){

       List<Order> orderList = orderMapper.getAllOrder();

       for(Order order:orderList){

           System.out.println(order);

       }

       //Order(order_id=1, order_no=1, product_id=1, user_id=1, order_num=2, order_amt=0.00, order_status=已发货, pay_status=支付完成, create_time=2022-08-21, update_time=2022-08-21, create_user=user1, update_user=user2)

       //Order(order_id=2, order_no=2, product_id=2, user_id=1, order_num=2, order_amt=10.00, order_status=已发货, pay_status=支付, create_time=2022-08-21, update_time=2022-08-21, create_user=user1, update_user=user2)

   }

}

注意事项


1.接口类的mapper名字和xml的名字一定要一致。


2.注意xml的命名空间,以及mapper类的方法名字和xml的方法id需要一样。


3.mapper类需要加上注解@Mapper或者在启动类加上@MapperScan(value = “com.elite.springboot.mapper”)


4.pom.xml中可能需要加入以下配置,编译的时候xml才会被打包。

<build>

<resources>

     <resource>

       <directory>src/main/resources</directory>

       <includes>

         <include>**/*.properties</include>

         <include>**/*.xml</include>

         <include>**/*.yml</include>

       </includes>

       <filtering>false</filtering>

     </resource>

 </resources>

</build>

5.mysql版本的问题,xml中表明需要加上 ``,否则报语法错误。**

注解实现

接口类

package com.elite.springboot.mapper;

import com.elite.springboot.entity.Order;

import org.apache.ibatis.annotations.Delete;

import org.apache.ibatis.annotations.Insert;

import org.apache.ibatis.annotations.Select;

import org.apache.ibatis.annotations.Update;

import java.util.List;

/**

* 使用注解进行

*/

public interface OrderAnnotationMapper {

   //保存数据

   @Insert("INSERT INTO `order`(`order_no`,`product_id`,`user_id`,`order_num`,`order_amt`,`order_status`,`pay_status`,`create_user`,`update_user`)\n" +

           " VALUES( #{order_no},#{product_id},#{user_id},#{order_num},#{order_amt},#{order_status},#{pay_status},#{create_user},#{update_user})")

   void insertOrder(Order order);

   //更新订单

   @Update("update `order` set  order_status = #{order_status} where order_id= #{order_id}")

   void updateOrderStatusById(Order order);

   //删除订单

   @Delete("DELETE FROM `order` WHERE order_id= #{order_id}")

   void delOrderById(Integer order_id);

   //获取订单

   @Select("select * from `order` where order_id= #{order_id}")

   Order getOneOrderById(Integer order_id);

   //获取所有的订单

   @Select("select * from `order`")

   List<Order> getAllOrder();

}

测试

package com.elite.springboot;

import com.elite.springboot.entity.Order;

import com.elite.springboot.mapper.OrderAnnotationMapper;

import lombok.extern.slf4j.Slf4j;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.test.context.junit4.SpringRunner;

import javax.annotation.Resource;

import java.math.BigDecimal;

import java.util.List;

/**

* 注解sql测试

*/

@Slf4j

@RunWith(SpringRunner.class)

@SpringBootTest

public class AppAnnotationTest {

   @Resource

   OrderAnnotationMapper orderAnnotationMapper;

   /**

    * 保存订单

    */

   @Test

   public void InserOrder(){

       Order order = new Order();

//        order.setOrder_no(2);

//        order.setProduct_id(2);

//        order.setUser_id(1);

//        order.setOrder_num(2);

//        order.setOrder_amt(BigDecimal.valueOf(20));

//        order.setOrder_status("下单");

//        order.setPay_status("未支付");

//        order.setCreate_user("annotation");

//        order.setUpdate_user("annotation");

       order.setOrder_no(4);

       order.setProduct_id(1);

       order.setUser_id(1);

       order.setOrder_num(2);

       order.setOrder_amt(BigDecimal.valueOf(20));

       order.setOrder_status("下单");

       order.setPay_status("支付");

       order.setCreate_user("annotation");

       order.setUpdate_user("annotation");

       orderAnnotationMapper.insertOrder(order);

   }

   /**

    * 更新订单状态

    */

   @Test

   public void updateOrderStatusById(){

       Order order = new Order();

       order.setOrder_id(3);

       order.setOrder_status("取消下单");

       orderAnnotationMapper.updateOrderStatusById(order);

   }

   /**

    * 查询订单

    */

   @Test

   public void getOneOrderById(){

       Order order = orderAnnotationMapper.getOneOrderById(3);

       System.out.println(order);

       //Order(order_id=3, order_no=2, product_id=2, user_id=1, order_num=2, order_amt=20.00, order_status=取消下单, pay_status=未支付, create_time=2022-08-21, update_time=2022-08-21, create_user=annotation, update_user=annotation)

   }

   /**

    * 查询所有订单

    */

   @Test

   public void getAllOrder(){

       List<Order> allOrder = orderAnnotationMapper.getAllOrder();

       for (Order order :allOrder){

           System.out.println(order);

       }

       //Order(order_id=1, order_no=1, product_id=1, user_id=1, order_num=2, order_amt=0.00, order_status=已发货, pay_status=支付完成, create_time=2022-08-21, update_time=2022-08-21, create_user=user1, update_user=user2)

       //Order(order_id=3, order_no=2, product_id=2, user_id=1, order_num=2, order_amt=20.00, order_status=取消下单, pay_status=未支付, create_time=2022-08-21, update_time=2022-08-21, create_user=annotation, update_user=annotation)

       //Order(order_id=4, order_no=4, product_id=1, user_id=1, order_num=2, order_amt=20.00, order_status=下单, pay_status=支付, create_time=2022-08-21, update_time=2022-08-21, create_user=annotation, update_user=annotation)

       //Order(order_id=5, order_no=4, product_id=1, user_id=1, order_num=2, order_amt=20.00, order_status=下单, pay_status=支付, create_time=2022-08-21, update_time=2022-08-21, create_user=annotation, update_user=annotation)

   }

   /**

    *删除订单

    */

   @Test

   public void delOrderById(){

       orderAnnotationMapper.delOrderById(5);

   }

}


相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
1月前
|
缓存 Java 数据库连接
深入探讨:Spring与MyBatis中的连接池与缓存机制
Spring 与 MyBatis 提供了强大的连接池和缓存机制,通过合理配置和使用这些机制,可以显著提升应用的性能和可扩展性。连接池通过复用数据库连接减少了连接创建和销毁的开销,而 MyBatis 的一级缓存和二级缓存则通过缓存查询结果减少了数据库访问次数。在实际应用中,结合具体的业务需求和系统架构,优化连接池和缓存的配置,是提升系统性能的重要手段。
53 4
|
1月前
|
SQL Java 数据库连接
spring和Mybatis的各种查询
Spring 和 MyBatis 的结合使得数据访问层的开发变得更加简洁和高效。通过以上各种查询操作的详细讲解,我们可以看到 MyBatis 在处理简单查询、条件查询、分页查询、联合查询和动态 SQL 查询方面的强大功能。熟练掌握这些操作,可以极大提升开发效率和代码质量。
60 3
|
2月前
|
Java 数据库连接 数据库
spring和Mybatis的逆向工程
通过本文的介绍,我们了解了如何使用Spring和MyBatis进行逆向工程,包括环境配置、MyBatis Generator配置、Spring和MyBatis整合以及业务逻辑的编写。逆向工程极大地提高了开发效率,减少了重复劳动,保证了代码的一致性和可维护性。希望这篇文章能帮助你在项目中高效地使用Spring和MyBatis。
32 1
|
3月前
|
Java 数据库连接 Maven
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和MyBatis Generator,使用逆向工程来自动生成Java代码,包括实体类、Mapper文件和Example文件,以提高开发效率。
162 2
mybatis使用一:springboot整合mybatis、mybatis generator,使用逆向工程生成java代码。
|
3月前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
91 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
|
3月前
|
前端开发 Java Apache
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个
本文详细讲解了如何整合Apache Shiro与Spring Boot项目,包括数据库准备、项目配置、实体类、Mapper、Service、Controller的创建和配置,以及Shiro的配置和使用。
615 1
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个
|
3月前
|
Java 关系型数据库 MySQL
springboot学习五:springboot整合Mybatis 连接 mysql数据库
这篇文章是关于如何使用Spring Boot整合MyBatis来连接MySQL数据库,并进行基本的增删改查操作的教程。
337 0
springboot学习五:springboot整合Mybatis 连接 mysql数据库
|
3月前
|
SQL Java 数据库连接
mybatis使用二:springboot 整合 mybatis,创建开发环境
这篇文章介绍了如何在SpringBoot项目中整合Mybatis和MybatisGenerator,包括添加依赖、配置数据源、修改启动主类、编写Java代码,以及使用Postman进行接口测试。
39 0
mybatis使用二:springboot 整合 mybatis,创建开发环境
|
3月前
|
Java 数据库连接 API
springBoot:后端解决跨域&Mybatis-Plus&SwaggerUI&代码生成器 (四)
本文介绍了后端解决跨域问题的方法及Mybatis-Plus的配置与使用。首先通过创建`CorsConfig`类并设置相关参数来实现跨域请求处理。接着,详细描述了如何引入Mybatis-Plus插件,包括配置`MybatisPlusConfig`类、定义Mapper接口以及Service层。此外,还展示了如何配置分页查询功能,并引入SwaggerUI进行API文档生成。最后,提供了代码生成器的配置示例,帮助快速生成项目所需的基础代码。
196 1
|
3月前
|
前端开发 Java 数据库连接
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学
本文是一份全面的表白墙/留言墙项目教程,使用SpringBoot + MyBatis技术栈和MySQL数据库开发,涵盖了项目前后端开发、数据库配置、代码实现和运行的详细步骤。
86 0
表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学