01-basic

  • 若依(RuoYi)是一款基于JavaEE技术的企业级快速开发平台
  • 让java开发者用少量代码,就能快速搭建和开发出各种管理系统工具
    • 快速构建
    • 通用模块
    • 代码生成

1. 若依搭建

1. 若依版本

1. 官方

若依官方针对不同开发需求提供了多个版本的框架,每个版本都有其独特的特点和适用场景:

  • 前后端混合版本:RuoYi结合了SpringBoot和Bootstrap的前端开发框架,适合快速构建传统的Web应用程序,其中前端和后端代码在同一项目中协同工作
  • 前后端分离版本:RuoYi-Vue利用SpringBoot作为后端开发框架,与Vue.js结合,实现了前后端分离的开发模式。这种架构有助于提高开发效率,前后端可以独立开发和部署,更适合现代化的Web应用开发
  • 微服务版本:RuoYi-Cloud基于Spring Cloud & Alibaba微服务架构,为构建大型分布式系统提供了完整的解决方案。它支持服务发现、配置管理、负载均衡等微服务特性,适合需要高可扩展性和高可用性的企业级应用
  • 移动端版本:RuoYi-App采用Uniapp进行开发,结合了Vue.js的优势,可以实现跨平台的移动端应用开发。一次编写,多端运行的能力使得它成为开发iOS和Android应用的理想选择
image-20240804152007404

2. 非官方

若依框架因其强大的功能和灵活性,吸引了众多第三方开发者基于其核心架构进行扩展和优化,从而形成了丰富的生态系统

2. RuoYi-Vue

官方推荐课程版本
JDK >= 1.8JDK 11
Mysql >= 5.7.0MySQL 8
Redis >= 3.0Redis 5(Win)
Maven >= 3.0Maven 3.6
Node >= 12Node 16(Vue3)

3. 运行后端项目

1. 初始化项目

  • 通过idea克隆若依源码,仓库地址:https://gitee.com/y_project/RuoYi-Vue.git

2. MySQL

  1. 创建数据库 create schema ry-vue;
  2. 执行下图的sql脚本文件,完成导入
image-20231115113710134
  1. 导入后 ry-vue 库内置30张表
1. 配置信息
  • ruoyi-admin 模块下,编辑 resources 目录下的 application-druid.yml,修改数据库连接
# 数据源配置
spring:
  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    driverClassName: com.mysql.cj.jdbc.Driver
    druid:
      # 主库数据源
      master:
        url: 数据库地址
        username: 数据库账号
        password: 数据库密码

3. Redis

  • 在redis解压目录下,执行redis-server.exe redis.windows.conf启动
  • ruoyi-admin模块下,resources目录下的application.yml,可以设置redis密码等相关信息

4. 项目运行

4. 运行前端项目

1. 初始化项目

# 克隆vue3项目
git clone https://gitee.com/ys-gitee/RuoYi-Vue3.git

# 通过vscode打开项目
code ./RuoYi-Vue3

2. 项目运行

# 安装依赖
npm install

# 建议不要直接使用 cnpm 安装依赖,会有各种诡异的 bug。可以通过如下操作解决 npm 下载速度慢的问题
npm install --registry=https://registry.npmmirror.com

# 启动服务
npm run dev

2. 入门案例

1. 功能需求

2. 步骤分析

  1. 准备课程表结构和数据sql文件,导入到数据库中
  2. 登录系统(系统工具 => 代码生成 => 导入课程表)
  3. 代码生成列表中找到课程表(可预览、编辑、同步、删除生成配置)
  4. 点击生成代码会得到一个 ruoyi.zip
  5. 执行sql文件导入菜单,按照包内目录结构复制到项目中即可

3. 代码生成

1. 提供课程表

  • 准备课程表结构和数据sql文件,导入到数据库中

2. 系统导入

  • 登录系统(系统工具 => 代码生成 => 导入课程表)
image-20240407145252756

3. 配置代码

  • 代码生成列表中找到课程表(可预览、编辑、同步、删除生成配置)
image-20240515194239631

4. 点击生成

  • 点击生成代码,得到一个ruoyi.zip
image-20240407171714267

解压后得到:后端代码、前端代码、菜单sql

4. 代码导入

1. 导入课程菜单

  • 执行sql脚本,导入菜单数据 sys_menu

2. 导入后端代码

  • 将生成的后端代码和mapper文件,导入ruoyi-admin模块中

3. 导入前端代码

  • 将生成的前端代码,导入ruoyi-ui模块中
image-20240515194033701

5. 访问测试

  • 代码生成器默认生成的课程管理模块在系统工具菜单下,打开测试CRUD功能
image-20240515194309391

3. 功能详解

我们将对若依的通用功能进行详解。本章内容分为三个重点部分:

1. 系统管理

1. 权限系统

1. RBAC
  • RBAC(基于角色的访问控制)是一种广泛使用的访问控制模型,通过角色来分配和管理用户的菜单权限
image-20240515194432239
2. 表关系
image-20240515194525518
image-20240515194550233
3. 案例
  • 创建新用户小智并关联《课研人员》角色,仅限《课程管理》和《统计分析》菜单访问
    1. 创建菜单
    2. 创建角色,并分配权限
    3. 创建用户,并关联角色
image-20240515194805143

2. 数据字典

1. 介绍
  • 若依内置的数据字典,用于维护系统中常见的静态数据。eg:性别、状态…
image-20240515194849692
  • 功能包括:字典类型管理、字典数据管理
