【Web应用】若依框架:基础篇14 源码阅读-后端代码分析-课程管理模块前后端代码分析

文章目录

  • 一、课程管理模块前端代码截图
  • 二、前端代码及分析
    • index.vue
    • course.js
  • 三、前端执行流程
      • 1. 组件初始化
      • 2. 查询操作
      • 3. 列表操作
      • 4. 对话框操作
      • 5. API 请求
      • 6. 执行流程总结
      • 关键点
  • 四、课程管理模块后端代码截图
  • 五、后端代码块
    • CourseController
    • ICourseService
    • CourseMapper
    • Course
  • 六、后端执行逻辑
      • 1. `Course` 类
      • 2. `CourseMapper` 接口
      • 3. `ICourseService` 接口
      • 4. `CourseController` 类
      • 调用关系总结
      • 示例流程
  • 七、前后端交互执行流程
      • **1. 初始化阶段**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **2. 查询操作**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **3. 新增课程**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **4. 修改课程**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **5. 删除课程**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **6. 导出操作**
        • **前端操作**
        • **后端处理**
        • **前端响应**
      • **7. 总结:前后端交互流程**
      • **关键点**


一、课程管理模块前端代码截图

在这里插入图片描述

二、前端代码及分析

index.vue

<template><div class="app-container"><el-form :model="queryParams" ref="queryRef" :inline="true" v-show="showSearch" label-width="68px"><el-form-item label="课程编码" prop="code"><el-inputv-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-optionv-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-inputv-model="queryParams.name"placeholder="请输入课程名称"clearable@keyup.enter="handleQuery"/></el-form-item><el-form-item label="适用人群" prop="applicablePerson"><el-inputv-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><el-row :gutter="10" class="mb8"><el-col :span="1.5"><el-buttontype="primary"plainicon="Plus"@click="handleAdd"v-hasPermi="['course:course:add']">新增</el-button></el-col><el-col :span="1.5"><el-buttontype="success"plainicon="Edit":disabled="single"@click="handleUpdate"v-hasPermi="['course:course:edit']">修改</el-button></el-col><el-col :span="1.5"><el-buttontype="danger"plainicon="Delete":disabled="multiple"@click="handleDelete"v-hasPermi="['course:course:remove']">删除</el-button></el-col><el-col :span="1.5"><el-buttontype="warning"plainicon="Download"@click="handleExport"v-hasPermi="['course:course:export']">导出</el-button></el-col><right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar></el-row><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><paginationv-show="total>0":total="total"v-model:page="queryParams.pageNum"v-model:limit="queryParams.pageSize"@pagination="getList"/><!-- 添加或修改课程管理对话框 --><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-optionv-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">
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)
const ids = ref([])
const single = ref(true)
const multiple = ref(true)
const total = ref(0)
const title = ref("")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" }],}
})const { queryParams, form, rules } = toRefs(data)/** 查询课程管理列表 */
function getList() {loading.value = truelistCourse(queryParams.value).then(response => {courseList.value = response.rowstotal.value = response.totalloading.value = false})
}// 取消按钮
function cancel() {open.value = falsereset()
}// 表单重置
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 = 1getList()
}/** 重置按钮操作 */
function resetQuery() {proxy.resetForm("queryRef")handleQuery()
}// 多选框选中数据
function handleSelectionChange(selection) {ids.value = selection.map(item => item.id)single.value = selection.length != 1multiple.value = !selection.length
}/** 新增按钮操作 */
function handleAdd() {reset()open.value = truetitle.value = "添加课程管理"
}/** 修改按钮操作 */
function handleUpdate(row) {reset()const _id = row.id || ids.valuegetCourse(_id).then(response => {form.value = response.dataopen.value = truetitle.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 = falsegetList()})} else {addCourse(form.value).then(response => {proxy.$modal.msgSuccess("新增成功")open.value = falsegetList()})}}})
}/** 删除按钮操作 */
function handleDelete(row) {const _ids = row.id || ids.valueproxy.$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>

course.js

import request from '@/utils/request'// 查询课程管理列表
export function listCourse(query) {return request({url: '/course/course/list',method: 'get',params: query})
}// 查询课程管理详细
export function getCourse(id) {return request({url: '/course/course/' + id,method: 'get'})
}// 新增课程管理
export function addCourse(data) {return request({url: '/course/course',method: 'post',data: data})
}// 修改课程管理
export function updateCourse(data) {return request({url: '/course/course',method: 'put',data: data})
}// 删除课程管理
export function delCourse(id) {return request({url: '/course/course/' + id,method: 'delete'})
}

三、前端执行流程

根据提供的 Vue 组件代码和 API 请求代码,我们可以分析出前端在执行过程中的主要步骤和逻辑。以下是详细的执行过程分析:

1. 组件初始化

  1. 模板渲染

    • 组件模板中定义了一个课程管理的界面,包括搜索表单、操作按钮、课程列表表格、分页组件以及添加/修改课程的对话框。
    • 使用了 Element Plus 组件库(如 el-formel-tableel-dialog 等)。
  2. 脚本初始化

    • 引入了课程管理的 API 请求函数(listCoursegetCoursedelCourseaddCourseupdateCourse)。
    • 定义了组件的响应式数据,包括课程列表(courseList)、查询参数(queryParams)、表单数据(form)、对话框状态(open)等。
    • 从字典中获取课程学科数据(course_subject)。
  3. 初始化请求

    • 调用 getList() 方法获取课程列表数据,渲染到表格中。

2. 查询操作

  1. 搜索表单

    • 用户可以在搜索表单中输入课程编码、选择课程学科、输入课程名称和适用人群。
    • 点击“搜索”按钮时,触发 handleQuery() 方法。
    • handleQuery() 方法将 pageNum 重置为 1,然后调用 getList() 方法。
  2. 重置表单

    • 点击“重置”按钮时,触发 resetQuery() 方法。
    • resetQuery() 方法重置查询表单,然后调用 handleQuery() 方法重新获取数据。

3. 列表操作

  1. 表格数据

    • 课程列表数据通过 el-table 渲染,每行数据包括课程 ID、编码、学科、名称、价格、适用人群和介绍。
    • 表格支持多选,选中数据后更新 idssinglemultiple 状态。
  2. 操作按钮

    • 新增:点击“新增”按钮时,触发 handleAdd() 方法,重置表单并打开对话框。
    • 修改:点击“修改”按钮时,触发 handleUpdate() 方法,获取课程详情并打开对话框。
    • 删除:点击“删除”按钮时,触发 handleDelete() 方法,弹出确认对话框,确认后调用 delCourse() 删除课程。
    • 导出:点击“导出”按钮时,触发 handleExport() 方法,调用 proxy.download() 导出课程数据。

4. 对话框操作

  1. 添加/修改课程

    • 对话框中包含课程表单,用户可以输入课程编码、选择学科、输入名称、价格、适用人群和介绍。
    • 点击“确定”按钮时,触发 submitForm() 方法。
    • submitForm() 方法验证表单数据,根据是否有 id 决定是新增还是修改课程,调用 addCourse()updateCourse() 方法。
    • 操作成功后,关闭对话框并重新获取课程列表。
  2. 取消操作

    • 点击“取消”按钮时,触发 cancel() 方法,关闭对话框并重置表单。

5. API 请求

  1. 请求函数
    • listCourse(query):发送 GET 请求,获取课程列表数据。
    • getCourse(id):发送 GET 请求,获取单个课程的详细信息。
    • addCourse(data):发送 POST 请求,新增课程。
    • updateCourse(data):发送 PUT 请求,修改课程。
    • delCourse(id):发送 DELETE 请求,删除课程。

6. 执行流程总结

  1. 组件初始化时,调用 getList() 获取课程列表。
  2. 用户通过搜索表单或分页组件触发查询操作,调用 getList() 更新数据。
  3. 用户点击操作按钮(新增、修改、删除、导出)时,触发相应的方法。
  4. 对话框中的表单提交时,调用 submitForm() 方法,根据操作类型调用对应的 API 请求。
  5. API 请求成功后,更新界面数据(如关闭对话框、重新获取列表等)。

关键点

  • 响应式数据:使用 refreactive 定义响应式数据,确保界面实时更新。
  • API 请求:通过 request 工具函数封装 HTTP 请求,简化代码。
  • 表单验证:使用 el-form 的验证规则确保用户输入的有效性。
  • 组件通信:通过 propsemit 实现父子组件通信(如 right-toolbar 组件)。

通过以上分析,可以清晰地了解该前端组件的执行过程和逻辑。

四、课程管理模块后端代码截图

在这里插入图片描述

五、后端代码块

CourseController

package com.ruoyi.course.controller;import java.util.List;
import javax.servlet.http.HttpServletResponse;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.ruoyi.common.annotation.Log;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.enums.BusinessType;
import com.ruoyi.course.domain.Course;
import com.ruoyi.course.service.ICourseService;
import com.ruoyi.common.utils.poi.ExcelUtil;
import com.ruoyi.common.core.page.TableDataInfo;/*** 课程管理Controller* * @author ruoyi* @date 2025-05-27*/
@RestController
@RequestMapping("/course/course")
public class CourseController extends BaseController
{@Autowiredprivate ICourseService courseService;/*** 查询课程管理列表*/@PreAuthorize("@ss.hasPermi('course:course:list')")@GetMapping("/list")public TableDataInfo list(Course course){startPage();List<Course> list = courseService.selectCourseList(course);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)@PostMappingpublic AjaxResult add(@RequestBody Course course){return toAjax(courseService.insertCourse(course));}/*** 修改课程管理*/@PreAuthorize("@ss.hasPermi('course:course:edit')")@Log(title = "课程管理", businessType = BusinessType.UPDATE)@PutMappingpublic 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));}
}

