SpringBoot2.x集成MongoDB兑徘,使用MongoTemplate操作

1.開發(fā)環(huán)境

IDEA刚盈、Maven、SpringBoot2.0.4挂脑、Jdk1.8藕漱、MongoDB4.0、MongoDB Compass Community崭闲、PostMan

2.MongoDB

MongoDB(來自于英文單詞“Humongous”肋联,中文含義為“龐大”)是可以應(yīng)用于各種規(guī)模的企業(yè)、各個(gè)行業(yè)以及各類應(yīng)用程序的開源數(shù)據(jù)庫刁俭¢先裕基于分布式文件存儲的數(shù)據(jù)庫。由C++語言編寫。旨在為WEB應(yīng)用提供可擴(kuò)展的高性能數(shù)據(jù)存儲解決方案侮繁。MongoDB是一個(gè)高性能虑粥,開源,無模式的文檔型數(shù)據(jù)庫鼎天,是當(dāng)前NoSql數(shù)據(jù)庫中比較熱門的一種舀奶。

MongoDB是一個(gè)介于關(guān)系數(shù)據(jù)庫和非關(guān)系數(shù)據(jù)庫之間的產(chǎn)品,是非關(guān)系數(shù)據(jù)庫當(dāng)中功能最豐富斋射,最像關(guān)系數(shù)據(jù)庫的育勺。他支持的數(shù)據(jù)結(jié)構(gòu)非常松散,是類似json的bjson格式罗岖,因此可以存儲比較復(fù)雜的數(shù)據(jù)類型涧至。Mongo最大的特點(diǎn)是他支持的查詢語言非常強(qiáng)大,其語法有點(diǎn)類似于面向?qū)ο蟮牟樵冋Z言桑包,幾乎可以實(shí)現(xiàn)類似關(guān)系數(shù)據(jù)庫單表查詢的絕大部分功能南蓬,而且還支持對數(shù)據(jù)建立索引。

傳統(tǒng)的關(guān)系數(shù)據(jù)庫一般由數(shù)據(jù)庫(database)哑了、表(table)赘方、記錄(record)三個(gè)層次概念組成,MongoDB是由數(shù)據(jù)庫(database)弱左、集合(collection)窄陡、文檔對象(document)三個(gè)層次組成。MongoDB對于關(guān)系型數(shù)據(jù)庫里的表拆火,但是集合中沒有列跳夭、行和關(guān)系概念,這體現(xiàn)了模式自由的特點(diǎn)们镜。

MySql MongoDB
數(shù)據(jù)庫 數(shù)據(jù)庫
集合
文檔
記錄 字段

3.配置開始

3.1 Maven 相關(guān)配置

在pox.xml文件中添加spring-boot-starter-data-mongodb引用

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-mongodb</artifactId>
</dependency>

3.2 資源文件yml配置

筆者這里使用yml配置方式币叹,配置時(shí)要注意縮進(jìn)!DO痢>备А!

server:
  port: 8031

spring:
  application:
    name: spirng-boot-mongodb
  data:
    mongodb:
      host: localhost   #同127.0.0.1
      port: 27017
      database: test    #指定操作的數(shù)據(jù)庫

3.3 創(chuàng)建實(shí)體類

import org.springframework.data.annotation.Id;
import java.util.Date;

/**
 * @author: huangyibo
 * @Date: 2019/1/30 23:33
 * @Description:圖書實(shí)體類
 */
public class Book {
    @Id
    private String id;
    //價(jià)格
    private Integer price;
    //書名
    private String name;
    //簡介
    private String info;
    //出版社
    private String publish;
    //創(chuàng)建時(shí)間
    private Date createTime;
    //修改時(shí)間
    private Date updateTime;

    //Getter嚼鹉、Setter省略....

3.4 創(chuàng)建service類

Service中主要來實(shí)現(xiàn)CURD的操作
此處需要說明的是Mongodb的修改操作大致有3種:

mongoTemplate.updateFirst操作:修改第一條
mongoTemplate.updateMulti操作:修改符合條件的所有
this.mongoTemplate.upsert操作:修改時(shí)如果不存在則添加.

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
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.stereotype.Service;

import java.util.Date;
import java.util.List;

/**
 * @author: huangyibo
 * @Date: 2019/1/30 23:33
 * @Description: mongo
 */

@Service
public class MongoDbService {
    private static final Logger logger = LoggerFactory.getLogger(MongoDbService.class);