image-20240515194953367
2. 表关系
image-20240515195031748
3. 案例
  • 将课程管理的学科字段改为数据字典维护
image-20240515211609621
  1. 添加字典类型和数据
image-20240515195114801
  1. 修改代码生成信息
image-20240515195134472
  1. 下载代码,导入前端
image-20240515195146589

3. 参数设置

  • 参数设置:对系统中的参数进行动态维护
image-20240515195404229
1. 关闭登录验证码
image-20240515195419521
2. 开启注册
image-20240811125624812

login.vue

// 注册开关
const register = ref(true);

4. 通知公告(半成品)

  • RuoYi的通知公告功能提供了一个方便的方式来发布和管理通知、公告和新闻等信息。管理员可以创建、编辑和删除通知(支持富文本编辑和附件上传)
  • 系统将信息发送给指定的用户、部门或角色。用户可以通过系统界面或电子邮件接收通知,从而确保信息及时传达 (这部分需要自己开发)
  • 通知公告功能有助于组织内部沟通和信息传递,提高了工作效率和信息共享
image-20240515195457616

5. 日志管理

1. 登录日志
  • 记录用户的登录信息,包括登录时间和地点(IP地址)
  • 帮助管理员监控登录行为,及时发现任何可疑的登录尝试
  • 同样提供搜索和筛选功能,方便查找特定用户的登录历史
image-20240515195520979
2. 操作日志
  • 记录用户在系统中的所有操作。eg:查看、修改数据等
  • 帮助管理员检查谁做了什么,以及何时做的,确保数据准确无误
  • 可以快速搜索和找到特定的操作记录,便于管理和审查
image-20240515195525287

2. 系统监控

1. 监控相关

  • 若依提供了一系列强大的监控工具,能够帮助开发者和运维快速了解应用程序的性能状态
image-20240515195822097
  1. 在线用户
    • 管理员可以看到当前谁在系统里,什么时候登录的,从哪里登录的,属于哪个部门
    • 如果有人没权限还赖着不走,管理员可以一键让他们下线,保证系统的安全
  2. 数据监控(集成Druid):
    • 管理员可以实时看到系统的各项指标。eg:资源使用情况,数据库状态等
    • 通过图表可以直观地看出系统是否健康,如果出现问题,系统会发出警报
image-20240811131646377
spring:
 datasource:
   type: com.alibaba.druid.pool.DruidDataSource
   driverClassName: com.mysql.cj.jdbc.Driver
   druid:
     statViewServlet:
       enabled: true
       # 设置白名单,不填则允许所有访问
       allow:
       url-pattern: /druid/*
       # 控制台管理用户名和密码
       login-username: ruoyi
       login-password: ******











 
 
  1. 服务监控
    • 管理员可以监控系统中各个服务是否正常运行,以及它们的性能指标
    • 如果服务出现问题,系统会立即通知管理员,并通过仪表板展示,方便管理员快速了解情况
image-20240811132038925
  1. 缓存监控
    • 管理员可以监控系统的缓存(Redis)使用情况。eg:缓存是否经常被用到,缓存的大小等
    • 系统还可以自动清理缓存,保持数据的新鲜度,如果缓存有问题,也会发出警报
image-20240811131956602

2. 定时任务

1. 介绍
  • 若依为定时任务功能提供方便友好的web界面,实现动态管理任务
image-20240515195859181
2. 案例
  • 每间隔5秒,控制台输出系统时间
image-20240515212217472
  1. 创建任务类。必须在ruoyi-quartz模块下
image-20240515200554019
/**
 * 定时任务调度测试
 */
@Component("ryTask")
public class RyTask {

    // ryTask.ryNoParams
    public void ryNoParams() {
        System.out.println("执行无参方法");
    }

    // ryTask.ryParams('ry')
    public void ryParams(String params) {
        System.out.println("执行有参方法:" + params);
    }

    // ryTask.ryMultipleParams('ry', true, 2000L, 316.50D, 100)
    public void ryMultipleParams(String s, Boolean b, Long l, Double d, Integer i) {
        System.out.println(StringUtils.format("执行多参方法: 字符串类型{},布尔类型{},长整型{},浮点型{},整形{}", s, b, l, d, i));
    }
}
  1. 添加任务规则
    • 任务名称:自定义。eg:定时查询任务状态
    • 任务分组:根据字典 sys_job_group 配置,可自行进行配置
    • 调用方法:
      • Bean调用示例:ryTask.ryParams('ry')
      • Class类调用示例:com.ruoyi.quartz.task.RyTask.ryParams('ry')
      • 参数说明:支持字符串,布尔类型,长整型,浮点型,整型
    • 执行表达式:可查询官方 cron 表达式介绍
    • 执行策略(服务器宕机间的任务):
      • 立即执行:中间没有执行的都补上
      • 执行一次:只执行最近宕机的任务
      • 放弃执行:放弃中间任务
    • 是否并发:是否需要多个任务间同时执行
image-20240515200608310
  1. 启动任务
image-20240515212230091
  1. 执行一次(操作按钮)。一般用于开发时的测试
  2. 调度日志
image-20240811134122492

3. 系统工具

1. 表单构建

1. 介绍
  • 允许用户通过拖放等可视化操作创建表单。eg:用来收集数据的表格或调查问卷
  • 可以自定义表单的各个部分。eg:添加不同的输入项和设置验证规则,无需编写代码
  • 提供了导出数据、导入数据、分享表单和设置权限的功能,方便数据管理和共享
