Koa基本使用
简介
koa 是由 Express 原班人马打造的,致力于成为一个更小、更富有表现力、更健壮的 Web 框架。使用 koa 编写 web 应用,通过组合不同的 generator,可以免除重复繁琐的回调函数嵌套,并极大地提升错误处理的效率。koa 不在内核方法中绑定任何中间件,它仅仅提供了一个轻量优雅的函数库,使得编写 Web 应用变得得心应手。
快速开始
安装Koa2
# 初始化package.json
npm init
# 安装koa2
npm install koa
hello world代码
const Koa = require('koa')
const app = new Koa()
app.use( async ( ctx ) => {
ctx.body = 'hello koa2' //json数据
})
app.listen(3000)
启动
node index.js
Koa对比Express
通常都会说 Koa 是洋葱模型,这重点在于中间件的设计。但是按照上面的分析,会发现 Express 也是类似的,不同的是Express 中间件机制使用了 Callback 实现,这样如果出现异步则可能会使你在执行顺序上感到困惑,因此如果我们想做接口耗时统计、错误处理 Koa 的这种中间件模式处理起来更方便些。最后一点响应机制也很重要,Koa 不是立即响应,是整个中间件处理完成在最外层进行了响应,而 Express 则是立即响应
更轻量
- koa 不提供内置的中间件;
- koa 不提供路由,而是把路由这个库分离出来了(koa/router)
Context对象
koa增加了一个Context
的对象,作为这次请求的上下文对象(在koa2中作为中间件的第一个参数传入)。同时Context
上也挂载了Request
和Response
两个对象。和Express类似,这两个对象都提供了大量的便捷方法辅助开发, 这样的话对于在保存一些公有的参数的话变得更加合情合理。
异步流程控制
Express采用callback
来处理异步, Koa v1采用generator
,Koa v2 采用async/await
。
generator
和async/await
使用同步的写法来处理异步,明显好于callback
和promise
,
中间件模型
Express基于connect
中间件,线性模型;
Koa中间件采用洋葱模型(对于每个中间件,在完成了一些事情后,可以非常优雅的将控制权传递给下一个中间件,并能够等待它完成,当后续的中间件完成处理后,控制权又回到了自己)
//同步
var express = require("express")
var app = express()
app.use((req,res,next)=>{
console.log(1)
next()
console.log(4)
res.send("hello")
})
app.use(()=>{
console.log(3)
})
app.listen(3000)
//异步
var express = require("express")
var app = express()
app.use(async (req,res,next)=>{
console.log(1)
await next()
console.log(4)
res.send("hello")
})
app.use(async ()=>{
console.log(2)
await delay(1)
console.log(3)
})
function delay(time){
return new Promise((resolve,reject)=>{
setTimeout(resolve,1000)
})
}
//同步
var koa = require("koa")
var app = new koa()
app.use((ctx,next)=>{
console.log(1)
next()
console.log(4)
ctx.body="hello"
})
app.use(()=>{
console.log(3)
})
app.listen(3000)
//异步
var koa = require("koa")
var app = new koa()
app.use(async (ctx,next)=>{
console.log(1)
await next()
console.log(4)
ctx.body="hello"
})
app.use(async ()=>{
console.log(2)
await delay(1)
console.log(3)
})
function delay(time){
return new Promise((resolve,reject)=>{
setTimeout(resolve,1000)
})
}
app.listen(3000)
路由
基本用法
var Koa = require("koa")
var Router = require("koa-router")
var app = new Koa()
var router = new Router()
router.post("/list",(ctx)=>{
ctx.body=["111","222","333"]
})
app.use(router.routes()).use(router.allowedMethods())
app.listen(3000)
router.allowedMethods
作用
请求方式
Koa-router 请求方式: get
、 put
、 post
、 patch
、 delete
、 del
,而使用方法就是 router.方式()
,比如 router.get()
和 router.post()
。而 router.all()
会匹配所有的请求方法。
var Koa = require("koa")
var Router = require("koa-router")
var app = new Koa()
var router = new Router()
router.get("/user",(ctx)=>{
ctx.body=["aaa","bbb","ccc"]
})
.put("/user/:id",(ctx)=>{
ctx.body={ok:1,info:"user update"}
})
.post("/user",(ctx)=>{
ctx.body={ok:1,info:"user post"}
})
.del("/user/:id",(ctx)=>{
ctx.body={ok:1,info:"user del"}
})
app.use(router.routes()).use(router.allowedMethods())
app.listen(3000)
拆分路由
routes/list.js
var Router = require("koa-router")
var router = new Router()
router.get("/",(ctx)=>{
ctx.body=["111","222","333"]
})
.put("/:id",(ctx)=>{
ctx.body={ok:1,info:"list update"}
})
.post("/",(ctx)=>{
ctx.body={ok:1,info:"list post"}
})
.del("/:id",(ctx)=>{
ctx.body={ok:1,info:"list del"}
})
module.exports = router
routes/index.js
var Router = require("koa-router")
var user = require("./user")
var list = require("./list")
var router = new Router()
router.use('/user', user.routes(), user.allowedMethods())
router.use('/list', list.routes(), list.allowedMethods())
module.exports = router
app.js
var Koa = require("koa")
var router = require("./routes")
var app = new Koa()
app.use(router.routes()).use(router.allowedMethods())
app.listen(3000)
路由前缀
router.prefix('/api')
router.use('/user', user.routes(), user.allowedMethods())
路由重定向
router.get("/home",(ctx)=>{
ctx.body="home页面"
})
//写法1
router.redirect('/', '/home'); //匹配到/重定向到/home
//写法2
router.get("/",(ctx, next)=>{
ctx.redirect("/home")
})
静态资源
const Koa = require('koa')
const path = require('path')
const static = require('koa-static')
const app = new Koa()
app.use(static(path.join( __dirname, "public")))
app.use( async ( ctx ) => {
ctx.body = 'hello world'
})
app.listen(3000)
获取请求参数
get参数
在Koa中,获取GET请求数据源头是Koa中request
对象中的query
方法或querystring
方法。
query
返回是格式化好的参数对象,querystring
返回的是请求字符串,由于ctx对request
的API有直接引用的方式,所以获取GET请求数据有两个途径。
- 从上下文中直接获取 请求对象
ctx.query
,返回如{ a:1, b:2 }
请求字符串ctx.querystring
,返回如a=1&b=2
- 从上下文的
request
对象中获取 请求对象ctx.request.query
,返回如{ a:1, b:2 }
请求字符串ctx.request.querystring
,返回如a=1&b=2
post参数
对于POST请求的处理,koa-bodyparser
中间件可以把koa2上下文的formData
数据解析到ctx.request.body
中
const bodyParser = require('koa-bodyparser')
// 使用ctx.body解析中间件
app.use(bodyParser())
ejs模板
安装模块
# 安装koa模板使用中间件
npm install --save koa-views
# 安装ejs模板引擎
npm install --save ejs
使用模板引擎
文件目录
├── package.json
├── index.js
└── view
└── index.ejs
index.js
文件
const Koa = require('koa')
const views = require('koa-views')
const path = require('path')
const app = new Koa()
// 加载模板引擎
app.use(views(path.join(__dirname, './view'), {
extension: 'ejs'
}))
app.use( async ( ctx ) => {
let title = 'hello koa2'
await ctx.render('index', {
title: 'hello world',
})
})
app.listen(3000)
./view/index.ejs
模板
<!DOCTYPE html>
<html>
<head>
<title><%= title %></title>
</head>
<body>
<h1><%= title %></h1>
<p>EJS Welcome to <%= title %></p>
</body>
</html>
cookie&session
cookie
Koa提供了从上下文直接读取、写入cookie
的方法
ctx.cookies.get(name, [options])
读取上下文请求中的cookie
ctx.cookies.set(name, value, [options])
在上下文中写入cookie
session
koa-session-minimal
适用于 Koa2 的session
中间件,提供存储介质的读写接口 。
const session = require('koa-session-minimal')
app.use(session({
key: 'SESSION_ID',
cookie: {
maxAge:1000*60
}
}))
app.use(async (ctx, next) => {
//排除login相关的路由和接口
if (ctx.url.includes("login")) {
await next()
return
}
if (ctx.session.user) {
//重新设置sesssion
ctx.session.mydate = Date.now()
await next()
} else {
ctx.redirect("/login")
}
})
图片上传
环境
koa
:用来起一个web服务器koa2-cors
: 解决跨域问题koa-router
: koa的路由处理koa-body
: koa参数的获取koa-static
: 静态资源配置@koa/multer
和multer
:图片上传的插件
代码结构
实现
-
第一步:用
koa
+koa-router
搭建一个简单的web服务//main.js const Koa = require('koa') // 引入koa const Router = require('koa-router') // 引入koa-router const { koaBody } = require('koa-body'); var router = new Router() router.get('/', async (ctx) => { ctx.type = 'html' ctx.body = '<h1>hello world!</h1>' }).post('/upload', async (ctx) => { ctx.body = 'ok' }) app.use(koaBody()) .use(router.routes()) .use(router.allowedMethods()) app.listen(3000)
-
接着我们新建一个upload文件夹,且在代码中加入静态内容的的代码
//mian.js 新增代码 const static = require('koa-static') const path = require('path') app.use(router.routes()) .use(router.allowedMethods()) .use(static(path.join(__dirname, './upload')))
此时假如你在upload文件夹下新增一张照片便可通过
http://localhost:3000/***.png
查看到了。(***:你自己新增的照片名称加后缀) -
此时新增一个
index.html
文件,且加入以下代码<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> </head> <body> <input type="file" class="file" name="avatar"> <button onclick="send()">上传</button> <script src="https://unpkg.com/axios/dist/axios.min.js"></script> <script> let formData = new FormData() document.querySelector('.file').addEventListener('change', function(e) { let files = e.target.files console.log(files) if (!files.length) return formData.append('file', files[0], files[0].name) }) function send(){ axios.post('http://localhost:3000/upload',formData,{ Headers:{ "Content-type":"multipart/form-data" } }) } </script> </body> </html>
选择图片且上传会发现存在跨域问题,那么发现问题解决问题直接上代码:
//mian.js新增代码 const cors = require('koa2-cors') //注意这个配置要在router前使用不然不生效 app.use(cors()) .use(koaBody()) .use(router.routes()) .use(router.allowedMethods()) .use(static(path.join(__dirname, './upload')) )
解决完跨域后,选择图片且上传,此时咱们已经拿到传过来的数据啦,重头戏来了:
@koa/multer
使用const multer = require('@koa/multer') const storage = multer.diskStorage({ destination: function (req, file, cb) { cb(null, './upload') }, filename: function (req, file, cb) { const fileFormat = (file.originalname).split('.') cb(null, Date.now() + '.' + fileFormat[fileFormat.length - 1]) } }) const upload = multer({ storage })
配置好后修改
/upload
router.post('/upload', upload.single('file'), async (ctx) => { console.log('ctx.file', ctx.file) })
注意:
upload.single('file')
,中的file
需要和上方的index.html
中的formData
字段一致
记录日志
const Koa = require('koa')
const Router = require('koa-router')
// 引入 koa-logger
const logger = require('koa-logger')
const app = new Koa()
const router = new Router()
// 使用 koa-logger 中间件
app.use(logger((str, args) => {
// console.log(str);
// console.log(args);
}))
router.get('/', ctx => {
ctx.body = '首页'
})
// 使用路由中间件
app.use(router.routes())
app.listen(3000, () => {
console.log('listen 3000 ok');
})
- 在注册
koa-logger
中间件时可以传递一个函数,该函数有2个参数 str
是一个字符串类型,在发生请求时str
包含 请求类型、请求路径信息,在发生响应时str
包含 响应状态码、响应时长、响应文件大小信息。args
是一个数组类型,在发生请求时会将请求类型、请求路径放在该数组中,在发生响应时会将响应状态码、响应时长、响应文件大小信息放入该数组中