    @Autowired
    private MongoTemplate mongoTemplate;

    /**
     * 保存對象
     *
     * @param book
     * @return
     */
    public String saveObj(Book book) {
        logger.info("--------------------->[MongoDB save start]");
        book.setCreateTime(new Date());
        book.setUpdateTime(new Date());
        mongoTemplate.save(book);
        return "添加成功";
    }


    /**
     * 查詢所有
     *
     * @return
     */
    public List<Book> findAll() {
        logger.info("--------------------->[MongoDB find start]");
        return mongoTemplate.findAll(Book.class);
    }


    /***
     * 根據(jù)id查詢
     * @param id
     * @return
     */
    public Book getBookById(String id) {
        logger.info("--------------------->[MongoDB find start]");
        Query query = new Query(Criteria.where("_id").is(id));
        return mongoTemplate.findOne(query, Book.class);
    }

    /**
     * 根據(jù)名稱查詢
     *
     * @param name
     * @return
     */
    public Book getBookByName(String name) {
        logger.info("--------------------->[MongoDB find start]");
        Query query = new Query(Criteria.where("name").is(name));
        return mongoTemplate.findOne(query, Book.class);
    }

    /**
     * 更新對象
     *
     * @param book
     * @return
     */
    public String updateBook(Book book) {
        logger.info("--------------------->[MongoDB update start]");
        Query query = new Query(Criteria.where("_id").is(book.getId()));
        Update update = new Update().set("publish", book.getPublish())
                .set("info", book.getInfo())
                .set("updateTime", new Date());
        //updateFirst 更新查詢返回結(jié)果集的第一條
        mongoTemplate.updateFirst(query, update, Book.class);
        //updateMulti 更新查詢返回結(jié)果集的全部
//        mongoTemplate.updateMulti(query,update,Book.class);
        //upsert 更新對象不存在則去添加
//        mongoTemplate.upsert(query,update,Book.class);
        return "success";
    }

    /***
     * 刪除對象
     * @param book
     * @return
     */
    public String deleteBook(Book book) {
        logger.info("--------------------->[MongoDB delete start]");
        mongoTemplate.remove(book);
        return "success";
    }

    /**
     * 根據(jù)id刪除
     *
     * @param id
     * @return
     */
    public String deleteBookById(String id) {
        logger.info("--------------------->[MongoDB delete start]");
        //findOne
        Book book = getBookById(id);
        //delete
        deleteBook(book);
        return "success";
    }
}

3.5 controller

import com.example.mqdemo.mongo.Book;
import com.example.mqdemo.mongo.MongoDbService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

import java.util.List;


/**
 * @author: huangyibo
 * @Date: 2019/1/30 23:31
 * @Description:
 */
@RestController
public class BaseController {

    @Autowired
    private MongoDbService mongoDbService;

    @PostMapping("/mongo/save")
    public String saveObj(@RequestBody Book book) {return mongoDbService.saveObj(book);}

    @GetMapping("/mongo/findAll")
    public List<Book> findAll() {return mongoDbService.findAll();}

    @GetMapping("/mongo/findOne")
    public Book findOne(@RequestParam String id) {return mongoDbService.getBookById(id);}

    @GetMapping("/mongo/findOneByName")
    public Book findOneByName(@RequestParam String name) {return mongoDbService.getBookByName(name);}

    @PostMapping("/mongo/update")
    public String update(@RequestBody Book book) {return mongoDbService.updateBook(book);}

    @PostMapping("/mongo/delOne")
    public String delOne(@RequestBody Book book) {return mongoDbService.deleteBook(book);}

