Egg框架应用Sequelize操作MySQL小结

Egg.js,是阿里开源的企业级 Node.js 框架。相比Express、Koa,Egg.js更为轻量,是Koa的增强,开发成本和效率也更为高效。

Sequelize,是一个广泛使用的 ORM 框架,它支持 MySQL、PostgreSQL、SQLite 和 MSSQL 等多个数据源。

一、安装配置插件

打开vscode终端安装egg-mysql,mysql2

npm install --save egg-sequelize mysql2

在 config/plugin.js 中引入 egg-sequelize 插件

exports.sequelize = {
  enable: true,
  package: 'egg-sequelize',
}

在 config/config.default.js 中编写 sequelize 配置

config.sequelize = {
  dialect: 'mysql',
  host: '123.45.67.890',
  port: 3306,
  database: 'test',
  username: 'root',
  password: '123456',
  // 配置数据库时间为东八区北京时间
  timezone: '+08:00',
  define: {  // model的全局配置
    timestamps: true,   // 添加create,update,delete时间戳
    paranoid: true,   // 添加软删除
    freezeTableName: true,  // 防止修改表名为复数
    underscored: false  // 防止驼峰式字段被默认转为下划线
  },
  // 打印日志
  logging: true,
  // 时间格式化
  dialectOptions: {
    dateStrings: true,
    typeCast: true
  }
};

注意点

1、日期显示格式异常

默认情况下查询的日期是这种样子2022-01-02T09:14:03.102Z,我们需要对它自动格式化才行。

dialectOptions: {
  dateStrings: true,
  typeCast: true 
}

这样就会格式化成酱紫:2022-01-04 10:39:56

二、模型配置

模型是 Sequelize 的本质. 模型是代表数据库中表的抽象. 在 Sequelize 中,它是一个 Model 的扩展类。

Sequelize 中的模型有一个名称,此名称不必与它在数据库中表示的表的名称相同。 通常,模型具有单数名称(例如,User),而表具有复数名称(例如,Users),这是完全可配置的。

我们先在model文件夹中创建一个parents.js文件定义parents模型:

'use strict';
/**
 * 家长表
 */
module.exports = app => {
  const { STRING, INTEGER, UUID, NOW, DATE, UUIDV4 } = app.Sequelize;
  const Parents = app.model.define('Parents', {
    id: { 
      type: UUID,      
      primaryKey: true,
      allowNull: false,
      defaultValue: UUIDV4,
      comment: '家长id'
    },
    name: {
      type: STRING(36),
      allowNull: false,
      comment: '家长姓名'
    },
    age: {
      type: INTEGER,
      allowNull: false,
      comment: '家长年龄'
    },
    createDate: {
      type: DATE,
      defaultValue: NOW,
      field: 'create_date',
      comment: '创建时间'
    },
    updateDate: {
      type: DATE,
      defaultValue: NOW,
      field: 'update_date',
      comment: '更新时间'
    }
  }, {
    // 去除createAt,updateAt
    timestamps: false,  
    // 实例对应的表名
    tableName: 'parents'
  });
  return Parents;
};

是否修改表名为复数配置

Squelize会将Parents默认转变为复数,也就是直接加s变为Parentss,很多时候这并不是我们想要的,可以对其进行修改。

1、在config.json文件中,添加如下配置

这是全局的,避免在每个模型中修设置。

"define": {
   "freezeTableName": true
}

2、对app.model.define的第三个参数进行配置

项目已经在config.json配置了,模型中已被注释掉。此外,我们可以在第三个参数中配置我们想要的表名tableName

添加默认值

在更新或添加数据的时候,默认值是有必要的。一般对id,创建时间(create_date),更新时间(updated_date)设置默认值。

id: { 
  defaultValue: UUIDV4
},
create_date: {
  defaultValue: NOW
},
update_date: {
  defaultValue: NOW
}

三、模型字段的数据类型

这里只展示部分数据类型,详细的请查阅文档。

1、字符串(String)

DataTypes.STRING      // VARCHAR(255)
DataTypes.STRING(36)      // VARCHAR(36)
DataTypes.TEXT      // TEXT

2、布尔(Boolean)

DataTypes.BOOLEAN      // TINYINT(1)

3、数字(Number)

DataTypes.INTEGER            // INTEGER
DataTypes.BIGINT(11)         // BIGINT(11)

DataTypes.FLOAT              // FLOAT
DataTypes.FLOAT(11)          // FLOAT(11)

4、日期