ICourseService

package com.ruoyi.course.controller;import java.util.List;
import javax.servlet.http.HttpServletResponse;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.ruoyi.common.annotation.Log;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.enums.BusinessType;
import com.ruoyi.course.domain.Course;
import com.ruoyi.course.service.ICourseService;
import com.ruoyi.common.utils.poi.ExcelUtil;
import com.ruoyi.common.core.page.TableDataInfo;/*** 课程管理Controller* * @author ruoyi* @date 2025-05-27*/
@RestController
@RequestMapping("/course/course")
public class CourseController extends BaseController
{@Autowiredprivate ICourseService courseService;/*** 查询课程管理列表*/@PreAuthorize("@ss.hasPermi('course:course:list')")@GetMapping("/list")public TableDataInfo list(Course course){startPage();List<Course> list = courseService.selectCourseList(course);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)@PostMappingpublic AjaxResult add(@RequestBody Course course){return toAjax(courseService.insertCourse(course));}/*** 修改课程管理*/@PreAuthorize("@ss.hasPermi('course:course:edit')")@Log(title = "课程管理", businessType = BusinessType.UPDATE)@PutMappingpublic 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));}
}

CourseMapper

package com.ruoyi.course.controller;import java.util.List;
import javax.servlet.http.HttpServletResponse;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.ruoyi.common.annotation.Log;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.enums.BusinessType;
import com.ruoyi.course.domain.Course;
import com.ruoyi.course.service.ICourseService;
import com.ruoyi.common.utils.poi.ExcelUtil;
import com.ruoyi.common.core.page.TableDataInfo;/*** 课程管理Controller* * @author ruoyi* @date 2025-05-27*/
@RestController
@RequestMapping("/course/course")
public class CourseController extends BaseController
{@Autowiredprivate ICourseService courseService;/*** 查询课程管理列表*/@PreAuthorize("@ss.hasPermi('course:course:list')")@GetMapping("/list")public TableDataInfo list(Course course){startPage();List<Course> list = courseService.selectCourseList(course);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)@PostMappingpublic AjaxResult add(@RequestBody Course course){return toAjax(courseService.insertCourse(course));}/*** 修改课程管理*/@PreAuthorize("@ss.hasPermi('course:course:edit')")@Log(title = "课程管理", businessType = BusinessType.UPDATE)@PutMappingpublic 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));}
}

