node.js学习笔记

node.js学习笔记

node

01

什么是node?

  • chrome V8 runtime
  • 事件驱动
  • 非阻塞的i/o

优点:处理高并发比较好

模块化

  • 内置模块(node中自带的模块)
    • 文件操作
    1. fs.readdir
    2. fs.mkdir(同步,异步)
    3. fs.rename
    4. fs.rmdir(只能删除空文件夹)
    • 错误信息处理 node中大多数秉持错误优先原则,回调中有err参数,同步代码为了不影响后续代码执行需要使用try {} catch (err) {}捕获错误。
  • 第三方模块
  • 自定义模块

自定义模块使用模块步骤

1.创建一个模块(一个js文件一个模块)

2.导出一个模块(module.exports = name)

3.引入一个模块并且调用

运行环境

REPL :在命令行执行的环境。

内置文件模块fs

文件操作

创建文件(覆盖写入)

  fs.writeFile(path,option,callback)

写入文件(追加写入)

  fs.appendFile(path,option,callback)

读取文件

fs.readFile(path,config,(err,msg)=>{
  //msg默认为buffer数据流 可以使用 msg.toString('utf8')转换数据
  //也可以使用config出配置‘utf8’
})

删除文件

fs.unlink(path,callback)

内置url模块

url : 统一资源定位符

url.parse();将url字符串转换为对象

url.format();将url对象转换为字符串

内置querystring模块

qs.parse(str,query1,query2);
//query1 表示以什么符号切分键值对 默认 &
//query2 表示以什么符号切分键与值 默认 = 
qs.stringify(obj,query1,query2);
//query1 表示以什么符号拼接键值对 默认 &
//query2 表示以什么符号拼接键与值 默认 =
qs.escape() //编码
 
qs.unescape() //解码

第三方模块

nodemailer 可以实现发邮件

"use strict";
const nodemailer = require("nodemailer");

// 创建一个发送器
let transporter = nodemailer.createTransport({
    host: "smtp.qq.com",
    port: 465,
    secure: true, // true for 465, false for other ports
    auth: {
        user: '572410049@qq.com', // 发送者邮箱地址
        pass: 'pwuepzqfdpjybeid' // 发送者SMTP密码
    }
});

// 配置发送信息对象
var emailconfig = {
    from: '572410049@qq.com', // 发送者邮箱地址
    to: "572410049@qq.com", // 接受者邮箱地址
    subject: "你好", // 主题
    text: "准备睡觉吧", // 内容 只能是纯文本 与 html 配置项只能存在一个
    // html: "<b>Hello world?</b>" // html body
}

// 调用sendMail()方法
transporter.sendMail(emailconfig);

爬虫案例

1.获取目标网站
2.分析网站内容
3.获取有效信息 下载或其他操作

http 模块

http模块负责发请求:

const http = require('https');
let html = '';
http.get(url,(res)=>{

  let { statusCode } = res; //状态码
  let contentType = res.headers['content-type']; //请求类型
  let err = null;
  if(statusCode !== 200){
    err = new Error('请求状态出错');
  }else if(!/^text\/html/.test(contentType)){
    err = new Error('请求类型出错');
  }

  if(err){
    console.log(err);//此处可以确定错误类型
    res.resume(); //重置缓存
    return false //终止程序
  }
  //---------------------------- 代码健壮判断抓取的是否为所需类型,和是否能访问目标地址 ----------------------------------------
  res.on('data',(chunk)={
    //chunk 为buffer数据流 需通过chunk.toString('utf8')转换
    //爬取中。。。
    html += chunk.toString('utf8')
  });
  res.on('end',()=>{
    console.log(html);
    //爬取结束。
  })
}).on('error',(e)=>{
  console.log('请求出错')
})

cheerio 模块

将一组html格式的字符串 转化为类之后可以通过jq的语法选中其中的元素。

const cheerio = require('cheerio');
//将一组html格式的字符串 转化为类之后可以通过jq的语法选中其中的元素。
var div = '<div> <img src="http://www.baidu.com"/> </div>'
const $ = cheerio.load(div);

console.log($('img').attr('src'));

实现抓取一个页面的图片