    @GetMapping("/mongo/delById")
    public String delById(@RequestParam String id) {return mongoDbService.deleteBookById(id);}
}

3.6.運(yùn)行測試

啟動項(xiàng)目邪意,打開postman開始接口調(diào)試,可以看到成功添加book對象反砌。返回添加成功雾鬼。


image.png

打開MongoDB Compass Community,連接本地MongoDB宴树,可以看到剛才添加的信息策菜。


image.png

其他接口這里就不一一測試了。

優(yōu)化使用
完成以上配置,我們springBoot集成MongoDB環(huán)境基本已經(jīng)搭建好了又憨。

但是在使用中會發(fā)現(xiàn)一個(gè)問題翠霍,假如要對數(shù)據(jù)庫操作多個(gè)對象,那豈不是每一個(gè)對象Service都需要寫一套增刪查改的方法蠢莺。

為了解決這一問題我們可以封裝一個(gè)通用的操作類來提高效率寒匙。

創(chuàng)建MongoDbDao類如下:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
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 java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.List;

/**
 * @author: huangyibo
 * @Date: 2019/1/31 0:08
 * @Description: mongoDB基礎(chǔ)方法封裝
 */
public abstract class MongoDbDao<T> {

    protected Logger logger = LoggerFactory.getLogger(MongoDbDao.class);

    /**
     * 反射獲取泛型類型
     *
     * @return
     */
    protected abstract Class<T> getEntityClass();

    @Autowired
    private MongoTemplate mongoTemplate;

    /***
     * 保存一個(gè)對象
     * @param t
     */
    public void save(T t) {
        logger.info("-------------->MongoDB save start");
        this.mongoTemplate.save(t);
    }

    /***
     * 根據(jù)id從幾何中查詢對象
     * @param id
     * @return
     */
    public T queryById(Integer id) {
        Query query = new Query(Criteria.where("_id").is(id));
        logger.info("-------------->MongoDB find start");
        return this.mongoTemplate.findOne(query, this.getEntityClass());
    }

    /**
     * 根據(jù)條件查詢集合
     *
     * @param object
     * @return
     */
    public List<T> queryList(T object) {
        Query query = getQueryByObject(object);
        logger.info("-------------->MongoDB find start");
        return mongoTemplate.find(query, this.getEntityClass());
    }

    /**
     * 根據(jù)條件查詢只返回一個(gè)文檔
     *
     * @param object
     * @return
     */
    public T queryOne(T object) {
        Query query = getQueryByObject(object);
        logger.info("-------------->MongoDB find start");
        return mongoTemplate.findOne(query, this.getEntityClass());
    }

    /***
     * 根據(jù)條件分頁查詢
     * @param object
     * @param start 查詢起始值
     * @param size  查詢大小
     * @return
     */
    public List<T> getPage(T object, int start, int size) {
        Query query = getQueryByObject(object);
        query.skip(start);
        query.limit(size);
        logger.info("-------------->MongoDB queryPage start");
        return this.mongoTemplate.find(query, this.getEntityClass());
    }

    /***
     * 根據(jù)條件查詢庫中符合條件的記錄數(shù)量
     * @param object
     * @return
     */
    public Long getCount(T object) {
        Query query = getQueryByObject(object);
        logger.info("-------------->MongoDB Count start");
        return this.mongoTemplate.count(query, this.getEntityClass());
    }

    /***
     * 刪除對象
     * @param t
     * @return
     */
    public int delete(T t) {
        logger.info("-------------->MongoDB delete start");
        return (int) this.mongoTemplate.remove(t).getDeletedCount();
    }

    /**
     * 根據(jù)id刪除
     *
     * @param id
     */
    public void deleteById(Integer id) {
        Criteria criteria = Criteria.where("_id").is(id);
        if (null != criteria) {
            Query query = new Query(criteria);
            T obj = this.mongoTemplate.findOne(query, this.getEntityClass());
            logger.info("-------------->MongoDB deleteById start");
            if (obj != null) {
                this.delete(obj);
            }
        }
    }

    /*MongoDB中更新操作分為三種
    * 1:updateFirst     修改第一條
    * 2:updateMulti     修改所有匹配的記錄
    * 3:upsert  修改時(shí)如果不存在則進(jìn)行添加操作
    * */

