Egg框架應(yīng)用Sequelize操作MySQL小結(jié)

Egg.js,是阿里開(kāi)源的企業(yè)級(jí) Node.js 框架寄纵。相比Express鳖敷、Koa,Egg.js更為輕量程拭,是Koa的增強(qiáng)定踱,開(kāi)發(fā)成本和效率也更為高效。

Sequelize恃鞋,是一個(gè)廣泛使用的 ORM 框架崖媚,它支持 MySQL、PostgreSQL恤浪、SQLite 和 MSSQL 等多個(gè)數(shù)據(jù)源至扰。

一、安裝配置插件

打開(kāi)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 中編寫(xiě) sequelize 配置

config.sequelize = {
  dialect: 'mysql',
  host: '123.45.67.890',
  port: 3306,
  database: 'test',
  username: 'root',
  password: '123456',
  // 配置數(shù)據(jù)庫(kù)時(shí)間為東八區(qū)北京時(shí)間
  timezone: '+08:00',
  define: {  // model的全局配置
    timestamps: true,   // 添加create,update,delete時(shí)間戳
    paranoid: true,   // 添加軟刪除
    freezeTableName: true,  // 防止修改表名為復(fù)數(shù)
    underscored: false  // 防止駝峰式字段被默認(rèn)轉(zhuǎn)為下劃線
  },
  // 打印日志
  logging: true,
  // 時(shí)間格式化
  dialectOptions: {
    dateStrings: true,
    typeCast: true
  }
};

注意點(diǎn)

1、日期顯示格式異常

默認(rèn)情況下查詢(xún)的日期是這種樣子2022-01-02T09:14:03.102Z阶祭,我們需要對(duì)它自動(dòng)格式化才行绷杜。

dialectOptions: {
  dateStrings: true,
  typeCast: true 
}

這樣就會(huì)格式化成醬紫:2022-01-04 10:39:56

二、模型配置

模型是 Sequelize 的本質(zhì). 模型是代表數(shù)據(jù)庫(kù)中表的抽象. 在 Sequelize 中濒募,它是一個(gè) Model 的擴(kuò)展類(lèi)鞭盟。

Sequelize 中的模型有一個(gè)名稱(chēng),此名稱(chēng)不必與它在數(shù)據(jù)庫(kù)中表示的表的名稱(chēng)相同瑰剃。 通常齿诉,模型具有單數(shù)名稱(chēng)(例如,User)晌姚,而表具有復(fù)數(shù)名稱(chēng)(例如粤剧,Users),這是完全可配置的挥唠。

我們先在model文件夾中創(chuàng)建一個(gè)parents.js文件定義parents模型:

'use strict';
/**
 * 家長(zhǎng)表
 */
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: '家長(zhǎng)id'
    },
    name: {
      type: STRING(36),
      allowNull: false,
      comment: '家長(zhǎng)姓名'
    },
    age: {
      type: INTEGER,
      allowNull: false,
      comment: '家長(zhǎng)年齡'
    },
    createDate: {
      type: DATE,
      defaultValue: NOW,
      field: 'create_date',
      comment: '創(chuàng)建時(shí)間'
    },
    updateDate: {
      type: DATE,
      defaultValue: NOW,
      field: 'update_date',
      comment: '更新時(shí)間'
    }
  }, {
    // 去除createAt抵恋,updateAt
    timestamps: false,  
    // 實(shí)例對(duì)應(yīng)的表名
    tableName: 'parents'
  });
  return Parents;
};

是否修改表名為復(fù)數(shù)配置

Squelize會(huì)將Parents默認(rèn)轉(zhuǎn)變?yōu)閺?fù)數(shù),也就是直接加s變?yōu)镻arentss宝磨,很多時(shí)候這并不是我們想要的弧关,可以對(duì)其進(jìn)行修改盅安。

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

這是全局的别瞭,避免在每個(gè)模型中修設(shè)置。

"define": {
   "freezeTableName": true
}

2株憾、對(duì)app.model.define的第三個(gè)參數(shù)進(jìn)行配置

項(xiàng)目已經(jīng)在config.json配置了蝙寨,模型中已被注釋掉。此外号胚,我們可以在第三個(gè)參數(shù)中配置我們想要的表名tableName

添加默認(rèn)值

在更新或添加數(shù)據(jù)的時(shí)候籽慢,默認(rèn)值是有必要的。一般對(duì)id猫胁,創(chuàng)建時(shí)間(create_date)箱亿,更新時(shí)間(updated_date)設(shè)置默認(rèn)值。

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