const http = require('https');
const fs = require('fs');
const cheerio = require('cheerio');
const request = require('request');
const reg = /(http|https):\/\/([\w.]+\/?)\S*/ig
let url = 'https://www.zcool.com.cn/';
let html = '';
http.get(url, (res) => {
    let { statusCode } = res; //状态码
    let contentType = res.headers['content-type']; //请求类型
    let err = null;
    if (statusCode !== 200) {
        err = new Error('请求状态错误')
        //请求状态错误
    } else if (!/^text\/html/.test(contentType)) {
        //请求类型错误
        err = new Error('请求类型错误')
    }
    if (err) {
        console.log(err);
        res.resume(); //重置缓存
        return;
    }
    res.on('data', (chunk) => {
        html += chunk.toString('utf8');
        // console.log('爬取中')
    });
    res.on('end', () => {
        console.log(html);
        const $ = cheerio.load(html)
        $('img').each((index, el) => {
            if (reg.test($(el).attr('src'))) { //判断是否是以http 或者https 开头的
                request($(el).attr('src')).pipe(  //下载到本地
                    fs.createWriteStream(`./node/${index}.png`).on('close', err => {
                        console.log('下载成功', err)
                    })
                )
            }
        })
    })
  }).on('error', (e) => {
     console.log('出错啦');
  })

02

通过express 框架书写api

1.安装 express

npm install express --save

模块(第三方模块)的引用 从当前目录的node_modules依次向上寻找。

服务器相关

服务器:

1.服务器本质就是一台电脑

2.需要服务器软件(apatch tomcat iis nginx node)

3.服务器ip和端口号 (80)

局域网:服务器通过(无线)网线连接 每一台电脑都被分配一个ip

外网:

ip:确定服务器主机的位置

port:是确定服务器里程序的位置

api接口

构成要素:ip port pathname method(get,post) data

  • 接收数据
    • get req.query
    • post req.body 需要body-parser模块
      • 注意数据格式
        • json (app.use(bodyParser.json()))
        • x-www-form-urlencoded (app.use(bodyParser.urlencoded(extended:false)))
        • formdata

中间件 middlewear

  • 内置中间件 static
  • 自定义中间件
  • 第三方中间件 (body-parser)

中间件使用一定要注意 next);

中间件有全局中间件,还有局部中间件,局部中间只走该中间件所在接口。

static 静态目录 (类似apche www目录)

path 模块 处理路径

对文件操作 必须使用绝对路径

__dirname 绝对路径

static 模块使用

app.use('./',express.static(path.join(__dirname,'./html')));

此时 可以通过http://localhost:3000/index.html 访问html文件夹下的index.html文件

node router

防止一个文件写过多接口造成不良影响,拆分接口到对应文件

server.js

const express = require('express');
const app = express();
const bodyParser = require('body-parser');

app.use(bodyParser.json());

app.use('/',(req,res,next)=>{
    if(req.query.token || req.body.token){
        next()
    }else{
        res.send('no token')
    }
});
//引入各个router
const userRouter = require('./router/user');
const customRouter = require('./router/custom');
//使用各个router
app.use('/user',userRouter);
app.use('/custom',customRouter);

app.listen(3000,()=>{
    console.log('server start');
})

router/user.js

const express = require('express');
const router = express.Router(); //实例化router

router.get('/login',(req,res)=>{
    res.send('login ok')
})

router.get('/logout',(req,res)=>{
    res.send('logout ok');
})

module.exports = router; //导出

非关系数据库 mongodb

