Node.jsJavaScript基础上发展起来的语言,所以前端开发者应该天生就会一点。一般我们会用它来做CLI工具或者Web服务器,做Web服务器也有很多成熟的框架,比如ExpressKoa。但是ExpressKoa都是对Node.js原生API的封装,所以其实不借助任何框架,只用原生API我们也能写一个Web服务器出来。本文要讲的就是不借助框架,只用原生API怎么写一个Web服务器。因为在我的计划中,后面会写ExpressKoa的源码解析,他们都是使用原生API来实现的。所以本文其实是这两个源码解析的前置知识,可以帮我们更好的理解ExpressKoa这种框架的意义和源码。本文仅为说明原生API的使用方法,代码较丑,请不要在实际工作中模仿!

本文可运行代码示例已经上传GitHub,大家可以拿下来玩玩:https://github.com/dennis-jiang/Front-End-Knowledges/tree/master/Examples/Node.js/HttpServer

Hello World

要搭建一个简单的Web服务器,使用原生的http模块就够了,一个简单的Hello World程序几行代码就够了:

const http = require('http')const port = 3000const server = http.createServer((req, res) => {  res.statusCode = 200  res.setHeader('Content-Type', 'text/plain')  res.end('Hello World')})server.listen(port, () => {  console.log(`Server is running on http://127.0.0.1:${port}/`)})

这个例子就很简单,直接用http.createServer创建了一个服务器,这个服务器也没啥逻辑,只是在访问的时候返回Hello World。服务器创建后,使用server.listen运行在3000端口就行。

这个例子确实简单,但是他貌似除了输出一个Hello World之外,啥也干不了,离我们一般使用的Web服务器还差了很远,主要是差了这几块:

1.不支持HTTP动词,比如GETPOST等2.不支持路由3.没有静态资源托管4.不能持久化数据

前面三点是一个Web服务器必备的基础功能,第四点是否需要要看情况,毕竟目前很多NodeWeb服务器只是作为一个中间层,真正跟数据库打交道做持久化的还是各种微服务,但是我们也应该知道持久化怎么做。

所以下面我们来写一个真正能用的Web服务器,也就是说把前面缺的几点都补上。

处理路由和HTTP动词

前面我们的那个Hello World也不是完全不能用,因为代码位置还是得在http.createServer里面,我们就在里面添加路由的功能。为了跟后面的静态资源做区分,我们的API请求都以/api开头。要做路由匹配也不难,最简单的就是直接用if条件判断就行。为了能拿到请求地址,我们需要使用url模块来解析传过来的地址。而Http动词直接可以用req.method拿到。所以http.createServer改造如下:

const url = require('url');const server = http.createServer((req, res) => {  // 获取url的各个部分  // url.parse可以将req.url解析成一个对象  // 里面包含有pathname和querystring等  const urlObject = url.parse(req.url);  const { pathname } = urlObject;  // api开头的是API请求  if (pathname.startsWith('/api')) {    // 再判断路由    if (pathname === '/api/users') {      // 获取HTTP动词      const method = req.method;      if (method === 'GET') {        // 写一个假数据        const resData = [          {            id: 1,            name: '小明',            age: 18          },          {            id: 2,            name: '小红',            age: 19          }        ];        res.setHeader('Content-Type', 'application/json')        res.end(JSON.stringify(resData));        return;      }    }  }});

现在我们访问/api/users就可以拿到用户列表了:

支持静态文件

上面说了API请求是以/api开头,也就是说不是以这个开头的可以认为都是静态文件,不同文件有不同的Content-Type,我们这个例子里面暂时只支持一种.jpg吧。其实就是给我们的if (pathname.startsWith('/api'))加一个else就行。返回静态文件需要:

1.使用fs模块读取文件。2.返回文件的时候根据不同的文件类型设置不同的Content-Type

所以我们这个else就长这个样子:

// ... 省略前后代码 ...else {  // 使用path模块获取文件后缀名  const extName = path.extname(pathname);  if (extName === '.jpg') {    // 使用fs模块读取文件    fs.readFile(pathname, (err, data) => {      res.setHeader('Content-Type', 'image/jpeg');      res.write(data);      res.end();    })  }}

然后我们在同级目录下放一个图片试一下:

数据持久化

数据持久化的方式有好几种,一般都是存数据库,少数情况下也有存文件的。存数据库比较麻烦,还需要创建和连接数据库,我们这里不好demo,我们这里演示一个存文件的例子。一般POST请求是用来存新数据的,我们在前面的基础上再添加一个POST /api/users来新增一条数据,只需要在前面的if (method === 'GET')后面加一个POST的判断就行:

// ... 省略其他代码 ...else if (method === 'POST') {  // 注意数据传过来可能有多个chunk  // 我们需要拼接这些chunk  let postData = '';  req.on('data', chunk => {    postData = postData + chunk;  })  req.on('end', () => {    // 数据传完后往db.txt插入内容    fs.appendFile(path.join(__dirname, 'db.txt'), postData, () => {      res.end(postData);  // 数据写完后将数据再次返回    });  })}

然后我们测试一下这个API:

再去看看文件里面写进去没有:

总结

到这里我们就完成了一个具有基本功能的web服务器,代码不复杂,但是对于帮我们理解Node web服务器的原理很有帮助。但是上述代码还有个很大的问题就是:代码很丑!所有代码都写在一堆,而且HTTP动词和路由匹配全部是使用if条件判断,如果有几百个API,再配合十来个动词,那代码简直就是个灾难!所以我们应该将路由处理HTTP动词静态文件数据持久化这些功能全部抽离出来,让整个应用变得更优雅,更好扩展。这就是ExpressKoa这些框架存在的意义,下一篇文章我们就去Express的源码看看他是怎么解决这个问题的,点个关注不迷路~

本文可运行代码示例已经上传GitHub,大家可以拿下来玩玩:https://github.com/dennis-jiang/Front-End-Knowledges/tree/master/Examples/Node.js/HttpServer

文章的最后,感谢你花费宝贵的时间阅读本文,如果本文给了你一点点帮助或者启发,请不要吝啬你的赞和GitHub小星星,你的支持是作者持续创作的动力。

作者博文GitHub项目地址:https://github.com/dennis-jiang/Front-End-Knowledges

作者掘金文章汇总:https://juejin.im/post/5e3ffc85518825494e2772fd

我也搞了个公众号[进击的大前端],不打广告,不写水文,只发高质量原创,欢迎关注~

api怎么写_使用Node.js原生API写一个web服务器相关推荐

  1. electron 打包把node代理服务打包进去_用 Node.js 官方镜像打包一个 express 服务

    最近感觉确实有必要了解一下 Docker,在 CI/CD 这个阶段很有用,所以记录一下. 先放上 Node.js 的官方 Docker 镜像. Docker Hub​hub.docker.com 需要 ...

  2. node aws 内存溢出_如何使用Node.js和AWS快速创建无服务器RESTful API

    node aws 内存溢出 by Mark Hopson 马克·霍普森(Mark Hopson) 如何使用Node.js和AWS快速创建无服务器RESTful API (How to quickly ...

  3. Rust: 基于 napi-rs 开发 Node.js 原生模块

    Rust: 基于 napi-rs 开发 Node.js 原生模块 文章目录 Rust: 基于 napi-rs 开发 Node.js 原生模块 完整代码示例 背景 & napi 环境/工具链准备 ...

  4. Vanilla Node.js REST API示例

    A Vanilla Node.js REST API without Frameworks such us Express | Engineering Education (EngEd) Progra ...

  5. 基于 Node.js + Koa 构建完整的 Web API (配置 ESLint 和使用 Airbnb 编码规范)

    主题内容:基于 Node.js + Koa 构建完整的 Web API (配置 ESLint 和使用 Airbnb 代码规范) 背景描述:上一篇 基于 Node.js + Koa 构建完整的 Web ...

  6. 使用Node.js+Koa 从零开始写个人博客系统——后端部分(一)

    使用Node.js+Koa 从零开始写个人博客系统系列 提示:在此文章中你可以学习到的内容如下: 1 如何使用Koa快速搭建项目 2 对Koa的核心组件Koa-Route的简单使用 3 3层架构思想 ...

  7. 十个书写Node.js REST API的最佳实践(上)

    收录待用,修改转载已取得腾讯云授权 原文:10 Best Practices for Writing Node.js REST APIs 我们会通过本文介绍下书写Node.js REST API的最佳 ...

  8. 【node.js后台api项目】(二)实现注册功能

    [node.js后台api项目](二)实现注册功能 一.实现思路 二.项目目录 三.代码编写 1.基本代码 2.数据合法性校验 3.注册路由处理函数 4.密码加密 一.实现思路 注册功能简单来说就是提 ...

  9. 【node.js后台api项目】(七)更新用户头像接口

    [node.js后台api项目](七)更新用户头像接口 1.接口相关信息 2. 定义路由和处理函数 3.验证提交的数据 4.实现更新用户基本信息功能 1.接口相关信息 路由: /my/update/a ...

最新文章

  1. oracle 学习笔记 Flashback drop
  2. 23. 进程并发控制之Semaphore
  3. VC++ ATL 学习总结
  4. 我在阿里收获的N个成长
  5. 技术人的七夕表白可以有多浪漫?
  6. bert 无标记文本 调优_使用BERT准确标记主观问答内容
  7. python爬虫requests源码链家_Python爬虫之---爬链家
  8. 【HDU5091】Beam Cannon,扫描线+线段树
  9. 2015年下半年的读书技术
  10. keil中下载按钮和调试按钮灰掉了
  11. 重新认识JavaScript面向对象: 从ES5到ES6
  12. python 录制网易云登陆_Github获8300星!用Python开发的一个命令行的网易云音乐
  13. 计算机桌面蓝字,教你电脑桌面图标有蓝色阴影怎么去掉
  14. 斐波那契数列c语言程序改错题,C语言习题004:斐波那契数列
  15. 移动拼图游戏(八数码问题) BFS版
  16. Redis记录:Invalid argument during startup: Failed to open the .conf file
  17. 怎样用html做学生成绩表,怎样用excel制作学生成绩单
  18. 记事本打开文件乱码的问题
  19. 星型模型和olap多维数据库
  20. python 批量下载小说

热门文章

  1. Java正成为COBOL的一部分-它将成为COBOL的一部分吗?
  2. Spring @Lazy批注用例
  3. 并发加对象锁_通用并发对象池
  4. maven 父maven_Maven的鸟瞰图
  5. SWT ScrolledComposite解释
  6. 在Sqoop中管理密码的关键提示
  7. 比较中的Commons VFS,SSHJ和JSch
  8. 单元测试编写_为什么要编写单元测试-测试技巧8
  9. Wildfly,Apache CXF和@SchemaValidation
  10. 但这是不可能的,或者无法发现JIT破坏了您的代码。