Course

package com.ruoyi.course.controller;import java.util.List;
import javax.servlet.http.HttpServletResponse;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import com.ruoyi.common.annotation.Log;
import com.ruoyi.common.core.controller.BaseController;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.enums.BusinessType;
import com.ruoyi.course.domain.Course;
import com.ruoyi.course.service.ICourseService;
import com.ruoyi.common.utils.poi.ExcelUtil;
import com.ruoyi.common.core.page.TableDataInfo;/*** 课程管理Controller* * @author ruoyi* @date 2025-05-27*/
@RestController
@RequestMapping("/course/course")
public class CourseController extends BaseController
{@Autowiredprivate ICourseService courseService;/*** 查询课程管理列表*/@PreAuthorize("@ss.hasPermi('course:course:list')")@GetMapping("/list")public TableDataInfo list(Course course){startPage();List<Course> list = courseService.selectCourseList(course);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)@PostMappingpublic AjaxResult add(@RequestBody Course course){return toAjax(courseService.insertCourse(course));}/*** 修改课程管理*/@PreAuthorize("@ss.hasPermi('course:course:edit')")@Log(title = "课程管理", businessType = BusinessType.UPDATE)@PutMappingpublic 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));}
}

六、后端执行逻辑

在这个项目结构中,course 包下的四个绿色类文件(CourseControllerCourseCourseMapperICourseService)通常在若依(RuoYi)框架中扮演不同的角色,它们之间的调用关系如下:

1. Course

  • 角色:领域模型(Domain Model)或实体类。
  • 作用:表示课程(Course)的基本信息,通常与数据库中的 course 表对应。它包含了课程的属性,如课程名称、描述、创建时间等。
  • 调用关系:不直接调用其他类,而是被其他类(如 CourseMapperCourseService)使用。

2. CourseMapper 接口

  • 角色:数据访问层(DAO)。
  • 作用:定义了与数据库交互的方法,如插入、查询、更新和删除课程信息。通常使用 MyBatis 框架,通过 XML 映射文件或注解实现 SQL 语句。
  • 调用关系
    • CourseService 调用,用于执行数据库操作。
    • 依赖于 Course 类,因为它的方法通常以 Course 对象作为参数或返回值。

3. ICourseService 接口

  • 角色:服务层接口。
  • 作用:定义了与课程相关的业务逻辑方法,如添加课程、查询课程列表等。它是对外暴露的业务接口。
  • 调用关系
    • CourseController 调用,用于处理业务逻辑。
    • 依赖于 CourseMapper,因为业务逻辑可能需要访问数据库。

4. CourseController

  • 角色:控制层(Controller)。
  • 作用:处理 HTTP 请求,接收前端传来的参数,调用 CourseService 的方法执行业务逻辑,并返回响应结果。
  • 调用关系
    • 调用 ICourseService 的方法,执行业务逻辑。
    • 依赖于 Course 类,因为请求参数和响应数据通常以 Course 对象的形式传递。

调用关系总结

  1. 前端请求CourseController
  2. CourseController → 调用 ICourseService 的方法。
  3. ICourseService 的实现类 → 调用 CourseMapper 的方法。
  4. CourseMapper → 执行数据库操作,操作 Course 对象。
  5. 最终结果通过 CourseController 返回给前端。

示例流程

假设有一个添加课程的请求:

  1. 前端发送添加课程的请求到 CourseController
  2. CourseController 调用 ICourseService.addCourse() 方法。
  3. CourseServiceImplICourseService 的实现类)调用 CourseMapper.insertCourse() 方法。
  4. CourseMapper 执行 SQL 插入语句,将课程信息存入数据库。
  5. 返回操作结果,最终由 CourseController 返回给前端。

这种分层架构(Controller-Service-Mapper-Domain)是典型的 Web 应用设计模式,有助于代码的模块化和可维护性。

七、前后端交互执行流程

我们可以详细讲解整个前后端交互的过程与步骤。以下是完整的交互流程:


1. 初始化阶段

前端操作
  1. 组件加载
    • Vue 组件初始化时,调用 getList() 方法。
    • getList() 方法调用 listCourse(queryParams) API 请求,向后端发送 GET 请求,携带查询参数(如 pageNumpageSizecode 等)。
后端处理
  1. 接收请求
    • 后端 CourseController 接收 /course/course/list 的 GET 请求。
  2. 业务逻辑
    • 调用 ICourseService.listCourse(queryParams) 方法,查询数据库获取课程列表。
  3. 返回数据
    • 返回查询结果,格式通常为 { rows: [...], total: number }
前端响应
  1. 数据渲染
    • 前端接收后端返回的数据,更新 courseListtotal,渲染到表格和分页组件。

2. 查询操作