image-20240515201246417
2. 案例
  • 通过表单构建工具,单独制作一个添加课程的表单页面
image-20240515212304226
  1. 制作表单并导出
image-20240515201358208
  1. 复制到前端工程。更改文件名称
image-20240515201427300
  1. 创建动态菜单。【菜单类型】【路由地址】【组件路径】
image-20240515201440877

2. 树表生成

  • 自动化工具,可以快速生成项目中常用的代码。eg:数据库操作类、后端控制器、前端页面等
  • 提供三种生成模板:单表、 树表、主子表(一对多),可以生成适用于 Spring BootMyBatis 等流行框架的代码,提高开发效率和代码质量
  • 树表是一种展示层级数据的表格,能展开折叠,清晰呈现父子关系,便于管理
image-20240515201619313
  • 代码生成配置主表实现细节:
image-20240515201640870

3. 系统接口

  • Swagger,能够自动生成 API 的同步在线文档,并提供Web界面进行接口调用和测试
  • https://swagger.io/specification/v3/
image-20240515201705896
image-20240811144912579
image-20240811144939530

4. 项目结构

1. 后端结构

com.ruoyi
├── ruoyi-admin      // 后台服务模块
│       └── web                           // 内置功能的表现层
│       └── RuoYiApplication              // 若依项目启动类
├── ruoyi-common     // 通用工具模块
│       └── annotation                    // 自定义注解
│       └── config                        // 全局配置
│       └── constant                      // 通用常量
│       └── core                          // 核心控制
│       └── enums                         // 通用枚举
│       └── exception                     // 通用异常
│       └── filter                        // 过滤器处理
│       └── utils                         // 通用类处理
│       └── xss                           // 自定义xss校验
├── ruoyi-framework  // 框架核心模块
│       └── aspectj                       // AOP配置
│       └── config                        // 系统配置
│       └── datasource                    // 多数据源配置
│       └── interceptor                   // 拦截器
│       └── manager                       // 异步处理
│       └── security                      // 权限控制
│       └── web                           // 前端控制
├── ruoyi-generator  // 代码生成模块(可移除)
├── ruoyi-quartz     // 定时任务模块(可移除)
├── ruoyi-system     // 系统代码模块
│       └── domain                        // 系统代码的实体类
│       └── mapper                        // 系统代码的持久层
│       └── service                       // 系统代码的业务层
image-20240515202429752

1. admin

  • 项目的启动入口
  • RuoYiApplication打jar包
  • RuoYiServletInitializer打war包
image-20240811150051655

2. common

  • 通用工具
  • core核心控制,BaseController统一Controller,统一返回结果,统一异常处理
image-20240811150441927

3. framework

  • 框架核心
image-20240811150922689

4. system

image-20240811151200619

2. 项目中配置

项目中的配置文件在 ruoyi-admin 模块下

image-20240811151330153
  • 最主要的两个配置文件:application.yml
# 项目相关配置
ruoyi:
  # 名称
  name: RuoYi
  # 版本
  version: 3.8.7
  # 版权年份
  copyrightYear: 2024
  # 文件路径 示例( Windows配置D:/ruoyi/uploadPath,Linux配置 /home/ruoyi/uploadPath)
  profile: D:/ruoyi/uploadPath
  # 获取ip地址开关
  addressEnabled: false
  # 验证码类型 math 数字计算 char 字符验证
  captchaType: math

# 开发环境配置
server:
  # 服务器的HTTP端口,默认为8080
  port: 8080
  servlet:
    # 应用的访问路径
    context-path: /
  tomcat:
    # tomcat的URI编码
    uri-encoding: UTF-8
    # 连接数满后的排队数,默认为100
    accept-count: 1000
    threads:
      # tomcat最大线程数,默认为200
      max: 800
      # Tomcat启动初始化的线程数,默认值10
      min-spare: 100

# 日志配置
logging:
  level:
    com.ruoyi: debug
    org.springframework: warn

# 用户配置
user:
  password:
    # 密码最大错误次数
    maxRetryCount: 5
    # 密码锁定时间(默认10分钟)
    lockTime: 10

# Spring配置
spring:
  # 资源信息
  messages:
    # 国际化资源文件路径
    basename: i18n/messages
  profiles:
    active: druid
  # 文件上传
  servlet:
    multipart:
      # 单个文件大小
      max-file-size: 10MB
      # 设置总上传的文件大小
      max-request-size: 20MB
  # 服务模块
  devtools:
    restart:
      # 热部署开关
      enabled: true
  # redis 配置
  redis:
    # 地址
    host: localhost
    # 端口,默认为6379
    port: 6379
    # 数据库索引
    database: 0
    # 密码
    password: ******
    # 连接超时时间
    timeout: 10s
    lettuce:
      pool:
        # 连接池中的最小空闲连接
        min-idle: 0
        # 连接池中的最大空闲连接
        max-idle: 8
        # 连接池的最大数据库连接数
        max-active: 8
        # #连接池最大阻塞等待时间(使用负值表示没有限制)
        max-wait: -1ms

# token配置
token:
  # 令牌自定义标识
  header: Authorization
  # 令牌密钥
  secret: *********************
  # 令牌有效期(默认30分钟)
  expireTime: 30

# MyBatis配置
mybatis:
  # 搜索指定包别名
  typeAliasesPackage: com.ruoyi.**.domain
  # 配置mapper的扫描,找到所有的mapper.xml映射文件
  mapperLocations: classpath*:mapper/**/*Mapper.xml
  # 加载全局的配置文件
  configLocation: classpath:mybatis/mybatis-config.xml

