目錄
- 一、mongoDB是什么?
- 1. mongo的體系結構
- 2. mongoDB的特點(或使用場景)
- 3. mongoDB與mysql、redis對比
- 4. mongoDB存儲原理
- 二、使用docker安裝mongo
- 1.安裝
- 2.創建用戶
- 3. 連接、測試
- 三、SpringBoot整合mongoDB
- 四、mongoDB原生使用
- 總結
一、mongoDB是什么?
MongoDB是一個NoSQL的非關系型數據庫 ,支持海量數據存儲,高性能的讀寫。
1. mongo的體系結構

- mongo中的集合相當于mysql中表的概念;
- mongo中的文檔相當于mysql中行的概念;
- mongo中的域相當于mysql中字段/列的概念;
2. mongoDB的特點(或使用場景)
- 支持存儲海量數據;(例如:直播中的打賞數據);
- 支持頻繁的數據讀寫;(例如:游戲道具);
- 數據安全性不高,存在數據誤差(丟失數據);
- mongoDB不支持多表操作,不支持事務;
- mongoDB使用Bson存儲格式,支持動態字段管理;
3. mongoDB與mysql、redis對比
與redis對比
1. redis純內存數據庫,內存不足時觸發淘汰策略,mongoDB使用內存加磁盤的存儲策略具有高擴展性;
2. mongoDB使用Bson存儲格式,支持動態字段管理方便擴展;
與mysql對比
1. mongoDB不支持多表操作,不支持事務;
2. mongoDB使用Bson存儲格式,支持動態字段管理;
查詢效率對比
Redis > MongoDB > MySQL
4. mongoDB存儲原理

- mongoDb采用內存加磁盤的方式存儲數據;
- mongoDb支持數據分片,當單一的服務器中磁盤不夠用的時候,還可以串聯其他服務器;
- 客戶端的請求到達內存時,先在日志中記錄下操作記錄,然后再去操作內存;
- 內存中的日志每10ms向磁盤中的日志進行同步一次,數據則每分鐘同步一次;
- 客戶端先去內存中查詢數據,內存中沒有再去查詢磁盤;
- 當客戶端寫入的時候,會先寫入到內存中,內存中寫入后請求直接返回,內存中的數據會根據同步策略同步到磁盤;
- 如果機器宕機,在重啟服務的時候會解析磁盤中的日志和磁盤中的數據進行對比,將未入到磁盤中的數據寫入磁盤,但可能會丟失10ms的數據;
二、使用docker安裝mongo
1.安裝
拉取mongo鏡像
docker pull mongo:4.4
創建mongo數據持久化目錄
mkdir -p /docker_volume/mongodb/data
運行容器
docker run -itd --name mongo -v /docker_volume/mongodb/data:/data/db -p 27017:27017 mongo:4.4 --auth
-v: 將宿主機的/docker_volume/mongodb/data映射到容器的/data/db目錄,將數據持久化到宿主機,以防止刪除容器后,容器內的數據丟失
–auth:需要密碼才能訪問容器服務
2.創建用戶
登錄mongo容器,并進入到【admin】數據庫
docker exec -it mongo mongo admin
創建一個用戶,mongo 默認沒有用戶
db.createUser({<!--{C}%3C!%2D%2D%20%2D%2D%3E--> user:'root',pwd:'123456',roles:[ {<!--{C}%3C!%2D%2D%20%2D%2D%3E--> role:'userAdminAnyDatabase', db: 'admin'},'readWriteAnyDatabase']});
【user:‘root’ 】:設置用戶名為root
【pwd:‘123456’】:設置密碼為123456
【role:‘userAdminAnyDatabase’】:只在admin數據庫中可用,賦予用戶所有數據庫的userAdmin權限
【db: ‘admin’】:可操作的數據庫
【‘readWriteAnyDatabase’】:賦予用戶讀寫權限
dbAdmin:允許用戶在指定數據庫中執行管理函數,如索引創建、刪除,查看統計或訪問system.profile
3. 連接、測試
連接mongo數據庫
db.auth('root', '123456')
測試數據庫,插入一條語句
db.user.insert({<!--{C}%3C!%2D%2D%20%2D%2D%3E-->"name":"zhangsan","age":18})
測試數據庫,查詢剛才插入的語句
db.user.find()

navicat連接測試