三弃秆、模型字段的數(shù)據(jù)類(lèi)型

這里只展示部分?jǐn)?shù)據(jù)類(lèi)型届惋,詳細(xì)的請(qǐng)查閱文檔。

1菠赚、字符串(String)

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

2脑豹、布爾(Boolean)

DataTypes.BOOLEAN      // TINYINT(1)

3、數(shù)字(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
}

四拌牲、數(shù)據(jù)查詢(xún)

1俱饿、新增

單個(gè)新增

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

parents實(shí)體的其他字段設(shè)置了默認(rèn)值,會(huì)返回新增數(shù)據(jù)的實(shí)體塌忽。

批量新增

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

2拍埠、查詢(xún)

const result = await this.ctx.model.Parents.findAll({
  limit: 10, // 當(dāng)頁(yè)條數(shù)
  offset: 0, // 開(kāi)始下標(biāo)
  order: [['create_date', 'desc']], // 排序規(guī)則
  where: {
    age: {
      [Op.gt]: 36
    },
    name: {
      [Op.like]: '王%'
    }
  },
  attributes: [ // 指定返回的屬性
    'id',
    'name',
    // 第一個(gè)參數(shù)為屬性,第二個(gè)參數(shù)為別名土居,返回?cái)?shù)據(jù)以別名返回
    ['create_date', 'createDate'] 
  ]
});

操作符

Sequelize提供了多種運(yùn)算符枣购,常見(jiàn)的都寫(xiě)在例子里了,其他的請(qǐng)查閱文檔擦耀。

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簡(jiǎn)寫(xiě)
    // 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實(shí)例
    // [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 方法僅計(jì)算數(shù)據(jù)庫(kù)中元素出現(xiàn)的次數(shù)迄损。

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账磺、模型查找

這種查詢(xún)方式效率會(huì)高很多芹敌。

findByPk

findByPk 方法使用提供的主鍵從表中僅獲得一個(gè)條目痊远。

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

findOne

findOne 方法獲得它找到的第一個(gè)條目(它可以滿(mǎn)足提供的可選查詢(xún)參數(shù))。

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

findAndCountAll

在處理與分頁(yè)有關(guān)的查詢(xún)時(shí)非常有用氏捞。

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

最后

歡迎關(guān)注我的公眾號(hào)【前端技術(shù)驛站】碧聪,多多交流,共同進(jìn)步液茎!
回復(fù) 100獲取100本圖靈圖書(shū)
回復(fù)react逞姿,vue,node獲取最新實(shí)戰(zhàn)視頻

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市捆等,隨后出現(xiàn)的幾起案子滞造,更是在濱河造成了極大的恐慌,老刑警劉巖栋烤,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谒养,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡明郭,警方通過(guò)查閱死者的電腦和手機(jī)买窟,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)薯定,“玉大人始绍,你說(shuō)我怎么就攤上這事』爸叮” “怎么了亏推?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)年堆。 經(jīng)常有香客問(wèn)我径簿,道長(zhǎng),這世上最難降的妖魔是什么嘀韧? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮缠捌,結(jié)果婚禮上锄贷,老公的妹妹穿的比我還像新娘。我一直安慰自己曼月,他們只是感情好谊却,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著哑芹,像睡著了一般炎辨。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上聪姿,一...
    開(kāi)封第一講書(shū)人閱讀 51,631評(píng)論 1 305
  • 那天碴萧,我揣著相機(jī)與錄音乙嘀,去河邊找鬼。 笑死破喻,一個(gè)胖子當(dāng)著我的面吹牛虎谢,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播曹质,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼婴噩,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了羽德?” 一聲冷哼從身側(cè)響起几莽,我...
    開(kāi)封第一講書(shū)人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎宅静,沒(méi)想到半個(gè)月后章蚣,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡坏为,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年究驴,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片匀伏。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡洒忧,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出够颠,到底是詐尸還是另有隱情熙侍,我是刑警寧澤,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布履磨,位于F島的核電站蛉抓,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏剃诅。R本人自食惡果不足惜巷送,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望矛辕。 院中可真熱鬧笑跛,春花似錦、人聲如沸聊品。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)翻屈。三九已至陈哑,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背惊窖。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工刽宪, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人爬坑。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓纠屋,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親盾计。 傳聞我的和親對(duì)象是個(gè)殘疾皇子售担,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容