全栈前端入门必看 koa2+mysql+vue+vant 构建简单版移动端博客
koa2+mysql+vue+vant 构建简单版移动端博客
具体内容展示
开始正文
github地址
<br/>
觉得对你有帮助的话,可以star一下^_^
必须安装:<br/>mysql <br/>node.jsvue-cli<br/>
目录结构
<br/><br/>
代码步骤
<br/>
在 app 目录下 打开 node 运行vue-cli vue init webpack 新建Vue项目<br/>
安装以下依赖模块:
<br/>
"axios": "^0.18.0",
"js-cookie": "^2.2.0",
"js-md5": "^0.7.3",
"nprogress": "^0.2.0",
"vant": "^1.1.15",
"vue": "^2.5.2",
"vue-router": "^3.0.1"
详细具体看github源码
在 koa2 目录下 打开 node 运行 npm init 编写信息<br/>
安装以下依赖模块: <br/>
"jsonwebtoken": "^8.3.0", // 生成token
"koa-bodyparser": "^4.2.1", // 解析requeset body
"koa-cors": "^0.0.16", // koa跨域
"koa-router": "^7.4.0", // koa-router
"koa-static": "^5.0.0", // koa静态文件
"koa2": "^2.0.0-alpha.7", // koa2
"mysql": "^2.16.0", // mysql
"uuid": "^3.3.2" // 生成userId
<br/>
在 koa2/config.js 进行 mysql 链接配置
// config.js 数据库配置
module.exports = { // mysql 配置
mysql: {
host: 'localhost', // 地址
user: 'root', // 用户账号
password: '', // 密码
database: 'test' // test库
},
port: 3001 // 监听端口
}
编写 sql 配置
在 koa2/mysql.js 进行编写 sql 语句
// mysql.js 编写sql语句
const mysql = require('mysql');
const config = require('./config.js');
var pool = mysql.createPool(config.mysql);
const query = function (sql, val) {
return new Promise((resolve, reject) => {
pool.getConnection(function (err, connection) {
if (err) {
reject(err)
} else {
connection.query(sql, val, (err, res) => {
if (err) {
reject(err)
} else {
resolve(res)
}
connection.release();
})
}
})
})
}
const createTable = (sql) => {
query(sql, [])
}
const usersTable = `CREATE TABLE IF NOT EXISTS users (
id VARCHAR(36) NOT NULL,
userName VARCHAR(16) NOT NULL,
passWord VARCHAR(16) NOT NULL,
avator VARCHAR(50) NOT NULL,
createTime VARCHAR(50) NOT NULL,
PRIMARY KEY (id)
)`;
const postsTable = `CREATE TABLE IF NOT EXISTS posts (
id INT NOT NULL AUTO_INCREMENT,
userName VARCHAR(100) NOT NULL,
userId VARCHAR(40) NOT NULL,
avator VARCHAR(100) NOT NULL,
title VARCHAR(100) NOT NULL,
content TEXT(0) NOT NULL,
hot VARCHAR(40) NOT NULL,
comments VARCHAR(40) NOT NULL,
createTime VARCHAR(100) NOT NULL,
PRIMARY KEY (id)
)`;
const commentTable = `CREATE TABLE IF NOT EXISTS comment (
id INT NOT NULL AUTO_INCREMENT,
userName VARCHAR(100) NOT NULL,
content TEXT(0) NOT NULL,
postId VARCHAR(40) NOT NULL,
avator VARCHAR(100) NOT NULL,
createTime VARCHAR(100) NOT NULL,
PRIMARY KEY (id)
)`;
// 建表
createTable(usersTable) // 用户表
createTable(postsTable) // 文章表
createTable(commentTable) // 评论表
const insetUser = (val) => { // 注册
let _sql = `INSERT INTO users (id, userName, passWord, avator, createTime) VALUES (?,?,?,?,?)`
return query(_sql, val)
}
const findUser = (val) => { // 查找所有User
let _sql = `SELECT * FROM users WHERE userName = '${val}'`
return query(_sql)
}
const createPosts = (val) => { // 新建posts
let _sql = `INSERT INTO posts (userName, userId, avator, title, content, hot, comments, createTime) VALUES (?,?,?,?,?,?,?,?)`
return query(_sql, val)
}
const updatePosts = (val) => { // 修改posts
let _sql = `UPDATE posts SET title=?, content=? WHERE id=?`
return query(_sql, val)
}
const updatePostsComment = (val) => { // 修改posts评论数量
let _sql = `UPDATE posts SET comments=? WHERE id=?`
return query(_sql, val)
}
const updatePostsHot = (val) => { // 修改posts查看人数
let _sql = `UPDATE posts SET hot=? WHERE id=?`
return query(_sql, val)
}
const postsList = (key, pg, size) => { // 查找所有posts
let _sql = `SELECT * FROM posts ${ key ? "WHERE title LIKE '%"+key+"%' " : ' '}ORDER BY createTime DESC limit ${pg * size} , ${size}`
return query(_sql)
}
const postDetail = (val) => { // 根据ID 查询 postsDetail
let _sql = `SELECT * FROM posts WHERE id = '${val}'`
return query(_sql)
}
const commentList = (val) => { // 获取留言列表
let _sql = `SELECT * FROM comment WHERE postId = '${val}' ORDER BY createTime DESC`
return query(_sql)
}
const createComment = (val) => { // 添加 留言
let _sql = `INSERT INTO comment (userName, content, postId, avator, createTime) VALUES (?,?,?,?,?)`
return query(_sql, val)
}
module.exports = {
insetUser,
findUser,
createPosts,
postsList,
updatePosts,
updatePostsComment,
updatePostsHot,
postDetail,
createComment,
commentList
}
编写 koa2 配置
在 koa2/app.js 进行编写 koa2 配置
// app.js
const path = require('path')
const Koa = require('koa2');
const router = require('koa-router');
var cors = require('koa-cors');
const bodyParser = require('koa-bodyparser');
const config = require('./config.js');
const server = require('koa-static');
const jwt = require('jsonwebtoken')
require('./mysql.js');
const app = new Koa();
app.use(server(
path.join(__dirname , './public')
)) // 设置静态文件
app.use(cors({
origin: 'http://localhost:8080', // 允许 loclhost:8080 访问
exposeHeaders: ['WWW-Authenticate', 'Server-Authorization'],
maxAge: 5,
credentials: true,
allowMethods: ['GET', 'POST'],
allowHeaders: ['Content-Type', 'Authorization', 'Accept'],
})
); // 设置跨域
app.use(bodyParser({
formLimit: '1mb'
})) // ctx body 中间件
app.use(require('./routers/signUp.js').routes()) // 注册
app.use(require('./routers/signIn.js').routes()) // 登录
app.use(require('./routers/createPosts.js').routes()) // 新建文章
app.use(require('./routers/postsList.js').routes()) // 搜索文章
app.use(require('./routers/postDetail.js').routes()) // 文章detail
app.use(require('./routers/updatePosts.js').routes()) // 修改文章
app.use(require('./routers/createComment.js').routes()) // 添加留言
app.use(require('./routers/commentList.js').routes()) // 获取留言
app.listen(config.port) // 监听端口
console.log('listen in localhost:' + config.port)
分析登录注册
详细分析一下 登录 注册
登录 -> 有账号 -> 校验密码 -> 成功 -> 返回Token
登录 -> 没有账号 -> 注册 -> 判断是否有账号 -> 没有-> sql注册
///////////////////-> 注册 -> 有账号 -> 去登录
注册<br/>
signUp.js
// signUp.js
const router = require('koa-router')();
const uuidV1 = require('uuid/v1'); // 生成13位 userId
const userModel = require('../mysql.js');
const fs = require('fs'); // 文件操作
// 注册
router.post('/signUp', async (ctx, next) => {
// post请求 从body中获取注册参数
let user = {
userName: ctx.request.body.userName,
passWord: ctx.request.body.passWord,
repeatPass: ctx.request.body.repeatPass,
avator: ctx.request.body.avator
}
await userModel.findUser(user.userName).then(async (res) => {
if (res.length) { // length > 1 说明 表中有数据
try {
throw Error('用户已存在')
} catch (err) {
console.log(err)
}
ctx.body = {
state: 0,
msg: '用户已存在!',
data: []
}
} else if (!user.userName || user.passWord !== user.repeatPass) {
ctx.body = {
state: 0,
msg: '密码输入错误',
data: []
}
} else { // 否者没有注册
//处理上传头像
let base64Data = user.avator.replace(/^data:image\/\w+;base64,/, "");
let dataBuffer = new Buffer(base64Data, 'base64');
let getName = Number(Math.random().toString().substr(3)).toString(36) + Date.now()
// 上传图片到 public/images 文件夹中
await fs.writeFile('./public/images/' + getName + '.png', dataBuffer, err => {
if (err) {
console.log(err);
return false
}
console.log('头像上传成功')
});
await userModel.insetUser([uuidV1(), user.userName, user.passWord, getName, new Date().getTime()]).then((res) => {
console.log('注册成功')
ctx.body = {
state: 1,
msg: '注册成功',
data: []
}
}).catch((err) => {
ctx.body = {
state: 0,
msg: err,
data: []
}
})
}
})
})
module.exports = router
登录signIn.js
// signIn.js
let router = require('koa-router')();
let userModel = require('../mysql.js');
const createToken = require('../token/createToken.js');
router.post('/signIn',async (ctx, next) => {
let user = {
userName: ctx.request.body.userName,
passWord: ctx.request.body.passWord
}
await userModel.findUser(user.userName).then((res) => {
if (!res.length) {
ctx.body = {
state: 0,
msg: '用户未注册!',
data: []
}
console.log('用户未注册')
} else {
if (res[0].passWord === user.passWord) {
let token = createToken(res[0]) // 创建token 存储用户id等重要信息
ctx.body = {
state: 1,
msg: '用户登录成功!',
data: [],
token
}
console.log('密码校验正确, 允许登录')
} else {
ctx.body = {
state: 0,
msg: '用户名或者密码错误!',
data: []
}
console.log('用户名或者密码错误')
}
}
}).catch((err) => {
ctx.body = {
state: 0,
msg: err,
data: []
}
})
})
module.exports = router
生成Token, 解析Token
createToken.js , checkToken.js 原理
// createToken.js
const jwt = require('jsonwebtoken');
// 创建token
//登录时:核对用户名和密码成功后,应用将用户的id 作为JWT Payload的一个属性
module.exports = function(user){
// jwt.sign 参数详情
//第一个是Payload,也就是用户信息(要注意payload不要传整个文档,Payload需要的是唯一且不变的数据,否则当Payload改变的时候需要重新下发token)。这里我们用文档的id,目的是唯一标识用户
// 第二个参数是密钥,也就是你生成Signature时所用到的加密密钥。要注意这里必须和创建jwt的时候传入的secret一致,因为服务端需要用创建时的secret来解密。
// 第三个参数则是设置一个token的过期时间,这里我们设置的是1天。
const token = jwt.sign({
userId: user.id,
userName: user.userName,
avator: user.avator
}, 'kuaifengle', { // "kuaifengle" 是校验码 解析时需要一致 才能取到 user 信息
expiresIn: '24h' //过期时间设置为24h 格式有(s, m, h , day)。那么decode这个token的时候得到的过期时间为 : 创建token的时间 + 设置的值
});
return token; 返回token 前端存在浏览器cookie 中
};
// checkToken.js
const jwt = require('jsonwebtoken');
// 接口访问必须要有Token (需要用户登录)
module.exports = async ( ctx, next ) => {
const authorization = ctx.get('Authorization'); // request 带过来的 token 存在浏览器的cookie中
if (authorization == '') {
ctx.body = {
state: 0,
msg: '用户未登录'
}
return false
}
const token = authorization;
let tokenContent;
try {
// 根据 "kuaifengle" 钥解析 token 判断是否失效
tokenContent = await jwt.verify(token, 'kuaifengle'); //如果token过期或验证失败,将抛出错误
// 存入ctx 中 next() 可以获取到设置的 userInfo 数据
ctx.userInfo = tokenContent
} catch (err) {
ctx.body = {
state: 0,
msg: '用户登录验证失效'
}
}
await next();
}
<br/>
其他的接口就不做解释了
项目打包后,放在 Koa2 / public / 下,就可以访问 localhost:3001 查看页面了
Github地址
都是基本的mysql语句操作,和业务逻辑, 具体看github源码
觉得对你有帮助的话,可以star一下 ^_^
全栈前端入门必看 koa2+mysql+vue+vant 构建简单版移动端博客的更多相关文章
- Koa2+MySQL+VUE+ElementIUI搭建简单的后台管理小系统
如题,前端入坑许久,还是写个小东西出来吧 想要搭建自己的一个后台管理,实现简单的增删改查,看起来很简单 其实是真的简单,没有想的那么难,我也就写了一个月吧, 当然是假的,其实也就每天一两个小时,花了大 ...
- uni-app高分开源电影项目源码案例分析,支持一套代码发布小程序、APP平台多个平台(前端入门必看)
uni-app-Video 一个优秀的uni-app案例,旨在帮助大家更快的上手uni-app,共同进步! Features 代码编写简洁,注释清晰,快速入门必备: 支持在线模糊搜索: 程序类目懒 ...
- 知数堂MYSQL优化课---CU论坛版主 DBA 博客
http://www.cnblogs.com/MYSQLZOUQI/category/546261.html
- spring boot + vue + element-ui全栈开发入门——spring boot后端开发
前言 本文讲解作为后端的spring boot项目开发流程,如果您还不会配置spring boot环境,就请点击<玩转spring boot——快速开始>,如果您对spring boot还 ...
- Java编程学习知识点分享 入门必看
Java编程学习知识点分享 入门必看 阿尔法颜色组成(alpha color component):颜色组成用来描述颜色的透明度或不透明度.阿尔法组成越高,颜色越不透明. API:应用编程接口.针对软 ...
- spring boot + vue + element-ui全栈开发入门——开篇
最近经常看到很多java程序员朋友还在使用Spring 3.x,Spring MVC(struts),JSP.jQuery等这样传统技术.其实,我并不认为这些传统技术不好,而我想表达的是,技术的新旧程 ...
- spring boot + vue + element-ui全栈开发入门——基于Electron桌面应用开发
前言 Electron是由Github开发,用HTML,CSS和JavaScript来构建跨平台桌面应用程序的一个开源库. Electron通过将Chromium和Node.js合并到同一个运行时环 ...
- spring boot + vue + element-ui全栈开发入门
今天想弄弄element-ui 然后就在网上找了个例子 感觉还是可以用的 第一步是完成了 果断 拿过来 放到我这里这 下面直接是连接 点进去 就可以用啊 本想着不用vue 直接导入连接 ...
- Liunx新手入门必看
安装CentOS(Linux的一个常用发行版本,互联网公司经常使用这个发行版)用到的软件: VMware_workstation_full_12.5.2.exe 虚拟机软件,虚拟机由这个软件安装.管理 ...
随机推荐
- MTSP问题
问题描述:m个旅行商去旅游 n个城市,规定都必须从同一个出发点出发,而且返回原出发点,需要将所有的城市遍历完毕,每个城市只能游历一次,但是为了路径最短可以路过这个城市多次.这个就是多旅行商问题.是在T ...
- 如何将jar包打包到本地maven仓库
--例如下载jar到本地(例如经常用到的oracle数据库驱动) --前提本地已将安装maven并配置好环境,cmd并切换到jar包的文件夹下,执行以下命令,注意DgroupId.DartifactI ...
- Spring Cloud服务的注册与发现(Eureka)
一.spring cloud简介 spring cloud 为开发人员提供了快速构建分布式系统的一些工具,包括配置管理.服务发现.断路器.路由.微代理.事件总线.全局锁.决策竞选.分布式会话等等.它运 ...
- 如何使用css伪类,实现div左上角出现封面等提示信息
HTML <div class="ui-cover-tip”><div> CSS .ui-cover-tip{ position: relative; width: ...
- android cpu affinity
暂时无法获取当前线程运行在哪个CPU上,待调查... int omask = 0; int nmask = 0xF0; static void affinity() { int err; int sy ...
- 吴裕雄--天生自然操作系统操作笔记:window10显示隐藏文件夹
基于安全考虑,操作系统会隐藏一些文件和文件夹,防止误删除操作.但有可能是个别人为了隐藏一些私密数据,也同样采取隐藏的方式.
- 林轩田机器学习基石笔记3—Types of Learning
上节课我们主要介绍了解决线性分类问题的一个简单的方法:PLA.PLA能够在平面中选择一条直线将样本数据完全正确分类.而对于线性不可分的情况,可以使用Pocket Algorithm来处理.本节课将主要 ...
- leetcode第24题:两两交换链表中的节点
通过分析,这属于数据结构类型题目,但涉及到多次交换,也需要算法知识. 首先,我想的是,将链表中节点相互交换. class Solution: def swapPairs(self, head: Lis ...
- jenkins使用(1)
术语:构建一次job指的是执行一次任务 注:到了公司,如果需要搭建jenkins环境可以找运维 jenkins使用: 创建视图 常用的两个配置: 新建任务: 可以选择构建后的步骤: 然后保存 图标状态 ...
- Kafka与RabbitMQ、ActiveMQ协议区别
对于Kafka与RabbitMQ.ActiveMQ协议,它们具体的区别如下: activemq: activemq支持主从复制.集群.但是集群功能看起来很弱,只有failover功能,即 ...