个人博客编写

后记

2022.12.2.4 : 30、此项目告一段落、

​ 编撰此博客本意里除去对找工作的帮助、更多地是想帮助未走过的人去探探路、总结经验、少走弯路、知识的宝贵不在于无价、而是无私、天下熙熙皆为利往、人们之所以对某事产生敬仰、究其原由不过是你做了、而我做不到或者不愿做。

​ 此篇博客算不上精品、很多处都是点到为止、能用即可、不求甚解、但在我看来、又有那么点好、谦虚点、这篇博客至少能帮你们在编写前后端分离的个人博客项目中、解决70%的问题、而且略有延伸、我这个人一点爱钻牛角尖、又恰逢强迫症、所以大多问题的解决方法都不会是似是而非、定然是知其然、知其所以然。

​ 闲话少叙、这篇博客主要写在使用springboot+vue编写前后端分离的个人博客项目中会遇到的困惑与解决方法、至于项目架构方面的思考可以选择性看、毕竟初期设想不可能与最终一致、所以多关注问题及其解决方案才是能收获最大的。

项目地址:http://8.134.81.146/

运行方法

1.只考虑开发环境下的运行,而不是上传至服务器的运行(两者有区别,末尾有解释)

  • 本项目将前端代码与后端代码分开编写与运行、后端负责处理数据、前端负责视图跳转、这也是当下对前后端分离的解释(刚开始我误解了前后端分离的意思,所以此处我说明一下)

  • 前端运行方式:因为使用vue脚手架vue-cli所以我们需要node.js、运行:npm run dev # 运行项目

    详细启动方式在往期博客中寻找:(4条消息) Vue笔记基础_邹飞鸣的博客-CSDN博客

  • 后端启动、需要jdk8、然后直接在idea中启动、就和普通的springboot项目启动一致

  • sql代码:直接是根目录下一sql结尾的文件,你可以复制文件内容然后在Mysql运行即可

GitHub地址

此项目已开源github:https://github.com/feimingabandon/blog

注:此博客集百家所长、有的解决方法下会标注参考文档、有的可能会有遗漏、望海涵、或联系我修改。

技术:SpringBoot、Mysql、Vue

要求:

1、前后端分离、有后台系统

2、

  • 数据库设计:

    • 用户表(user):user_id(自增、非空、序号与博客表对应(一对多))、

      user_name(用户名)、

      user_password(密码)、

      user_permissions(权限:1(管理员)、2(普通用户)、3(游客))、

      nickname(昵称)、

      mail(用户邮箱)、

      head(用户头像)、

      register_time(注册时间)、

      birthday(用户生日)、

      age(用户年龄)

      注:普通用户可添加文章、但需要经过管理员审核、游客只能查看后台无法进行操作。游客可评论(需要审核)

    • 博客表(blog):

      • 主要存储博客内容与关联评论
      • 字段设计
        • blog_id(自增)、
        • title(标题)、
        • blog_Summary(摘要)
        • blog_content(博客内容)、
        • user_id(表示是哪个用户发的博客)、
        • audit(审核状态:0(通过)、1(未通过)、2(未审核))、
        • like(点赞数,可多次点赞)、
        • release_time(发布时间)、
        • mtime(修改时间)、
        • browse(浏览量)、
        • top(置顶:0:置顶,1:不置顶)
    • 评论表(comment)

      • 存储评论,包括二级评论,且暂且只设计二级评论回去再改良
      • 字段设计
        • comment_id(自增)、
        • blog_id(表示属于哪个博客)、
        • level(1:表示父级为博客,2:表示是评论下的评论(即回复))、
        • level2(默认为-1,即不是二级评论则为可,如果是二级评论则记录父级评论id)
        • user_name(发表用户)、
        • release_time(发布时间)、
        • content(评论内容)、
        • audit(审核状态(0(通过)、1(未通过)、2(未审核))、
        • like(点赞数,可多次点赞))
    • 分类(标签)表(tags)

      • 用来对文章进行分类、一个文章可有多个标签。
      • 我们建立两张表,一张只写各种类型的标签,另一张写文章与标签的映射关系,如一篇文章有两个标签,那么在第二张表中就记录两行,记录格式:id、blog_id、tag_id。这样写的两行中id自增、blog_id不变、tag_id一行写一种即可。
      • 字段设计
        • id(自增)、tag(表示标签,所以一行代表一个标签,然后后期还可以添加标签)
    • 文章标签映射表(tags_blog)(不使用映射,直接在博客表中存储tag,多个tag以逗号分隔)

      • 让标签与文章形成多对一的形式
      • 字段
        • id(自增)、blog_id(文章id)、tag_id(标签id)

    标签功能可以不设计如上两表,而是在博客表中添加个字段,存储所有标签,标签与标签间逗号分隔

3、功能

  • 博客大全、博客查询、博客详情、用户登陆、后台管理、
  • 整个页面分为页首、页脚、页左、页右、只有中间内容显示会改变,是博客详情或者博客查询结果或大全
  • 页首放侧边栏、页脚放联系方式、页左放玩偶、页右放回到最上面
  • 前端使用 vuesax ui、饿了么、等等ui框架
  • 图片存储在后端

前端

1、页首

  • 菜单
  • 白 / 黑 切换
  • 看板娘
  • 音乐播放器

2、页脚

3、首页

4、详情页

5、登陆页

6.Vuesax的使用

1、安装

# install
npm install vuesax # OR yarn add vuesax
# 上面npm出错可换成cnpm

2、使用

Vuesax 是一个 Vuejs 库。要使用它,请添加以下代码:

import Vue from 'vue'
import Vuesax from 'vuesax'import 'vuesax/dist/vuesax.css' //Vuesax styles
Vue.use(Vuesax, {// options here
})

问题

1.背景图片不能全屏的问题

<template >
<div id="building"></div>
</template><style scoped>
#building {background: url("../assets/test01.jpg") ;  /* "https://api.ixiaowai.cn/api/api.php" */height: 100%;width: 100%;position: fixed;background-size: 100% 100%;top: 0;  /* 这里是设置与顶部的距离*/left: 0;  /* 这里是设置与左边的距离*/
}
</style>

2、内容长度超出div让滑动显示

在相应div下设置此参数

<div style="overflow-y:scroll;overflow-x:hidden;height:100%">

3、添加一言

创建 vue 文件,

<template><!-- 请注意,以下的示例包含超链接,您可能需要手动配置样式使其不变色。如果您嫌麻烦,可以移除。 --><p id="hitokoto"><a id="hitokoto_text">,获取中...</a></p>
</template><script >
fetch('https://v1.hitokoto.cn?c=i').then(response => response.json()).then(data => {const hitokoto = document.getElementById('hitokoto_text')hitokoto.innerText = data.hitokoto}).catch(console.error)
</script>

4、打字机效果

因为与一言结合使用,所以在一言基础上修改了代码

参考地址:vue实现打字机动画_远方_ry的博客-CSDN博客

全部代码:

<template><!-- 请注意,以下的示例包含超链接,您可能需要手动配置样式使其不变色。如果您嫌麻烦,可以移除。 --><p ><a  class='typewriter'>{{typewriter}}</a></p>
</template><script >
export default {name: "Poetry",data () {return {typewriter: '',i: 0,timer: 0,str: 'Hi, i´m a web Designer' // 本质上是设置这里的值为自己需要的即可,使用this.str}},mounted () {this.typeing()// 这块内容是一言的,相当于获取一个文本并赋值给this.str变量fetch('https://v1.hitokoto.cn?c=i').then(response => response.json()).then(data => {this.str = data.hitokoto //  data.hitokoto这个是需要一言网页返回后才有值}).catch(console.error)},// 这里应该是动态增减的实现methods: {typeing () {if (this.i <= this.str.length) {this.typewriter = this.str.slice(0, this.i++) + '\'this.timer = setTimeout(() => {this.typeing()}, 150)} else {clearTimeout(this.timer)}}}}
</script>

5、效果:一张背景图占全屏,背景图有诗句、下滑才是正文

1、如果处理不好会出现正文被背景图覆盖了一部分,本质上是块级元素大小未处理好。

我们先定义一个总块级元素< div>、此块级元素范围占全部网页(包括下拉)、然后在里面设置第一个 < div>,此元素存放背景图和诗句,大小只需要设置height: 100%; 即可,即高度占一整个屏幕。然后在此div下在创建的div里面的内容就会挨揍背景图往下输出而不是被背景图覆盖

结构:

 <div  class="building"><div style="height: 100%" ><Poetry id="poetry_text"></Poetry></div><div><br><h1 >我是大帅比</h1><h1>我是大帅比</h1></div></div>
</script>
<style scoped>
.building {background: url("../assets/test01.jpg") ;  /* "https://api.ixiaowai.cn/api/api.php" */height: 100%;width: 100%;position: absolute; /*fixed:绝对定位,absolute:固定在那,滑动时位置改变*/background-size: 100% 100%;top: 0;  /* 这里是设置与顶部的距离*/left: 0;  /* 这里是设置与左边的距离*/
}#poetry_text{color: aliceblue; // 白色
}</style>

6、文字垂直居中

我们知道左右居中很方便text-align: center; 即可,但是如果想以百分比形式去垂直居中,可以使用表格属性

// 父级需要被设置的属性,宽高display、很重要的三属性
.poetry_text{height: 100%;width: 100%;color: aliceblue; text-align: center;display: table;/*父元素设置表格属性*/
}
// 子级需要被设置的属性、一定要父级设置了display子级设置才起作用
.portry_text_span{display: table-cell;/*img设置成表格元素属性*/vertical-align: middle;/*两个display设置后这个属性就起作用*/
}=====<div class="poetry_text"><Poetry class="portry_text_span"></Poetry></div>

注:此操作是在父级范围内垂直居中,如果父级范围出现问题可能导致偏差

6.2、文字最下面居中

在上面的基础上我们设置position: absolute;bottom: 7px;width:100%;属性即可,width属性一定要100%

  <div class="poetry_text"><div class="portry_text_span"><p><Poetry  ></Poetry></p><p style="position: absolute;bottom: 7px;width:100%;font: 30px '华文行楷'">︾︾︾o(>﹏<)o不要向下滑动︾︾︾</p></div>

7、添加动态樱花

直接在index.html文件中添加

<script src="./sakura.js"></script>

注意js文件要保存下来,然后确定好路径就可以了

js文件:

//樱花 sakuravar stop, staticx;
var img = new Image();
img.src = "";// 樱花数量 (添加)
var sakuraNum = 21;
// 樱花越界限制次数, -1不做限制,无限循环 (添加)
var limitTimes = -1;// 定义限制数组 (添加)
var limitArray = new Array(sakuraNum);
for(var index = 0;index < sakuraNum;index++){limitArray[index] = limitTimes;
}// 定义樱花, idx 是修改添加的
function Sakura(x, y, s, r, fn, idx) {this.x = x;this.y = y;this.s = s;this.r = r;this.fn = fn;this.idx = idx;
}// 绘制樱花
Sakura.prototype.draw = function(cxt) {cxt.save();var xc = 40 * this.s / 4;cxt.translate(this.x, this.y);cxt.rotate(this.r);cxt.drawImage(img, 0, 0, 40 * this.s, 40 * this.s)cxt.restore();
}// 修改樱花位置,模拟飘落.
Sakura.prototype.update = function() {this.x = this.fn.x(this.x, this.y);this.y = this.fn.y(this.y, this.y);this.r = this.fn.r(this.r);// 如果樱花越界, 重新调整位置if(this.x > window.innerWidth || this.x < 0 ||this.y > window.innerHeight || this.y < 0) {// 如果樱花不做限制if (limitArray[this.idx] == -1) {this.r = getRandom('fnr');if(Math.random() > 0.4) {this.x = getRandom('x');this.y = 0;this.s = getRandom('s');this.r = getRandom('r');} else {this.x = window.innerWidth;this.y = getRandom('y');this.s = getRandom('s');this.r = getRandom('r');}}// 否则樱花有限制else {if (limitArray[this.idx] > 0) {this.r = getRandom('fnr');if(Math.random() > 0.4) {this.x = getRandom('x');this.y = 0;this.s = getRandom('s');this.r = getRandom('r');} else {this.x = window.innerWidth;this.y = getRandom('y');this.s = getRandom('s');this.r = getRandom('r');}// 该越界的樱花限制数减一limitArray[this.idx]--;}}}
}SakuraList = function() {this.list = [];
}
SakuraList.prototype.push = function(sakura) {this.list.push(sakura);
}// list update 方法
SakuraList.prototype.update = function() {for(var i = 0, len = this.list.length; i < len; i++) {this.list[i].update();}
}// list draw 方法
SakuraList.prototype.draw = function(cxt) {for(var i = 0, len = this.list.length; i < len; i++) {this.list[i].draw(cxt);}
}
SakuraList.prototype.get = function(i) {return this.list[i];
}
SakuraList.prototype.size = function() {return this.list.length;
}// 位置随机策略
function getRandom(option) {var ret, random;switch(option) {case 'x':ret = Math.random() * window.innerWidth;break;case 'y':ret = Math.random() * window.innerHeight;break;case 's':ret = Math.random();break;case 'r':ret = Math.random() * 6;break;case 'fnx':random = -0.5 + Math.random() * 1;ret = function(x, y) {return x + 0.5 * random - 1.7;};break;case 'fny':random = 1.5 + Math.random() * 0.7ret = function(x, y) {return y + random;};break;case 'fnr':random = Math.random() * 0.03;ret = function(r) {return r + random;};break;}return ret;
}// 樱花入口
function startSakura() {requestAnimationFrame = window.requestAnimationFrame ||window.mozRequestAnimationFrame ||window.webkitRequestAnimationFrame ||window.msRequestAnimationFrame ||window.oRequestAnimationFrame;var canvas = document.createElement('canvas'),cxt;staticx = true;canvas.height = window.innerHeight;canvas.width = window.innerWidth;canvas.setAttribute('style', 'position: fixed;left: 0;top: 0;pointer-events: none;');canvas.setAttribute('id', 'canvas_sakura');document.getElementsByTagName('body')[0].appendChild(canvas);cxt = canvas.getContext('2d');var sakuraList = new SakuraList();// sakuraNum 樱花个数 (原版为50个)for(var i = 0; i < sakuraNum; i++) {var sakura, randomX, randomY, randomS, randomR, randomFnx, randomFny;randomX = getRandom('x');randomY = getRandom('y');randomR = getRandom('r');randomS = getRandom('s');randomFnx = getRandom('fnx');randomFny = getRandom('fny');randomFnR = getRandom('fnr');sakura = new Sakura(randomX, randomY, randomS, randomR, {x: randomFnx,y: randomFny,r: randomFnR}, i);sakura.draw(cxt);sakuraList.push(sakura);}stop = requestAnimationFrame(function() {cxt.clearRect(0, 0, canvas.width, canvas.height);// 修改樱花位置逻辑sakuraList.update();// 画出修改后的樱花sakuraList.draw(cxt);// 递归 修改位置, 画出修改后的樱花stop = requestAnimationFrame(arguments.callee);})
}window.onresize = function() {var canvasSnow = document.getElementById('canvas_snow');// canvasSnow 在改变浏览器大小的时候会为null (修改空指针异常), 不过在改变大小时体验稍差if (canvasSnow) {canvasSnow.width = window.innerWidth;canvasSnow.height = window.innerHeight;}
}img.onload = function() {startSakura();
}// 没看懂哪里调用了, 应该是关闭樱花特效的方法. 还请大佬们解释自己是怎么使用的.
function stopp() {if(staticx) {var child = document.getElementById("canvas_sakura");child.parentNode.removeChild(child);window.cancelAnimationFrame(stop);staticx = false;} else {startSakura();}
}

8、使用Element框架

npm 安装

推荐使用 npm 的方式安装,它能更好地和 webpack 打包工具配合使用。

npm i element-ui -S

引入 Element(针对npm安装)

完整引入

1、在引入 Element 时,可以传入一个全局配置对象。该对象目前支持 sizezIndex 字段。size 用于改变组件的默认尺寸,zIndex 设置弹框的初始 z-index(默认值:2000)。

在 main.js 中写入以下内容:

import Vue from 'vue';
import ElementUI from 'element-ui';
import 'element-ui/lib/theme-chalk/index.css';
import App from './App.vue';//Vue.use(Element, { size: 'small', zIndex: 3000 });Vue.use(ElementUI);new Vue({el: '#app',render: h => h(App)
});

