前后端分离-小项目-3前后端交互
第一步编写前端页面,第二步搭建后端环境,现在开始第三步,继续完善前端功能
完善“添加”按钮功能
Ajax异步请求安装
在前端项目安装ajax。在Terminal输入:npm i axios -S
点击“添加”按钮,弹出Dialog对话框
设置对话框里面的内容——表单
对话框选用Dialog
嵌入表单menu,二者一起构成了新增对话框
注意v-model,:model,变量,方法的设置及格式
<!--添加对话框--><el-dialog text v-model="dialogVisible" width="30%" title="提示"><!--添加表单--><el-form :model="furn" label-width="120px" style="width:80%"><el-form-item label="商品名"><el-input v-model="furn.name" /></el-form-item><el-form-item label="制造商"><el-input v-model="furn.maker" /></el-form-item><el-form-item label="价格"><el-input v-model="furn.price" /></el-form-item><el-form-item label="销量"><el-input v-model="furn.sales" /></el-form-item><el-form-item label="库存"><el-input v-model="furn.stock" /></el-form-item></el-form><template #footer><span class="dialog-footer"><el-button @click="dialogVisible = false">取消</el-button><el-button type="primary" @click="dialogVisible = false">确定</el-button></span></template></el-dialog>
创建axios request
命名成了myRequest,便于区别request
// 引入axios
import axios from 'axios';
// 通过axios创建对象-request,用于发送请求到后端
const myRequest=axios.create({timeout: 5000
})
// request拦截器处理
myRequest.interceptors.request.use(config=>{config.headers['Content-Type']='application/json;charset=utf-8'return config
},error => {return Promise.reject(error)
})
export default myRequest
跨域异常:AxiosError/跨源
修改vue.config.js
module.exports = {devServer: {port: 10000,proxy:{ //设置代理,必填"/api":{ //设置拦截器,格式:/+名字target:"http://localhost:8080" ,//目标地址,就是'/api'要替换的地址changeOrigin:true, //设置是否同源,设为是。实现跨域pathRewrite:{ //路径重写'/api':'' //选择忽略拦截器里面的单词}}}}
}
HomeView.vue修改ajax的访问url:"localhost:8080/save"改为。前后端即可通信
myRequest.post("/api/save",this.furn).then(
显示所有家居信息
思路分析
完成后台代码从mapper->service->controller,并使用Postman对代码进行测试
因为使用搭建的环境,mybatis-plus已经完成了mapper\service层,直接完成controller即可
@RequestMapping("/list")public Result listFurns(){List<Furn> furns = furnService.list();return Result.success(furns);}
完成前台代码,使用axios发送http请求,返回所有家居数据,将数据绑定显示
methods:{list(){myRequest.get("/api/list").then(res=>{console.log("res=",res);this.tableData=res.data.data;})},
调用显示数据的方法。VUE生命周期
created(){this.list()},
增加 处理响应后的结果
主要是练习这种处理方式。这儿的用处不大
实际产生的作用就是把response.data的数据层级拿出来了一层,达成下面这个效果
this.tableData=res.data.data=>this.tableData=res.data;前端HomeView.vue
// response拦截器,返回结果处理
myRequest.interceptors.response.use(response=>{let res=response.dataif(response.config.responseType==='blob'){return res;}if(typeof res==='string'){res=res? JSON.parse(res):res}return res;
})
在save()方法增加list(),完成刷新
之前在list()方法里面加入了list(),导致循环,致使view页面不能显示
list()方法加在res=>{}括号里面,在页面能立马看到更新效果。原因是Ajax的异步机制
Ajax有通讯,运行慢,=>{}而括号外的代码在本机执行快,就导致list更新在新数据到来之前就执行了,导致看不到更新效果。
save(){...this.list() //新增},
回显家具信息,并完成修改
思路分析
完成后台代码从mapper->service->controller,并对代码进行测试
破案了。updateById()这个方法来自于接口IService,它是谁实现的呢?它自己!接口的默认实现
@PutMapping("/update")public Result updateFurn(@RequestBody Furn furn){log.info("待修改的furn="+furn);furnService.updateById(furn);return Result.success();}
完成前台代码,回显家居信息[方式1:直接将点击的表格当前行的数据进行回显,方式2(先思考,尝试完成):根据当前行的id(家居id),到db查询对应的数据,进行回显],再使用axios发送http请求,更新数据,将数据绑定显示
获取当前行数据的方法,就是在该行@Click=handleEdit(scope.row),在方法区,使用该方法,带入参数(row)即可获取。插槽机制
row对象转换成JSON格式的过程
JSON.parse(JS0N.stringify(row))就是对行数据进行深拷贝
这样点击的表格当前行的数据和弹出框的数据是独立的
handleEdit(row){// 输出原始rowconsole.log("row1=",row)// 字符串化rowconsole.log("row2=",JSON.stringify(row))// JSON格式rowconsole.log("row3=",JSON.parse(JSON.stringify(row)))},
回显当前行数据。方式1:
methods:{handleEdit(row){this.form= JSON.parse(JSON.stringify(row)) //获取当前行数据。并放到表单this.dialogVisible=true //显示表单},
从后端拿到数据再回显。方式2:
修改当前行数据,以及成功状态返回。同样也存在Ajax异步,更新list()要放在then()里面
save(){if(this.form.id){//此时执行更新myRequest.put("/api/update",this.form).then(res=>{if(res.code==="200"){//更新成功this.$message({type: "success",message: res.msg})}else{//更新失败this.$message({type:"error",message:"更新数据失败"})}this.dialogVisible = falsethis.list()})}else{//此时执行添加myRequest.post("/api/save",this.form).then(res=>{// res就是后端输出给前端的结果console.log("res",res)this.dialogVisible = falsethis.list()})}},
删除家居项
思路分析
完成后台代码从mapper->service->controller,并对代码进行测试
@DeleteMapping("/del/{id}")public Result deleteFurn(@PathVariable Integer id){furnService.removeById(id);return Result.success();}
完成前台代码,使用axios发送http请求,删除数据,将数据绑定显示
注意删除的方法是用@confirm在<el-popconfirm>绑定的,而不是在<el-button>用@click。
<template #default="scope"><el-button link type="primary" size="small" @click="handleEdit(scope.row)">编辑</el-button><el-popconfirm title="确定删除这条记录?" @confirm="handleDel(scope.row)"><template #reference><el-button link type="primary" size="small">删除</el-button></template></el-popconfirm></template>
handleDel()方法的具体实现
完成后用list()进行刷新。同样注意位置,避免ajax异步导致刷新异常。
handleDel(row){myRequest.delete("/api/del/"+row.id).then(res=>{if(res.code==="200"){this.$message({type:"success",message:res.msg})}else{this.$message({type:"error",message:res.msg})}this.list()})}
分页功能
需求分析
1.显示共多少记录
2.可以设置每项显示几条
3.点击第几页,显示对应数据
思路分析
1.后台使用MyBatis-plus分页插件完成查询
2.修改FurnController,增加处理分页显示代码
3.完成前台代码,加入分页导航,并将分页请求和后台接口结合
创建config/MybatisplusConfig.java配置类。配置分页插件
@Configuration
public class MybatisPlusConfig {@Beanpublic MybatisPlusInterceptor mybatisPlusInterceptor(){//需要注入的对象MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();//添加分页插件//设置数据库类型interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));return interceptor;}
}
增加分页插件的调用处理
application.yml配置输出日志。便于调试
mybatis-plus:configuration:log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
完成后端分页功能
@RequestMapping("/page")public Result page(@RequestParam(defaultValue = "1") Integer pageNum,@RequestParam(defaultValue = "5") Integer pageSize){Page<Furn> furnPage = furnService.page(new Page<>(pageNum, pageSize));return Result.success(furnPage);}
增加前端分页控件,并在数据池完善参数。
<div><el-paginationv-model:current-page="currentPage"v-model:page-size="pageSize":page-sizes="[5, 10]"layout="total, sizes, prev, pager, next, jumper":total="totalCount"@size-change="handleSizeChange"@current-change="handleCurrentChange"/></div>
完成控件功能
完善分页长度,显示指定页码内容的方法。注意方法参数名是自定义的,由控件传入
methods:{handleSizeChange(pageSize){this.pageSize=pageSizethis.list()},handleCurrentChange(currentPage){this.currentPage=currentPagethis.list()},
采用分页方法显示,重写显示列表
注意url添加参数的格式。而且参数名需要和后端接收参数名一致。
注意返回接收数据时,表数据和总条数分别来自于records和total。记不得可以利用console的输出,自己手动查询变量名称后完善。
list(){// 采用分页方式更新数据列表myRequest.get("/api/page",{params:{pageNum:this.currentPage,pageSize:this.pageSize}}).then(res=>{console.log("res=",res)this.tableData=res.data.recordsthis.totalCount=res.data.total})},
切换数据源为DruidDataSource
检查在pom.xml是否已经配置
<dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.1.17</version></dependency>
创建配置类,完成注入
@Configuration
@Slf4j
public class DruidDataSourceConfig {@Bean@ConfigurationProperties("spring.datasource")public DataSource dataSource(){DruidDataSource druidDataSource = new DruidDataSource();log.info("数据源={}",druidDataSource.getClass());return druidDataSource;}
}
注意@ConfigurationProperties("spring.datasource")是已经配置在application.yml的
spring:datasource:url: jdbc:mysql://localhost:3306/furn_ssm?useSSL=true&useUnicode=true&charactorEncoding=utf-8password: rootusername: rootdriver-class-name: com.mysql.cj.jdbc.Driver
带条件检索的分页显示
思路分析
完成后台代码从mapper->service->controller,并对代码进行测试
构造用的是Wrappers的静态方法,而不是new ...
带条件的分页查找。page(分页,条件queryWrapper)
@RequestMapping("/search")public Result pageBySearch(@RequestParam(defaultValue = "1")Integer pageNum,@RequestParam(defaultValue = "5")Integer pageSize,@RequestParam(defaultValue = "")String search){QueryWrapper<Furn> queryWrapper = Wrappers.query();if(StringUtils.hasText(search)){queryWrapper.like("name",search);}Page<Furn> furnPage =furnService.page(new Page<>(pageNum, pageSize), queryWrapper);return Result.success(furnPage);}
完成前台代码,使用axios发送http请求,完成带条件查询分页显示
前端按键添加响应方法
<el-button @click="list">查询</el-button>
注意参数的增加
list(){//改用带条件的分页查询myRequest.get("/api/search",{params:{pageNum:this.currentPage,pageSize:this.pageSize,search:this.search}}).then(res=>{...
相关文章:
前后端分离-小项目-3前后端交互
第一步编写前端页面,第二步搭建后端环境,现在开始第三步,继续完善前端功能完善“添加”按钮功能Ajax异步请求安装在前端项目安装ajax。在Terminal输入:npm i axios -S点击“添加”按钮,弹出Dialog对话框设置对话框里面…...
Spring如何整合MyBatis框架?使用XML及java类的配置方式
前言 Spring文章链接: 从头到尾Spring概念,家族,核心技术使用,事务这一篇就够了!!!_千小半的博客-CSDN博客_spring最新技术 mybatis文章链接: MyBatis框架入门(含实例)_mybatis sqlsession创建和关闭_千小…...
第七届蓝桥杯省赛——8冰雹数(递归)
题目:任意给定一个正整数N,如果是偶数,执行: N / 2如果是奇数,执行: N * 3 1生成的新的数字再执行同样的动作,循环往复。通过观察发现,这个数字会一会儿上升到很高,一会…...
Android 10.0 设置静态ip重启后获取不到ip的修复
1.概述 在定制化开发中,对于设置静态ip以后可以正常使用,但是遇到一个新问题 就是开机以后,获取不到ip 地址,这就有点不正常了,获取不到ip 就自然连不上网了,所以要分析问题所在解决问题 2.设置静态ip重启后获取不到ip的修复的核心代码 frameworks/opt/net/ethernet/java…...
mysql笔记
基础 概念 数据库体系结构的三级模式为:外模式、概念模式和内模式。 内模式:存储模式,对数据的物理结构和存储方式的描述。提供数据定义语言定义的。如顺序还是索引存储(将概念模式定义的数据进行组织存储,达到较好…...
华为OD机试 - 最多等和不相交连续子序列(Python)| 真题+思路+考点+代码+岗位
最多等和不相交连续子序列 题目 给定一个数组,我们称其中连续的元素为连续子序列,称这些元素的和为连续子序列的和。 数组中可能存在几组连续子序列,组内的连续子序列互不相交且有相同的和。 求一组连续子序列,组内子序列的数目最多。输出这个数目。 输入 第一行输入为…...
第四届宁波网安市赛训练题
Crypto 散乱的密文 8fd4a4c94gf15{50}l72d3提示了2 1 6 5 3 4,我们直接横向排列 2165348fd4a4c94gf15{50}l72d3 按顺序竖着抄下来fc1l84f}a45dg034{2d957,然后栅栏解密,注意这里是W型栅栏解密,行数6 flag:flag{52048c453d794df1} 综合解密…...
Windows中MySQL 8.x版本忘记密码如何重设
Windows中MySQL 8.x版本忘记密码如何重置 文章目录Windows中MySQL 8.x版本忘记密码如何重置一、前言二、重置密码操作1、停止MySQL服务2、以安全模式启动MySQL服务3、无密码登录mysql4、重置登录密码5、验证密码是否重置成功三、最后我想说一、前言 好久之前在电脑下载的MySQL…...
【信管12.1】信息文档管理与配置管理
信息文档管理与配置管理对于项目管理来说,文档非常重要,如果是传统的工程行业项目的话,仅仅标书就是几百上千页的。相对来说,其实信息系统开发项目已经好很多了。另外就是配置项,它是比文档更大的一个概念,…...
一文搞懂Linux的标准输出/错误重定向
前言 今天在写一个脚本时,需要将shell命令和可执行程序的输出重定向在某一个log文件中,但是遇到了点小问题,索性就研究下输出重定向到底怎么回事。 Linux系统,有一个非常重要概念,就是一切皆文件。在使用shell脚本时&a…...
【OJ】计数的梦
📚Description: Bessie 处于半梦半醒的状态。过了一会儿,她意识到她好像在数羊,不能入睡。Bessie的大脑反应灵敏,仿佛真实地看到了她数过的一个又一个数。她开始注意每一个数码:每一个数码在计数的过程中出现过多少次…...
【项目实战】MySQL使用CONCAT字符串拼接函数实现与特殊字符的拼接
一、需求说明 因为有新功能需要上生产环境,总有一些乱七八糟的兼容历史数据的活要去做,比如以下。 需要批量的更新数据库中某个字段(如id列中原来是ABCDEFG,需要改成[“ABCDEFG”]), 没错,就是…...
OpenCV实战(11)——形态学变换详解
OpenCV实战(11)——形态学变换详解0. 前言1. 腐蚀和膨胀运算1.1 腐蚀和膨胀基础1.2 使用形态学滤波器执行图像腐蚀和膨胀运算2. 开运算和闭运算2.1 使用形态学滤波器执行图像开运算和闭运算3. 形态学变换应用3.1 使用形态学滤波器检测边缘3.2 使用形态学…...
SPI协议详解(Standard SPI、Dual SPI和Queued SPI)
1、标准SPI 1.1、SPI接口的引脚 (1)SCLK:时钟线; (2)MOSI(master output slave input):主设备输出,从设备输入,单向传输; (3)MISO(master input slave output):主设备输入,从设备输…...
【代码随想录二刷】Day15-二叉树-C++
代码随想录二刷Day15 今日任务 层序遍历 226.翻转二叉树 101.对称二叉树 语言:C 层序遍历 102.二叉树的层序遍历 class Solution { public:vector<vector<int>> levelOrder(TreeNode* root) {vector<vector<int>> res;if(root NULL) …...
C++为什么能重夺年度语言?
目录一、爷青回1、年初依旧很多大新闻,其中一条就是TIOBE把年度编程语言颁给了C。2、这是什么概念?那一年Java的流行指数是14%。二、C为什么衰落三、C为什么重新流行1、C为什么重新流行起来了呢?2、C究竟做对了什么呢?3、根本原因…...
视频监控实时接入——以海康威视为例(2023.2.16)
海康威视实时视频监控接入学习 2023.2.16引言1、视频协议简介1.1 RTSP——Real Time Streaming Protocol(实时流传输协议)1.2 RTMP——Real Time Messaging Protocol(实时消息传输协议)1.3 HLS——HTTP Live Streaming(…...
推荐系统[一]:超详细知识介绍,一份完整的入门指南,解答推荐系统是什么。
1. 推荐算法的初步理解 如果说互联网的目标就是连接一切,那么推荐系统的作用就是建立更加有效率的连接,推荐系统可以更有效率的连接用户与内容和服务,节约了大量的时间和成本。 1.1 推荐系统主要解决问题 任务一:挖掘长尾:帮助用户找到想要的物品(音乐、商品、新闻),…...
新手小白入门必看!如何批量注册Twitter账号?
Twitter是目前海外比较流行的社媒营销平台,所以很多从事跨境电商行业的朋友都需要利用多个Twitter账号来推广营销,但是注册和管理多个Twitter账号其实并不是简单的事情。龙哥将会在这里详细讲讲该如何批量注册并且让这些账号不会因为关联被封号ÿ…...
虚拟环境的创建以及labelme的使用教程
本来打算是将这两部分分开的,但写完虚拟环境的创建似乎字数太少了,不过二者有关联,所以就放一起了。简单介绍一下,虚拟环境的创建有win11系统已经Ubuntu系统,labelme教程包括了下载及其使用的全部流程,以及…...
CSS中的BFC详细讲解(易懂)
带你用最简单的方式理解最全面的BFC~~~1.先了解最常见定位方案普通流元素按照其在 HTML 中的先后位置至上而下布局行内元素水平排列,直到当行被占满然后换行,块级元素则会被渲染为完整的一个新行所有元素默认都是普通流定位浮动元素首先按照普通流的位置…...
华为3面,官网显示面试通过了...开始泡池子,进入漫长等待期
背景: 现在双非本科,非计算机科班,有算法方面的奖,有嵌入式开发经历,官网显示面试通过,短信说录用情况在十个工作日内告知,看别人的说法应该是泡池子了。 全程视频面试,一天面完三…...
【新2023】华为OD机试 - 构成的正方形数量(Python)
构成的正方形数量 题目 输入 N 个互不相同的二维整数坐标, 求这 N 个坐标可以构成的正方形数量。(内积为零的两个向量垂直) 输入 第一行输入为 N,N 代表坐标数量,N为正整数。N <= 100 之后的 K 行输入为坐标 x y以空格分隔,x, y 为整数, -10 <= x, y <= 10 输…...
ElasticSearch之RestClient操作索引库和文档
前言:上文介绍了使用DSL语言操作索引库和文档,本篇文章将介绍使用Java中的RestClient来对索引库和文档进行操作。 希望能够加深自己的印象以及帮助到其他的小伙伴儿们😉😉。 如果文章有什么需要改进的地方还请大佬不吝赐教&#x…...
Lp正则化
一、L1 和 L2范数(norm)A norm is a mathematical thing that is applied to a vector. The norm of a vector maps vector values to values in [0,∞). In machine learning, norms are useful because they are used to express distances: this vect…...
云原生 -- Docker进阶(Docker-compose,Docker网络简单介绍)
Dockerfile的构建过程 每条保留字段必须为大写字母。Dockerfile每行只支持一条指令,但是每条指令可以带多个参数,并且每条保留字指令后面至少要带有一个参数。从上到下依次执行。每条指令都会创建一个新的镜像层,并提交新的镜像。 大致流程…...
taskset命令:让进程运行在指定CPU上
1. 操作场景 taskset命令,可用于进程的CPU调优,可以把云服务器上运行的某个进程,指定在某个CPU上工作。 本节操作指导用户使用taskset命令让进程运行在指定CPU上。 2. 操作步骤 2.1. 执行如下命令,查看云服务器CPU核数。 cat …...
Pod基本概念与Pod应用生命周期
Pod是一个逻辑抽象概念,kubernetes创建和管理的最小单元,一个Pod由一个容器或多个容器组成。特点:一个Pod可以理解为是一个应用实例,提供服务Pod中容器始终部署在一个Node上Pod中容器共享网络、存储资源Pod主要用法:运…...
DDL 数据定义语言
DDL 数据定义语言 目录概述一、库的管理1、库的创建2、库的修改【一般不修改,容易出现错误】3、库的删除二、表的管理【重要】1、表的创建2、表的修改3、表的删除4、表的复制 【可以跨库复制】练习题概述 数据定义语言 库和表的管理 一、库的管理 创建、修改、删除…...
设计模式概述
1. 概念 设计模式概念的提出: 设计模式最早于1977年在建筑设计行业中被 克里斯托夫亚历山大(Christopher Alexander) 在他的著作 《建筑模式语言:城镇、建筑、构造》 中提出。 软件工程界在1990年开始了设计模式话题的研…...
网站建设基本范例/深圳seo优化seo优化
记得在年初MVP 峰会上Luis Cabrera 在一次WPF的Session 中向MVP们介绍了一些Surface 2.0 的相关工作,以及Surface 2.0 设备的测试视频,由于NDA原因没有更多的透露详细信息。 如Luis Cabrera 几天前在Blog 里所说“Next week: The Microsoft Surface SDK …...
东台建设企业网站/如何优化搜索引擎的搜索功能
数据结构是以某种形式将数据组织在一起的集合,它不仅存储数据,还支持访问和处理数据的操作。算法是为求解一个问题需要遵循的、被清楚指定的简单指令的集合。下面是自己整理的常用数据结构与算法相关内容,如有错误,欢迎指出。 为了…...
html主页面模板/seo推广关键词公司
互动直播、线上会议、在线医疗和在线教育是实时音视频技术应用的重要场景,而这些场景对高可用、高可靠、低延时有着苛刻的要求,很多团队在音视频产品开发过程中会遇到各种各样的问题。例如:流畅性,如果在视频过程中频繁卡顿&#…...
wordpress主动提交百度/济南网络优化网站
今年9月10号华为开发者大会中,华为正式开源发布了鸿蒙os 2.0系统,鸿蒙终于开放给开发者,目前只能用来做指挥屏/手表/车机。华为手机版本的鸿蒙系统则要到今年的12月才提供。在9月10号我就去鸿蒙gitee上的开源代码库中参观,主要是看…...
建设美食网站的意义/友情链接站长平台
链接:https://www.nowcoder.com/questionTerminal/fe298c55694f4ed39e256170ff2c205f?toCommentId12726161&ran685 来源:牛客网 某商店规定:三个空汽水瓶可以换一瓶汽水,允许向老板借空汽水瓶(但是必须要归还&am…...
设计制作个人网站/国内网络销售平台有哪些
学习内容简单查询汇总分析复杂查询多表查询如何提高SQL查询效率简单查询创建学校数据库的表查找学生查询姓‘猴’的学生名单查询姓名中最后一个字是‘猴’的学生名单查询姓名中带‘猴’的学生名单select * from student where 姓名 like 猴%;select * from student where 姓名 …...