前端操作
  1. 用户输入查询条件
    • 用户在搜索表单中输入课程编码、选择学科等。
  2. 触发查询
    • 点击“搜索”按钮,调用 handleQuery() 方法。
    • handleQuery() 重置 pageNum 为 1,并调用 getList()
后端处理
  1. 接收请求
    • 后端接收新的查询参数,重复初始化阶段的查询逻辑。
  2. 返回数据
    • 返回符合查询条件的课程列表。
前端响应
  1. 更新界面
    • 前端更新 courseListtotal,表格重新渲染。

3. 新增课程

前端操作
  1. 打开对话框
    • 点击“新增”按钮,调用 handleAdd() 方法,打开对话框并重置表单。
  2. 提交表单
    • 用户填写课程信息,点击“确定”按钮,调用 submitForm() 方法。
    • submitForm() 验证表单数据,调用 addCourse(form) API 请求,向后端发送 POST 请求,携带课程数据。
后端处理
  1. 接收请求
    • 后端 CourseController 接收 /course/course 的 POST 请求。
  2. 业务逻辑
    • 调用 ICourseService.addCourse(course) 方法,将课程数据插入数据库。
  3. 返回结果
    • 返回操作结果(如成功或失败信息)。
前端响应
  1. 成功处理
    • 前端接收成功响应,关闭对话框,调用 getList() 重新获取课程列表。
  2. 失败处理
    • 显示错误提示(如 proxy.$modal.msgError("新增失败"))。

4. 修改课程

前端操作
  1. 打开对话框
    • 点击“修改”按钮,调用 handleUpdate(row) 方法。
    • 调用 getCourse(id) API 请求,获取课程详情。
  2. 填充表单
    • 后端返回课程详情,填充到表单中。
  3. 提交表单
    • 用户修改信息后点击“确定”,调用 submitForm() 方法。
    • submitForm() 验证数据,调用 updateCourse(form) API 请求,向后端发送 PUT 请求,携带更新后的课程数据。
后端处理
  1. 接收请求
    • 后端 CourseController 接收 /course/course 的 PUT 请求。
  2. 业务逻辑
    • 调用 ICourseService.updateCourse(course) 方法,更新数据库中的课程数据。
  3. 返回结果
    • 返回操作结果。
前端响应
  1. 成功处理
    • 前端接收成功响应,关闭对话框,调用 getList() 重新获取课程列表。
  2. 失败处理
    • 显示错误提示。

5. 删除课程

前端操作
  1. 触发删除
    • 点击“删除”按钮,调用 handleDelete(row) 方法。
    • 弹出确认对话框,用户确认后调用 delCourse(ids) API 请求,向后端发送 DELETE 请求,携带课程 ID。
后端处理
  1. 接收请求
    • 后端 CourseController 接收 /course/course/{id} 的 DELETE 请求。
  2. 业务逻辑
    • 调用 ICourseService.deleteCourseById(id) 方法,从数据库中删除课程数据。
  3. 返回结果
    • 返回操作结果。
前端响应
  1. 成功处理
    • 前端接收成功响应,调用 getList() 重新获取课程列表。
  2. 失败处理
    • 显示错误提示。

6. 导出操作

前端操作
  1. 触发导出
    • 点击“导出”按钮,调用 handleExport() 方法。
    • 调用 proxy.download() 方法,向后端发送导出请求,携带查询参数。
后端处理
  1. 接收请求
    • 后端 CourseController 接收 /course/course/export 的请求。
  2. 业务逻辑
    • 调用 ICourseService.exportCourse(queryParams) 方法,生成 Excel 文件。
  3. 返回文件
    • 返回文件流,前端下载。
前端响应
  1. 下载文件
    • 前端接收文件流,触发浏览器下载。

7. 总结:前后端交互流程

  1. 前端发起请求
    • 用户操作(如查询、新增、修改、删除、导出)触发前端 API 请求。
  2. 后端处理请求
    • 后端 Controller 接收请求,调用 Service 层处理业务逻辑,操作数据库。
  3. 后端返回响应
    • 返回 JSON 数据或文件流。
  4. 前端处理响应
    • 成功时更新界面数据或触发下载。
    • 失败时显示错误提示。