# PageHelper分页插件
pagehelper:
  helperDialect: mysql
  supportMethodsArguments: true
  params: count=countSql

# Swagger配置
swagger:
  # 是否开启swagger
  enabled: true
  # 请求前缀
  pathMapping: /dev-api

# 防止XSS攻击
xss:
  # 过滤开关
  enabled: true
  # 排除链接(多个用逗号分隔)
  excludes: /system/notice
  # 匹配链接
  urlPatterns: /system/*,/monitor/*,/tool/*

3. 模块依赖关系

image-20240515202002135

4. 前端结构

ruoyi-vue3
├── bin                        // 执行脚本
├── html                       // IE低版本提示页
├── node_modules               // 第三方依赖库
├── public                     // 公共资源
│   ├── favicon.ico            // favicon图标
├── src                        // 源代码
│   ├── api                    // 所有请求
│   ├── assets                 // 静态资源
│   ├── components             // 全局公用组件
│   ├── directive              // 全局指令
│   ├── layout                 // 布局
│   ├── plugins                // 通用插件
│   ├── router                 // 路由配置
│   ├── store                  // 状态管理
│   ├── utils                  // 全局公用方法
│   ├── views                  // 视图组件
│   ├── App.vue                // 入口组件
│   ├── main.js                // 入口文件
│   ├── permission.js          // 权限管理
│   └── settings.js            // 系统配置
├── vite                       // 构建工具
├── .env.development           // 开发环境配置
├── .env.production            // 生产环境配置
├── .env.staging               // 测试环境配置
├── .gitignore                 // git 忽略项
├── index.html                 // 入口页面
├── package.json               // 项目配置文件(相当于pom.xml)
└── vue.config.js              // Vue项目的配置信息(相当于application.yml)
image-20240515202411640

5. 表结构介绍

  • ruoyi-vue 数据库设计包含了多个表结构,用于支持系统的各种功能模块
  • 这些表根据功能和用途进行分类,以便在后期使用时能够快速定位和理解
image-20240515202517972
ruoyi-vue

5. 源码阅读

  • 前后端代码分析
  • 前后端代码交互流程

1. 前端代码

src/views/course/course/index.vue

<template>
    <div class="app-container">

        <!-- 搜索表单 start -->
        <el-form :model="queryParams" ref="queryRef" :inline="true" v-show="showSearch" label-width="68px">
            <el-form-item label="课程编码" prop="code">
                <el-input
                    v-model="queryParams.code"
                    placeholder="请输入课程编码"
                    clearable
                    @keyup.enter="handleQuery"
                />
            </el-form-item>
            <el-form-item label="课程学科" prop="subject">
                <el-select v-model="queryParams.subject" placeholder="请选择课程学科" clearable>
                    <el-option
                        v-for="dict in course_subject"
                        :key="dict.value"
                        :label="dict.label"
                        :value="dict.value"
                    />
                </el-select>
            </el-form-item>
            <el-form-item label="课程名称" prop="name">
                <el-input
                    v-model="queryParams.name"
                    placeholder="请输入课程名称"
                    clearable
                    @keyup.enter="handleQuery"
                />
            </el-form-item>
            <el-form-item label="适用人群" prop="applicablePerson">
                <el-input
                    v-model="queryParams.applicablePerson"
                    placeholder="请输入适用人群"
                    clearable
                    @keyup.enter="handleQuery"
                />
            </el-form-item>
            <el-form-item>
                <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
                <el-button icon="Refresh" @click="resetQuery">重置</el-button>
            </el-form-item>
        </el-form>
        <!-- 搜索表单 end -->

        <!-- 按钮区域 start -->
        <el-row :gutter="10" class="mb8">
            <el-col :span="1.5">
                <el-button
                    type="primary"
                    plain
                    icon="Plus"
                    @click="handleAdd"
                    v-hasPermi="['course:course:add']"
                >新增
                </el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button
                    type="success"
                    plain
                    icon="Edit"
                    :disabled="single"
                    @click="handleUpdate"
                    v-hasPermi="['course:course:edit']"
                >修改
                </el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button
                    type="danger"
                    plain
                    icon="Delete"
                    :disabled="multiple"
                    @click="handleDelete"
                    v-hasPermi="['course:course:remove']"
                >删除
                </el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button
                    type="warning"
                    plain
                    icon="Download"
                    @click="handleExport"
                    v-hasPermi="['course:course:export']"
                >导出
                </el-button>
            </el-col>
            <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
        </el-row>
        <!-- 按钮区域end -->

        <!-- 数据展示表格 start -->
        <el-table v-loading="loading" :data="courseList" @selection-change="handleSelectionChange">
            <el-table-column type="selection" width="55" align="center"/>
            <el-table-column label="课程id" align="center" prop="id"/>
            <el-table-column label="课程编码" align="center" prop="code"/>
            <el-table-column label="课程学科" align="center" prop="subject">
                <template #default="scope">
                    <dict-tag :options="course_subject" :value="scope.row.subject"/>
                </template>
            </el-table-column>
            <el-table-column label="课程名称" align="center" prop="name"/>
            <el-table-column label="价格" align="center" prop="price"/>
            <el-table-column label="适用人群" align="center" prop="applicablePerson"/>
            <el-table-column label="课程介绍" align="center" prop="info"/>
            <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
                <template #default="scope">
                    <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)"
                               v-hasPermi="['course:course:edit']">修改
                    </el-button>
                    <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)"
                               v-hasPermi="['course:course:remove']">删除
                    </el-button>
                </template>
            </el-table-column>
        </el-table>
        <!-- 数据展示表格 end -->

        <!-- 分页区域 start -->
        <pagination
            v-show="total>0"
            :total="total"
            v-model:page="queryParams.pageNum"
            v-model:limit="queryParams.pageSize"
            @pagination="getList"
        />
        <!-- 分页区域 end -->

        <!-- 添加或修改课程管理对话框 -->
        <el-dialog :title="title" v-model="open" width="500px" append-to-body>
            <el-form ref="courseRef" :model="form" :rules="rules" label-width="80px">
                <el-form-item label="课程编码" prop="code">
                    <el-input v-model="form.code" placeholder="请输入课程编码"/>
                </el-form-item>
                <el-form-item label="课程学科" prop="subject">
                    <el-select v-model="form.subject" placeholder="请选择课程学科">
                        <el-option
                            v-for="dict in course_subject"
                            :key="dict.value"
                            :label="dict.label"
                            :value="dict.value"
                        ></el-option>
                    </el-select>
                </el-form-item>
                <el-form-item label="课程名称" prop="name">
                    <el-input v-model="form.name" placeholder="请输入课程名称"/>
                </el-form-item>
                <el-form-item label="价格" prop="price">
                    <el-input v-model="form.price" placeholder="请输入价格"/>
                </el-form-item>
                <el-form-item label="适用人群" prop="applicablePerson">
                    <el-input v-model="form.applicablePerson" placeholder="请输入适用人群"/>
                </el-form-item>
                <el-form-item label="课程介绍" prop="info">
                    <el-input v-model="form.info" placeholder="请输入课程介绍"/>
                </el-form-item>
            </el-form>
            <template #footer>
                <div class="dialog-footer">
                    <el-button type="primary" @click="submitForm">确 定</el-button>
                    <el-button @click="cancel">取 消</el-button>
                </div>
            </template>
        </el-dialog>
    </div>
</template>

<script setup name="Course">


// 引入后端api接口
import {listCourse, getCourse, delCourse, addCourse, updateCourse} from "@/api/course/course";
// 获取当前实例代理对象,用于访问组件数据、方法
const {proxy} = getCurrentInstance();
// 获取课程学科的数据字典
const {course_subject} = proxy.useDict('course_subject');
// 列表数据
const courseList = ref([]);
// 是否显示弹框
const open = ref(false);
// 是否显示加载状态
const loading = ref(true);
// 是否显示搜索栏
const showSearch = ref(true);
// 复选框,被选中id的数组
const ids = ref([]);
// 复选框,是否单选,用于高亮修改、删除按钮
const single = ref(true);
// 复选框,是否多选,仅高亮删除按钮
const multiple = ref(true);
// 总(记录)条数
const total = ref(0);
// 用于区分新增、修改对话框标题
const title = ref("");
// 定义reactive响应式对象
const data = reactive({
    // 新增或修改表单数据
    form: {},
    // 搜索条件参数
    queryParams: {
        pageNum: 1,
        pageSize: 10,
        code: null,
        subject: null,
        name: null,
        applicablePerson: null,
    },
    // 表单校验规则
    rules: {
        code: [
            {required: true, message: "课程编码不能为空", trigger: "blur"}
        ],
        subject: [
            {required: true, message: "课程学科不能为空", trigger: "change"}
        ],
        name: [
            {required: true, message: "课程名称不能为空", trigger: "blur"}
        ],
        price: [
            {required: true, message: "价格不能为空", trigger: "blur"}
        ],
        applicablePerson: [
            {required: true, message: "适用人群不能为空", trigger: "blur"}
        ],
        info: [
            {required: true, message: "课程介绍不能为空", trigger: "blur"}
        ],
    }
});
// 将data对象的三个属性,转换为ref响应式对象
const {queryParams, form, rules} = toRefs(data);

/** 查询课程管理列表 */
function getList() {
    loading.value = true;
    listCourse(queryParams.value).then(response => {
        courseList.value = response.rows;
        total.value = response.total;
        loading.value = false;
    });
}