以上代码便完成了 Element 的引入。需要注意的是,样式文件需要单独引入。

9、Uncaught SyntaxError: Unexpected token '<'

今天在运行 Vue 项目时,发现报了一个 Uncaught SyntaxError: Unexpected token '<' 错误,如图

解决办法:把放在 Vue项目 src/assets里边的资源文件放到 public 文件夹下来引用

2.1 assets文件夹与static文件夹的区别

区别一:assets文件是src下的,所以最后运行时需要进行打包,而static文件不需要打包就直接放在最终的文件中了

区别二:assets中的文件在vue中的template/style下用…/这种相对路径的形式进行引用,在script下必须用@import的方式引入,而static下的文件在.vue中的任何地方只需使用…/这种相对路径的方式引入,

参考文档:解决 Uncaught SyntaxError: Unexpected token ‘<‘ 错误解决方法_优雅哥cc的博客-CSDN博客

10、诗词显示两行

问题:之前是只有一行所以使用行内标签< span>、但如果有两个行内标签它们会并排显示,即会尽量显示在一行内,所以我使用style="display:block" 将行内标签强制变为块元素,块元素就是一个占一行

11、博客常见左中右布局

只需要在父级< div>标签下设置一个div标签,然后根据盒子模型,将margin:auto值设置成这样

padding-right:20px:在div里面的内容到边框距离

<div style="top: 0;left: 0;width: 100%;height: 100%;"><!--中间--><div style="width: 70%;margin:auto;"></div>
</div>

12、背景色只有一个屏幕大小被渲染了,下拉后就无了

大概率是设置了高度为100%,把高度删除然后设置为: overflow:hidden

意思:HTML溢出;

所以我们设置足够多的空间让它不溢出

height:给元素指定高度,一旦元素的内容超出这个高度就会溢出
min-height:给元素设置最小高度,当内容少的时候,元素是这个高度,当内容超出时,元素高度会自动适应内容,不会出现内容溢出情况。

overflow 属性规定如何处理如何处理不符合元素框的内容。

属性:

overflow:hidden:内容会被修剪,但浏览器不会显示供查看内容的滚动条

overflow:visible:内容不修剪,会呈现在元素框之外

overflow:scroll:会修剪,但是浏览器会显示滚动条以查看其余内容

overflow:auto:由浏览器决定如何显示,如果需要,则显示滚动条

13、看板娘

注:此插件那用但是偶尔会出现人物加载失败的情况,具体原因就是人物模型是在线加载的,有时候网站会打不开,可以将人物模型下载至本地
1.去 GitHub 下载插件 :看板娘插件

下载完成后将 assets 整个文件复制在项目的静态目录下(static文件夹下)

2、导入jQuery、这里为了方便使用链接,可自行下载 jQuery 文件

<script src="http://libs.baidu.com/jquery/2.1.4/jquery.min.js"></script>

3、在index.html中引入文件并初始化、结合上面的jquery后的整体index文件为:

 <!DOCTYPE html>