三、SpringBoot整合mongoDB
導入坐標
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-mongodb</artifactId>
<version>2.3.9.RELEASE</version>
</dependency>
添加yml配置
spring:
data:
mongodb:
uri: mongodb://192.156.136.168:27017/testdb
username: root
password: 123456
編寫實體類
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.bson.types.ObjectId;
import org.springframework.data.annotation.Id;
import org.springframework.data.mongodb.core.mapping.Document;
import org.springframework.data.mongodb.core.mapping.Field;
@Data
@AllArgsConstructor
@NoArgsConstructor
//指定實體類和數據庫文檔的映射關系 默認實體類名 數據庫如果沒有該文檔,會自動創建
@Document(value="tb_person")
public class Person {
@Id
private ObjectId id; //mongoDB推薦使用ID
//指定屬性名和數據庫域的映射關系 默認屬性名
@Field("person_name")
private String name;
private int age;
private String address;
}
1.@Document(value=“tb_person”) :指定實體類和數據庫文檔的映射關系 默認實體類名 數據庫如果沒有該文檔,會自動創建
2. @Field(“person_name”): //指定屬性名和數據庫域的映射關系 默認屬性名
測試類
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.domain.Sort;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.query.Criteria;
import org.springframework.data.mongodb.core.query.Query;
import org.springframework.data.mongodb.core.query.Update;
import org.springframework.test.context.junit4.SpringRunner;
/**
* 多條件查詢
*/
@Test
public void find() {
//設置查詢條件 age小于30,且person_name="張三"
Criteria criteria = Criteria.where("age").lt(30)
.and("person_name").is("張三");
//設置查詢條件
Query query = new Query(criteria);
//查詢
List<Person> list = mongoTemplate.find(query, Person.class);
for (Person person : list) {
System.out.println(person);
}
}
/**
* 分頁查詢
*/
@Test
public void findPage() {
//設置查詢條件 age小于30,且person_name="張三"
Criteria criteria = Criteria.where("age").lt(30)
.and("person_name").is("張三");
//根據條件 查詢總數
Query queryCount = new Query(criteria);
long count = mongoTemplate.count(queryCount, Person.class);
//查詢當前頁的數據列表, 查詢第二頁,每頁查詢2條
Query queryLimit = new Query(criteria)
.with(Sort.by(Sort.Order.desc("age")))
.limit(2)//每頁查詢條數
.skip(2); //從第幾頁開始 (page-1)*size
List<Person> list = mongoTemplate.find(queryLimit, Person.class);
for (Person person : list) {
System.out.println(person);
}
}
/**
* 更新數據
*/
@Test
public void update() {
//設置查詢條件 age小于30,且person_name="張三"
Criteria criteria = Criteria.where("person_name").is("王五");
//設置更新條件
Query query = new Query(criteria);
//設置更新數據
Update update = new Update();
update.set("age", 16);
mongoTemplate.upsert(query, update, Person.class);
}
/**
* 保存
*/
@Test
public void save() {
Person person = new Person();
person.setName("張三");
person.setAge(18);
mongoTemplate.save(person);
}
/**
* 刪除數據
*/
@Test
public void dlete() {
mongoTemplate.remove(Query.query(Criteria.where("person_name").is("張三")), Person.class);
}
mongoDB索引
提示:1 :升序索引 -1 :降序索引
#查看索引
db.user.getIndexes()
#創建索引
#db.user.createIndex({'age':1})
四、mongoDB原生使用
新增
db.tb_person.insert({person_name: "陳六", age: 16})
修改
-- 普通修改
db.tb_person.update({age: 16}, {$set: {person_name: "張三"}})
修改格式:db.collection.update(query, update, [ upsert: boolean, multi: boolean, writeConcern: document])
db.tb_person.update({age: 16}, {$set: {person_name: "張三"}}, {upsert: true, multi: true})
-- upsert (默認false ) : 可選,如果不存在update的記錄,是否保存。true為保存。
-- multi(默認false ) : 可選,默認只更新第一條記錄。true:更新所有匹配數據
-- writeConcern :可選,拋出異常的級別
刪除
-- 普通刪除
db.tb_person.remove({person_name: "張三"})
刪除格式:db.collection.remove(query, update, [ justOne: boolean, writeConcern: document])
db.tb_person.remove({person_name: "張三"}, false)
-- justOne: (可選)true:刪除第一個匹配記錄,false:刪除所有
-- writeConcern :(可選)拋出異常的級別
查詢
-- 查詢person_name=張三 或者 年齡 18
db.tb_person.find({$or:[{person_name: "張三"},{age:18 }]})
-- 分頁查詢 以年齡升序排序 跳過第1條數據,查詢后面2條數據
db.tb_person.find().sort({age:1}).limit(2).skip(1)
-- 查詢年齡小于等于21的數據,以年齡降序排序
db.tb_person.find({age:{$lte:21}}).sort({age:-1})
常用查詢條件
| 中文 | 符號 |
|---|---|
| 小于 | $lt: |
| 大于 | $gt: |
| 小于或等于 | $lte: |
| 大于或等于 | $gte: |
| 不等于 | $ne: |