// 取消按钮
function cancel() {
    open.value = false;
    reset();
}

// 表单重置
function reset() {
    form.value = {
        id: null,
        code: null,
        subject: null,
        name: null,
        price: null,
        applicablePerson: null,
        info: null,
        createTime: null,
        updateTime: null
    };
    proxy.resetForm("courseRef");
}

/** 搜索按钮操作 */
function handleQuery() {
    queryParams.value.pageNum = 1;
    getList();
}

/** 重置按钮操作 */
function resetQuery() {
    proxy.resetForm("queryRef");
    handleQuery();
}

// 多选框选中数据
function handleSelectionChange(selection) {
    ids.value = selection.map(item => item.id);
    single.value = selection.length != 1;
    multiple.value = !selection.length;
}

/** 新增按钮操作 */
function handleAdd() {
    reset();
    open.value = true;
    title.value = "添加课程管理";
}

/** 修改按钮操作 */
function handleUpdate(row) {
    reset();
    const _id = row.id || ids.value
    getCourse(_id).then(response => {
        form.value = response.data;
        open.value = true;
        title.value = "修改课程管理";
    });
}

/** 提交按钮 */
function submitForm() {
    proxy.$refs["courseRef"].validate(valid => {
        if (valid) {
            if (form.value.id != null) {
                updateCourse(form.value).then(response => {
                    proxy.$modal.msgSuccess("修改成功");
                    open.value = false;
                    getList();
                });
            } else {
                addCourse(form.value).then(response => {
                    proxy.$modal.msgSuccess("新增成功");
                    open.value = false;
                    getList();
                });
            }
        }
    });
}