DataTypes.DATE          // DATETIME 适用于 mysql/sqlite

5、UUID

Sequelize 使用 Sequelize.UUIDV1 或 Sequelize.UUIDV4 生成UUID作为主键。

{
  type: DataTypes.UUID,
  defaultValue: Sequelize.UUIDV4 // 或 Sequelize.UUIDV1
}

四、数据查询

1、新增

单个新增

const parents = await Parents.create({
  name: '王大锤'
});

parents实体的其他字段设置了默认值,会返回新增数据的实体。

批量新增

const parents = await Parents.bulkCreate({
  name: '王大锤'
}, {
  name: '山呱呱'
});

2、查询

const result = await this.ctx.model.Parents.findAll({
  limit: 10, // 当页条数
  offset: 0, // 开始下标
  order: [['create_date', 'desc']], // 排序规则
  where: {
    age: {
      [Op.gt]: 36
    },
    name: {
      [Op.like]: '王%'
    }
  },
  attributes: [ // 指定返回的属性
    'id',
    'name',
    // 第一个参数为属性,第二个参数为别名,返回数据以别名返回
    ['create_date', 'createDate'] 
  ]
});

操作符

Sequelize提供了多种运算符,常见的都写在例子里了,其他的请查阅文档。

const result = await this.ctx.model.Parents.findAll({
  where: {
    age: {
      // [Op.gt]: 36,
      // [Op.eq]: 39, // = 39
      // [Op.ne]: 39 // != 39
      // [Op.gt]: 32 // > 32
      // [Op.gte]: 32 // >= 32
      // [Op.lt]: 32 // < 32
      // [Op.lte]: 32 // <= 32
      // [Op.between]: [32, 35],  // BETWEEN 32 AND 35
      // [Op.notBetween]: [32, 35], // NOT BETWEEN 32 AND 35
      // [Op.in]: [36, 38], // IN [36, 38]
      // [Op.notIn]: [36, 38],  // NOT IN [36, 38]  
      
      // 组合
      // [Op.or]: {
      //   [Op.lt]: 36,
      //   [Op.eq]: 60
      // }

    },

    // Op.in简写
    // age: [32, 39], 同使用 `age: { [Op.in]: [32, 39] }`
    
    name: {
      // [Op.like]: '王%',
      // [Op.like]: '%hat', // LIKE '%hat'
      // [Op.notLike]: '%hat',  // NOT LIKE '%hat'
      // [Op.startsWith]: 'hat',  // LIKE 'hat%'
      // [Op.endsWith]: 'hat',  // LIKE '%hat'
    },

    // Op.not实例
    // [Op.not]: [{
    //   age: [36,37,38]
    // }, {
    //   name: {
    //     [Op.like]: '王%'
    //   }
    // }],

    create_date: {
      [Op.lt]: new Date(),
      [Op.gt]: new Date(new Date() - 24 * 60 * 60 * 1000)
    }
  }
});

3、更新

const effectedNum = await this.ctx.model.Parents.update({
  name: '王小锤'
}, {
  where: {
    id: '123456789'
  }
})

4、删除

const effectedNum = await this.ctx.model.Parents.destroy({
  where: '123456789'
})

5、count

count 方法仅计算数据库中元素出现的次数。

const effectedNum = await this.ctx.model.Parents.count({
  where: {
    age: {
      [Op.gt]: 25
    }
  }
})

6、max, min 和 sum

await this.ctx.model.Parents.max('age'); // 63

await this.ctx.model.Parents.max('age', {
  where: {
    age: {[Op.lt]: 40}
  }
}); // 39

await this.ctx.model.Parents.sum('age'); // 1027

7、模型查找

这种查询方式效率会高很多。

findByPk

findByPk 方法使用提供的主键从表中仅获得一个条目。

await this.ctx.model.Parents.findByPk('12');

findOne

findOne 方法获得它找到的第一个条目(它可以满足提供的可选查询参数)。

await Project.findOne({ where: { age: 40 } });

findAndCountAll

在处理与分页有关的查询时非常有用。

const { count, rows } = await this.ctx.model.Parents.findAndCountAll({
  where: {
    name: {
      [Op.like]: '王%'
    }
  },
  offset: 10,
  limit: 0
});
console.log(count);
console.log(rows);

最后

欢迎关注我的公众号【前端技术驿站】,多多交流,共同进步!
回复 100获取100本图灵图书
回复reactvue,node获取最新实战视频

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

推荐阅读更多精彩内容