关键点

  • API 设计:前后端通过 RESTful API 交互,明确请求路径、方法和参数。
  • 数据格式:通常使用 JSON 格式传输数据,导出时使用文件流。
  • 错误处理:前后端均需处理异常情况(如网络错误、数据验证失败)。
  • 用户体验:前端通过加载状态(如 v-loading)和提示信息(如 msgSuccess)优化用户体验。

通过以上流程,前后端协同工作,实现了课程管理的完整功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/pingmian/84078.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入理解系统:UML类图

UML类图 类图&#xff08;class diagram&#xff09; 描述系统中的对象类型&#xff0c;以及存在于它们之间的各种静态关系。 正向工程&#xff08;forward engineering&#xff09;在编写代码之前画UML图。 逆向工程&#xff08;reverse engineering&#xff09;从已有代码建…

DeepSeek12-Open WebUI 知识库配置详细步骤

&#x1f4da; Open WebUI 知识库配置详细步骤&#xff08;中英文对照&#xff09; &#x1f310; 界面语言切换 # 首次登录后切换语言&#xff1a; 1. 点击左下角用户头像 → Settings 2. 在 "General" 选项卡找到 "Language" 3. 选择 中文(简体)/Engli…

Python网络设备批量配置脚本解析

目录 脚本概述 代码解析 导入模块 日志配置 核心函数config_device 主程序逻辑 使用说明 脚本优化建议 完整代码 脚本概述 这是一个使用Python编写的网络设备批量配置脚本&#xff0c;主要功能是通过SSH协议批量登录多台网络设备&#xff08;如路由器、交换机等&…

Z-FOLD: A Frustratingly Easy Post-Training Quantization Scheme for LLMs

文章目录 摘要1 引言2 相关工作2.1 量化2.2 大型语言模型的量化 3 Z-FOLD3.1 新引入的参数 ζ3.2 参数整合&#xff08;ζ 折叠&#xff09;3.3 使用校准集的微调 4 实验4.1 实验设置4.2 与其他方法的比较4.3 Z-FOLD 的泛化能力4.4 Z-FOLD 的可移植性4.5 消融研究 5 结论6 限制…

交流电机深度解析:从基础到实战的全面指南

简介 交流电机作为现代工业中不可或缺的动力设备,广泛应用于各个领域。本文将带你深入了解交流电机,从最基础的概念和工作原理开始,逐步介绍其类型、结构、关键参数等基础知识。同时,我们会探讨交流电机在企业级开发研发中的技术实战,包括控制技术、调速方法、建模与仿真…

【靶场】XXE-Lab xxe漏洞

前言 学习xxe漏洞,搭了个XXE-Lab的靶场 一、搭建靶场 现在需要登录,不知道用户名密码,先随便试试抓包 二、判断是否存在xxe漏洞 1.首先登录抓包 看到xml数据解析,由此判断和xxe漏洞有关,但还不确定xxe漏洞是否存在。 2.尝试xxe 漏洞 判断是否存在xxe漏洞 A.send to …

【C++特殊工具与技术】优化内存分配(三):operator new函数和opertor delete函数

目录 一、基础概念&#xff1a;operator new与operator delete的本质 1.1 标准库提供的operator new接口 1.2 标准库operator delete的接口 1.3 关键特性总结 二、new表达式与operator new的调用链解析 2.1 new表达式的底层步骤 2.2 示例&#xff1a;观察new表达式的调用…

