NodeJS仿WebApi路由示例
发布时间 - 2026-01-10 23:20:27 点击率:次用过WebApi或Asp.net MVC的都知道微软的路由设计得非常好,十分方便,也十分灵活。虽然个人看来是有的太灵活了,team内的不同开发很容易使用不同的路由方式而显得有点混乱。 不过这不是重点,我在做Node项目的时候就觉得不停的用use(...)来指定路由路径很烦人,所以用Typescript写了这个基于Koa和Koa-router的路由插件,可以简单实现一些类似WebApi的路由功能。

目标是和WebApi一样:
1.加入的controller会自动加入路由。
2.也可以通过path()手动指定路由。
3.可以定义http method, 如GET或POST等。
4.Api的参数可以指定url里的query param、path param以及body等。
包已经上传到npm中,npm install webapi-router 安装,可以先看看效果:
第一步,先设置controllers的目录和url的固定前缀
所有的controller都在这目录下,这样会根据物理路径自动算出路由。 url的固定前缀就是host和路由之间的,比如localhost/api/v2/user/name,api/v2就是这个固定前缀。
import { WebApiRouter } from 'webapi-router';
app.use(new WebApiRouter().router('sample/controllers', 'api'));
第二步是controller都继承自BaseController
export class TestController extends BaseController
{
}
第三步给controller的方法加上装饰器
@POST('/user/:name')
postWithPathParam(@PathParam('name') name: string, @QueryParam('id') id: string, @BodyParam body: any) {
console.info(`TestController - post with name: ${name}, body: ${JSON.stringify(body)}`);
return 'ok';
}
@POST里的参数是可选的,空的话会用这个controller的物理路径做为路由地址。
:name是路径里的变量,比如 /user/brook, :name就是brook,可以在方法的参数里用@PathParam得到
@QueryParam可以得到url里?后的参数
@BodyParam可以得到Post上来的body
是不是有点WebApi的意思了。
现在具体看看是怎么实现的
实现过程其实很简单,从上面的目标入手,首先得到controllers的物理路径,然后还要得到被装饰器装饰的方法以及它的参数。
装饰器的目的在于要得到是Get还是Post等,还有就是指定的Path,最后就是把node request里的数据赋值给方法的参数。
核心代码:
得到物理路径
initRouterForControllers() {
//找出指定目录下的所有继承自BaseController的.js文件
let files = FileUtil.getFiles(this.controllerFolder);
files.forEach(file => {
let exportClass = require(file).default;
if(this.isAvalidController(exportClass)){
this.setRouterForClass(exportClass, file);
}
});
}
从物理路径转成路由
private buildControllerRouter(file: string){
let relativeFile = Path.relative(Path.join(FileUtil.getApiDir(), this.controllerFolder), file);
let controllerPath = '/' + relativeFile.replace(/\\/g, '/').replace('.js','').toLowerCase();
if(controllerPath.endsWith('controller'))
controllerPath = controllerPath.substring(0, controllerPath.length - 10);
return controllerPath;
}
装饰器的实现
装饰器需要引入reflect-metadata库
先看看方法的装饰器,@GET,@POST之类的,实现方法是给装饰的方法加一个属性Router,Router是个Symbol,确保唯一。 然后分析装饰的功能存到这个属性中,比如Method,Path等。
export function GET(path?: string) {
return (target: BaseController, name: string) => setMethodDecorator(target, name, 'GET', path);
}
function setMethodDecorator(target: BaseController, name: string, method: string, path?: string){
target[Router] = target[Router] || {};
target[Router][name] = target[Router][name] || {};
target[Router][name].method = method;
target[Router][name].path = path;
}
另外还有参数装饰器,用来给参数赋上request里的值,如body,param等。
export function BodyParam(target: BaseController, name: string, index: number) {
setParamDecorator(target, name, index, { name: "", type: ParamType.Body });
}
function setParamDecorator(target: BaseController, name: string, index: number, value: {name: string, type: ParamType}) {
let paramTypes = Reflect.getMetadata("design:paramtypes", target, name);
target[Router] = target[Router] || {};
target[Router][name] = target[Router][name] || {};
target[Router][name].params = target[Router][name].params || [];
target[Router][name].params[index] = { type: paramTypes[index], name: value.name, paramType: value.type };
}
这样装饰的数据就存到对象的Router属性上,后面构建路由时就可以用了。
绑定路由到Koa-router上
上面从物理路径得到了路由,但是是以装饰里的参数路径优先,所以先看看刚在存在原型里的Router属性里有没有Path,有的话就用这个作为路由,没有Path就用物理路由。
private setRouterForClass(exportClass: any, file: string) {
let controllerRouterPath = this.buildControllerRouter(file);
let controller = new exportClass();
for(let funcName in exportClass.prototype[Router]){
let method = exportClass.prototype[Router][funcName].method.toLowerCase();
let path = exportClass.prototype[Router][funcName].path;
this.setRouterForFunction(method, controller, funcName, path ? `/${this.urlPrefix}${path}` : `/${this.urlPrefix}${controllerRouterPath}/${funcName}`);
}
}
给controller里的方法参数赋上值并绑定路由到KoaRouter
private setRouterForFunction(method: string, controller: any, funcName: string, routerPath: string){
this.koaRouter[method](routerPath, async (ctx, next) => { await this.execApi(ctx, next, controller, funcName) });
}
private async execApi(ctx: Koa.Context, next: Function, controller: any, funcName: string) : Promise<void> { //这里就是执行controller的api方法了
try
{
ctx.body = await controller[funcName](...this.buildFuncParams(ctx, controller, controller[funcName]));
}
catch(err)
{
console.error(err);
next();
}
}
private buildFuncParams(ctx: any, controller: any, func: Function) { //把参数具体的值收集起来
let paramsInfo = controller[Router][func.name].params;
let params = [];
if(paramsInfo)
{
for(let i = 0; i < paramsInfo.length; i++) {
if(paramsInfo[i]){
params.push(paramsInfo[i].type(this.getParam(ctx, paramsInfo[i].paramType, paramsInfo[i].name)));
} else {
params.push(ctx);
}
}
}
return params;
}
private getParam(ctx: any, paramType: ParamType, name: string){ // 从ctx里把需要的参数拿出来
switch(paramType){
case ParamType.Query:
return ctx.query[name];
case ParamType.Path:
return ctx.params[name];
case ParamType.Body:
return ctx.request.body;
default:
console.error('does not support this param type');
}
}
这样就完成了简单版的类似WebApi的路由.
源码下载:webapi-router_jb51.rar
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# NodeJS仿WebApi
# nodejs
# webapi
# node.js中使用q.js实现api的promise化
# 实现一个完整的Node.js RESTful API的示例
# 详解本地Node.js服务器作为api服务器的解决办法
# node.js实现微信JS-API封装接口的示例代码
# 浅析Node.js 中 Stream API 的使用
# Node.js原生api搭建web服务器的方法步骤
# 30分钟用Node.js构建一个API服务器的步骤详解
# 零基础之Node.js搭建API服务器的详解
# NodeJS基础API搭建服务器详细过程记录
# 深入分析node.js的异步API和其局限性
# 浅析Node在构建超媒体API中的作用
# Node.js API详解之 tty功能与用法实例分析
# 就用
# 可以得到
# 绑定
# 是个
# 我在
# 目录下
# 在这
# 是怎么
# 很容易
# 微软
# 可以通过
# 用了
# 这不是
# 写了
# 非常好
# 可选
# 用过
# 时就
# 要得
# 拿出来
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
IOS倒计时设置UIButton标题title的抖动问题
如何在宝塔面板创建新站点?
详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)
香港服务器WordPress建站指南:SEO优化与高效部署策略
Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】
Laravel如何使用Livewire构建动态组件?(入门代码)
如何注册花生壳免费域名并搭建个人网站?
Laravel如何创建和注册中间件_Laravel中间件编写与应用流程
如何在企业微信快速生成手机电脑官网?
Laravel怎么实现验证码功能_Laravel集成验证码库防止机器人注册
Angular 表单中正确绑定输入值以确保提交与验证正常工作
如何利用DOS批处理实现定时关机操作详解
微信小程序制作网站有哪些,微信小程序需要做网站吗?
linux写shell需要注意的问题(必看)
如何在阿里云域名上完成建站全流程?
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置
免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?
大型企业网站制作流程,做网站需要注册公司吗?
,交易猫的商品怎么发布到网站上去?
Android利用动画实现背景逐渐变暗
Laravel如何处理和验证JSON类型的数据库字段
如何在服务器上配置二级域名建站?
UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】
laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法
Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧
C语言设计一个闪闪的圣诞树
家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?
什么是javascript作用域_全局和局部作用域有什么区别?
Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作
高防服务器如何保障网站安全无虞?
js实现点击每个li节点,都弹出其文本值及修改
大同网页,大同瑞慈医院官网?
Laravel如何使用Blade组件和插槽?(Component代码示例)
如何确保西部建站助手FTP传输的安全性?
Laravel怎么在Controller之外的地方验证数据
网站制作价目表怎么做,珍爱网婚介费用多少?
jQuery中的100个技巧汇总
如何快速搭建高效简练网站?
佐糖AI抠图怎样调整抠图精度_佐糖AI精度调整与放大细化操作【攻略】
Python3.6正式版新特性预览
如何用y主机助手快速搭建网站?
大学网站设计制作软件有哪些,如何将网站制作成自己app?
html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】
如何在阿里云虚拟机上搭建网站?步骤解析与避坑指南
如何在宝塔面板中修改默认建站目录?
Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层
如何在云指建站中生成FTP站点?
微信小程序 canvas开发实例及注意事项
Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优