1.mongodb 下载 (下载速度慢问题解决mongodb下载慢问题

2.安装 建议参考菜鸟教程 mongodb安装 菜鸟教程mongodb windows安装

指令

mongodb 数据库名

mongod 命令行启动数据库

mongo 命令行操作数据库指令

mongoose node操作mongodb的插件

promise

为什么要是用promise?

大量的异步操作如果需要顺序执行通过回调函数会形成回调地狱

通过promise解决回调地狱

1.封装promise函数

//使用形式
//返回promise
function test(){
  return new Promise((resolve,reject)=>{
    //异步处理
    //成功的时候
    resolve();
    //失败的时候
    reject();
  })
}

2.根据顺序,形成链式调用
test().then().then();

在第一个then中需再return出一个promise对象

3.捕获错误

示例代码:

const fs = require('fs');
//回调形式
// fs.stat('./aaa.js',(err,stats)=>{ 
//     if(err){
//         console.log('文件不存在')
//     }else{
//         console.log('文件存在');
//         fs.unlink('./aaa.js',(err)=>{
//             if(err){
//                 console.log('删除错误');
//             }else{
//                 console.log('删除正确');
//             }
//         })
//     }
// })

//promise形式
function fileIsExsit(){
    return new Promise((resolve,reject)=>{
        fs.stat('./aaa.js',(err,stats)=>{
            if(err){
                console.log('不存在')
                reject('err')
            }else{
                console.log('存在')
                resolve('yes')
            }
        })
    })
}

function filedel(){
    return new Promise((resolve,reject)=>{
        fs.unlink('./aaa.js',(err)=>{
            if(err){
                reject('删除失败')
            }else{
                resolve('删除成功')
            }
        })
    })
}

fileIsExsit()
.then((data)=>{
    console.log(data,'------------then')
    return filedel();
})
.then((data)=>{
    console.log(data,'----then')
    //如何手动终止then?
    throw new Error('手动抛出错误,为了手动终止then执行');
})
.then((a)=>{
    console.log(111);
})
.catch((data)=>{
    console.log(data,'------------catch')
})

mongoose

mongoose:为了方便操作数据库而诞生的一个模块

1.安装mongoose:

npm install mongoose

2.引入模块,连接数据库

const mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/users',{ useNewUrlParser: true,useUnifiedTopology: true })
//第一个参数为本地数据库,第二个参数为mongoose.connect()方法新增的参数,否则会warning

3.创建数据库连接对象

var db = mongoose.connection;
db.on('error',console.error.bind(console,'connection error:')) //监听错误
db.on('open',function(){ //监听连接成功
  console.log('数据库连接成功');
})

4.创建schema对象

var userSchema = new mongoose.Schema({
  name:String,
  age:Number,
  sex:{
    type:Number,
    default:0
  },  
})

5.创建model结构

//将schema对象转换为数据模型
//将schema对象与集合关联('集合名',schema对象);注意:集合名mongoose会自动转换成复数形式因此建议集合名使用复数形式
var User = mongoose.model('users',userSchema);

6.操作数据库

//增 
//insertMany([options]) options 可为数组 可为对象 是追加操作,会追加到当前表的数据末尾
// User.insertMany([{name:'tll',age:18},{name:'zt',age:18,sex:0},{name:'ah',age:18,sex:0}])
// .then((data)=>{
//     console.log(data);
//     console.log('插入成功');
// })
// .catch((data)=>{
//     console.log('插入失败');
// })

// 删
// deleteOne([option]) 删除一条数据,option为筛选条件,当有多条数据符合该条件时,只删除第一个查到的,当option为空时默认删除第一条 
// deleteMany([option]) 删除多条数据,option为筛选条件,当options为空时删除该集合所有数据 

// User.deleteMany({name:'zt'})
// .then((data)=>{
//     console.log(data,'删除所有成功')
// })
// .catch((data)=>{
//     console.log(data,'删除所有失败');
// })

// 改
// update(query,update) query为查询条件,update为修改后的数据 如果匹配到多条数据只修改第一条
// updateMany(query,update) 批量修改query为查询条件,update为修改后的数据 
// updateOne(query,update) 与update()相同,只修改匹配到的第一条数据
// User.update({sex:1},{age:90})
// .then((data)=>{
//     console.log(data,'更新成功');
// })
// .catch((data)=>{
//     console.log(data,'更新失败')
// })

// 查
/*
mongodb 条件操作符
$gt >
$lt <
$gte >=
$lte <=
$ne !=
$eq =
模糊搜索
{title:/教/} title中包含教的文档
{title:/^教/} title中以教开头的文档
{title:/教$/} title中以教结尾的文档
*/

//find(query) // 查询符合条件的所有数据 
//findOne() //查询符合条件的第一条数据
//findById() //通过id查找一条数据,该id为mongodb自动生成的id 在查找前会对id进行数据转换
// User.findById("5e70dcf88963670128d1c20e")
// .then((data)=>{
//     console.log(data,'查询成功')
// })
// .catch((data)=>{
//     console.log(data,'查询失败')
// })

使用mongodb + nodeJs实现一个登录注册逻辑

1.项目结构

- server.js   //服务
- db  //数据库管理
  - model 
    - userModel.js //创建userModel对象
  - connect.js  //连接数据库
- router //路由
  - useRouter.js //接口具体逻辑

2.项目需求分析

1.注册

先判断数据库中是否有userName,有则返回账号已存在,注册失败,否则写入数据库,标识注册成功。

2.登录

根据userName 和 password 判断数据库中是否存在该条数据,存在则表示登录成功,否则表示失败。

server.js

const express = require('express');
const db = require('./db/connect');
const app = express();
const userRouter = require('./router/userRouter');
const bodyParser = require('body-parser');
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({extended:false}))

