node.js中 mysql 增删改查操作及async,await处理实例分析


Posted in Javascript onFebruary 11, 2020

本文实例讲述了node.js中 mysql 增删改查操作及async,await处理。分享给大家供大家参考,具体如下:

要对mysql进行操作,我们需要安装一个mysql的库。

一、安装mysql库

npm install mysql --save

二、对mysql进行简单查询操作

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//连接数据库
conn.connect(function (err) {
  if (err) {
    throw err;
  }
  console.log('连接成功');
});
//查询数据库
conn.query('select * from tb_user', function (err, data, field) {
  if (err) {
    throw err;
  }
  //data表示结果集数据,是一个数组
  console.log(data);
  data.forEach(function (value) {
    console.log(value.id, value.user_name, value.addr);
  });
  //表字段的详细信息
  console.log(field);
});
//关闭数据库连接
conn.end();

二、对mysql进行增删改操作

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//连接数据库
conn.connect(function (err) {
  if (err) {
    throw err;
  }
  console.log('连接成功');
});
//插入数据,query()方法可以对sql语句进行参数绑定,用?号作为占位符。
conn.query('insert into tb_user values(null, ?, ?)', ['xxx', 'xxx'], function (err, data) {
  if (err) {
    throw err;
  }
  if (data && data.affectedRows) {
    console.log('插入数据成功,id为', data.insertId);
  }
});
//修改数据
conn.query('update tb_user set user_name = ? where id = ?', ['ggg', 7], function (err, data) {
  if (err) {
    throw err;
  }
  if (data && data.affectedRows) {
    console.log('修改数据成功');
  }
});
//删除数据
conn.query('delete from tb_user where id = ?', [5], function (err, data) {
  if (err) {
    throw err;
  }
  if (data && data.affectedRows) {
    console.log('删除数据成功');
  }
});
//关闭数据库连接
conn.end();

三、使用mysql连接池来优化对数据库的操作

频繁的连接和断开mysql是比较消耗资源的,我们可以创建一个连接池,复用连接池中的连接,提高效率。