/** 删除按钮操作 */
function handleDelete(row) {
    const _ids = row.id || ids.value;
    proxy.$modal.confirm('是否确认删除课程管理编号为"' + _ids + '"的数据项?').then(function () {
        return delCourse(_ids);
    }).then(() => {
        getList();
        proxy.$modal.msgSuccess("删除成功");
    }).catch(() => {
    });
}

/** 导出按钮操作 */
function handleExport() {
    proxy.download('course/course/export', {
        ...queryParams.value
    }, `course_${new Date().getTime()}.xlsx`)
}

// 页面加载时执行-查询课程管理列表
getList();
</script>










 











































 








 


























 








 
 
 
 
 







 


 












 









































































































 

















































































































1. 模拟网速很慢

image-20240811163145614

2. 后端代码

image-20240811222053389

1. CourseController

  • ruoyi-admin 模块下这个类
/**
 * 课程管理Controller
 */
@RestController
@RequestMapping("/course/course")
public class CourseController extends BaseController {
    @Autowired
    private ICourseService courseService;

    /**
     * 查询课程管理列表
     */
    @PreAuthorize("@ss.hasPermi('course:course:list')")
    @GetMapping("/list")
    public TableDataInfo list(Course course) {
        // 1. 开启分页
        startPage();
        // 2. 查询课程列表
        List<Course> list = courseService.selectCourseList(course);
        // 3. 返回表格分页数据对象
        return getDataTable(list);
    }

    /**
     * 导出课程管理列表
     */
    @PreAuthorize("@ss.hasPermi('course:course:export')")
    @Log(title = "课程管理", businessType = BusinessType.EXPORT)
    @PostMapping("/export")
    public void export(HttpServletResponse response, Course course) {
        List<Course> list = courseService.selectCourseList(course);
        ExcelUtil<Course> util = new ExcelUtil<Course>(Course.class);
        util.exportExcel(response, list, "课程管理数据");
    }

    /**
     * 获取课程管理详细信息
     */
    @PreAuthorize("@ss.hasPermi('course:course:query')")
    @GetMapping(value = "/{id}")
    public AjaxResult getInfo(@PathVariable("id") Long id) {
        return success(courseService.selectCourseById(id));
    }

    /**
     * 新增课程管理
     */
    @PreAuthorize("@ss.hasPermi('course:course:add')")
    @Log(title = "课程管理", businessType = BusinessType.INSERT)
    @PostMapping
    public AjaxResult add(@RequestBody Course course) {
        return toAjax(courseService.insertCourse(course));
    }

    /**
     * 修改课程管理
     */
    @PreAuthorize("@ss.hasPermi('course:course:edit')")
    @Log(title = "课程管理", businessType = BusinessType.UPDATE)
    @PutMapping
    public AjaxResult edit(@RequestBody Course course) {
        return toAjax(courseService.updateCourse(course));
    }

    /**
     * 删除课程管理
     */
    @PreAuthorize("@ss.hasPermi('course:course:remove')")
    @Log(title = "课程管理", businessType = BusinessType.DELETE)
    @DeleteMapping("/{ids}")
    public AjaxResult remove(@PathVariable Long[] ids) {
        return toAjax(courseService.deleteCourseByIds(ids));
    }

}

2. BaseController

  • Controller继承了BaseController,其中BaseController详细定义如下:
image-20240515203205623

3. TableDataInfo

  • 分页查询统一返回对象:表格分页数据对象
image-20240515203308811

4. AjaxResult

  • 增、删、改、查统一返回对象:操作消息提醒
image-20240515203414483

5. BaseEntity

  • 所有实体类默认继承的 BaseEntity 基类
image-20240515203531887

6. 权限注解

  • @PreAuthorize 注解是 Spring Security 框架中用来做权限检查的
  • 运行方法前先验证权限,权限够就放行,不够就拦截
image-20240515203653598
  • 权限控制流程图
image-20240515203730129
image-20240515203733871

3. 前后端交互流程

  • 课程管理列表查询

1. 接口文档

image-20240515203815442

2. 跨域

  • 在前端开发中,跨域是一个常见的问题,特别是在使用Vue框架进行开发时
  • 跨域是指在浏览器中发送的Ajax请求的目标地址与当前页面的地址不在同一个域下,这会导致浏览器的同源策略产生限制,从而阻止了跨域请求的发送。可以通过代理服务器来解决这个问题
image-20240515203903598
  • 代理服务器是位于客户端和目标服务器之间的一台服务器,它接收客户端发送的请求,并将请求转发给目标服务器。通过在代理服务器上进行请求转发,可以绕过浏览器的同源策略限制,从而实现跨域请求

vue.config.js

image-20240515203936120

course.js

import request from '@/utils/request'

// 查询课程管理列表
export function listCourse(query) {
  return request({
    url: '/course/course/list',
    method: 'get',
    params: query
  })
}

request.js

// 创建axios实例
const service = axios.create({
    // axios中请求配置有baseURL选项,表示请求URL公共部分
    baseURL: import.meta.env.VITE_APP_BASE_API,
    // 超时
    timeout: 10000
})

