4.2MongoDB和Spring整合

MongoDB现在用的非常非常多,如何和Spring整合也是经常碰到的问题。
Spring提供了MongoTemplate这样一个模板类的实现方法,简化了具体操作。

下面讲一下具体实现:

添加依赖

<dependency>
      <groupId>org.springframework.data</groupId>
      <artifactId>spring-data-mongodb</artifactId>
      <version>1.10.3.RELEASE</version>
</dependency>

其余Spring相关的忽略

Spring的配置applicationContext-mongo.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:util="http://www.springframework.org/schema/util"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:mongo="http://www.springframework.org/schema/data/mongo"
       xsi:schemaLocation="
        http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/data/mongo http://www.springframework.org/schema/data/mongo/spring-mongo.xsd">
    <context:property-placeholder
            ignore-unresolvable="true" location="classpath:/mongodb.properties"/>

    <mongo:mongo-client id="mongoClient" host="${mongo.host}" port="${mongo.port}">
        <!-- credentials="${mongo.user}:${mongo.pwd}@${mongo.defaultDbName}"-->
    </mongo:mongo-client>

    <mongo:db-factory id="mongoDbFactory"
                      dbname="${mongo.database}"
                      mongo-ref="mongoClient"/>
    <!-- 默认Mongodb类型映射 -->
    <bean id="defaultMongoTypeMapper" class="org.springframework.data.mongodb.core.convert.DefaultMongoTypeMapper">
        <constructor-arg name="typeKey">
            <!-- 这里设置为空,可以把 spring data mongodb 多余保存的_class字段去掉 -->
            <null/>
        </constructor-arg>
    </bean>

    <mongo:repositories base-package="com.critc.mongo"/>
    <!-- 自动扫描以下包的有Doucment注解的类 -->
    <mongo:mapping-converter id="mappingConverter" base-package="com.critc.mongo.model"
                             type-mapper-ref="defaultMongoTypeMapper">
    </mongo:mapping-converter>

    <!-- Mongodb的模板 -->
    <bean id="mongoTemplate" class="org.springframework.data.mongodb.core.MongoTemplate">
        <constructor-arg name="mongoDbFactory" ref="mongoDbFactory"/>
        <constructor-arg name="mongoConverter" ref="mappingConverter"/>
    </bean>
</beans>

mongo:mongo-client是定义MongoDB的客户端连接,需要hostport参数,如果需要账号密码的话,需要增加credentials配置

mongo:db-factory配置连接工厂,指定具体的连接数据库,本例默认为test

defaultMongoTypeMapper默认Mongodb类型映射

mongo:mapping-converterMongoDB的实体映射

mongoTemplate这是最主要的,定义模板类,依赖连接工厂和实体映射

这里举一个article的增删改查的例子。

相关实体Article.java

@Document(collection = "article_info")
public class Article {
    @Id
    private String id;//id
    @Field("title")
    private String title;//标题
    @Field("url")
    private String url;//链接
    @Field("author")
    private String author;//作者
    @Field("tags")
    private List<String> tags;//tag 标签
    @Field("visit_count")
    private Long visitCount;//访问次数
    @Field("add_time")
    private Date addTime;//添加时间
// get set方法省略

@Document(collection = "article_info")这个注解和Hibernate的注解Entiry非常相似,就是定义一个文档,对象MongoDB存储的Collection的名称是article_info

@Id指该字段是主键,不能缺少

@Field("add_time")指该字段映射MongoDB的实际字段,如果一致可以省略

ArticleRepository实际访问接口


@Repository("ArticleRepository")
public interface ArticleRepository extends PagingAndSortingRepository<Article, String> {

    //分页查询
    public Page<Article> findAll(Pageable pageable);


    //根据author查询
    public List<Article> findByAuthor(String author);
    
    //根据作者和标题查询
    public List<Article> findByAuthorAndTitle(String author, String title);
    
    //忽略参数大小写
    public List<Article> findByAuthorIgnoreCase(String author);
    
    //忽略所有参数大小写
    public List<Article> findByAuthorAndTitleAllIgnoreCase(String author, String title);
    
    //排序
    public List<Article> findByAuthorOrderByVisitCountDesc(String author);
    public List<Article> findByAuthorOrderByVisitCountAsc(String author);
    
    //自带排序条件
    public List<Article> findByAuthor(String author, Sort sort);
    
}

Spring的data repository封装了一套增删改查的方法,就和JPA实现的一样,ArticleRepository继承PagingAndSortingRepository,就集成了常用的增删改查方法,比如save、findOne、exists、findAll、delete等等,可以采用默认实现方式来完成常用的增删改查操作。

测试上述各个方法ArticleRepositoryTest.java

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath*:applicationContext-mongo.xml"})
public class ArticleRepositoryTest {

    @Autowired
    private ArticleRepository articleRepository;