<html><head><meta charset="utf-8"><meta name="viewport" content="width=device-width,initial-scale=1.0"><title>blog_vue</title><link rel="stylesheet" type="text/css" href="static/assets/waifu.css"/></head><body><!-- 看板娘 --><script src="http://libs.baidu.com/jquery/2.1.4/jquery.min.js"></script><script src="static/assets/waifu-tips.js"></script><script src="./static/assets/autoload.js"></script><script src="./static/assets/live2d.js"></script><!--初始化数据,也可以在waifu-tips.js文件中初始化,最后调用方法initModel--><script type="text/javascript">live2d_settings['modelId'] = 1;live2d_settings['modelTexturesId'] = 87;initModel("/static/assets/waifu-tips.json")</script></body>
</html>
  • 定制属于你的看板娘

    • 修改 waifu-tips.js 顶部的设置参数 (或初始化前设置
    • 修改 waifu-tips.json,定制看板娘提示语,打造专属看板娘

设置参数

Tips: waifu-tips.js 已自带默认参数,如无特殊要求可跳过

  • 后端接口

    • live2d_settings['modelAPI']
      看板娘 API 地址,默认值 '//live2d.fghrsh.net/api/'
    • live2d_settings['tipsMessage']
      提示语读取路径,默认值 'waifu-tips.json' (一般在 initModel 时指定)
    • live2d_settings['hitokotoAPI']
      一言 API 接口,可选 'lwl12.com''hitokoto.cn''jinrishici.com' (古诗词)
  • 默认模型
    • live2d_settings['modelId']
      默认模型(分组) ID,可在 Demo 页 [F12] 呼出 控制台(Console) 找到
    • live2d_settings['modelTexturesId']
      默认材质(模型) ID,可在 Demo 页 [F12] 呼出 控制台(Console) 找到
  • 工具栏设置
    • live2d_settings['showToolMenu'], 显示工具栏, true | false
    • live2d_settings['canCloseLive2d'], 关闭看板娘 按钮,true | false
    • live2d_settings['canSwitchModel'], 切换模型 按钮, true | false
    • live2d_settings['canSwitchTextures'], 切换材质 按钮, true | false
    • live2d_settings['canSwitchHitokoto'], 切换一言 按钮, true | false
    • live2d_settings['canTakeScreenshot'], 看板娘截图 按钮,true | false
    • live2d_settings['canTurnToHomePage'], 返回首页 按钮, true | false
    • live2d_settings['canTurnToAboutPage'],跳转关于页 按钮,true | false
  • 模型切换模式
    • live2d_settings['modelStorage'],记录 ID (刷新后恢复),true | false
    • live2d_settings['modelRandMode'],模型切换,可选 'rand' (随机) | 'switch' (顺序)
    • live2d_settings['modelTexturesRandMode'],材质切换,可选 'rand' | 'switch'
  • 提示消息选项
    • live2d_settings['showHitokoto'],空闲时一言,true | false
    • live2d_settings['showF12Status'],控制台显示加载状态,true | false
    • live2d_settings['showF12Message'],提示消息输出到控制台,true | false
    • live2d_settings['showF12OpenMsg'],控制台被打开触发提醒,true | false
    • live2d_settings['showCopyMessage'],内容被复制触发提醒,true | false
    • live2d_settings['showWelcomeMessage'],进入面页时显示欢迎语,true | false
  • 看板娘样式设置
    • live2d_settings['waifuSize'],看板娘大小,例如 '280x250''600x535'
    • live2d_settings['waifuTipsSize'],提示框大小,例如 '250x70''570x150'
    • live2d_settings['waifuFontSize'],提示框字体,例如 '12px''30px'
    • live2d_settings['waifuToolFont'],工具栏字体,例如 '14px''36px'
    • live2d_settings['waifuToolLine'],工具栏行高,例如 '20px''36px'
    • live2d_settings['waifuToolTop'],工具栏顶部边距,例如 '0px''-60px'
    • live2d_settings['waifuMinWidth']
      面页小于 指定宽度 隐藏看板娘,例如 'disable' (停用),'768px'
    • live2d_settings['waifuEdgeSide']
      看板娘贴边方向,例如 'left:0' (靠左 0px),'right:30' (靠右 30px)
    • live2d_settings['waifuDraggable']
      拖拽样式,可选 'disable' (禁用),'axis-x' (只能水平拖拽),'unlimited' (自由拖拽)
    • live2d_settings['waifuDraggableRevert'],松开鼠标还原拖拽位置,true | false
  • 其他杂项设置
    • live2d_settings['l2dVersion'],当前版本 (无需修改)
    • live2d_settings['l2dVerDate'],更新日期 (无需修改)
    • live2d_settings['homePageUrl'],首页地址,可选 'auto' (自动),'{URL 网址}'
    • live2d_settings['aboutPageUrl'],关于页地址,'{URL 网址}'
    • live2d_settings['screenshotCaptureName'],看板娘截图文件名,例如 'live2d.png'

定制提示语

Tips: waifu-tips.json 已自带默认提示语,如无特殊要求可跳过

  • "waifu"
    

    系统提示

    • "console_open_msg" 控制台被打开提醒(支持多句随机)
    • "copy_message" 内容被复制触发提醒(支持多句随机)
    • "screenshot_message" 看板娘截图提示语(支持多句随机)
    • "hidden_message" 看板娘隐藏提示语(支持多句随机)
    • "load_rand_textures" 随机材质提示语(暂不支持多句)
    • "hour_tips" 时间段欢迎语(支持多句随机)
    • "referrer_message" 请求来源欢迎语(不支持多句)
    • "referrer_hostname" 请求来源自定义名称(根据 host,支持多句随机)
    • "model_message" 模型切换欢迎语(根据模型 ID,支持多句随机)
    • "hitokoto_api_message",一言 API 输出模板(不支持多句随机)
  • "mouseover" 鼠标触发提示(根据 CSS 选择器,支持多句随机)

  • "click" 鼠标点击触发提示(根据 CSS 选择器,支持多句随机)

  • "seasons" 节日提示(日期段,支持多句随机)

14、定制鼠标样式的坑

1、定制鼠标样式:

body{/*鼠标样式改变*/cursor: url("static/pence.ico"),auto;}

2、鼠标点击效果

<!--鼠标点击效果--><script src="https://cdn.jsdelivr.net/gh/TRHX/CDN-for-itrhx.com@3.0.8/js/maodian.js"></script>

坑:

1.图片不能大于32*32

2.尽量是ico格式

3.逗号后面加auto

附:

在线修改图片大小网址:在线图片大小修改器,图片尺寸修改,格式转换【免费】

png等格式转ico:PNG转ICO - 在线转换图标文件

15、悬挂的喵:点击回到顶部

1.把下面html的内容与你的index.html内容进行比较,差不多就是如下格式:

<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">然后是大量的css内容
<style>...
</style>
然后导入jquery
<script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.js"></script>
最后导入js代码
<script>$(function () {...})</script>

上面要导入的详细信息在下面的代码中。

<!DOCTYPE html>
<html lang="zh"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><meta http-equiv="X-UA-Compatible" content="ie=edge"><title>返回顶部(小猫特效)</title><style>@media (max-width:1000px) {.back-to-top {display: none !important;}}@-webkit-keyframes wrench {0% {-webkit-transform: rotate(-12deg);transform: rotate(-12deg)}8% {-webkit-transform: rotate(12deg);transform: rotate(12deg)}10% {-webkit-transform: rotate(24deg);transform: rotate(24deg)}18%,20% {-webkit-transform: rotate(-24deg);transform: rotate(-24deg)}28%,30% {-webkit-transform: rotate(24deg);transform: rotate(24deg)}38%,40% {-webkit-transform: rotate(-24deg);transform: rotate(-24deg)}48%,50% {-webkit-transform: rotate(24deg);transform: rotate(24deg)}58%,60% {-webkit-transform: rotate(-24deg);transform: rotate(-24deg)}68% {-webkit-transform: rotate(24deg);transform: rotate(24deg)}100%,75% {-webkit-transform: rotate(0deg);transform: rotate(0deg)}}@keyframes wrench {0% {-webkit-transform: rotate(-12deg);-ms-transform: rotate(-12deg);transform: rotate(-12deg)}8% {-webkit-transform: rotate(12deg);-ms-transform: rotate(12deg);transform: rotate(12deg)}10% {-webkit-transform: rotate(24deg);-ms-transform: rotate(24deg);transform: rotate(24deg)}18%,20% {-webkit-transform: rotate(-24deg);-ms-transform: rotate(-24deg);transform: rotate(-24deg)}28%,30% {-webkit-transform: rotate(24deg);-ms-transform: rotate(24deg);transform: rotate(24deg)}38%,40% {-webkit-transform: rotate(-24deg);-ms-transform: rotate(-24deg);transform: rotate(-24deg)}48%,50% {-webkit-transform: rotate(24deg);-ms-transform: rotate(24deg);transform: rotate(24deg)}58%,60% {-webkit-transform: rotate(-24deg);-ms-transform: rotate(-24deg);transform: rotate(-24deg)}68% {-webkit-transform: rotate(24deg);-ms-transform: rotate(24deg);transform: rotate(24deg)}100%,75% {-webkit-transform: rotate(0deg);-ms-transform: rotate(0deg);transform: rotate(0deg)}}.faa-parent.animated-hover:hover>.faa-wrench,.faa-wrench.animated,.faa-wrench.animated-hover:hover {-webkit-animation: wrench 2.5s ease infinite;animation: wrench 2.5s ease infinite;transform-origin-x: 90%;transform-origin-y: 35%;transform-origin-z: initial}.faa-parent.animated-hover:hover>.faa-wrench.faa-fast,.faa-wrench.animated-hover.faa-fast:hover,.faa-wrench.animated.faa-fast {-webkit-animation: wrench 1.2s ease infinite;animation: wrench 1.2s ease infinite}.faa-parent.animated-hover:hover>.faa-wrench.faa-slow,.faa-wrench.animated-hover.faa-slow:hover,.faa-wrench.animated.faa-slow {-webkit-animation: wrench 3.7s ease infinite;animation: wrench 3.7s ease infinite}@-webkit-keyframes ring {0% {-webkit-transform: rotate(-15deg);transform: rotate(-15deg)}2% {-webkit-transform: rotate(15deg);transform: rotate(15deg)}4% {-webkit-transform: rotate(-18deg);transform: rotate(-18deg)}6% {-webkit-transform: rotate(18deg);transform: rotate(18deg)}8% {-webkit-transform: rotate(-22deg);transform: rotate(-22deg)}10% {-webkit-transform: rotate(22deg);transform: rotate(22deg)}12% {-webkit-transform: rotate(-18deg);transform: rotate(-18deg)}14% {-webkit-transform: rotate(18deg);transform: rotate(18deg)}16% {-webkit-transform: rotate(-12deg);transform: rotate(-12deg)}18% {-webkit-transform: rotate(12deg);transform: rotate(12deg)}100%,20% {-webkit-transform: rotate(0deg);transform: rotate(0deg)}}@keyframes ring {0% {-webkit-transform: rotate(-15deg);-ms-transform: rotate(-15deg);transform: rotate(-15deg)}2% {-webkit-transform: rotate(15deg);-ms-transform: rotate(15deg);transform: rotate(15deg)}4% {-webkit-transform: rotate(-18deg);-ms-transform: rotate(-18deg);transform: rotate(-18deg)}6% {-webkit-transform: rotate(18deg);-ms-transform: rotate(18deg);transform: rotate(18deg)}8% {-webkit-transform: rotate(-22deg);-ms-transform: rotate(-22deg);transform: rotate(-22deg)}10% {-webkit-transform: rotate(22deg);-ms-transform: rotate(22deg);transform: rotate(22deg)}12% {-webkit-transform: rotate(-18deg);-ms-transform: rotate(-18deg);transform: rotate(-18deg)}14% {-webkit-transform: rotate(18deg);-ms-transform: rotate(18deg);transform: rotate(18deg)}16% {-webkit-transform: rotate(-12deg);-ms-transform: rotate(-12deg);transform: rotate(-12deg)}18% {-webkit-transform: rotate(12deg);-ms-transform: rotate(12deg);transform: rotate(12deg)}100%,20% {-webkit-transform: rotate(0deg);-ms-transform: rotate(0deg);transform: rotate(0deg)}}.faa-parent.animated-hover:hover>.faa-ring,.faa-ring.animated,.faa-ring.animated-hover:hover {-webkit-animation: ring 2s ease infinite;animation: ring 2s ease infinite;transform-origin-x: 50%;transform-origin-y: 0;transform-origin-z: initial}.faa-parent.animated-hover:hover>.faa-ring.faa-fast,.faa-ring.animated-hover.faa-fast:hover,.faa-ring.animated.faa-fast {-webkit-animation: ring 1s ease infinite;animation: ring 1s ease infinite}.faa-parent.animated-hover:hover>.faa-ring.faa-slow,.faa-ring.animated-hover.faa-slow:hover,.faa-ring.animated.faa-slow {-webkit-animation: ring 3s ease infinite;animation: ring 3s ease infinite}@-webkit-keyframes vertical {0% {-webkit-transform: translate(0, -3px);transform: translate(0, -3px)}4% {-webkit-transform: translate(0, 3px);transform: translate(0, 3px)}8% {-webkit-transform: translate(0, -3px);transform: translate(0, -3px)}12% {-webkit-transform: translate(0, 3px);transform: translate(0, 3px)}16% {-webkit-transform: translate(0, -3px);transform: translate(0, -3px)}20% {-webkit-transform: translate(0, 3px);transform: translate(0, 3px)}100%,22% {-webkit-transform: translate(0, 0);transform: translate(0, 0)}}@keyframes vertical {0% {-webkit-transform: translate(0, -3px);-ms-transform: translate(0, -3px);transform: translate(0, -3px)}4% {-webkit-transform: translate(0, 3px);-ms-transform: translate(0, 3px);transform: translate(0, 3px)}8% {-webkit-transform: translate(0, -3px);-ms-transform: translate(0, -3px);transform: translate(0, -3px)}12% {-webkit-transform: translate(0, 3px);-ms-transform: translate(0, 3px);transform: translate(0, 3px)}16% {-webkit-transform: translate(0, -3px);-ms-transform: translate(0, -3px);transform: translate(0, -3px)}20% {-webkit-transform: translate(0, 3px);-ms-transform: translate(0, 3px);transform: translate(0, 3px)}100%,22% {-webkit-transform: translate(0, 0);-ms-transform: translate(0, 0);transform: translate(0, 0)}}.faa-parent.animated-hover:hover>.faa-vertical,.faa-vertical.animated,.faa-vertical.animated-hover:hover {-webkit-animation: vertical 2s ease infinite;animation: vertical 2s ease infinite}.faa-parent.animated-hover:hover>.faa-vertical.faa-fast,.faa-vertical.animated-hover.faa-fast:hover,.faa-vertical.animated.faa-fast {-webkit-animation: vertical 1s ease infinite;animation: vertical 1s ease infinite}.faa-parent.animated-hover:hover>.faa-vertical.faa-slow,.faa-vertical.animated-hover.faa-slow:hover,.faa-vertical.animated.faa-slow {-webkit-animation: vertical 4s ease infinite;animation: vertical 4s ease infinite}@-webkit-keyframes horizontal {0% {-webkit-transform: translate(0, 0);transform: translate(0, 0)}6% {-webkit-transform: translate(5px, 0);transform: translate(5px, 0)}12% {-webkit-transform: translate(0, 0);transform: translate(0, 0)}18% {-webkit-transform: translate(5px, 0);transform: translate(5px, 0)}24% {-webkit-transform: translate(0, 0);transform: translate(0, 0)}30% {-webkit-transform: translate(5px, 0);transform: translate(5px, 0)}100%,36% {-webkit-transform: translate(0, 0);transform: translate(0, 0)}}@keyframes horizontal {0% {-webkit-transform: translate(0, 0);-ms-transform: translate(0, 0);transform: translate(0, 0)}6% {-webkit-transform: translate(5px, 0);-ms-transform: translate(5px, 0);transform: translate(5px, 0)}12% {-webkit-transform: translate(0, 0);-ms-transform: translate(0, 0);transform: translate(0, 0)}18% {-webkit-transform: translate(5px, 0);-ms-transform: translate(5px, 0);transform: translate(5px, 0)}24% {-webkit-transform: translate(0, 0);-ms-transform: translate(0, 0);transform: translate(0, 0)}30% {-webkit-transform: translate(5px, 0);-ms-transform: translate(5px, 0);transform: translate(5px, 0)}100%,36% {-webkit-transform: translate(0, 0);-ms-transform: translate(0, 0);transform: translate(0, 0)}}.faa-horizontal.animated,.faa-horizontal.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-horizontal {-webkit-animation: horizontal 2s ease infinite;animation: horizontal 2s ease infinite}.faa-horizontal.animated-hover.faa-fast:hover,.faa-horizontal.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-horizontal.faa-fast {-webkit-animation: horizontal 1s ease infinite;animation: horizontal 1s ease infinite}.faa-horizontal.animated-hover.faa-slow:hover,.faa-horizontal.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-horizontal.faa-slow {-webkit-animation: horizontal 3s ease infinite;animation: horizontal 3s ease infinite}@-webkit-keyframes flash {0%,100%,50% {opacity: 1}25%,75% {opacity: 0}}@keyframes flash {0%,100%,50% {opacity: 1}25%,75% {opacity: 0}}.faa-flash.animated,.faa-flash.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-flash {-webkit-animation: flash 2s ease infinite;animation: flash 2s ease infinite}.faa-flash.animated-hover.faa-fast:hover,.faa-flash.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-flash.faa-fast {-webkit-animation: flash 1s ease infinite;animation: flash 1s ease infinite}.faa-flash.animated-hover.faa-slow:hover,.faa-flash.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-flash.faa-slow {-webkit-animation: flash 3s ease infinite;animation: flash 3s ease infinite}@-webkit-keyframes bounce {0%,10%,100%,20%,50%,80% {-webkit-transform: translateY(0);transform: translateY(0)}40%,60% {-webkit-transform: translateY(-15px);transform: translateY(-15px)}}@keyframes bounce {0%,10%,100%,20%,50%,80% {-webkit-transform: translateY(0);-ms-transform: translateY(0);transform: translateY(0)}40%,60% {-webkit-transform: translateY(-15px);-ms-transform: translateY(-15px);transform: translateY(-15px)}}.faa-bounce.animated,.faa-bounce.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-bounce {-webkit-animation: bounce 2s ease infinite;animation: bounce 2s ease infinite}.faa-bounce.animated-hover.faa-fast:hover,.faa-bounce.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-bounce.faa-fast {-webkit-animation: bounce 1s ease infinite;animation: bounce 1s ease infinite}.faa-bounce.animated-hover.faa-slow:hover,.faa-bounce.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-bounce.faa-slow {-webkit-animation: bounce 3s ease infinite;animation: bounce 3s ease infinite}@-webkit-keyframes spin {0% {-webkit-transform: rotate(0deg);transform: rotate(0deg)}100% {-webkit-transform: rotate(359deg);transform: rotate(359deg)}}@keyframes spin {0% {-webkit-transform: rotate(0deg);-ms-transform: rotate(0deg);transform: rotate(0deg)}100% {-webkit-transform: rotate(359deg);-ms-transform: rotate(359deg);transform: rotate(359deg)}}.faa-parent.animated-hover:hover>.faa-spin,.faa-spin.animated,.faa-spin.animated-hover:hover {-webkit-animation: spin 1.5s linear infinite;animation: spin 1.5s linear infinite}.faa-parent.animated-hover:hover>.faa-spin.faa-fast,.faa-spin.animated-hover.faa-fast:hover,.faa-spin.animated.faa-fast {-webkit-animation: spin .7s linear infinite;animation: spin .7s linear infinite}.faa-parent.animated-hover:hover>.faa-spin.faa-slow,.faa-spin.animated-hover.faa-slow:hover,.faa-spin.animated.faa-slow {-webkit-animation: spin 2.2s linear infinite;animation: spin 2.2s linear infinite}@-webkit-keyframes float {0% {-webkit-transform: translateY(0);transform: translateY(0)}50% {-webkit-transform: translateY(-6px);transform: translateY(-6px)}100% {-webkit-transform: translateY(0);transform: translateY(0)}}@keyframes float {0% {-webkit-transform: translateY(0);-ms-transform: translateY(0);transform: translateY(0)}50% {-webkit-transform: translateY(-6px);-ms-transform: translateY(-6px);transform: translateY(-6px)}100% {-webkit-transform: translateY(0);-ms-transform: translateY(0);transform: translateY(0)}}.faa-float.animated,.faa-float.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-float {-webkit-animation: float 2s linear infinite;animation: float 2s linear infinite}.faa-float.animated-hover.faa-fast:hover,.faa-float.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-float.faa-fast {-webkit-animation: float 1s linear infinite;animation: float 1s linear infinite}.faa-float.animated-hover.faa-slow:hover,.faa-float.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-float.faa-slow {-webkit-animation: float 3s linear infinite;animation: float 3s linear infinite}@-webkit-keyframes pulse {0% {-webkit-transform: scale(1.1);transform: scale(1.1)}50% {-webkit-transform: scale(0.8);transform: scale(0.8)}100% {-webkit-transform: scale(1.1);transform: scale(1.1)}}@keyframes pulse {0% {-webkit-transform: scale(1.1);-ms-transform: scale(1.1);transform: scale(1.1)}50% {-webkit-transform: scale(0.8);-ms-transform: scale(0.8);transform: scale(0.8)}100% {-webkit-transform: scale(1.1);-ms-transform: scale(1.1);transform: scale(1.1)}}.faa-parent.animated-hover:hover>.faa-pulse,.faa-pulse.animated,.faa-pulse.animated-hover:hover {-webkit-animation: pulse 2s linear infinite;animation: pulse 2s linear infinite}.faa-parent.animated-hover:hover>.faa-pulse.faa-fast,.faa-pulse.animated-hover.faa-fast:hover,.faa-pulse.animated.faa-fast {-webkit-animation: pulse 1s linear infinite;animation: pulse 1s linear infinite}.faa-parent.animated-hover:hover>.faa-pulse.faa-slow,.faa-pulse.animated-hover.faa-slow:hover,.faa-pulse.animated.faa-slow {-webkit-animation: pulse 3s linear infinite;animation: pulse 3s linear infinite}.faa-parent.animated-hover:hover>.faa-shake,.faa-shake.animated,.faa-shake.animated-hover:hover {-webkit-animation: wrench 2.5s ease infinite;animation: wrench 2.5s ease infinite}.faa-parent.animated-hover:hover>.faa-shake.faa-fast,.faa-shake.animated-hover.faa-fast:hover,.faa-shake.animated.faa-fast {-webkit-animation: wrench 1.2s ease infinite;animation: wrench 1.2s ease infinite}.faa-parent.animated-hover:hover>.faa-shake.faa-slow,.faa-shake.animated-hover.faa-slow:hover,.faa-shake.animated.faa-slow {-webkit-animation: wrench 3.7s ease infinite;animation: wrench 3.7s ease infinite}@-webkit-keyframes tada {0% {-webkit-transform: scale(1);transform: scale(1)}10%,20% {-webkit-transform: scale(.9) rotate(-8deg);transform: scale(.9) rotate(-8deg)}30%,50%,70% {-webkit-transform: scale(1.3) rotate(8deg);transform: scale(1.3) rotate(8deg)}40%,60% {-webkit-transform: scale(1.3) rotate(-8deg);transform: scale(1.3) rotate(-8deg)}100%,80% {-webkit-transform: scale(1) rotate(0);transform: scale(1) rotate(0)}}@keyframes tada {0% {-webkit-transform: scale(1);-ms-transform: scale(1);transform: scale(1)}10%,20% {-webkit-transform: scale(.9) rotate(-8deg);-ms-transform: scale(.9) rotate(-8deg);transform: scale(.9) rotate(-8deg)}30%,50%,70% {-webkit-transform: scale(1.3) rotate(8deg);-ms-transform: scale(1.3) rotate(8deg);transform: scale(1.3) rotate(8deg)}40%,60% {-webkit-transform: scale(1.3) rotate(-8deg);-ms-transform: scale(1.3) rotate(-8deg);transform: scale(1.3) rotate(-8deg)}100%,80% {-webkit-transform: scale(1) rotate(0);-ms-transform: scale(1) rotate(0);transform: scale(1) rotate(0)}}.faa-parent.animated-hover:hover>.faa-tada,.faa-tada.animated,.faa-tada.animated-hover:hover {-webkit-animation: tada 2s linear infinite;animation: tada 2s linear infinite}.faa-parent.animated-hover:hover>.faa-tada.faa-fast,.faa-tada.animated-hover.faa-fast:hover,.faa-tada.animated.faa-fast {-webkit-animation: tada 1s linear infinite;animation: tada 1s linear infinite}.faa-parent.animated-hover:hover>.faa-tada.faa-slow,.faa-tada.animated-hover.faa-slow:hover,.faa-tada.animated.faa-slow {-webkit-animation: tada 3s linear infinite;animation: tada 3s linear infinite}@-webkit-keyframes passing {0% {-webkit-transform: translateX(-50%);transform: translateX(-50%);opacity: 0}50% {-webkit-transform: translateX(0%);transform: translateX(0%);opacity: 1}100% {-webkit-transform: translateX(50%);transform: translateX(50%);opacity: 0}}@keyframes passing {0% {-webkit-transform: translateX(-50%);-ms-transform: translateX(-50%);transform: translateX(-50%);opacity: 0}50% {-webkit-transform: translateX(0%);-ms-transform: translateX(0%);transform: translateX(0%);opacity: 1}100% {-webkit-transform: translateX(50%);-ms-transform: translateX(50%);transform: translateX(50%);opacity: 0}}.faa-parent.animated-hover:hover>.faa-passing,.faa-passing.animated,.faa-passing.animated-hover:hover {-webkit-animation: passing 2s linear infinite;animation: passing 2s linear infinite}.faa-parent.animated-hover:hover>.faa-passing.faa-fast,.faa-passing.animated-hover.faa-fast:hover,.faa-passing.animated.faa-fast {-webkit-animation: passing 1s linear infinite;animation: passing 1s linear infinite}.faa-parent.animated-hover:hover>.faa-passing.faa-slow,.faa-passing.animated-hover.faa-slow:hover,.faa-passing.animated.faa-slow {-webkit-animation: passing 3s linear infinite;animation: passing 3s linear infinite}@-webkit-keyframes passing-reverse {0% {-webkit-transform: translateX(50%);transform: translateX(50%);opacity: 0}50% {-webkit-transform: translateX(0%);transform: translateX(0%);opacity: 1}100% {-webkit-transform: translateX(-50%);transform: translateX(-50%);opacity: 0}}@keyframes passing-reverse {0% {-webkit-transform: translateX(50%);-ms-transform: translateX(50%);transform: translateX(50%);opacity: 0}50% {-webkit-transform: translateX(0%);-ms-transform: translateX(0%);transform: translateX(0%);opacity: 1}100% {-webkit-transform: translateX(-50%);-ms-transform: translateX(-50%);transform: translateX(-50%);opacity: 0}}.faa-parent.animated-hover:hover>.faa-passing-reverse,.faa-passing-reverse.animated,.faa-passing-reverse.animated-hover:hover {-webkit-animation: passing-reverse 2s linear infinite;animation: passing-reverse 2s linear infinite}.faa-parent.animated-hover:hover>.faa-passing-reverse.faa-fast,.faa-passing-reverse.animated-hover.faa-fast:hover,.faa-passing-reverse.animated.faa-fast {-webkit-animation: passing-reverse 1s linear infinite;animation: passing-reverse 1s linear infinite}.faa-parent.animated-hover:hover>.faa-passing-reverse.faa-slow,.faa-passing-reverse.animated-hover.faa-slow:hover,.faa-passing-reverse.animated.faa-slow {-webkit-animation: passing-reverse 3s linear infinite;animation: passing-reverse 3s linear infinite}@-webkit-keyframes burst {0% {opacity: .6}50% {-webkit-transform: scale(1.8);transform: scale(1.8);opacity: 0}100% {opacity: 0}}@keyframes burst {0% {opacity: .6}50% {-webkit-transform: scale(1.8);-ms-transform: scale(1.8);transform: scale(1.8);opacity: 0}100% {opacity: 0}}.faa-burst.animated,.faa-burst.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-burst {-webkit-animation: burst 2s infinite linear;animation: burst 2s infinite linear}.faa-burst.animated-hover.faa-fast:hover,.faa-burst.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-burst.faa-fast {-webkit-animation: burst 1s infinite linear;animation: burst 1s infinite linear}.faa-burst.animated-hover.faa-slow:hover,.faa-burst.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-burst.faa-slow {-webkit-animation: burst 3s infinite linear;animation: burst 3s infinite linear}@-webkit-keyframes falling {0% {-webkit-transform: translateY(-50%);transform: translateY(-50%);opacity: 0}50% {-webkit-transform: translateY(0%);transform: translateY(0%);opacity: 1}100% {-webkit-transform: translateY(50%);transform: translateY(50%);opacity: 0}}@keyframes falling {0% {-webkit-transform: translateY(-50%);-ms-transform: translateY(-50%);transform: translateY(-50%);opacity: 0}50% {-webkit-transform: translateY(0%);-ms-transform: translateY(0%);transform: translateY(0%);opacity: 1}100% {-webkit-transform: translateY(50%);-ms-transform: translateY(50%);transform: translateY(50%);opacity: 0}}.faa-falling.animated,.faa-falling.animated-hover:hover,.faa-parent.animated-hover:hover>.faa-falling {-webkit-animation: falling 2s linear infinite;animation: falling 2s linear infinite}.faa-falling.animated-hover.faa-fast:hover,.faa-falling.animated.faa-fast,.faa-parent.animated-hover:hover>.faa-falling.faa-fast {-webkit-animation: falling 1s linear infinite;animation: falling 1s linear infinite}.faa-falling.animated-hover.faa-slow:hover,.faa-falling.animated.faa-slow,.faa-parent.animated-hover:hover>.faa-falling.faa-slow {-webkit-animation: falling 3s linear infinite;animation: falling 3s linear infinite}.topButton {display: none !important}#goToTop {display: none !important}.back-to-top {cursor: pointer;position: fixed;right: 80px;top: -900px;z-index: 2;width: 70px;height: 900px;background: url("http://zhouql.vip/cdn/images/scroll.png");transition: all .5s ease-in-out;opacity: 1;}</style><script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.js"></script>
</head><body style="height: 2000px;"><div class="back-to-top cd-top faa-float animated cd-is-visible" style="top: -900px;"></div><script>$(function () {$(window).scroll(function () {var scroHei = $(window).scrollTop();if (scroHei > 500) {$('.back-to-top').css('top', '-200px');// $('.back-to-top').fadeIn();} else {$('.back-to-top').css('top', '-999px');// $('.back-to-top').fadeOut();}})$('.back-to-top').click(function () {$('body,html').animate({scrollTop: 0}, 600);})})</script>
</body></html>

15、css中的单位:效果之一,文字随着页面大小变化而变化

单位 解析
px 绝对单位。像素
em 相对单位。相对于其父元素的font-size,如父元素font-size: 12px;那么子元素1em = 12px,1em = 24px,…
rem 相对单位。相对于其根元素()的font-size,如html{font-size: 12px};那么页面中1em = 12px,1em = 24px,…
vw 相对单位。相对页面视口的宽度,页面总宽度为100vw,页面宽度的1% = 1vw
vh 相对单位。相对页面视口的高度,页面总高度为100vw,页面高度的1% = 1vh
vmin 相对单位。相对于页面视口的宽度与高度,取其最小值;如页面宽1000px,高1200px,那么1vmin = 10px
vmax 相对单位。相对于页面视口的宽度与高度,取其最大值;如页面宽1000px,高1200px,那么1vmin = 12px
% 相对单位。相对于父元素的宽度;如父元素宽100px,那么1% = 1px
in 绝对单位,寸
cm 绝对单位,厘米
mm 绝对单位,毫米
pt 绝对单位,point,大约 1/72寸
pc 绝对单位,pica,大约6pt,1/6寸

16、安装axios

文档:vue-axios|axios中文网 | axios (axios-js.com)

npm install --save axios vue-axios

将下面代码加入入口文件:

import Vue from 'vue'
import axios from 'axios'
import VueAxios from 'vue-axios'Vue.use(VueAxios, axios)

你可以按照以下方式使用:

Vue.axios.get(api).then((response) => {console.log(response.data)
})this.axios.get(api).then((response) => {console.log(response.data)
})this.$http.get(api).then((response) => {console.log(response.data)
})

可以通过向 axios 传递相关配置来创建请求

axios(config)
// 发送 POST 请求
axios({method: 'post',url: '/user/12345',data: {firstName: 'Fred',lastName: 'Flintstone'}
});
// 获取远端图片
axios({method:'get',url:'http://bit.ly/2mTM3nY',responseType:'stream'
}).then(function(response) {response.data.pipe(fs.createWriteStream('ada_lovelace.jpg'))
});

注可能出现错误,所有使用axios的方法更正为:

main.js 文件中写:

Vue.prototype.$axios = axios

在需要使用时:

this.$axios.get('url').then(response=>{// data1是已经写好了与json格式一样的变量,用来接收、response.data是接收的数据this.data1 = response.data});

17、博客界面初始化:获取所有博客的特点信息

1.要通过 json 格式去获取返回的博文大全的数据,这里暂时没有后端,所以我们写一个json文件 临时存放数据

注:注意接收的变量中字段名要与 json 中的一样
0、此处是数据库中对应表的字段

- blog_id                      (自增)、
- title                          (标题)、
- blog_Summary            (摘要)
- blog_content             (博客内容)、// 博文内容暂时还不需要
- user_id                     (表示是哪个用户发的博客)、
- comment_id              (对应评论表中id)、
- audit                         (审核状态:0(通过)、1(未通过)、2(未审核))、
- like                          (点赞数,可多次点赞)、
- release_time              (发布时间)、
- mtime                      (修改时间)、
- browse                     (浏览量)、
- top                           (置顶:0:置顶,1:不置顶)

2、先写好json文件data_blog.json


{"success": 200,"blog_data": [{"id": "1","title": "标题","blog_Summary": "摘要","user_id": "表示是哪个用户发的博客","comment_count": "评论数量,类中没有需要再查","like": "点赞数,可多次点赞","release_time": "发布时间","browse": "浏览量","top": "置顶:0:置顶,1:不置顶"},{"id":"2","title": "书单","blog_Summary": "读书使人得到一种优雅和风味,这就是读书的整个目的,而只有抱着这种目的的读书才可以叫做艺术。一人读书的目的并不是要“改进心智”,因为当他开始想要改进心智的时候,一切读书的乐趣便丧失净尽了。","user_id": "2","comment_count": "14","like": "85","release_time": "2019-01-21","browse": "9954","top": "0"},{"id": "3","title": "多态到底有什么用?","blog_Summary": "结论:多态可有可无但又至关重要,在不考虑代码今后的发展时,多态是无用的,如果考虑以后代码的修改与增删,多态是能大大提高代码的扩展性与复用的。想搞清楚有什么用之前,先说说什么是多态:多态可以理解为一个事物可以拥有多种形态(自己理解的,后面会讲为啥)规范定义:1.需要有","user_id": "3","comment_count": "99","like": "100","release_time": "2022-11-17","browse": "170","top": "0"},{"id":"4","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id": "5","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id":"6","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id":"7","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id":"8","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id": "9","title": "单例模式在多线程下的数据修改问题(即线程不安全),spring中是如何保证单例的线程安全问题的?","blog_Summary": "原因:在步骤3进入且准备更改步骤1的值时此时步骤2的线程获取到了步骤1的值且在步骤3赋值完成且输出后线程的值才输出,所以导致了线程在步骤1的值已经修改的情况下还能获取到步骤1的值。面试官原问题是:在单例模式下,类A获取单例对象且修改对象中的属性值,然后类B也获取对象也修","user_id": "4","comment_count": "9","like": "10","release_time": "2022-11-15","browse": "263","top": "1"},{"id": "9","title": "十、二、八、十六进制间转换(包含负数间进制转换)原码、补码、反码定义","blog_Summary": "十、二、八、十六进制间转换(包含负数间进制转换)0.0、有符号的 1 字节表示范围。\u200B 我们知道 1字节 (byte) = 8 位(bit)(即8位二进制,如:11111111)、1024个字节 = 1 K 、1024 K = 1M\u200B 所以,一字节可以表示的范围就有 0 ~ 256 (无符号数),而有符号数表示的范围则是: -128 ~ 127\u200B","user_id": "10","comment_count": "2","like": "7","release_time": "2021-05-18","browse": "3189","top": "1"}]}

3、我们要接收到 json 的数据,就要先定义好一个和 json 一模一样格式的变量去接收

data(){return{data1:{success:200,blog_data:[{id:"",title: "",blog_Summary:"",user_id: "",comment_count:"",like: "",release_time:"",browse: "",top: ""}]}}}

4、编写axios、因为是一进去就要有的数据,所以使用vue的钩子函数:mounted、然后在里面写异步通信:axios函数

mounted() {// 钩子函数this.$axios.get('http://localhost:8080/static/data/data_blog.json').then(response=>{// 这句话的意思是,我从返回的数据中选择blog_data的数据赋值给data1变量(response.data代表所有返回的数据)this.data1 = response.data.blog_data});
}

5、显示数据、因为 json 数据不止一条,所以我们使用vue的for去输出所有数据、我从项目代码中截取的,加了些样式

<div v-for="i in this.data1"><!--一篇博客--><div style="margin: 0 0 52px 0;padding-top: 49px;"><!--存储标题和置顶--><div><!--标题,用 a 标签--><a style="font-size: 2.5vw;color: #212121;transition: color .3s ease;font-family: Merriweather,'Open Sans','Microsoft Jhenghei','Microsoft Yahei',sans-serif;font-weight:bold;">{{i.title}}</a><!--置顶:里面没有值除非传入,值为 0 时代表需要置顶--><span v-if='i.top=="0"' style="font-size: 1vw;color: #fff;background: #6fa3ef;border-radius: 3px;padding: 1px 3px;">置顶</span></div><!--存储时间,阅读,评论,推荐(推荐值即点赞数)--><span style="font-size: 0.9vw;color: #888;">&nbsp;&nbsp;<!--发表--><i></i>◕ &nbsp;发表于 {{i.release_time}}&nbsp;&nbsp;&nbsp;<!--阅读--><i></i>♒ &nbsp;阅读:{{i.browse}}&nbsp;&nbsp;&nbsp;<!--评论--><i></i>♩ &nbsp;评论:{{i.comment_count}}&nbsp;&nbsp;&nbsp;<!--推荐--><i></i>❤ &nbsp;推荐:{{i.like}}</span><!--存储摘要和阅读全文的文字--><div style="font-size:1.1vw;padding-top: 5px;color: #212121"><!--摘要-->摘要:{{i.blog_Summary}}<!--阅读全文--><router-link to="" style="display:block;padding-top: 25px;font-weight: bolder;opacity: 0.9">阅读全文 »</router-link></div></div><!--分割线--><hr></div>

18、对登陆数据验证

原文:在vue中使用rules的定义和校验规则_蓝木蓝木的博客-CSDN博客_rules vue

在vue中使用rules的定义和校验规则

表单内容里面定义属性:

<Form ref="rulesForm"  :model="rulesForm"  :label-width="100" :rules="rules"><FormItem label="名称" prop="name"><Input v-model="rulesForm.name" placeholder="名称"/></FormItem>
</Form>

在data()里面写具体的规则:

rules {  name: [ { type: 'string',required: true,message: "名称必填", trigger: 'blur'}, {max: 30,message: "名称长度不能超过30位" }]}

rules:与上文 ‘表单内容’ 中 表单的 :rules 属性值相同
prop:与规则中的name属性相同
验证内容:必填,blur是失去焦点验证,max是最大长度验证(min最小长度)

直接用pattern进行匹配验证:

name: [ { pattern: pattern验证, required: true, message: "提示信息", trigger: "blur" }]

前端Vue中常用rules校验规则:

1、是否合法IP地址:

pattern:/^(\d{1,2}|1\d\d|2[0-4]\d|25[0-5]).(\d{1,2}|1\d\d|2[0-4]\d|25[0-5]).(\d{1,2}|1\d\d|2[0-4]\d|25[0-5]).(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])$/,

2.是否手机号码或者固话
pattern:/^((0\d{2,3}-\d{7,8})|(1[34578]\d{9}))$/,

3.是否身份证号码
pattern:/(\d{15}$)|(^\d{18}$)|(\d{17}(\d|X|x)$)/,

4.是否邮箱
pattern:/^([a-zA-Z0-9]+[-_.]?)+@[a-zA-Z0-9]+.[a-z]+$/,

5.整数填写
pattern:/^-?[1-9]\d*$/,

6.正整数填写
pattern:/1\d*$/,

7.小写字母
pattern:/2+$/,

8.大写字母
pattern:/3+$/,

9.大小写混合
pattern:/4+$/,

10.多个8位数字格式(yyyyMMdd)并以逗号隔开
pattern:/^\d{8}(,\d{8})*$/,

11.数字加英文,不包含特殊字符
pattern:/5+$/,

12.前两位是数字后一位是英文
pattern:/^\d{2}[a-zA-Z]+$/,

13.密码校验(6-20位英文字母、数字或者符号(除空格),且字母、数字和标点符号至少包含两种)
pattern:/(?![\d]+$)(?![a-zA-Z]+$)(?![\da-zA-Z]+KaTeX parse error: Got function '\u' with no arguments as superscript at position 5: )([^\̲u̲4e00-\u9fa5\s])…/,

14.中文校验
pattern:/6+$/,

19、vue在第二次跳转同一路由跳转数据不更新

原因

使用router-view时,如果在相同路由之间跳转,默认在跳转路由时会采用缓存策略,并不会刷新当前路由组件。即mounted,beforeDestory等钩子函数并不会触发。vue 同一路由跳转不走生命周期,导致数据不更新。

解决办法

办法一:

使用watch 监听路由变化。手动更新数据。

办法二:

使用 <router-view :key="$route.fullPath"/> 就是每次跳转加个随机值,那么就不一样了

办法三:

把你mounted(){} 里面执行的办法在 activated(){} 里面在执行一遍,完美解决。

这里选择第二种、其他方法暂时没搞明白

1、我们一般显示组件是用

<router-view />

2、我们将其修改为:

<router-view :key="key1" />

3、即每次组件跳转都要带key1的值,key1的值使用计算属性去设置,全部代码为:

<template><div id="app"> <router-view :key="key1" /> </div>
</template><script>
export default {name: 'App',data(){return{}},computed: { // 计算属性key1(){return this.$route.path + Math.random(); // 生成以时间为标准的随机数}}
}

20、编写登陆界面

注:登陆界面还是自己写吧,丑就丑了点

问题:我想使用别人的登陆界面,或者是一些UI框架,但无一例外,他们使用的css样式中设置了lang="scss" 属性,导致在项目打包发布时会却一些模块,当你安装模块后,又会显示你某些模块的版本有问题,当你一个个改版本后,发现你项目的模块彻底崩盘了,运行不了。所以慎用这个属性

<style lang="scss" scoped>

最后还是白嫖了一个登陆界面,自己就加了一个背景图片

<template><div class="building"><el-form ref="loginForm" :model="form" :rules="rules" label-width="80px" class="login-box"><h3 class="login-title">欢迎登录</h3><el-form-item label="账号" prop="username"><el-input type="text" placeholder="请输入账号" v-model="form.username"/></el-form-item><el-form-item label="密码" prop="password"><el-input type="password" placeholder="请输入密码" v-model="form.password"/></el-form-item><el-form-item><el-button type="primary" v-on:click="onSubmit('loginForm')">登录</el-button></el-form-item></el-form><el-dialogtitle="温馨提示":visible.sync="dialogVisible"width="30%":before-close="handleClose"><span>请输入账号和密码</span><span slot="footer" class="dialog-footer"><el-button type="primary" @click="dialogVisible = false">确 定</el-button></span></el-dialog></div></template><script>
export default {name: "Land",data() {return {form: {username: '',password: ''},// 表单验证,需要在 el-form-item 元素中增加 prop 属性rules: {username: [{required: true, message: '账号不可为空', trigger: 'blur'}],password: [{required: true, message: '密码不可为空', trigger: 'blur'}]},// 对话框显示和隐藏dialogVisible: false}},methods: {onSubmit(formName) {// 为表单绑定验证功能this.$refs[formName].validate((valid) => {if (valid) {// 使用 vue-router 路由到指定页面,该方式称之为编程式导航this.$router.push("/");} else {this.dialogVisible = true;return false;}});}}
}
</script><style  scoped>
.login-box {border: 1px solid #DCDFE6;width: 350px;margin: 180px auto;padding: 35px 35px 15px 35px;border-radius: 5px;-webkit-border-radius: 5px;-moz-border-radius: 5px;box-shadow: 0 0 25px #909399;
}.login-title {text-align: center;margin: 0 auto 40px auto;color: #303133;
}
.building {background: url("https://imgapi.xl0408.top/index.php") ;  /* API接口大全:https://imgapi.xl0408.top/index.php http://prlrr.com/wallpaper-API/  https://img.r10086.com/  https://api.r10086.com/img-api.php?type=动漫综合1 "https://api.ixiaowai.cn/api/api.php" "../assets/test01.jpg"*/height: 100%;width: 100%;position: absolute; /*fixed:绝对定位,absolute:固定在那,滑动时位置改变*/background-size: 100% 100%;top: 0;  /* 这里是设置与顶部的距离*/left: 0;  /* 这里是设置与左边的距离*/
}
</style>

21、vue 回到上一个页面

参考:[vue项目如何实现返回上一页 - 腾讯云开发者社区-腾讯云 (tencent.com)]

vue 返回上一页有两种方法:

如果使用的是 vue-router ,this.$router.go(-1) 就可以回到上一页。 history.go(-1) 是回到浏览器上一页。

但是由于 Vue 应用是单页应用,浏览器的访问历史未必和 Vue 的浏览历史相同。

还有一点,就是使用 router 跳转的时候,Vue 不会重新加载 CSS 。比如从 A 页面跳到 B 页面,会沿用 A 页面中的 CSS 样式,我在 A 页面中设置了 .content 的上边距是 20px ,B 页面没有设置边距,但如果从 A 页面跳到 B 页面,B 页面中的 .content 也会带有 20px 的上边距。

解决办法就是给 style 标签添加 scope 属性。

Vue的 style 中使用 scope 属性,浏览器渲染后,会给每个组件中的元素增加自定义属性,浏览器渲染样式时会变成 data-v-xxx

这也是 scoped 的工作原理,所以在子组件中写的元素,只有子组件中的自定义属性,而父组件中加的样式,最终浏览器渲染时是找不到对应的元素的,(因为父组件中样式给出的自定义属性是不一致的),所以子组件中的样式没办法在父组件中修改。

这样也就解决了,页面跳转上个页面的 CSS 样式也被带过来的问题。

22、Vue页面跳转后位置不在顶部的解决办法

注:使用方式1好像会导致悬挂的喵失效,方式2就不会

方式1:注:在中间增加下面代码,不要在末尾,会不生效

router.beforeEach((to, from, next) => {    // chromedocument.body.scrollTop = 0// firefoxdocument.documentElement.scrollTop = 0// safariwindow.pageYOffset = 0next()
})或者下面代码:// 跳转后返回顶部router.afterEach((to,from,next) => {window.scrollTo(0,0);})

方式二:如果需要某个页面跳转回顶部,仅需要在当前页面(page)合适的位置加入一下代码即可:

 // chromedocument.body.scrollTop = 0// firefoxdocument.documentElement.scrollTop = 0// safariwindow.pageYOffset = 0

方式三:在路由的index.js中,加入以下代码也可实现路由改变定位到顶部的效果,savedPosition当且仅当通过浏览器的前进/后退按钮触发时才可用,代码如下:

scrollBehavior(to, from, savedPosition) {if (savedPosition) {return savedPosition}return {x: 0,y: 0}
}

至此前端告一小段落了,要开始写后端了

后端

  • 使用Mysql数据库,使用SpringBoot框架
  • 整合 MyBatis
  • 使用 Druid 数据库连接池 (有日志监控)
  • 使用 SpringSecurity (Spring 的安全框架,使用其中的身份认证与授权功能,即权限控制 )
  • 使用 token
  • 使用 Swagger 功能,即实时 API 接口文档功能
  • 使用 JWT 进行基于 Token 的前后端分离用户认证(和上面的安全框架择其一,具体再了解)

先写要用到的技术,至于功能慢慢看。

后端编写时问题

1、编写数据库表

1.varchar相当于String类型,能存储的汉字数看是使用什么编码,utf8(2个为一个汉字),所以varchar(30)只能输入15个汉字

2.String 可以存储 4G 的数据

3.微型文本可使用 tinytext 类型 ,2^8-1

4.text 文本串 2^16-1 保存大文本

5.大型文本可使用 mediumtext 类型,百万级别

6.在使用 MyBatis 时,数据库字段名与实体类成员变量名保持一致,但数据库中 user_id 可在实体类中写成 userId,MyBatis 会自动识别。
注:不能识别,但是在编写sql语句时可以给字段加别名,别名设置为你实体类中对应的成员变量名就行

7.图片类型存储地址,存储在后端中,且前端访问是通过链接,而不是相对地址或者决对地址,相当于前端跨域请求了一个图片

用户表/博客表/标签表/评论表:
注:如下字段不代表最终字段、有些名字改了一点,有些字段属性更改了,具体看GitHub上的源码

/*[15:35:33][0 ms]*/ CREATE DATABASE `blog_springboot_vue`CHARACTER SET utf8 COLLATE utf8_general_ci;
/*[15:35:34][0 ms]*/ SHOW DATABASES;
/*[15:35:34][0 ms]*/ USE `blog_springboot_vue`;
/*[15:35:39][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[15:35:46][0 ms]*/ SHOW CHARSET;
/*[15:36:10][0 ms]*/ SHOW COLLATION;
/*[15:36:10][0 ms]*/ SHOW COLLATION;
/*[15:52:15][511 ms]*/ CREATE TABLE `blog_springboot_vue`.`user`( `id` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '用户id', `name` VARCHAR(30) NOT NULL COMMENT '用户名', `pwd` VARCHAR(30) NOT NULL COMMENT '密码', `nickname` VARCHAR(30) NOT NULL COMMENT '昵称', `root` INT(4) UNSIGNED NOT NULL DEFAULT 2 COMMENT '权限(0:管理员,1:普通用户)', `email` VARCHAR(30) NOT NULL COMMENT '邮箱', `head` VARCHAR(100) COMMENT '头像地址', `registertime` VARCHAR(30) DEFAULT '0000-00-00' COMMENT '注册时间', `birthday` VARCHAR(30) DEFAULT '0000-00-00' COMMENT '生日', `age` INT(4) UNSIGNED DEFAULT 18 COMMENT '年龄', PRIMARY KEY (`id`) ) ENGINE=INNODB CHARSET=utf8 COLLATE=utf8_general_ci;
/*[15:52:18][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[15:52:18][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[15:52:18][0 ms]*/ SHOW CHARSET;
/*[15:52:18][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'user';
/*[15:52:18][0 ms]*/ SHOW CHARSET;
/*[15:52:18][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`user`;
/*[15:52:18][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`user`;
/*[15:52:18][0 ms]*/ SHOW COLLATION;
/*[15:52:34][0 ms]*/ SHOW COLLATION;
/*[15:52:34][13 ms]*/ SHOW COLLATION;
/*[16:08:19][935 ms]*/ CREATE TABLE `blog_springboot_vue`.`blog`( `id` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '文章id', `title` VARCHAR(100) NOT NULL COMMENT '文章标题', `Summary` TINYTEXT COMMENT '摘要', `content` MEDIUMTEXT COMMENT '文章全文', `user_id` INT(4) NOT NULL COMMENT '哪个用户的文章', `audit` INT(4) NOT NULL COMMENT '审核状态:0(通过)、1(未通过)、2(未审核)', `like` INT(10) NOT NULL COMMENT '点赞数,可多次点赞', `release_time` VARCHAR(30) NOT NULL COMMENT '发布时间', `mtime` VARCHAR(30) DEFAULT '0' COMMENT '修改时间', `browse` INT(10) NOT NULL DEFAULT 0 COMMENT '浏览量', `top` INT(4) NOT NULL DEFAULT 1 COMMENT '0:置顶,1:不置顶', PRIMARY KEY (`id`) ) ENGINE=INNODB CHARSET=utf8 COLLATE=utf8_general_ci;
/*[16:08:21][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'blog';
/*[16:08:21][0 ms]*/ SHOW CHARSET;
/*[16:08:22][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`blog`;
/*[16:08:22][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`blog`;
/*[16:08:22][0 ms]*/ SHOW COLLATION;
/*[16:08:22][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:09:19][1417 ms]*/ ALTER TABLE `blog_springboot_vue`.`user` CHANGE `name` `name` VARCHAR(100) CHARSET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '用户名', CHANGE `pwd` `pwd` VARCHAR(100) CHARSET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '密码', CHANGE `nickname` `nickname` VARCHAR(100) CHARSET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '昵称', CHANGE `email` `email` VARCHAR(100) CHARSET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '邮箱', CHANGE `head` `head` VARCHAR(200) CHARSET utf8 COLLATE utf8_general_ci NULL COMMENT '头像地址', CHANGE `registertime` `registertime` VARCHAR(50) CHARSET utf8 COLLATE utf8_general_ci DEFAULT '0000-00-00' NULL COMMENT '注册时间', CHANGE `birthday` `birthday` VARCHAR(50) CHARSET utf8 COLLATE utf8_general_ci DEFAULT '0000-00-00' NULL COMMENT '生日';
/*[16:09:21][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'user';
/*[16:09:21][0 ms]*/ SHOW CHARSET;
/*[16:09:21][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`user`;
/*[16:09:21][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`user`;
/*[16:09:21][0 ms]*/ SHOW COLLATION;
/*[16:09:21][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:09:40][133 ms]*/ ALTER TABLE `blog_springboot_vue`.`blog` CHANGE `release_time` `release_time` VARCHAR(50) CHARSET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '发布时间', CHANGE `mtime` `mtime` VARCHAR(50) CHARSET utf8 COLLATE utf8_general_ci DEFAULT '0' NULL COMMENT '修改时间';
/*[16:09:42][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'blog';
/*[16:09:42][0 ms]*/ SHOW CHARSET;
/*[16:09:42][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`blog`;
/*[16:09:42][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`blog`;
/*[16:09:42][0 ms]*/ SHOW COLLATION;
/*[16:09:42][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:09:49][0 ms]*/ SHOW CHARSET;
/*[16:09:58][0 ms]*/ SHOW COLLATION;
/*[16:09:58][0 ms]*/ SHOW COLLATION;
/*[16:16:59][1090 ms]*/ CREATE TABLE `blog_springboot_vue`.`comment`( `id` INT(4) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '评论id', `blog_id` INT(4) NOT NULL COMMENT '属于哪个博客', `level` INT(4) NOT NULL COMMENT '1:表示父级为博客,2:表示是评论下的评论(即回复)', `level2` INT(4) NOT NULL DEFAULT -1 COMMENT '默认为-1,如果是二级评论则记录父级评论id', `user_id` INT(4) UNSIGNED NOT NULL COMMENT '发布者id', `time` VARCHAR(50) COMMENT '发布时间', `content` TINYTEXT COMMENT '评论内容', `audit` INT(4) NOT NULL DEFAULT 2 COMMENT '审核状态(0(通过)、1(未通过)、2(未审核)', `like` INT(10) NOT NULL DEFAULT 0 COMMENT '点赞数', PRIMARY KEY (`id`) ) ENGINE=INNODB CHARSET=utf8 COLLATE=utf8_general_ci;
/*[16:17:02][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:17:02][0 ms]*/ SHOW CHARSET;
/*[16:17:02][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'comment';
/*[16:17:02][0 ms]*/ SHOW CHARSET;
/*[16:17:02][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`comment`;
/*[16:17:02][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`comment`;
/*[16:17:02][0 ms]*/ SHOW COLLATION;
/*[16:17:34][0 ms]*/ SHOW COLLATION;
/*[16:17:34][0 ms]*/ SHOW COLLATION;
/*[16:18:24][461 ms]*/ ALTER TABLE `blog_springboot_vue`.`comment` CHANGE `id` `id` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '评论id', CHANGE `blog_id` `blog_id` INT(10) NOT NULL COMMENT '属于哪个博客', CHANGE `user_id` `user_id` INT(10) UNSIGNED NOT NULL COMMENT '发布者id';
/*[16:18:26][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'comment';
/*[16:18:26][0 ms]*/ SHOW CHARSET;
/*[16:18:26][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`comment`;
/*[16:18:26][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`comment`;
/*[16:18:26][0 ms]*/ SHOW COLLATION;
/*[16:18:26][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:18:45][396 ms]*/ ALTER TABLE `blog_springboot_vue`.`blog` CHANGE `user_id` `user_id` INT(10) NOT NULL COMMENT '哪个用户的文章';
/*[16:18:47][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'blog';
/*[16:18:47][0 ms]*/ SHOW CHARSET;
/*[16:18:47][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`blog`;
/*[16:18:47][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`blog`;
/*[16:18:47][0 ms]*/ SHOW COLLATION;
/*[16:18:47][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:20:35][1413 ms]*/ ALTER TABLE `blog_springboot_vue`.`blog` ADD COLUMN `tag` VARCHAR(100) NULL COMMENT '文章标签' AFTER `top`;
/*[16:20:38][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'blog';
/*[16:20:38][0 ms]*/ SHOW CHARSET;
/*[16:20:38][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`blog`;
/*[16:20:38][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`blog`;
/*[16:20:38][0 ms]*/ SHOW COLLATION;
/*[16:20:38][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:21:34][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:22:08][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE';
/*[16:22:39][574 ms]*/ CREATE TABLE `blog_springboot_vue`.`tags`( `id` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '标签id', `tag` VARCHAR(30) COMMENT '标签名称', PRIMARY KEY (`id`) ) ENGINE=INNODB CHARSET=utf8 COLLATE=utf8_general_ci;
/*[16:22:42][0 ms]*/ SHOW TABLE STATUS FROM `blog_springboot_vue` LIKE 'tags';
/*[16:22:42][0 ms]*/ SHOW CHARSET;
/*[16:22:42][0 ms]*/ SHOW FULL FIELDS FROM `blog_springboot_vue`.`tags`;
/*[16:22:42][0 ms]*/ SHOW KEYS FROM `blog_springboot_vue`.`tags`;
/*[16:22:42][0 ms]*/ SHOW COLLATION;
/*[16:22:43][0 ms]*/ SHOW FULL TABLES FROM `blog_springboot_vue` WHERE table_type = 'BASE TABLE'; 




2、创建SpringBoot项目、导入依赖

暂时只导入,mysql、web、jdbc、MyBatis、swagger、druid、log4j。

剩下的需要再补

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.zou</groupId><artifactId>blog_springboot</artifactId><version>0.0.1-SNAPSHOT</version><name>blog_springboot</name><description>Demo project for Spring Boot</description><properties><java.version>1.8</java.version><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding><spring-boot.version>2.3.7.RELEASE</spring-boot.version></properties><dependencies><!--swagger接口API--><!-- https://mvnrepository.com/artifact/io.springfox/springfox-swagger2 --><dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>2.9.2</version></dependency><!-- https://mvnrepository.com/artifact/io.springfox/springfox-swagger-ui --><dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger-ui</artifactId><version>2.9.2</version></dependency><!-- https://mvnrepository.com/artifact/com.alibaba/druid数据源 --><dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.1.21</version></dependency><!--log4j 数据源会使用到此日志依赖--><dependency><groupId>log4j</groupId><artifactId>log4j</artifactId><version>1.2.17</version></dependency><!--jdbc--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jdbc</artifactId></dependency><!--web--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!--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><scope>runtime</scope></dependency><!--测试--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope><exclusions><exclusion><groupId>org.junit.vintage</groupId><artifactId>junit-vintage-engine</artifactId></exclusion></exclusions></dependency></dependencies><dependencyManagement><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-dependencies</artifactId><version>${spring-boot.version}</version><type>pom</type><scope>import</scope></dependency></dependencies></dependencyManagement><build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><version>3.8.1</version><configuration><source>1.8</source><target>1.8</target><encoding>UTF-8</encoding></configuration></plugin><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><version>2.3.7.RELEASE</version><configuration><mainClass>com.zou.BlogSpringbootApplication</mainClass></configuration><executions><execution><id>repackage</id><goals><goal>repackage</goal></goals></execution></executions></plugin></plugins></build></project>

2、设置配置文件

因为功能没开始写,所以有些地方的配置是不对的、如:myBatis,后续会改、配置文件使用 yaml 格式,原配置文件需要删除

spring:application:name: blog_springbootdatasource:driver-class-name: com.mysql.jdbc.Driverusername: rootpassword: 123456#?serverTimezone=UTC解决时区的报错url: jdbc:mysql://localhost:3306/blog_springboot_vue?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTCtype: com.alibaba.druid.pool.DruidDataSource # 自定义数据源#Spring Boot 默认是不注入这些属性值的,需要自己绑定#druid 数据源专有配置initialSize: 5minIdle: 5maxActive: 20maxWait: 60000timeBetweenEvictionRunsMillis: 60000minEvictableIdleTimeMillis: 300000validationQuery: SELECT 1 FROM DUALtestWhileIdle: truetestOnBorrow: falsetestOnReturn: falsepoolPreparedStatements: true#配置监控统计拦截的filters,stat:监控统计、log4j:日志记录、wall:防御sql注入#如果允许时报错  java.lang.ClassNotFoundException: org.apache.log4j.Priority#则导入 log4j 依赖即可,Maven 地址:https://mvnrepository.com/artifact/log4j/log4jfilters: stat,wall,log4jmaxPoolPreparedStatementPerConnectionSize: 20useGlobalDataSourceStat: trueconnectionProperties: druid.stat.mergeSql=true;druid.stat.slowSqlMillis=500
server:port: 8081# 整合mybatismybatis:type-aliases-package: com.zou.pojo # 实体类的包路径mapper-locations: classpath:mybatis/mapper/*.xml # xml文件路径,classpath表示resources文件路径

3.编写实体类

略过

4、定义dao层的mapper接口和mapper.xml文件中的sql语句

略过

5、为啥插入,删除,修改、MyBatis的返回值是int

mybatis 执行 insert,update,delete 返回的都是实际操作影响的行数

insert: 插入n条记录,返回影响行数n。(n>=1,n为0时实际为插入失败)

update: 更新n条记录,返回影响行数n。(n>=0)

delete: 删除n条记录,返回影响行数n。(n>=0)

6、只需要部分字段

解决方法:

注:这种方法虽然可以,但如果不需要的字段是基本数据类型,它是可以有默认值的,所以我可以不管它,就不用再通过下面这个方法去写

一、.xml文件中,resultMap的 type改为 java.util.HashMap 即可,其余代码都不用写, 亲测可用,xml文件中代码如下:

<!-- 返回类型  -->
<resultMap id="testRestMap" type="java.util.HashMap"><result column="province_name" jdbcType="VARCHAR" property="provinceName" /><result column="area_name" jdbcType="VARCHAR" property="areaName" /><result column="lane_num" jdbcType="VARCHAR" property="laneNum" /><result column="road_name" jdbcType="VARCHAR" property="roadName" />
</resultMap><!--sql查询  -->
<select id="selTest"  resultMap="testRestMap">select province_name,area_name,lane_num,road_name from  site_infowhere 1=1and del_flag='0'
</select>

7、MyBatis parameterType和resultType区别

1.resultType

用来设置接收数据库返回结果的类型,即使返回多条结果,也只需要设置结果类型即可,多条结果会以List形式返回

一般用来从数据库提取数据

2.parameterType

用于将信息存储进数据库,接收接口方法的参数类型

如:我们通过id查询数据,所以这个parameterType的值设置为id的类型

3、当我们需要通过id查询数据时

要同时设置返回值类型和传入参数类型。

4、为什么dao层是在接口上注解,而service是在实现类上(实例化bean的注解)

问题:我们一般在dao层的接口上使用注解

@Repository // 对象交给spring管理。

而service则是在实现类上注解,为什么呢?不能也在接口上注解?

结果:不能、会报如下错误,具体就是:接口不能实例化,所以我们使用@Autowired注入的变量会失败,因为bean不存在

Error starting ApplicationContext. To display the conditions report re-run your application with 'debug' enabled.
2022-11-24 21:57:27.819 ERROR 7860 --- [           main] o.s.b.d.LoggingFailureAnalysisReporter   : ***************************
APPLICATION FAILED TO START
***************************Description:Field blogService in com.zou.controller.BlogController required a bean of type 'com.zou.service.BlogService' that could not be found.The injection point has the following annotations:- @org.springframework.beans.factory.annotation.Autowired(required=true)Action:Consider defining a bean of type 'com.zou.service.BlogService' in your configuration.

那为什么dao层可以注解到接口上?

1、有MyBatis在,无论你在不在接口上注解,它都会实例化为且给IoC托管

因为:最终发现 MapperScannerConfigurer 帮我们做了实例化bean的工作。

在Spring配置Mybatis的文件中我们可以看到如下代码:
[html] view plain copy 在CODE上查看代码片派生到我的代码片

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="org.tarena.note.dao">
</property>

MapperScannerConfigurer,让它扫描特定的包,自动帮我们成批地创建映射器。这样就大大减少了配置的工作量。
basePackage属性是让你为映射器接口文件设置基本的包路径。可以使用分号或逗号作为分隔符设置多于一个的包路径。每个映射器都会在指定的包路径中递归地被搜索到。被发现的映射器将会使用spring对自动侦测组件默认的命名策略来命名。也就是说,如果没有发现注解,它就会使用映射器的非大写的非完全限定类名。如果发现了@Component或JSR-330@Named注解,它会获取名称。

5、在SpringBoot的xml文件中编写sql语句时字段名需要使用``括起来

不然会报错:

### The error may involve defaultParameterMap
### The error occurred while setting parameters

6、使用 JSONObject 类

是第三方类库,所以要导入、导入依赖:

<dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.47</version>
</dependency>

使用:

编码:

public void testJson() {JSONObject object = new JSONObject();//stringobject.put("string","string");//intobject.put("int",2);//booleanobject.put("boolean",true);//arrayList<Integer> integers = Arrays.asList(1,2,3);object.put("list",integers);//nullobject.put("null",null);System.out.println(object);
}

解码:

public void testJson2() {JSONObject object = JSONObject.parseObject("{\"boolean\":true,\"string\":\"string\",\"list\":[1,2,3],\"int\":2}");//stringString s = object.getString("string");System.out.println(s);//intint i = object.getIntValue("int");System.out.println(i);//booleanboolean b = object.getBooleanValue("boolean");System.out.println(b);//listList<Integer> integers = JSON.parseArray(object.getJSONArray("list").toJSONString(),Integer.class);integers.forEach(System.out::println);//nullSystem.out.println(object.getString("null"));}

7、解决跨域问题

跨域:Web的一种保护机制(同源策略限制),所谓同源(即指在同一个域)就是两个页面具有相同的协议(protocol),主机(host)和端口号(port。

所以当我们将前端代码与后端代码放在不同文件夹下且两者都开启运行后,它们拥有不同访问路径,例如:

前端:localhost:\ \8080

后端:localhost:\ \8081

此时使用Axios异步通信时就会失败、若只是前端传输基本数据,不传输cookie时,只需要在Springboot中设置一个配置类即可(注:我项目中前后端都配置了跨域才成功,可能和传输的数据有关)

也可以直接在controller层的方法上面加@CrossOrigin注解

@Configuration
public class CrosConfig implements WebMvcConfigurer {@Override
public void addCorsMappings(CorsRegistry registry) {registry.addMapping("/**")
.allowedOrigins("*")
.allowedMethods("GET","HEAD","POST","DELETE","OPTIONS","PUT")
.allowCredentials(true)
.maxAge(3600)
.allowedHeaders("*");
}
}

如果需要传输cookie,则需要在vue中设置

在vue项目中config目录下的index.js文件中设置跨域

proxyTable: {'/apis':{ // 当前端请求以这个开头时target: 'http://localhost:8089',//这里是你的sprintboot项目中的后台接口changeOrigin: true,//允许跨域pathRewrite: {//重写路径'^/apis':'/apis'//写'/apis'就相当于写'http://localhost:8089/apis'}}

8、前端传递json数据给后端

POST类型

前端

  methods:{ // 这是vue中定义多个方法的关键字insertPost(){// 定义一个方法,让点击按钮的响应函数为执行这个方法this.$axios({url:'api/blog/addBlog', // 地址要注意,我这是在前端配置了跨域的,真实访问地址为:http://localhost:8081/blog/addBlogmethod:'post',//发送格式为jsondata:JSON.stringify( // stringfy是为了将json对象变成字符串{"id": "","title": "这是前端传入的","summary": "这是前端传入的这是前端传入的","content": "这是前端传入的这是前端传入的这是前端传入的","userId": "1","audit": "0","like": "99","releaseTime": new Date(), // 获取当前时间"mtime": "","browse": "33333","top": (Math.random()*100) % 2,"tag": "前端传入"}),headers: {'Content-Type': 'application/json'}// 重要,设置传递的数据为json格式}).then(rep=>{console.log(rep.data)})}}

后端接收

@RequestBody 注解表示会匹配传递过来的json数据,参数进行匹配,即与Blog中的成员变量匹配并调用set方法进行值注入。

// 控制层进行传输
@RestController//使用的效果是将方法返回的对象直接在浏览器上展示成json格式.
@RequestMapping("/blog")
public class BlogController {// 插入一篇博客,添加@PostMapping("/addBlog")public int insertBlog(@RequestBody  Blog blog) {System.out.println("插入了博客"+(i++));System.out.println(blog);// 调用插入数据库的方法return blogService.insertBlog(blog);}
}

9、后端传递json给前端

前端使用get请求,因为是后端给前端数据,前端不需要传递数据

后端

// 控制层进行传输
@RestController//使用的效果是将方法返回的对象直接在浏览器上展示成json格式.
@RequestMapping("/blog")
public class BlogController {// 查询所有博客@GetMapping("/queryBlog")public JSONObject selectBlog() {System.out.println("查询了博客"+(i++));// 查询数据库并接收数据List<Blog> blogs = blogService.selectBlog();// 生成json数据并返回JsonData jsonData = new JsonData(new JSONObject());// 放入数据:return jsonData.json(State.success,blogs);}
}

后端会传递json的格式

"blog_message": 200,"blog_data": [{"id": "","title": "","summary": "","content": "","userId": "","audit": "","like": "","releaseTime": "","mtime": "","browse": "","top": "","tag": ""}

前端

1、先准备一个接收的模板,除了data1的名字可以改以外,剩下的名字要严格与传递过来的json数据的key值一样,注意,下面的所有变量值不能为 null,你可以设置值,或者像我一样,但是不能这样写:id:null,尽量不要这样写,有可能会出错,不是一定会出错。

data(){return{data1:{blog_message:"",blog_data:[{id:"",title: "",summary:"",content:"",userId: "",audit:"",like:"",releaseTime:""mtime:"",browse: "",top: "",tag:""}]}}
}

2、使用Axios接收

// 钩子函数
mounted() {this.$axios.get('api/blog/queryBlog').then(response=>{this.data1 = response.data.blog_data // 因为我们只需要blog_data里面的数据,所以blog_message数据没取console.log(this.data1)});},

11、使用get方式传递变量值给后端

参考文档:3-6.restFul传递方式/Axios_冥丿城的博客-CSDN博客_axios restful风格传参

1.get 方式传值有两种方法

第一种 restFul 风格:http://localhost:8081/blog/selectIdBlog/1
第二种 正常 :http://localhost:8081/blog/selectIdBlog?id=1

第一种:restFul 风格

前端

注:一定要记得将url地址标记起来的不是单引号 ‘’ 而是 ``、搞错了就无法传递变量,会当成字符串处理,然后记得将变量类型与后端接收的一致

注:传入的变量类型一定要转换,parseInt是转换为int,password.toString()是转换为字符串

    a(){// 通过id查询文章并显示// 需要将id从字符串形式变为int:  parseInt()this.$axios.get(`http://localhost:8081/blog/selectIdBlog/${parseInt(this.$route.params.id)}`).then(rep=>{console.log(rep.data)})}

后端

// 通过 id 查询博客@GetMapping("/blog/selectIdBlog/{id}") public Blog selectIdBlog(@PathVariable int id) {System.out.println("查询的id为:"+id+"次数:"+this.i);return blogService.selectIdBlog(id);}

最终url为(假设id为1)http://localhost:8081/blog/selectIdBlog/1

第二种:普通风格

前端

// 通过id查询文章并显示this.$axios.get('http://localhost:8081/blog/selectIdBlog',{params:{id:this.$route.params.id //假设id值为1}}).then(rep=>{console.log(rep.data)})

最终的url是:http://localhost:8081/blog/selectIdBlog?id=1

后端

// 通过 id 查询博客@GetMapping("/selectIdBlog") // 这个参数是 required 确定在 API 中的参数中是否必须要输出参数。public Blog selectIdBlog(@RequestParam(value = "id",required = true) int id) {System.out.println("查询的id为:"+id+"次数:"+this.i);return blogService.selectIdBlog(id);}

10. 前后端传值

掌握前后端传值的正确姿势,是你开始 CRUD 的第一步!

1. @PathVariable@RequestParam

@PathVariable用于获取路径参数,@RequestParam用于获取查询参数。

举个简单的例子:

@GetMapping("/klasses/{klassId}/teachers")
public List<Teacher> getKlassRelatedTeachers(@PathVariable("klassId") Long klassId,@RequestParam(value = "type", required = false) String type ) {...
}

如果我们请求的 url 是:/klasses/123456/teachers?type=web

那么我们服务获取到的数据就是:klassId=123456,type=web

2. @RequestBody

用于读取 Request 请求(可能是 POST,PUT,DELETE,GET 请求)的 body 部分并且Content-Type 为 application/json 格式的数据,接收到数据之后会自动将数据绑定到 Java 对象上去。系统会使用HttpMessageConverter或者自定义的HttpMessageConverter将请求的 body 中的 json 字符串转换为 java 对象。

我用一个简单的例子来给演示一下基本使用!

我们有一个注册的接口:

@PostMapping("/sign-up")
public ResponseEntity signUp(@RequestBody @Valid UserRegisterRequest userRegisterRequest) {userService.save(userRegisterRequest);return ResponseEntity.ok().build();
}

UserRegisterRequest对象:

@Data
@AllArgsConstructor
@NoArgsConstructor
public class UserRegisterRequest {@NotBlankprivate String userName;@NotBlankprivate String password;@NotBlankprivate String fullName;
}

我们发送 post 请求到这个接口,并且 body 携带 JSON 数据:

{"userName":"coder","fullName":"shuangkou","password":"123456"}

这样我们的后端就可以直接把 json 格式的数据映射到我们的 UserRegisterRequest 类上。

从0到1编写个人博客项目使用springboot+vue(前后端分离) 到 购买服务器上传项目 到 GitHub开源项目、此过程下所遇问题及解决方法,至少你帮你少走70%弯路相关推荐

  1. SpringBoot+vue前后端分离博客项目

    SpringBoot+vue前后端分离博客项目 Java后端接口开发 1.前言 2.新建Springboot项目 3.整合mybatis plus 第一步:导入jar包 第二步:然后去写配置文件: 第 ...

  2. 大二期末作孽(SpringBoot+Vue前后端分离博客社区(重构White Hole))

    文章目录 前言 目录 效果演示 前言 由于时间关系,完成度确实不高,而且不签只是完成了客户端,当然目前这样也是已经可以正常使用了,当然有点勉强.不过后续还是会不断的去更新维护的,不过大体的架构是这样的 ...

  3. 从0搭建一个Springboot+vue前后端分离项目(一)安装工具,创建项目

    从0搭建一个Springboot+vue前后端分离项目(二)使用idea进行页面搭建+页面搭建 参考学习vue官网文档 https://v3.cn.vuejs.org/guide/installati ...

  4. 超详细!4小时开发一个SpringBoot+vue前后端分离博客项目!!

    小Hub领读: 前后端分离的博客项目终于出来啦,真是花了好多心思录制咧.文末直接进入B站看视频哈! 作者:吕一明 项目代码:https://github.com/MarkerHub/vueblog 项 ...

  5. Jeecg-Boot 2.0.0 版本发布,基于Springboot+Vue 前后端分离快速开发平台

    Jeecg-Boot 2.0.0 版本发布,前后端分离快速开发平台 Jeecg-Boot项目简介 源码下载 升级日志 Issues解决 v1.1升级到v2.0不兼容地方 系统截图 Jeecg-Boot ...

  6. 基于Java+SpringBoot+Vue前后端分离博客系统设计与实现

    博主介绍:✌全网粉丝3W+,全栈开发工程师,从事多年软件开发,在大厂呆过.持有软件中级.六级等证书.可提供微服务项目搭建与毕业项目实战✌ 博主作品:<微服务实战>专栏是本人的实战经验总结, ...

  7. 从0搭建一个Springboot+vue前后端分离项目(七)完善前台与后台的联系,完善功能接口

    将之前前端写的表格内容,与数据库新建的表进行对应 启动项目 把写死的TableDa数据删掉.因为最终要从后台读取 完善新增接口 <el-button type="primary&quo ...

  8. 从0搭建一个Springboot+vue前后端分离项目(四)利用Element框架搭建页面主体部分表格与侧边栏

    https://element-plus.gitee.io/zh-CN/component/menu.html 导航栏部分 选定menu菜单,选择竖型样式的 复制一下 <el-menudefau ...

  9. 从0搭建一个Springboot+vue前后端分离项目(六)后台编写配置类与接口

    新建一个包,里面放入一些常用的配置类 引入mybatis-plus包与插件 引入包 前往官网查看 https://baomidou.com/pages/226c21/#%E5%88%9D%E5%A7% ...

最新文章

  1. [JS][前端]修改文件input为button样式
  2. 云数据中心选址PK:微软第一,IBM第二,谷歌最少
  3. 蓝牙最新版本6.0_FIIL新品 T1 Lite,蓝牙5.2加持下的TWS耳机
  4. 内核程序员的职位面试技巧
  5. python中的reduce函数用法
  6. 英国-新加坡联合博士奖学金(人工智能、数字医疗方向)
  7. 大数据实战之环境搭建(三)
  8. 韦东山嵌入式学习心得
  9. 【电力电子技术AC-DC】单相桥式晶闸管整流电路(阻性/阻感性负载)simulink仿真
  10. 苹果手机怎么设置流量限制
  11. java guardedby_java并发编程之Guarded Suspention
  12. Client can't access Jboss server, the port is not accessable.
  13. 理解偏差(Bias)和方差(Variance)
  14. pg日期转周_PostgreSQL 日期/时间类型
  15. kubelet 证书轮换失败的解决方案
  16. IRIS 2021 技术文档 First Look--技术概要:.NET Object Persistence with XEP
  17. 一张图看懂UML类图
  18. PT100分度表分享
  19. 顶点计划Ⅰ最终调查报告
  20. CAD教程:CAD建筑户型图纸还能这么画?

热门文章

  1. 写给想通过程序员转型为项目经理的人
  2. 斐波那契数列(循环)
  3. Python http.server 服务器
  4. bilibili DR_CAN 现代控制理论 and 非线性控制理论 and 浙大 最优控制
  5. 思维精进01:罗辑思维2019跨年演讲--小趋势
  6. 小程序未来发展趋势怎样?2020最新趋势分析
  7. MOSS--SharePoint 开发学习一些总结
  8. java过滤器修改响应,在过滤器中实现修改http请求体和响应体
  9. Android调用传感器和震动
  10. 大家是怎么学好python啊??