// request拦截器
service.interceptors.request.use(config => {
    // 是否需要设置 token
    const isToken = (config.headers || {}).isToken === false
    // 是否需要防止数据重复提交
    const isRepeatSubmit = (config.headers || {}).repeatSubmit === false
    if (getToken() && !isToken) {
        config.headers['Authorization'] = 'Bearer ' + getToken() // 让每个请求携带自定义token 请根据实际情况自行修改
    }
    // get请求映射params参数
    if (config.method === 'get' && config.params) {
        let url = config.url + '?' + tansParams(config.params);
        url = url.slice(0, -1);
        config.params = {};
        config.url = url;
    }
    if (!isRepeatSubmit && (config.method === 'post' || config.method === 'put')) {
        const requestObj = {
            url: config.url,
            data: typeof config.data === 'object' ? JSON.stringify(config.data) : config.data,
            time: new Date().getTime()
        }
        const requestSize = Object.keys(JSON.stringify(requestObj)).length; // 请求数据大小
        const limitSize = 5 * 1024 * 1024; // 限制存放数据5M
        if (requestSize >= limitSize) {
            console.warn(`[${config.url}]: ` + '请求数据大小超出允许的5M限制,无法进行防重复提交验证。')
            return config;
        }
        const sessionObj = cache.session.getJSON('sessionObj')
        if (sessionObj === undefined || sessionObj === null || sessionObj === '') {
            cache.session.setJSON('sessionObj', requestObj)
        } else {
            const s_url = sessionObj.url;                // 请求地址
            const s_data = sessionObj.data;              // 请求数据
            const s_time = sessionObj.time;              // 请求时间
            const interval = 1000;                       // 间隔时间(ms),小于此时间视为重复提交
            if (s_data === requestObj.data && requestObj.time - s_time < interval && s_url === requestObj.url) {
                const message = '数据正在处理,请勿重复提交';
                console.warn(`[${s_url}]: ` + message)
                return Promise.reject(new Error(message))
            } else {
                cache.session.setJSON('sessionObj', requestObj)
            }
        }
    }
    return config
}, error => {
    console.log(error)
    Promise.reject(error)
})

// 响应拦截器
service.interceptors.response.use(res => {
        // 未设置状态码则默认成功状态
        const code = res.data.code || 200;
        // 获取错误信息
        const msg = errorCode[code] || res.data.msg || errorCode['default']
        // 二进制数据则直接返回
        if (res.request.responseType === 'blob' || res.request.responseType === 'arraybuffer') {
            return res.data
        }
        if (code === 401) {
            if (!isRelogin.show) {
                isRelogin.show = true;
                ElMessageBox.confirm('登录状态已过期,您可以继续留在该页面,或者重新登录', '系统提示', {
                    confirmButtonText: '重新登录',
                    cancelButtonText: '取消',
                    type: 'warning'
                }).then(() => {
                    isRelogin.show = false;
                    useUserStore().logOut().then(() => {
                        location.href = '/index';
                    })
                }).catch(() => {
                    isRelogin.show = false;
                });
            }
            return Promise.reject('无效的会话,或者会话已过期,请重新登录。')
        } else if (code === 500) {
            ElMessage({message: msg, type: 'error'})
            return Promise.reject(new Error(msg))
        } else if (code === 601) {
            ElMessage({message: msg, type: 'warning'})
            return Promise.reject(new Error(msg))
        } else if (code !== 200) {
            ElNotification.error({title: msg})
            return Promise.reject('error')
        } else {
            return Promise.resolve(res.data)
        }
    },
    error => {
        console.log('err' + error)
        let {message} = error;
        if (message == "Network Error") {
            message = "后端接口连接异常";
        } else if (message.includes("timeout")) {
            message = "系统接口请求超时";
        } else if (message.includes("Request failed with status code")) {
            message = "系统接口" + message.substr(message.length - 3) + "异常";
        }
        ElMessage({message: message, type: 'error', duration: 5 * 1000})
        return Promise.reject(error)
    }
)

 







 


















































 




















































.env.development

# 页面标题
VITE_APP_TITLE = ooxx系统

# 开发环境配置
VITE_APP_ENV = 'development'

# 若依管理系统/开发环境
VITE_APP_BASE_API = '/dev-api'
image-20240812204509249

6. 二次开发

  • 实现外卖管理系统业务功能开发

1. 模块定制

1. RuoYi框架修改器

  • 若依框架修改器是一个可以一键修改RuoYi框架包名、项目名等的工具
  • 地址:https://gitee.com/lpf_project/RuoYi-MT/releases
  • 将项目打成压缩包zip,直接修改即可
image-20240503154426874
image-20240503154346025

2. 新建业务模块

1. 新建子模块
  • 在父工程下创建 ruoyi-waimai 子模块,在 pom.xml 中导入核心模块依赖
<dependency>
    <groupId>com.ruoyi</groupId>
    <artifactId>ruoyi-framework</artifactId>
</dependency>
2. 版本锁定
  • RuoYi-Vue 父工程 pom.xml 中进行版本锁定
<dependency>
    <groupId>com.ruoyi</groupId>
    <artifactId>ruoyi-waimai</artifactId>
    <version>${ruoyi.version}</version>
</dependency>
3. 添加模块依赖
  • ruoyi-admin 模块 pom.xml 中添加模块依赖
<dependency>
    <groupId>com.ruoyi</groupId>
    <artifactId>ruoyi-waimai</artifactId>
</dependency>

2. 菜品管理

  • 利用若依代码生成器(主子表模板),生成菜品管理的前后端代码
image-20240515205011305

1. 代码生成

  1. 准备SQL并导入数据库
