node使用Koa2搭建web項(xiàng)目的方法
隨著Node.js的日益火熱,各種框架開始層出不窮的涌現(xiàn)出來,Node.js也開始逐漸的被應(yīng)用到處理服務(wù)端請求的場景中。搭建Web項(xiàng)目的框架也隨之開始出現(xiàn)——express、koa、koa2、egg等,當(dāng)然要了解其好壞還是要自己去啃源碼的。本文將不會涉及到源碼,只是帶領(lǐng)初學(xué)者簡單了解下Koa2的基本使用,歡迎大家在評論中互相交流學(xué)習(xí)。
注意:koa2使用了ES7的語法,所以使用時(shí)請升級Node版本到最新。了解更詳細(xì)的源碼信息可以到git上的koajs/koa去了解
1. 項(xiàng)目目錄結(jié)構(gòu)

2. 代碼邏輯解析
2.1. 包結(jié)構(gòu)文件
[package.json]
{
"name": "weixin-node-koa",
"version": "1.0.0",
"description": "node.js with koa2",
"private": true,
"dependencies": {
"koa": "^2.0.0",
"koa-router": "^7.0.0",
"mysql":"2.13.0"
},
"scripts": {
"start": "node app.js"
},
"engines": {
"node": ">=6.0.0"
},
"author": "Fly",
"license": "CENTERM"
}
2.2. 啟動入口文件
[app.js]
const Koa = require('koa');
const app = new Koa();
const router2controller = require('./app/router2controller.js');
const config = require('./config/config.local.js');
app.use(router2controller());
app.listen(config.port);
console.log("Server started and listen on port " + config.port);
如果請求的報(bào)文體是XML格式,可以添加下面的代碼自動解析報(bào)文(注意引用koa-xxx的版本要與koa2對應(yīng))
const Koa = require('koa');
const app = new Koa();
const router2controller = require('./app/router2controller.js');
const config = require('./config/config.local.js');
//start接收到的xml數(shù)據(jù)請求單獨(dú)解析存儲
const xmlParser = require('koa-xml-body');
app.use(xmlParser()).use((ctx,next) => {
ctx.data = ctx.request.body;
return next();
});
//end
app.use(router2controller());
app.listen(config.port);
console.log("Server started and listen on port " + config.port);
從代碼看到引入了一個(gè)router2controller.js的文件,這個(gè)文件是完成前端請求到具體處理方法的路由過程
2.3. 路由器文件
[router2controller.js]
該類將會自動掃描controller文件夾中的文件來加載請求映射,不需要挨個(gè)請求單獨(dú)配置
koa-router原生提供方法如下:
router
.get('/', async (ctx,next) => {
this.body = 'Hello World!';
})
.post('/users', async (ctx,next) => {
//TODO
})
.put('/users/:id', async (ctx,next) => {
//TODO
})
.del('/users/:id', async (ctx,next) => {
//TODO
});
自動掃描controller包實(shí)現(xiàn)方法如下
const fs = require('fs');
const router = require('koa-router')();
function addMapping(router, mapping) {
for (var url in mapping) {
if (url.startsWith('GET ')) {
var path = url.substring(4);
router.get(path, mapping[url]);
console.log(`register URL mapping: GET ${path}`);
} else if (url.startsWith('POST ')) {
var path = url.substring(5);
router.post(path, mapping[url]);
console.log(`register URL mapping: POST ${path}`);
} else if (url.startsWith('PUT ')) {
var path = url.substring(4);
router.put(path, mapping[url]);
console.log(`register URL mapping: PUT ${path}`);
} else if (url.startsWith('DELETE ')) {
var path = url.substring(7);
router.del(path, mapping[url]);
console.log(`register URL mapping: DELETE ${path}`);
} else {
console.log(`invalid URL: ${url}`);
}
}
}
function addControllers(router, dir) {
fs.readdirSync(__dirname + '/' + dir).filter((f) => {
return f.endsWith('.js');
}).forEach((f) => {
console.log(`process controller: ${f}...`);
let mapping = require(__dirname + '/' + dir + '/' + f);
addMapping(router, mapping);
});
}
module.exports = function (dir) {
var controllersDir = dir || 'controller';
addControllers(router, controllersDir);
return router.routes();
};
2.4. 控制器
[userController.js]
***Controller.js是用來處理具體請求信息以及返回?cái)?shù)據(jù)的,userController.js中處理了GET請求獲取用戶信息,POST請求保存用戶信息
const userService = require('./../service/userService.js');
var getUserinfo = (ctx, next) => {
let query = ctx.query;
let userId = query.id;
let userInfo = userService.getUserById(userId);
let html = '<html><body>'
+ '<div> userinfo: ' + userInfo + '</div>'
+ '</body></html>';
ctx.response.type ='text/html';
ctx.response.body = html;
};
var saveUserinfo = (ctx, next) => {
const requestString = ctx.data;
//TODO數(shù)據(jù)處理
Console.log(requestString);
};
module.exports = {
'GET /getUserinfo': getUserinfo,
'POST /saveUserinfo': saveUserinfo
};
2.5. 數(shù)據(jù)處理
[userService.js]
處理封裝從***Dao.js獲取到的數(shù)據(jù)返回給Controller
const userDao = require('./../dao/userDao.js');
var getUserById = async (userId) => {
var users = userDao.getUserById(userId);
var responseContent = '';
for(let user of users) {
reaponseContent += '姓名:' + user.name + ' |';
reaponseContent += '年齡:' + user.age + ' |';
reaponseContent += '身高:' + user.height + '<br />';
}
return responseContent;
}
module.exports = {
getUserById : getUserById
};
2.6. 數(shù)據(jù)獲取
[userDao.js]
通過請求傳入?yún)?shù)來獲取user數(shù)據(jù)
const mysql = require('./../utils/mysqlUtil.js');
var getUserById = async (userId) => {
let mysqlOptions = {
sql : 'select * from table_user where user_id = ?',
args : [userId]
};
var users = await mysql.execQuery(mysqlOptions);
if(users.length == 0) {
return null;
} else {
return users;
}
};
module.exports = {
getUserById : getUserById
};
2.7. 數(shù)據(jù)庫操作
[mysqlUtil.js]
包含了數(shù)據(jù)庫連接池控制,連接建立、釋放管理,執(zhí)行Dao發(fā)起的數(shù)據(jù)庫操作請求
const mysql = require('mysql');
const config = require('./../../config/config.local.js');
var connectionPool = mysql.createPool({
'host' : config.database.host,
'port':config.database.port,
'user' : config.database.user,
'password' : config.database.password,
'database' : config.database.database,
'charset': config.database.charset,
'connectionLimit': config.database.connectionLimit,
'supportBigNumbers': true,
'bigNumberStrings': true
});
var release = connection => {
connection.end(function(error) {
if(error) {
console.log('Connection closed failed.');
} else {
console.log('Connection closed succeeded.');
}
});
};
var execQuery = sqlOptions => {
var results = new Promise((resolve, reject) => {
connectionPool.getConnection((error,connection) => {
if(error) {
console.log("Get connection from mysql pool failed !");
throw error;
}
var sql = sqlOptions['sql'];
var args = sqlOptions['args'];
if(!args) {
var query = connection.query(sql, (error, results) => {
if(error) {
console.log('Execute query error !');
throw error;
}
resolve(results);
});
} else {
var query = connection.query(sql, args, function(error, results) {
if(error) {
console.log('Execute query error !');
throw error;
}
resolve(results);
});
}
connection.release(function(error) {
if(error) {
console.log('Mysql connection close failed !');
throw error;
}
});
});
}).then(function (chunk) {
return chunk;
});
return results;
};
module.exports = {
release : release,
execQuery : execQuery
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- node koa2實(shí)現(xiàn)上傳圖片并且同步上傳到七牛云存儲
- nodejs之koa2請求示例(GET,POST)
- node+koa2+mysql+bootstrap搭建一個(gè)前端論壇
- nodejs中Express與Koa2對比分析
- nodejs6下使用koa2框架實(shí)例
- 阿里大于短信驗(yàn)證碼node koa2的實(shí)現(xiàn)代碼(最新)
- Node.js環(huán)境下Koa2添加travis ci持續(xù)集成工具的方法
- Node.js Koa2使用JWT進(jìn)行鑒權(quán)的方法示例
- node(koa2) web應(yīng)用模塊介紹詳解
- Node.js中Koa2在控制臺輸出請求日志的方法示例
相關(guān)文章
node koa2實(shí)現(xiàn)上傳圖片并且同步上傳到七牛云存儲
這篇文章主要介紹了node koa2實(shí)現(xiàn)上傳圖片并且同步上傳到七牛云存儲,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07
express中創(chuàng)建 websocket 接口及問題解答
本文主要介紹了express中創(chuàng)建 websocket 接口及問題解答,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05
Node.js前后端交互實(shí)現(xiàn)用戶登陸的實(shí)踐
本文主要介紹了Node.js前后端交互實(shí)現(xiàn)用戶登陸的實(shí)踐,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12
nodeJS代碼實(shí)現(xiàn)計(jì)算交社保是否合適
本文通過nodejs的一個(gè)具體示例來對比分析現(xiàn)階段我們交社保合不合適,主要是對nodejs的一個(gè)小的應(yīng)用,當(dāng)然大家也可以改成其他語言的,程序猿們,來算算吧。2015-03-03

