nodejs基础

Buffer: 处理二进制数据

二进制数据的获取

  • 文件数据
  • 网络数据

二进制数据的可读性:base64

Base64是一种基于64个可打印字符来表示二进制数据的方法。

转换前 11111111,11111111,11111111

转换后 00111111, 00111111, 00111111, 00111111

十进制 63 63 63 63

对应码表中的值 ////

图片
let buf = Buffer.alloc(3, 0xff)

/* 输出 //// */
console.log(
    buf.toString('base64')
)

二进制数据的显示

显示成base64
buffer.toString('base64')
显示成utf8编码的字符串
buffer.toString('utf8')
显示成16进制
buffer.toString('hex')

创建自定义Buffer

通过utf8编码的字符串创建
Buffer.from('text', 'utf8')
通过base64创建
Buffer.from('MTIz', 'base64')
生成并初始化一个Buffer
Buffer.alloc(3, 0xff)
数据格式转换

实现16进制、base64、utf8的自由转换

Buffer.from('text', 'utf8').toString('base64')

Stream

Writable 可写流

向目标写入数据

图片
const fs = require('fs')

let writable = fs.createWriteStream('./log')

writable.write('some text\n')
writable.end('end\n')

注意:如果写入数据时,writable.write(data)返回false说明缓冲区已满,需要等待drain事件触发后,再继续写入。

Readable 可读流

从目标读取数据

两种读取数据的模式

主动读取
图片

注意:缓冲区数据为空时readable.read()会返回null,建议配合readable事件表示。

被动接收

只要缓冲区中存在数据,就会触发data事件

const fs = require('fs')

let readable = fs.createReadStream('./log')

let bufs = []

// 绑定data事件后,流对象才会触发该事件
readable.on('data', buf => {
    bufs.push(buf)
})

readable.on('end', () => {
    console.log(Buffer.concat(bufs).toString())
})

pipe

连接可读流和可写流

const http = require('http')
const fs = require('fs')

http.createServer((req, res) => {
    fs.createReadStream('./index.html').pipe(res)
}).listen(3001)

Duplex and Transform Streams

图片
gulp.src('client/templates/*.jade')
  .pipe(jade())
  .pipe(minify())
  .pipe(gulp.dest('build/minified_templates'));

http模块

发送请求

const http = require('http')

// 可写流
let writable = http.request({
    protocol: 'http:',
    host: 'localhost',
    method: 'POST',
    headers: {
        'content-type': 'application/json'
    },
    path: '/test.json'
}, onResponse)

// 与请求头中的数据类型保持一致
writable.write(
    JSON.stringify({
        a: 1,
        b: 2
    })
)

// 结束流
writable.end()

/**
 * @param {Stream.Readable} readable
 */
function onResponse (readable) {
    let bufs = []
    readable.on('data', buf => {
        // 不能转成字符
        bufs.push(buf)
    })

    readable.on('end', () => {
        console.log(Buffer.concat(bufs).toString())
    })
}

http服务器

const http = require('http')

const server = http.createServer((readable, writable) => {

    // 从readable中获取数据

    writable.writeHead(200, {
        'content-type': 'application/json'
    })

    writable.write(
        JSON.stringify({
            code: 200,
            msg: 'success'
        })
    )
    writable.end()
})

server.listen(3001)

bigpipe

可以让浏览器在请求完全结束前提前开始页面渲染

const http = require('http')

const server = http.createServer(async (readable, writable) => {

    writable.writeHead(200, {
        'content-type': 'text/html'
    })

    await sleep(1000)
    writable.write('<div>a</div>')

    await sleep(1000)
    writable.write('<div>b</div>')

    await sleep(1000)
    writable.write('<div>c</div>')

    writable.end()
})

server.listen(3001)

function sleep (ms) {
    return new Promise(resolve => {
        setTimeout(() => resolve(), ms)
    })
}

express

每个中间件是一个函数。

每个到达服务器的请求都会交给声明的中间件依次处理。

每个中间件都可以选择直接向客户端返回数据或是将请求继续交给下一个中间件处理。

图片
const express = require('express')

const app = express()

app.use((req, res, next) => {
    console.log(req.originalUrl)
    next()
})

app.use((req, res, next) => {
    res.writeHead(200, {
        'content-type': 'text/html'
    })
    res.write('<div>hello world</div>')
    res.end()
})

app.use((req, res) => {
    console.log('不会执行')
})

app.listen(3001)

koa

每个中间件都可以选择直接返回上一个中间件或是将请求继续交给下一个中间件处理。

图片
const Koa = require('koa')

const app = new Koa()