[c#]判定当前软件是否用管理员权限打开

有时一些软件的逻辑中需要使用管理员权限对某些文件进行修改时&#xff0c;那么该软件在执行或者打开的场合&#xff0c;就需要用使用管理员身份运行才能达到效果。那么在c#里&#xff0c;如何判定该软件是否是对管理员身份运的呢&#xff1f; 1.取得当前的windows用户。 2.取得…

如果在main中抛出异常,该如何处理

#采用 setDefaultUncaughtExceptionHandler 进行全局兜底 public static void main(String[] args) { Thread.setDefaultUncaughtExceptionHandler((thread, ex) -> { System.err.println("全局捕获异常: " ex.getMessage()); ex.printStackTrace(); System.exi…

HBM 读的那些事

如下所示&#xff0c;为HBM读的时序。注意这里说的HBM是和HBM3是有区别的. RL 的配置,是通过MR2来实现的 WDQS貌似和CK同频。这幅图告诉你&#xff0c;WDQS和CK的源头是一样的&#xff0c;都来自PLL&#xff0c;而且中间没有经过倍频操作。所以两者频率基本是一致的。这是HBM的…

省略号和可变参数模板

本文主要介绍如何展开可变参数的参数包 1.C语言的va_list展开可变参数 #include <iostream> #include <cstdarg>void printNumbers(int count, ...) {// 声明va_list类型的变量va_list args;// 使用va_start将可变参数写入变量argsva_start(args, count);for (in…

三十五、面向对象底层逻辑-Spring MVC中AbstractXlsxStreamingView的设计

在Web应用开发中&#xff0c;大数据量的Excel导出功能是常见需求。传统Apache POI的XSSF实现方式在处理超大数据集时&#xff0c;会因全量加载到内存导致OOM&#xff08;内存溢出&#xff09;问题。Spring MVC提供的AbstractXlsxStreamingView通过流式处理机制&#xff0c;有效…

【大模型:知识图谱】--3.py2neo连接图数据库neo4j

【图数据库】--Neo4j 安装_neo4j安装-CSDN博客 需要打开图数据库Neo4j&#xff0c; neo4j console 目录 1.图数据库--连接 2.图数据库--操作 2.1.创建节点 2.2.删除节点 2.3.增改属性 2.4.建立关系 2.5.查询节点 2.6.查询关系 3.图数据库--实例 1.图数据库--连接 fr…

基于dify的营养分析工作流:3分钟生成个人营养分析报告

你去医院做体检&#xff0c;需要多久拿到体检报告呢&#xff1f;医院会为每位病人做一份多维度的健康报告吗&#xff1f;"人工报告需1小时/份&#xff1f;数据误差率高达35%&#xff1f;传统工具无法个性化&#xff1f; Dify工作流AI模型的组合拳&#xff0c;正在重塑健康…

Web后端基础(基础知识)

BS架构&#xff1a;Browser/Server&#xff0c;浏览器/服务器架构模式。客户端只需要浏览器&#xff0c;应用程序的逻辑和数据都存储在服务端。 优点&#xff1a;维护方便缺点&#xff1a;体验一般 CS架构&#xff1a;Client/Server&#xff0c;客户端/服务器架构模式。需要单独…

MySQL(56)什么是复合索引?

复合索引&#xff08;Composite Index&#xff09;&#xff0c;也称为多列索引&#xff0c;是在数据库表的多列上创建的索引。它可以提高涉及多个列的查询性能&#xff0c;通过组合多个列的值来索引数据。复合索引特别适用于需要同时过滤多列的查询。 复合索引的优点 提高多列…

高并发下的缓存击穿/雪崩解决方案

有效解决缓存击穿和雪崩的方法包括&#xff1a;1. 使用互斥锁处理缓存击穿&#xff1b;2. 采用熔断器模式防止雪崩&#xff1b;3. 实施缓存预热和降级策略&#xff1b;4. 利用分片和多级缓存分散请求压力。这些方法各有优劣&#xff0c;需根据实际业务场景灵活调整和结合使用。…

【读论文】OpenAI o3与o4系统模型技术报告解读

回顾一下,4月16日,OpenAI发布了一份关于其o系列新模型——OpenAI o3和OpenAI o4-mini——的System Card。这份文档不仅揭示了这两款模型在推理能力和工具使用方面的显著进步,也详细阐述了其训练方法、数据来源、安全评估以及在图像理解生成、数学推理等多个核心领域的表现。…

第1课、LangChain 介绍

LangChain 介绍 LangChain 是一个以大语言模型&#xff08;LLM, Large Language Model&#xff09;为核心的开发框架&#xff0c;旨在帮助开发者高效地将如 GPT-4 等大型语言模型与外部数据源和计算资源集成&#xff0c;构建智能化应用。 1.1 工作原理 如上图所示&#xff…

【论文阅读28】-CNN-BiLSTM-Attention-(2024)

本文把滑坡位移序列拆开、筛优质因子&#xff0c;再用 CNN-BiLSTM-Attention 来动态预测每个子序列&#xff0c;最后重构出总位移&#xff0c;预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵&#xff08;S…