    /**
     * 修改匹配到的第一條記錄
     * @param srcObj
     * @param targetObj
     */
    public void updateFirst(T srcObj, T targetObj){
        Query query = getQueryByObject(srcObj);
        Update update = getUpdateByObject(targetObj);
        logger.info("-------------->MongoDB updateFirst start");
        this.mongoTemplate.updateFirst(query,update,this.getEntityClass());
    }

    /***
     * 修改匹配到的所有記錄
     * @param srcObj
     * @param targetObj
     */
    public void updateMulti(T srcObj, T targetObj){
        Query query = getQueryByObject(srcObj);
        Update update = getUpdateByObject(targetObj);
        logger.info("-------------->MongoDB updateFirst start");
        this.mongoTemplate.updateMulti(query,update,this.getEntityClass());
    }

    /***
     * 修改匹配到的記錄,若不存在該記錄則進(jìn)行添加
     * @param srcObj
     * @param targetObj
     */
    public void updateInsert(T srcObj, T targetObj){
        Query query = getQueryByObject(srcObj);
        Update update = getUpdateByObject(targetObj);
        logger.info("-------------->MongoDB updateInsert start");
        this.mongoTemplate.upsert(query,update,this.getEntityClass());
    }

    /**
     * 將查詢條件對象轉(zhuǎn)換為query
     *
     * @param object
     * @return
     * @author Jason
     */
    private Query getQueryByObject(T object) {
        Query query = new Query();
        String[] fileds = getFiledName(object);
        Criteria criteria = new Criteria();
        for (int i = 0; i < fileds.length; i++) {
            String filedName = (String) fileds[i];
            Object filedValue = getFieldValueByName(filedName, object);
            if (filedValue != null) {
                criteria.and(filedName).is(filedValue);
            }
        }
        query.addCriteria(criteria);
        return query;
    }

    /**
     * 將查詢條件對象轉(zhuǎn)換為update
     *
     * @param object
     * @return
     * @author Jason
     */
    private Update getUpdateByObject(T object) {
        Update update = new Update();
        String[] fileds = getFiledName(object);
        for (int i = 0; i < fileds.length; i++) {
            String filedName = (String) fileds[i];
            Object filedValue =getFieldValueByName(filedName, object);
            if (filedValue != null) {
                update.set(filedName, filedValue);
            }
        }
        return update;
    }

    /***
     * 獲取對象屬性返回字符串?dāng)?shù)組
     * @param o
     * @return
     */
    private static String[] getFiledName(Object o) {
        Field[] fields = o.getClass().getDeclaredFields();
        String[] fieldNames = new String[fields.length];

        for (int i = 0; i < fields.length; ++i) {
            fieldNames[i] = fields[i].getName();
        }

        return fieldNames;
    }

    /***
     * 根據(jù)屬性獲取對象屬性值
     * @param fieldName
     * @param o
     * @return
     */
    private static Object getFieldValueByName(String fieldName, Object o) {
        try {
            String e = fieldName.substring(0, 1).toUpperCase();
            String getter = "get" + e + fieldName.substring(1);
            Method method = o.getClass().getMethod(getter, new Class[0]);
            return method.invoke(o, new Object[0]);
        } catch (Exception var6) {
            return null;
        }
    }
}

我們將mongoDB常用的CURD操作封裝為通用的父類躏将,然后在不同的業(yè)務(wù)場景下繼承該類锄弱,通過泛型和反射獲取到正在操作的實(shí)體類。

比如我們可以將之前的Book實(shí)體類的CURD類進(jìn)行改造

創(chuàng)建BookMongoDbDao類繼承MongoDbDao

import org.springframework.stereotype.Repository;

/**
 * @author: huangyibo
 * @Date: 2019/1/31 0:12
 * @Description:
 */
@Repository
public class BookMongoDbDao extends MongoDbDao<Book> {
    @Override
    protected Class<Book> getEntityClass() {
        return Book.class;
    }
}

接下來我們可以改造Book的Service類
原始版本:

@Service
public class BookMongoDbService {
    private static final Logger logger = LoggerFactory.getLogger(BookMongoDbService.class);

    @Autowired
    private MongoTemplate mongoTemplate;