    /**
     * 新增记录
     */
    @Test
    public void add() {
        //增加一条记录
        Article article = new Article();
        article.setId("1");
        article.setTitle("MongoTemplate的基本使用");
        article.setAuthor("kcy");
        article.setUrl("//www.greatytc.com/");
        article.setTags(Arrays.asList("java", "mongodb", "spring"));
        article.setVisitCount(0L);
        article.setAddTime(new Date());
        articleRepository.save(article);

        //批量添加
        List<Article> articles = new ArrayList<>(10);
        for (int i = 0; i < 10; i++) {
            Article article2 = new Article();
            article2.setId(String.valueOf(i + 1));
            article2.setTitle("MongoTemplate的基本使用");
            article2.setAuthor("kcy");
            article2.setUrl("//www.greatytc.com" + i);
            article2.setTags(Arrays.asList("java", "mongodb", "spring"));
            article2.setVisitCount(0L);
            article2.setAddTime(new Date());
            articles.add(article2);
        }
        articleRepository.save(articles);
    }

    /**
     * 修改记录,修改id为1的访问次数+1
     */
    @Test
    public void update() {
        Article article = articleRepository.findOne("1");
        article.setVisitCount(article.getVisitCount() + 1);
        articleRepository.save(article);
    }

    /**
     * 批量修改,查看author为kcy的统一修改为kcy2
     */
    @Test
    public void batchUpdate() {
        List<Article> articles = articleRepository.findByAuthor("kcy");
        articles.forEach(article -> {
            article.setAuthor("kcy2");
        });
        articleRepository.save(articles);
    }

    /**
     * 删除记录,删除id为10的
     */
    @Test
    public void delete() {
        Article article = articleRepository.findOne("10");
        articleRepository.delete(article);
    }

    @Test
    public void batchDelete() {
        List<Article> articles = articleRepository.findByAuthor("kcy2");
        articleRepository.delete(articles);
    }

    /**
     * 查询所有
     *
     * @author 孔垂云
     */
    @Test
    public void findAll() {
        Iterable<Article> articles = articleRepository.findAll();
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 根据author查询
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthor() {
        List<Article> articles = articleRepository.findByAuthor("kcy");
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 按照author和title查询
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorAndTitle() {
        List<Article> articles = articleRepository.findByAuthorAndTitle("kcy", "MongoTemplate的基本使用");
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 根据作者查询,忽略大小写
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorIgnoreCase() {
        List<Article> articles = articleRepository.findByAuthorIgnoreCase("JASON");
        articles.forEach(article -> {
            System.out.println(article.getId());
        });
    }

    /**
     * 忽略所有参数的大小写
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorAndTitleAllIgnoreCase() {
        List<Article> articles = articleRepository.findByAuthorAndTitleAllIgnoreCase("KCY", "MONGOTEMPLATE的基本使用");
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 根据author查询,并且以访问次数降序排序显示
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorOrderByVisitCountDesc() {
        List<Article> articles = articleRepository.findByAuthorOrderByVisitCountDesc("kcy");
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }


    /**
     * 根据作者查询,并且以访问次数升序排序显示
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorOrderByVisitCountAsc() {
        List<Article> articles = articleRepository.findByAuthorOrderByVisitCountAsc("kcy");
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 自带排序条件
     *
     * @author 孔垂云
     */
    @Test
    public void findByAuthorBySort() {
        List<Article> articles = articleRepository.findByAuthor("kcy", new Sort(Direction.ASC, "VisitCount"));
        articles.forEach(article -> {
            System.out.println(article.toString());
        });
    }

    /**
     * 分页查询所有,并且排序
     */
    @Test
    public void findByPage() {
        int page = 1;
        int size = 2;
        Pageable pageable = new PageRequest(page, size, new Sort(Direction.ASC, "VisitCount"));
        Page<Article> pageInfo = articleRepository.findAll(pageable);
        //总数量
        System.out.println(pageInfo.getTotalElements());
        //总页数
        System.out.println(pageInfo.getTotalPages());
        for (Article article : pageInfo.getContent()) {
            System.out.println(article.toString());
        }
    }
}

上面一段代码较长,基本上MongoDB常用的各种例子都讲清楚了,比如增加、批量增加、修改、删除、按id查找、按标题查询、分页等等。

源码下载

本工程详细源码

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

推荐阅读更多精彩内容

  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,773评论 6 342
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,629评论 18 139
  • application的配置属性。 这些属性是否生效取决于对应的组件是否声明为Spring应用程序上下文里的Bea...
    新签名阅读 5,358评论 1 27
  • 请分别对好奇心日报 的登录、注册、找回密码流程进行梳理。 1. 注册流程(体验版本:Android4.3系统,v3...
    朔月之舟阅读 325评论 0 0
  • Brew 安装 ffmpeg,以下命令可以直接安装一个较为完整的ffmpeg。把用的到的大部分都开了。也可以根据自...
    鱿鱼鉄板燒阅读 11,757评论 4 50