image-20240515204819091
  1. 配置代码生成信息
image-20240515204845464
image-20240812211428339
image-20240812211456200
  1. 下载代码并导入项目
image-20240515204901273

2. 升级改造

1. 图片上传组件

由于之前的图片访问是本地的路径和服务,需要发起请求才能获取图片,现在使用了OSS,图片可直接访问,无需再次访问后端服务,所以前端的图片访问逻辑我们需要修改

  • 修改文件位置:src/components/imageUpload/index.vue
  • 如果图片路径是以http开头的,则不走后台服务访问,直接访问OSS,之前的不变,如下图
image-20240813203838901
image-20240515205705604
image-20240807182047595
2. 一对多级联(通义)
const dishFlavorListSelector = ref([
    {name: '忌口', value: ["不要葱", "不要蒜", "不要香菜", "不要辣"]},
    {name: '辣度', value: ["不辣","微辣","中辣","重辣"]},
    {name: '甜味', value: ["无糖","少糖","半糖","多糖"]}
]);
<el-table-column label="口味名称" prop="name" width="150">
    <template #default="scope">
        <!-- 下拉框选择口味名称(通义) -->
        <el-select v-model="scope.row.name" placeholder="请选择口味名称" @change="changeSelector(scope.row)">
            <el-option
                v-for="dict in dishFlavorListSelector"
                :key="dict.name"
                :label="dict.name"
                :value="dict.name"
            ></el-option>
        </el-select>
    </template>
</el-table-column>



 
 
 
 
 
 
 
 


const dishFlavorValue = ref([]);

// 改变口味,更新口味列表
function changeSelector(row) {
    // 清空当前行value(通义)
    row.value = [];
    // 根据选中的name查找静态数据的value(通义)
    dishFlavorValue.value = dishFlavorListSelector.value.find(item => (item.name === row.name)).value
}







 

<el-table-column label="口味列表" prop="value" width="350">
    <template #default="scope">
        <!-- 下拉框选择口味列表 -->
        <el-select v-model="scope.row.value" placeholder="请选择口味列表" multiple
                   @focus="forceFlavorName(scope.row)" style="width: 90%">
            <el-option
                v-for="item in dishFlavorValue"
                :key="item"
                :label="item"
                :value="item"
            ></el-option>
        </el-select>
    </template>
</el-table-column>



 
 
 
 
 
 
 
 
 


/** 提交按钮 */
function submitForm() {
    proxy.$refs["dishRef"].validate(valid => {
        if (valid) {
            form.value.dishFlavorList = dishFlavorList.value;

            if (form.value.dishFlavorList != null) {
                // 将list中的对象的某个字段转为json字符串(通义)
                form.value.dishFlavorList.forEach(item => {
                    item.value = JSON.stringify(item.value);
                });
            }

            if (form.value.id != null) {
                updateDish(form.value).then(response => {
                    proxy.$modal.msgSuccess("修改成功");
                    open.value = false;
                    getList();
                });
            } else {
                addDish(form.value).then(response => {
                    proxy.$modal.msgSuccess("新增成功");
                    open.value = false;
                    getList();
                });
            }
        }
    });
}






 
 
 
 
 
 

















/** 修改按钮操作 */
function handleUpdate(row) {
    reset();
    const _id = row.id || ids.value
    getDish(_id).then(response => {
        form.value = response.data;
        if (response.data.dishFlavorList != null) {
            // list集合中,对象属性的字符串转为json
            response.data.dishFlavorList.forEach(item => {
                item.value = JSON.parse(item.value);
            })
        }
        dishFlavorList.value = response.data.dishFlavorList;
        open.value = true;
        title.value = "修改菜品管理";
    });
}






 
 
 
 
 
 





function forceFlavorName(row) {
    dishFlavorValue.value = dishFlavorListSelector.value.find(item => (item.name === row.name)).value
}

 

3. 页面调整

image-20240515211842252
image-20240515211848147

1. 浏览器标签页icon、标题

  • 找到资料中的logo图标,替换前端项目中的 public 文件夹,删除原有的 favicon.ico,把新拷贝过来的logo更名为 favicon.ico 即可
image-20240503154918697
  • 找到根目录下的 index.html 文件,把title更换为自己的内容即可
image-20240503154945668

2. 系统页面中的logo、标题

  • 找到资料中的logo文件,替换 src/assets/logo/logo.png 文件
image-20240503155059313
  • 若依的系统页面标题引用的是开发环境的配置,只需要修改开发的环境的 VITE_APP_TITLE 属性即可
image-20240503155126721

3. 去除源码 & 文档

image-20240503155305048

4. 主题和自定义图标

  • 在目前的前端项目中,已经提供了非常便利的操作方式,可以切换主题的风格
  • 操作:点击右上角的头像,可以找到布局设置,只能个人设置,不能控制全局
img
  • 在前端代码中也有对应的操作,更好主题风格文件位置:src/setting.js
image-20240515210823709
  • 更换主题颜色文件位置:src/store/modules/settings.js
image-20240515210930466image-20240515211041187
  • 将下载好的图标复制到 src/assets/icons/svg 目录下,就可以给指定菜单设置图标了
image-20240503155635595

5. 登录页面中标题、背景图

  • 登录名称和背景图,可以直接找到登录的组件进行修改即可
  • 组件位置:src/views/login.vue
image-20240510110514167
image-20240503155545404

7. 通义灵码

image-20240813202951276
image-20240813202841462
image-20240813203536830