app.use(async (ctx, next) => {
    let start = Date.now()
    await next()

    console.log(`${ctx.request.url}:${Date.now() - start}ms`)
})

app.use(async (ctx, next) => {
    ctx.response.status = 200
    ctx.response.headers['content-type'] = 'text/html'
    await sleep(1000)
    ctx.body = 'hello world'
})

app.use(async (ctx, next) => {
    console.log('不会执行')
})

app.listen(3001)

function sleep(ms) {
    return new Promise(resolve => {
        setTimeout(() => resolve(), ms)
    })
}

常用中间件

为服务器增加静态资源访问和缓存的能力
File Serving

解析HTTP请求体中的数据
Body Parsing

日志记录
Logging

多进程

单个nodejs进程无法有效利用多核CPU

// index.js

const cp = require('child_process')

for (let i = 0; i < 5; i++) {
    
    let ins = cp.fork('./sub.js')

    ins.on('message', msg => {
        console.log('master get message: ' + JSON.stringify(msg))
    })

    ins.send({
        text: 'message sent by master'
    })
}
// sub.js

console.log('sub: ' + process.pid)

process.on('message', msg => {
    console.log('sub get message: ' + JSON.stringify(msg))

    process.send({
        text: 'message sent by ' + process.pid
    })

})

cluster

解决的问题:多个nodejs无法监听同一个端口。

模块实现了多个进程同时监听同一个端口的功能。

const cluster = require('cluster');
const http = require('http');
const numCPUs = require('os').cpus().length;

if (cluster.isMaster) {
  console.log(`Master ${process.pid} is running`);

  // Fork workers.
  for (let i = 0; i < numCPUs; i++) {
    cluster.fork();
  }

  cluster.on('exit', (worker, code, signal) => {
    console.log(`worker ${worker.process.pid} died`);
  });
} else {
  // Workers can share any TCP connection
  // In this case it is an HTTP server
  http.createServer((req, res) => {
    res.writeHead(200);
    res.end('hello world\n');
  }).listen(8000);

  console.log(`Worker ${process.pid} started`);
}

pm2

多进程管理

根据服务器CPU的核数启动多个实例

实例意外退出时会自动重启实例

$ pm2 start index.js -i -1

进程监控

$ pm2 monit

图片

开发模式

$ pm2 start index.js --watch

部署

提供远程部署nodejs服务器的功能。

pm2 通过ssh登录远程服务器,执行部署操作。

/* deploy.json*/
{
    "apps": [
        {
            "name": "server",
            "script": "index.js",
            "instances": 1,
            "exec_mode": "cluster"
        }
    ],
    "deploy": {
        "server1": {
            "user": "root",
            "host": [
                "127.0.0.1"
            ],
            "ref": "origin/master",
            "repo": "https://github.com/user/project.git",
            "path": "/root/code",
            "post-deploy": "git pull && yarn && pm2 startOrRestart deploy.json",
            "pre-setup": "rm -rf /root/code"
        }
    }
}

第一次部署,pm2 会执行pre-setup,删除指定目录

pm2 deploy deploy.json server1 setup

再次部署,pm2 会执行post-deploy,从git拉取最新代码、安装依赖、重新启动程序

pm2 deploy deploy.json server1

eggjs

eggjs

继承自Koa, 增加了一组目录规范和一些常见问题的解决方案,包括应用部署、定时任务、日志、单元测试、CSRF防御、数据库操作等。

多进程模型

图片

服务器性能

wrk

一个http压测工具

Mac中安装方式:brew install wrk

8个线程、建立200个连接,连续请求30s

$ wrk -t8 -c200 -d30s --latency "http://localhost:3001"

测试服务器在一段时间内持续接收到大量请求时的延迟情况。

图片
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容

  • NodeJs 用于服务器端开发,是执行js的ECMAScript标准,不包含DOM和BOM,但是部分API可以使用...
    Is_JackieZhang阅读 2,071评论 1 3
  • 一.URL网址解析的好帮手(node API -- URL) url的方法1.parse:分析、解析eg: url...
    qqqc阅读 317评论 0 0
  • NodeJS基础原理 异步IO原理浅析及优化方案 异步IO的好处(输入输出input output) 前端通过异步...
    扶搏森阅读 1,830评论 0 3
  • 文件系统 在Node中,与文件系统的交互是非常重要的,服务器的本质就是将本地的文件发送到远程的客户端 Buffer...
    Is_JackieZhang阅读 584评论 0 0
  • 什么是NodeJS JS是脚本语言,脚本语言都需要一个解析器才能运行。对于写在HTML页面里的JS,浏览器充当了解...
    MrZhou_b216阅读 403评论 0 2