const mysql = require('mysql');
//创建数据库连接池
let pool = mysql.createPool({
  //连接数量,默认是10
  connectionLimit: 20,
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//pool.query()方法可以自动的帮我们在连接池中获取可用连接
pool.query('select * from tb_user', function (err, data) {
  if (err) {
    throw err;
  }
  data.forEach(function (value) {
    console.log(value.id, value.user_name, value.addr);
  });
});
//当然我们也可以手动获取可用连接
pool.getConnection(function (err, conn) {
  if (err) {
    throw err;
  }
  conn.query('select * from `order`', function (err, data) {
    if (err) {
      throw err;
    }
    data.forEach(function (value) {
      console.log(value.id, value.order_id, value.user_id);
    });
    //连接用完之后,需要释放,重新放回连接池中。
    //注意这里并没有销毁该连接,该连接仍然可用,但需要重新获取
    conn.release();
  });
});
//从连接池中获取连接时,将触发该事件
pool.on('acquire', function (conn) {
  console.log('获取连接', conn.threadId);
});
//在连接池中建立新连接时,将触发该事件
pool.on('connection', function (conn) {
  console.log('建立新连接', conn.threadId);
});
//等待可用连接时,将触发该事件
pool.on('enqueue', function () {
  console.log('等待可用连接');
});
//当连接释放回池中时,触发该事件
pool.on('release', function (conn) {
  console.log('连接被释放回池中', conn.threadId);
});
//结束池中所有的连接,不然node.js的事件循环会一直保持
setTimeout(function () {
  pool.end(function (err) {
    console.log('关闭连接池');
    console.log(err);
  });
}, 3000);

四、按流的方式进行查询

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
let query = conn.query('select * from tb_user');
//Query类继承自Sequence,而Sequence继承自EventEmitter
//所以Query类的实例是可以监听事件
//发生错误时
query.on('error', function (err) {
  console.log(err);
});
//获取查询字段信息
query.on('fields', function (fields) {
  console.log(fields);
});
//获取查询结果
query.on('result', function (result) {
  //暂停获取结果
  conn.pause();
  //跟流的pause()和resume()很类似,控制获取数据的频率。
  setTimeout(function () {
    console.log(result);
    //恢复获取结果
    conn.resume();
  }, 1000);
});
//查询结束
query.on('end', function () {
  console.log('查询结束');
});
conn.end();

通过query.stream()方法返回一个可读流来获取数据

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//从一个查询中获取一个可读流
let qs = conn.query('select * from tb_user').stream({highWaterMark: 2});
let result = [];
qs.on('data', function (data) {
  result.push(data);
});
qs.on('end', function () {
  console.log('查询结束');
  console.log(result);
});
conn.end();

五、mysql的事务处理

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//连接数据库
conn.connect(function (err) {
  if (err) {
    throw err;
  }
  console.log('连接成功');
});
//开启一个事务
conn.beginTransaction(function (err) {
  if (err) {
    throw err;
  }
  conn.query('update account set money = money - 50 where name = ?', ['A'], function (err, data) {
    if (err) {
      //如果有错误则回滚
      return conn.rollback(function () {
        throw err;
      });
    }
    conn.query('update account set money = money + 50 where name = ?', ['B'], function (err, data) {
      if (err) {
        //如果有错误则回滚
        return conn.rollback(function () {
          throw err;
        });
      }
      //提交事务
      conn.commit(function (err) {
        if (err) {
          //如果有错误则回滚
          return conn.rollback(function () {
            throw err;
          });
        }
        console.log('处理成功');
        conn.end();
      });
    });
  });
});

六、解决mysql嵌套回调的问题

有些时候我们的操作需要上一个操作的结果,这样会导致比较深的嵌套问题,为了解决可以使用async和await来解决,而async和await又是基于promise的。

const mysql = require('mysql');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
function query(conn, sql, params = []) {
  if (!conn) {
    return;
  }
  return new Promise(function (resolve, reject) {
    conn.query(sql, params, function (err, data) {
      if (err) {
        reject(err);
      } else {
        resolve(data);
      }
    });
  });
}
(async function () {
  let result = await query(conn, 'select * from tb_user');
  console.log(result);
  let row = await query(conn, 'select * from tb_user where id = ?', [result[0].id]);
  console.log(row);
  conn.end();
})();

当然我们还可以使用 util.promiseify() 进行包装。

const mysql = require('mysql');
const util = require('util');
//创建数据库连接
let conn = mysql.createConnection({
  //主机地址
  host: '127.0.0.1',
  //用户名
  user: 'root',
  //密码
  password: '123456',
  //数据库
  database: 'test',
  //端口
  port: 3306,
  //字符集
  charset: 'utf8'
});
//注意通过util.promisify进行包装的函数,必须满足
//1、函数的最后一个参数是回调函数
//2、回调函数的参数为(err, result),前者是错误,后者是正常结果
//注意这里不要重新创建一个变量,不然会报错。
conn.query = util.promisify(conn.query);
(async function () {
  let result = await conn.query('select * from tb_user');
  console.log(result);
  let row = await conn.query('select * from tb_user where id = ?', [result[0].id]);
  console.log(row);
  conn.end();
})();

希望本文所述对大家node.js程序设计有所帮助。

Javascript 相关文章推荐
jQuery动画animate方法使用介绍
May 06 Javascript
JavaScript实现N皇后问题算法谜题解答
Dec 29 Javascript
详解JavaScript的策略模式编程
Jun 24 Javascript
JQuery中attr属性和jQuery.data()学习笔记【必看】
May 18 Javascript
Bootstrap里的文件分别代表什么意思及其引用方法
May 01 Javascript
jquery加载单文件vue组件的方法
Jun 20 jQuery
Node解决简单重复问题系列之Excel内容的获取
Jan 02 Javascript
react中实现搜索结果中关键词高亮显示
Jul 31 Javascript
JavaScript制作3D旋转相册
Aug 02 Javascript
es6数组之扩展运算符操作实例分析
Apr 25 Javascript
js数组中去除重复值的几种方法
Aug 03 Javascript
vue-cli3访问public文件夹静态资源报错的解决方式
Sep 02 Javascript
基于Angular 8和Bootstrap 4实现动态主题切换的示例代码
Feb 11 #Javascript
原生js实现点击轮播切换图片
Feb 11 #Javascript
node.js中process进程的概念和child_process子进程模块的使用方法示例
Feb 11 #Javascript
小程序如何定位所在城市及发起周边搜索
Feb 11 #Javascript
JS+DIV实现拖动效果
Feb 11 #Javascript
vue项目中使用particles实现粒子背景效果及遇到的坑(按钮没有点击响应)
Feb 11 #Javascript
原生js拖拽实现图形伸缩效果
Feb 10 #Javascript
You might like
《DOTA3》开发工作已经开始 《DOTA3》将代替《DOTA2》
2021/03/06 DOTA
晋城吧对DiscuzX进行的前端优化要点
2010/09/05 PHP
php之readdir函数用法实例
2014/11/13 PHP
PHP中使用substr()截取字符串出现中文乱码问题该怎么办
2015/10/21 PHP
php获取给定日期相差天数的方法分析
2017/02/20 PHP
phpStudy配置多站点多域名和多端口的方法
2017/09/01 PHP
firefox浏览器不支持innerText的解决方法
2013/08/07 Javascript
JS实现Enter键跳转及控件获得焦点
2013/08/12 Javascript
Jquery实现由下向上展开效果的例子
2014/12/08 Javascript
JS+CSS实现可拖动的弹出提示框
2015/02/16 Javascript
js实现下拉列表选中某个值的方法(3种方法)
2015/12/17 Javascript
动态更新highcharts数据的实现方法
2016/05/28 Javascript
layer弹出层框架alert与msg详解
2017/03/14 Javascript
angular2+nodejs实现图片上传功能
2017/03/27 NodeJs
Webpack性能优化 DLL 用法详解
2017/08/10 Javascript
微信小程序实现分享到朋友圈功能
2018/07/19 Javascript
vue中动态select的使用方法示例
2019/10/28 Javascript
vue中 数字相加为字串转化为数值的例子
2019/11/07 Javascript
Js on及addEventListener原理用法区别解析
2020/07/11 Javascript
详解Python迭代和迭代器
2016/03/28 Python
Django 忘记管理员或忘记管理员密码 重设登录密码的方法
2018/05/30 Python
python中@property和property函数常见使用方法示例
2019/10/21 Python
Django框架表单操作实例分析
2019/11/04 Python
详解CSS3新增的背景属性
2019/12/25 HTML / CSS
荷兰鞋类购物网站:Donelli
2019/05/24 全球购物
橄榄树药房:OLIVEDA
2019/09/01 全球购物
C#面试题
2016/05/06 面试题
护理工作感言
2014/01/16 职场文书
《可爱的动物》教学反思
2014/02/22 职场文书
小组名称和口号
2014/06/09 职场文书
七一讲话心得体会
2014/09/05 职场文书
违反交通法规检讨书
2014/09/10 职场文书
学雷锋献爱心倡议书
2015/04/27 职场文书
百年孤独读书笔记
2015/06/29 职场文书
Angular CLI发布路径的配置项浅析
2021/03/29 Javascript
i7 6700处理器相当于i5几代
2022/04/19 数码科技