nodejs - MongoDB 学习笔记

一、简介

1、MongoDB 是什么

MongoDB 是一个基于分布式文件存储的数据库,官方地址 https://www.mongodb.com/

2、数据看是什么

数据库(DataBase)是按照数据结构来组织、存储和管理数据的应用程序。

3、数据库的作用

主要作用是 管理数据,对数据进行 增(c)、删(d)、改(u)、查(r)

4、数据库管理数据的特点

  1. 速度更快
  2. 扩展性更强
  3. 安全性更强

5、为什么选择 MongoDB

操作语法与 JavaScript 类似,容易上手,学习成本低

二、核心概念

  • 数据库(database):是一个数据仓库,数据库服务下可以创建很多数据库,数据库可以存放很多集合(类似于js中的数组)
  • 集合(collection):类似于 JS 中的数组,在集合中可以存放很多文档
  • 文档(document):是数据库中的最小单位,类似于js中的对象

在这里插入图片描述
JSON 文件示例:

{ "accounts": [ { "id": "3-YLju5f3", "title": "买电脑", "time": "2023-02-08", "type": "-1", "account": "5500", "remarks": "为了上网课" },{ "id": "3-YLju5f4", "title": "请女朋友吃饭", "time": "2023-02-08", "type": "-1", "account": "214", "remarks": "情人节聚餐" },],"users":[ { "id": 1, "name": "zhangsan","age": 18 },{ "id": 2, "name": "lisi", "age": 20 },{ "id": 3, "name": "wangwu", "age": 22 } ] 
}

上面 json 文件中:

  • 一个 json 文件好比是一个 数据库,一个 MongoDB 服务下可以有 n 个数据库
  • json 文件中的 一级属性的数据值 好比是 集合(accounts、users)
  • 数组中的对象好比是 文档
  • 对象中的属性有时也称为字段

一般情况下:

  • 一个项目使用 一个数据库
  • 一个集合会存储同一种类型的数据

三、下载安装与启动

下载地址: https://www.mongodb.com/try/download/community

mac 参考 mongodb 下载安装
windows 参考 视频教程

mac 中全局变量配置完以后,mongod 启动服务的命令必须要在 bin 目录下才能使用,mongo 启动数据库命令,可以全局使用。

1、数据库命令

  1. 显示所有的数据库
    show dbs
  2. 切换到指定的数据库,如果数据库不存在会自动创建数据库
    use 数据库名
  3. 显示当前所在数据库
    db
  4. 删除当前数据库
    use 数据库名
    db.dropDatabase()

2、集合命令

  1. 创建集合
    db.createCollection('集合名称')
  2. 显示当前数据库中所有集合
    show collections
  3. 删除某个集合
    db.集合名.drop()
  4. 重命名
    db.集合名.renameCollection('newName')

3、文档命令

  1. 插入文档
    db.集合名.insert(文档对象)
    db.account.insert({name:'wangwu', age: 20})
  2. 查询文档
    db.集合名.find(查询条件)
    db.account.find({name:'wangwu'})
    没有查询条件,就查集合里面所有的文档
  3. 更新文档
    db.集合名.update(查询条件,新的文档)
    db.account.update({name:'lisi'},{name:'lisi222'})
    上面两个 新文档 会覆盖 旧文档
    db.集合名.update(查询条件, {$set:更新文档})
    db.account.update({name:'jeck'},{$set:{age:22}})
    上面命令是只修改某一项或多项,不会直接覆盖
  4. 删除文档
    db.集合名.remove(查询条件)
    db.account.remove({name:'lisi222'})

4、应用场景

4-1、新增

  • 用户注册
  • 发布视频
  • 发布商品
  • 发朋友圈
  • 发评论
  • 发微博
  • 发弹幕

4-2、删除

  • 删除评论
  • 删除商品
  • 删除文章
  • 删除视频
  • 删除微博

4-3、更新

  • 更新个人信息
  • 修改商品价格
  • 修改文章内容