app.use('/user',userRouter);

app.listen(3000,()=>{
    console.log('server start');
})

connect.js

const mongoose = require('mongoose');
mongoose.connect('mongodb://localhost/users',{ useNewUrlParser: true,useUnifiedTopology: true });
const db = mongoose.connection;
db.on('error',()=>{
})
db.once('open',()=>{
    console.log('connect ok');
})

userModel.js

const mongoose = require('mongoose');

var userModel = new mongoose.Schema({
    us:{
        type:String,
        default:''
    },
    ps:{
        type:String,
        default:'',
    },
    uid:{
        type:String,
        default:''
    }
})
var user = mongoose.model('users',userModel);
module.exports = user;

userRouter.js

const express = require('express');
const router = express.Router();
const User = require('../db/model/userModel');
router.post('/reg',(req,res)=>{
    let{us,ps} = req.body;
    if(!us || !ps){
        res.send({
            err:-1,
            msg:"参数错误"
        })
    }else{
        User.find({us})
        .then((data)=>{
            if(data.length >= 1){
                res.send({
                    err:-1,
                    msg:"该用户已注册"
                })
            }else{
              return User.insertMany({us,ps,uid:us+ps})
            }
        })
        .then((data)=>{
            if(data){
                res.send({
                    err:0,
                    msg:'注册成功'
                })
            }
            console.log(data,'insert')
        })
        .catch((err)=>{
            res.send({
                err:-1,
                msg:'系统内部错误,请稍后重试'
            })
        })
    }
})
router.post('/login',(req,res)=>{
    let {us,ps} = req.body;
    if(!us || !ps){
        res.send({
            err:-1,
            msg:'参数错误'
        })
    }else{
        User.find({us,ps})
        .then((data)=>{
            console.log(data,'50000');
            if(data){
                res.send({
                    err:0,
                    msg:'注册成功',
                    us:data[0].us,
                    uid:data[0].uid
                })
            }
        })
        .catch((err)=>{
            res.send({
                err:-1,
                msg:'系统内部错误'
            })
        })
    }

})
module.exports = router;

跨域问题

ajax 同源策略 协议 域名 端口号

解决跨域:

  • 1.cors:后端配合
    • express cors
    • app.all()
app.all("*",function(req,res,next){ //设置允许跨域的域名“*”代表允许所有
    //允许的请求头
    res.header("Access-control-Allow-Origin","*");
    //设置允许的请求方式
    res.header("Access-Control-Allow-Methods","DELETE,PUT,POST,GET,OPTIONS");
    if(req.method.toLowerCase() == 'options'){
        res.send(200)
    }else{
        next();
    }
})

上传图片

npm install multer --save

//后端
var storage = multer.diskStorage({
    destination:function(req,file,cb){
        cb(null,'./uploads'); //定义文件存放地址
    },
    filename:function(req,file,cb){ //定义上传的文件名
        let types = file.originalname.split('.');
        let type = types[types.length - 1]; //取后缀
        cb(null,file.fieldname+'-'+Date.now()+parseInt(Math.random()*9999)+'.'+type);
    }
})


var upload = multer({storage:storage}); //调用

router.post('/imgs',upload.single('keys'),(req,res)=>{  //这个keys 属于约定,前后端都得知道
    const {mimetype,size,filename} = req.file;
    let types = ['jpg','png','jpeg','gif'];
    let imgtype = mimetype.split('/')[1];
    if(size>102400){  //上传大小判断
        res.send({
            err:-1,
            msg:'图片尺寸超过10k'
        });
        return;
    }else if(types.indexOf(imgtype)<=-1){  //文件类型判断
        res.send({
            err:-1,
            msg:"图片类型错误"
        })
        return;
    }else{
        res.send({
            err:0,
            msg:"上传成功",
            url:`/img/${filename}` //上传成功后将地址返回给前端
        })
    }
})

//前端

        $('button').click(function(){
            let formData = new FormData();
            let file = $('#file')[0].files[0];
            formData.append('keys',file);

            $.ajax('http://localhost:3000/upload/imgs',{
                method:'POST',
                cache:false,
                data:formData,
                processData:false,
                contentType:false,
                success:function(data){
                    console.log(data);
                    $('img').attr('src',`http://localhost:3000${data.url}`)
                }
            })

            console.log(file);
        })
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容