    /**
     * 保存對象
     *
     * @param book
     * @return
     */
    public String saveObj(Book book) {
        logger.info("--------------------->[MongoDB save start]");
        book.setCreateTime(new Date());
        book.setUpdateTime(new Date());
        mongoTemplate.save(book);
        return "添加成功";
    }

    //其他操作方法......
}

改造后:

@Service
public class BookMongoDbService {
    private static final Logger logger = LoggerFactory.getLogger(BookMongoDbService.class);

    //注入新的CURD操作類
    @Autowired
    private BookMongoDbDao bookMongoDbDao;

    /**
     * 保存對象
     *
     * @param book
     * @return
     */
    public String saveObj2(Book book) {
        book.setCreateTime(new Date());
        book.setUpdateTime(new Date()); 
        //調(diào)用bookMongoDbDao父類中的添加方法
        bookMongoDbDao.save(book);
        return "添加成功";
    }
}

改造后的saveObj2方法的效果與以前的一致祸憋,但是大大的提高了開發(fā)效率会宪。不需要重復(fù)的編寫CURD的方法。

總結(jié)
通過以上的配置已經(jīng)完成springboot與mongoDB集成環(huán)境的初步搭建蚯窥,當(dāng)然了MongoDB在springboot中的使用不僅于此掸鹅,還有更多的功能和更優(yōu)雅的使用方式等待著我們?nèi)グl(fā)掘。

原文出處:https://blog.csdn.net/qq_33619378/article/details/81544711

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拦赠,一起剝皮案震驚了整個(gè)濱河市巍沙,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌荷鼠,老刑警劉巖赎瞎,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異颊咬,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)牡辽,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進(jìn)店門喳篇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人态辛,你說我怎么就攤上這事麸澜。” “怎么了奏黑?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵炊邦,是天一觀的道長。 經(jīng)常有香客問我熟史,道長馁害,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任蹂匹,我火速辦了婚禮碘菜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己忍啸,他們只是感情好仰坦,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著计雌,像睡著了一般悄晃。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上凿滤,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天妈橄,我揣著相機(jī)與錄音,去河邊找鬼鸭巴。 笑死眷细,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的鹃祖。 我是一名探鬼主播溪椎,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼恬口!你這毒婦竟也來了校读?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤祖能,失蹤者是張志新(化名)和其女友劉穎歉秫,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體养铸,經(jīng)...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡雁芙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了钞螟。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片兔甘。...
    茶點(diǎn)故事閱讀 38,716評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖鳞滨,靈堂內(nèi)的尸體忽然破棺而出洞焙,到底是詐尸還是另有隱情,我是刑警寧澤拯啦,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布澡匪,位于F島的核電站,受9級特大地震影響褒链,放射性物質(zhì)發(fā)生泄漏唁情。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一甫匹、第九天 我趴在偏房一處隱蔽的房頂上張望荠瘪。 院中可真熱鬧夯巷,春花似錦、人聲如沸哀墓。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽篮绰。三九已至后雷,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間吠各,已是汗流浹背臀突。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留贾漏,地道東北人候学。 一個(gè)月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像纵散,于是被迫代替她去往敵國和親梳码。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評論 2 350

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

  • 關(guān)于Mongodb的全面總結(jié) MongoDB的內(nèi)部構(gòu)造《MongoDB The Definitive Guide》...
    中v中閱讀 31,913評論 2 89
  • 一伍掀、MongoDB簡介 1.概述 ? MongoDB是一個(gè)基于分布式文件存儲的數(shù)據(jù)庫掰茶,由C++語言編寫。旨在為WE...
    鄭元吉閱讀 976評論 0 2
  • 烏金的故事,我不知道如何去想把兔,也不知道如何去寫沪伙。 三天前,我發(fā)消息給烏金县好。烏子围橡,我想以自己為原型,寫篇短篇小說...
    青YANGi閱讀 578評論 11 6
  • 跑第一公里時(shí)聘惦,步子抬不起來,呼吸困難儒恋,感覺要跑完3公里好難啊~今天跑完是不是會渾身酸痛善绎,接下來幾天都會特別難熬吧~...
    春曉草雷閱讀 415評論 0 1