4-4、查询

  • 商品列表
  • 视频列表
  • 朋友圈列表
  • 微博列表
  • 搜索功能

二、Mongoose

1、介绍

Mongoose 是一个对象文档模型库,官网 http://www.mongoosejs.net/添加链接描述

2、作用

方便使用代码操作 mongodb 数据库

3、使用流程

  1. 安装 mongoose npm install mongoose
  2. 导入 mongoose
// 导入 mongoose
// 1、导入 mogoose
const mongoose = require('mongoose')
const { type } = require('os')// 2、连接 mongdb 服务器
mongoose.connect('mongodb://127.0.0.1:27017/bilibili')// 3、设置回调
mongoose.connection.once('open', () => {// console.log('连接成功')// 4、创建文档的结构对象// let BookSchema = new mongoose.Schema({//   name: String,//   author: String,//   author_gemder: String,//   price: Number,//   is_hot: Boolean,//   tags: Array,//   create_time: Date// })// 字段校验let BookSchema = new mongoose.Schema({name: {type: String,     // 设置类型require: true,    // 设置必填项unique: true      //  设置唯一值},author: {type: String,     // 设置类型default: '佚名'   // 设置默认值},author_gemder: {type: String,enum: ['男', '女', '未知']  // 设置枚举值(值只能从这里取其一)},price: Number,is_hot: Boolean,tags: Array,create_time: Date})// 5、创建模型对象  对文档操作的封装对象(可以完成对文档的增删改查操作)// 第一个参数 book 是集合(数组)名称,第二个参数 BookSchema 是结构对象let BookModel = mongoose.model('book', BookSchema)// 6、新增基本对象// 如果属性跟上面定义的属性不统一,在写入数据库时会忽略这个键值对// 如果属性值的类型跟上面定义的类型不同意,写入时会报错,写入失败BookModel.create({name: '西游记',author: '吴承恩',price: 19.9,is_hot: true,tags: ['言情','都市','恐怖','玄幻'],create_time: new Date(),author_gemder: '男'}).then((data) => {// 输出成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})
})mongoose.connection.on('error', () => {console.log('连接失败')
})mongoose.connection.on('close', () => {console.log('关闭连接')
})// setTimeout(() => {
//   // 关闭数据库连接
//   mongoose.disconnect()
// }, 3000)

4、字段类型

文档结构可选的常用字段类型列表:
在这里插入图片描述

5、字段值验证

Mongoose 有一些内建验证器,可以对字段值进行验证

5-1、必填项

title: { type: String, required: true // 设置必填项 
},

5-2、默认值

author: { type: String, default: '匿名' //默认值 
},

5-3、枚举值

gender: { type: String, enum: ['男','女'] //设置的值必须是数组中的 
},

5-4、唯一值

username: { type: String, unique: true 
},

unique 需要 重建集合 才能有效果
永远不要相信用户的输入

6、增删改查

数据库基本操作包括四个,增加(create),删除(delete),修改(update),查(read),其操作都是在mongoose.connection.once('open', () => {})回调中进行的

6-1、增加

插入一条:

// 1、导入 mogoose
const mongoose = require('mongoose')// 2、连接 mongdb 服务器
mongoose.connect('mongodb://127.0.0.1:27017/bilibili')// 3、设置回调
mongoose.connection.once('open', () => {// console.log('连接成功')// 4、创建文档的结构对象let BookSchema = new mongoose.Schema({name: String,author: String,price: Number})// 5、创建模型对象  对文档操作的封装对象(可以完成对文档的增删改查操作)// 第一个参数 book 是集合(数组)名称,第二个参数 BookSchema 是结构对象let BookModel = mongoose.model('book', BookSchema)// 6、新增基本对象// 如果属性跟上面定义的属性不统一,在写入数据库时会忽略这个键值对// 如果属性值的类型跟上面定义的类型不同意,写入时会报错,写入失败BookModel.create({name: '西游记',author: '吴承恩',price: 19.9}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})
})mongoose.connection.on('error', () => {console.log('连接失败')
})mongoose.connection.on('close', () => {console.log('关闭连接')
})// setTimeout(() => {
//   // 关闭数据库连接
//   mongoose.disconnect()
// }, 3000)

插入多条:

BookModel.insertMany([ { name:'华为', color:'灰色', price:2399, tags:['电量大','屏幕大','信号好'] },{ name:'小米', color:'白色', price:2099, tags:['电量大','屏幕大','信号好'] } ],(data)=>{ console.log('写入成功');mongoose.connection.close(); 
})

6-2、删除

删除一条:

  BookModel.deleteOne({_id:'669f4719bbf7718b48b790ed'}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

删除多条:

BookModel.deleteMany({is_hot: false}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

6-3、文档更新

单条文档更新 BookModel.updateOne({}, {}).then()
多条文档更新BookModel.updateMany({}, {}).then()

//1. 安装 mongoose
//2. 导入 mongoose
const mongoose = require('mongoose');//设置 strictQuery 为 true
mongoose.set('strictQuery', true);//3. 连接 mongodb 服务                        数据库的名称
mongoose.connect('mongodb://127.0.0.1:27017/bilibili');//4. 设置回调
// 设置连接成功的回调  once 一次   事件回调函数只执行一次
mongoose.connection.once('open', () => {//5. 创建文档的结构对象//设置集合中文档的属性以及属性值的类型let BookSchema = new mongoose.Schema({name: String,author: String,price: Number,is_hot: Boolean});//6. 创建模型对象  对文档操作的封装对象    mongoose 会使用集合名称的复数, 创建集合let BookModel = mongoose.model('novel', BookSchema);//7. 更新文档// 更新单个文档 updateOne 接收两个参数,第一个是条件,第二个是新的对象// BookModel.updateOne({name: '西游记'}, {price: 9.9}).then((data) => {// // 输出 插入成功后的文档对象// console.log(data)// // 7、关闭数据库连接(项目运行过程中,不会添加该代码)// mongoose.disconnect()// }).catch(res => {// console.error('捕捉到了错误信息'+res)// )// 更新多个文档BookModel.updateMany({is_hot: false}, {is_hot: true}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})});// 设置连接错误的回调
mongoose.connection.on('error', () => {console.log('连接失败');
});//设置连接关闭的回调
mongoose.connection.on('close', () => {console.log('连接关闭');
});

6-4、查询文档

查询一条数据:BookModel.findOne({name:'23'}).then()
根据 ID 查询数据BookModel.findById(‘’).then()
批量查询数据:BookModel.find({}).then()(有条件就根据条件查询,没条件就查所有的)

//1. 安装 mongoose
//2. 导入 mongoose
const mongoose = require('mongoose');//设置 strictQuery 为 true
mongoose.set('strictQuery', true);//3. 连接 mongodb 服务                        数据库的名称
mongoose.connect('mongodb://127.0.0.1:27017/bilibili');//4. 设置回调
// 设置连接成功的回调  once 一次   事件回调函数只执行一次
mongoose.connection.once('open', () => {//5. 创建文档的结构对象//设置集合中文档的属性以及属性值的类型let BookSchema = new mongoose.Schema({name: String,author: String,price: Number,is_hot: Boolean});//6. 创建模型对象  对文档操作的封装对象    mongoose 会使用集合名称的复数, 创建集合let BookModel = mongoose.model('novel', BookSchema);//7. 读取文档// 按照条件读取文档BookModel.findOne({name:'狂飙'}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})// 通过 id 读取文档// BookModel.findById('669f4ef193ea628ad18eeb33')//.then((data) => {//  console.log(data)//  mongoose.disconnect()//}).catch(res => {// console.error('捕捉到了错误信息'+res)// })// 读取所有文档// BookModel.find().then((data) => {// console.log(data)// mongoose.disconnect()//}).catch(res => {// console.error('捕捉到了错误信息'+res)// })});// 设置连接错误的回调
mongoose.connection.on('error', () => {console.log('连接失败');
});//设置连接关闭的回调
mongoose.connection.on('close', () => {console.log('连接关闭');
});

7、条件控制

7-1、运算符

在 mongodb 中不能使用 >, <, >=,<=,!==等运算符,需要使用替代符号

  • > 使用 $gt
  • < 使用 $lt
  • >= 使用 $gte
  • <= 使用 $lte
  • !== 使用 $ne
// 查询价格小于 20 的文档
BookModel.find({price:{$lt: 20}}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

7-2、逻辑运算符

$or 逻辑或
$and 逻辑与

  // 逻辑运算 - 查询 name 为 余华 或者 刘慈欣 的文档
BookModel.find({$or: [{author: '余华'},{author: '刘慈欣'}]}).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

7-3、正则匹配

条件中可以直接使用 js 的正则语法,通过正则可以进行模糊查询

// 正则匹配 - 查询 name 包含 ‘三’ 的文档
BookModel.find({name: //})
BookModel.find({name: new RegExp('三')})

8、个性化读取

8-1、字段筛选

取前端需要展示的字段,_id 会默认取出来,,如果不需要,要设为 0

// 字段筛选(取某些字段,0 不要的字段,1 要获取的字段)
BookModel
.find()
.select({name: 1, author: 1, price: 1, _id: 0})
.then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

8-2、数据排序

// 数据排序,1:升序,-1:倒序
BookModel
.find()
.select({name:1, price:1, _id: 0})
.sort({price: 1})
.then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

8-3、数据截取

常用来做分页,skip 为 (页码 - 1) * limit 中的值

// 数据截取,limit 接收一个参数,为要截取的数据的长度,skip接收一个参数,表示在截取前,要跳过几个数据,常用作分页BookModel.find().select({name:1, price:1, _id: 0}).sort({price: 1}).skip(3).limit(3).then((data) => {// 输出 插入成功后的文档对象console.log(data)// 7、关闭数据库连接(项目运行过程中,不会添加该代码)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})

9、模块化

在这里插入图片描述
config.js

module.exports = {DBHOST: '127.0.0.1',DBPORT: '27017',DBNAME: 'bilibili'
}

db.js

module.exports = function (success, error = () => {console.log('连接失败了啊')}) {// 1、导入 mogooseconst mongoose = require('mongoose')const { DBHOST, DBPORT, DBNAME } = require('../config/config')// 2、连接 mongdb 服务器mongoose.connect(`mongodb://${DBHOST}:${DBPORT}/${DBNAME}`)// 3、设置回调mongoose.connection.once('open', () => {success()})mongoose.connection.on('error', () => {error()})mongoose.connection.on('close', () => {console.log('关闭连接')})
}

BookModel.js

const mongoose = require('mongoose')let MovieSchema = mongoose.Schema({name: String,director: String
})
let MovieModel = new mongoose.model('movie', MovieSchema)module.exports = MovieModel

book.js

const db = require('./db/db')
const BookModel = require('./models/BookModel')db(() => {BookModel.create({name: '阿勒泰',author: '阿勒泰',price: 88.8}).then((data) => {console.log(data)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})
})

MovieModel.js

const mongoose = require('mongoose')let MovieSchema = new mongoose.Schema({name: String,director: String
})
let MovieModel = mongoose.model('movie', MovieSchema)module.exports = MovieModel

movie.js

const db = require('./db/db')
const MovieModel = require('./models/MovieModel')db(() => {MovieModel.create({name: '第二十条',director: '张艺谋'}).then((data) => {console.log(data)mongoose.disconnect()}).catch(res => {console.error('捕捉到了错误信息'+res)})
})

四、图形化管理工具

我们可以使用图形化的管理工具来对 Mongodb 进行交互,这里演示两个图形化工具

  • Robo 3T 免费 https://github.com/Studio3T/robomongo/releases
  • Navicat 收费 https://www.navicat.com.cn/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/51166.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

RedHat9 | Ansible 编写循环和条件任务

环境版本说明 RedHat9 [Red Hat Enterprise Linux release 9.0]Ansible [core 2.13.3]Python [3.9.10]jinja [3.1.2] 1. 利用循环迭代任务 通过利用循环&#xff0c;管理员无需编写多个使用同一模块的任务。Ansible支持使用loop关键字对一组项目迭代任务&#xff0c;通过配置…

音视频入门基础:WAV专题(3)——FFmpeg源码中,判断某文件是否为WAV音频文件的实现

一、引言 通过FFmpeg命令&#xff1a; ./ffmpeg -i XXX.wav 可以判断出某个文件是否为WAV格式的音频文件&#xff1a; 所以FFmpeg是怎样判断出某个文件是否为WAV格式的音频文件呢&#xff1f;它内部其实是通过wav_probe函数来判断的。从文章《FFmpeg源码&#xff1a;av_prob…

Spring Boot集成OpenPDF和Freemarker实现PDF导出功能并附水印

&#x1f604; 19年之后由于某些原因断更了三年&#xff0c;23年重新扬帆起航&#xff0c;推出更多优质博文&#xff0c;希望大家多多支持&#xff5e; &#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Mi…

关于@JsonSerialize序列化与@JsonDeserialize反序列化注解的使用(密码加密与解密举例)

注&#xff1a;另一种方式参考 关于TableField中TypeHandler属性&#xff0c;自定义的类型处理器的使用&#xff08;密码加密与解密举例&#xff09;http://t.csdnimg.cn/NZy4G 1.简介 1.1 序列化与反序列化 学习注解之前&#xff0c;我们可以先了解一下什么是序列化与反序列…

关于Redis持久化和集群模式(主从,哨兵,去中心化)使用和介绍

持久化: 持久化介绍 把内存中的数据存储到磁盘的过程。同时也可以把磁盘中的数据加载到内存中 持久化实现 redis实现持久化的方式提供了两种: RDB快照模式&#xff0c;数据备份和恢复速度快。 缺点: 数据完整性差。数据可能丢失多。 AOF日志追加: 数据完整性高。 缺点: 数据备…

遵义网站建设安全性保证

随着互联网的发展和普及&#xff0c;网站的建设成为了一个重要的工作&#xff0c;也是企业宣传的重要渠道。然而&#xff0c;随之而来的安全问题也是不容忽视的。为了保证遵义网站建设的安全性&#xff0c;我们需要采取一系列的措施。 首先&#xff0c;要选择合适的服务器和主机…

【Unity2D 2022:Data】读取csv格式文件的数据

一、创建csv文件 1. 打开Excel&#xff0c;创建xlsx格式文件 2. 编辑卡牌数据&#xff1a;这里共写了两类卡牌&#xff0c;第一类是灵物卡&#xff0c;具有编号、卡名、生命、攻击四个属性&#xff1b;第二类是法术卡&#xff0c;具有编号、卡名、效果三个属性。每类卡的第一…

算法题目整合5

文章目录 118. 小 y 删数字119. 小红的字符串切割120. 小红的数字匹配115. 组装手机116. 小欧的卡牌111. 构造二阶行列式112. 挑战boss 118. 小 y 删数字 题目描述 给定一个长度为 n 的数组&#xff0c;数组元素为 a1, a2, . . , an&#xff0c;每次能删除任意 a 的任意一位&…

(39)智能电池

文章目录 前言 1 通过任务规划器进行设置 2 补充信息 3 限制条件 4 参数说明 前言 虽然还不是很普遍&#xff0c;但智能电池更容易从飞行器上安装和拆卸&#xff0c;并且能够提供更多关于电池状态的信息&#xff0c;包括容量、单个电池电压、温度等。 ArduPilot 支持几种…

在Vue程序中,如何检测用户的登录状态并跳转到对应页面

在一个Vue应用程序中&#xff0c;你可以通过以下步骤来检测用户的登录状态并跳转到对应的页面。假设你使用的是Vue Router来管理路由&#xff0c;并且用户的登录状态保存在Vuex或localStorage中。 步骤一&#xff1a;在Vuex中管理用户的登录状态 首先&#xff0c;确保你在Vue…

开发环境搭建——Node.js

在启动前端项目的时候我们通常会用到Node.js&#xff0c;下面是对Node.js的下载安装以及配置的讲解 一、Node.js的安装 1.1、通过Node.js官网下载&#xff1a;Node.js — Run JavaScript Everywhere 下载后双击.msi安装文件后一直点击下一步即可 1.2、配置node 1.2.1、查看…

HTML meta

<meta>标签用于提供html文档的元信息&#xff08;metadata&#xff09;。这些信息不会显示在页面上&#xff0c;但会被浏览器或搜索引擎用来识别页面的编码方式、关键字、描述、作者信息、刷新时间等。 基本语法 <meta name"属性名" content"属性值&q…

锅总介绍技术标准基金会及组织

技术标准基金会及组织有哪些&#xff1f;中国主导的有哪些&#xff1f;它们之间有何关联&#xff1f;希望本文能帮您解答&#xff01; 一、主要的基金会和组织 以下是一些主要的基金会和组织&#xff0c;它们致力于开源软件和技术标准的发展&#xff1a; Linux Foundation 简…

js 替换json中的转义字符 \

例如有以下字符串 "\"{\\\"account\\\":\\\"66\\\",\\\"name\\\":\\\"66\\\"}\"" 想得到如下字符串 {"account":"66","name":"66"} 执行替换字符串 "\"{…

verilog数据自动扩展位宽问题

Verilog的比较运算中&#xff0c;需要以左右两边运算结果的最大值为参考进行扩展位宽 module test;reg [1 : 0] b ; reg [1 : 0] a ; reg [1 : 0] c ; wire a0;assign a0 (a b) > c; initial begin b d3; a d3; c d3; #2000; $finish; end endmodule如代码中所示&…

组队学习——决策树(以泰坦尼克号公共数据集为例)

本次我们挑战的数据集为泰坦尼克号公共数据集&#xff0c;为了降低难度&#xff0c;我们在原有数据集的基础上进行了优化&#xff0c;具体数据集介绍如下&#xff1a; 在这里也介绍一下数据的含义吧 数据介绍&#xff1a; Survived&#xff1a;是否存活&#xff08;label&#…

paraFoam 运行 报错 usr/lib/x86_64-linux-gnu/libQt5Core.so 已解决

在日常项目开发中。使用ubuntu 视图开发的时候。报错 缺少 libQt5Core 核心组件&#xff01; whereis libQt5Core.so.5sudo strip --remove-section.note.ABI-tag /usr/lib/x86_64-linux-gnu/libQt5Core.so.5 完美解决&#xff0c;并且能正常打开&#xff0c;前提是&#xff0c…

简化mybatis @Select IN条件的编写

最近从JPA切换到Mybatis&#xff0c;使用无XML配置&#xff0c;Select注解直接写到interface上&#xff0c;发现IN条件的编写相当麻烦。 一般得写成这样&#xff1a; Select({"<script>","SELECT *", "FROM blog","WHERE id IN&quo…

redis面试(一)String底层剖析

前言 今天开始更新一些redis相关的知识点&#xff0c;初步计划是redis的数据类型&#xff0c;以及redis分布式锁 本章节主要是redis的String类型 字符串SDS 一般情况下我们认为的redis 字符串就是String&#xff0c;但是我这边要说的是底层String类型。 redis底层是C语言&a…

【python】python生活管理费系统(源码+论文)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…