Spring整合mongodb详解

来源:互联网 发布:天气在线软件下载 编辑:程序博客网 时间:2024/05/22 05:29

一、前言

  1. MongoDB 是一个可扩展的、高性能的、开源的NoSQL数据库,跟传统的数据库不一样,MongoDB并不是将数据存储在表中,他将数据结构化为一个类似于JSON的文档中。这篇文章就是展示如何使用Java基于MongoDB和Spring Data创建一个CRUD应用。
  2. Spring Data for MongoDB                                                                                                                 Spring Data for MongoDB提供了一个类似于基于Sping编程模型的NoSQL数据存储。Spring Data for MongoDB提供了很多特性,它使很多MongoDB的Java开发者解放了很多。MongoTemplate helper类支持通用的Mongo操作。它整合了文档和POJO之间的对象映射。通常,他会转换数据库访问异常到Spring中的异常结构。使用起来非常的方便。

MongoDB的API提供了DBObject接口来实现BSONObject的操作方法,BasicDBObject是具体实现。但是并没有提供DBObjectBeanObject的转换。

代码要处理Bean中的各种类型,并且要控制持久化时的深度:

  • 基础类型(int, float, boolean...)
  • 基础扩展类型(Integer, Float, Boolean)
  • 枚举
  • ObjectId
  • 普通对象 extends Object
  • 业务对象(拥有ObjectId的对象)
  • 容器List, Map, Set(处理元素时要把上面的类型再处理一遍)

以上这些东西,spring-data-mongo  都可以实现

参考质料  

  • http://projects.spring.io/spring-data-mongodb/
  • http://docs.spring.io/spring-data/mongodb/docs/current/reference/html/
  • http://docs.mongodb.org/manual/core/introduction/
  • spring官方提供的整合mongodb      spring-data-mongodb-reference.pdf

 本项目是亲测过的

二、项目

  1. 所依赖的jar包                                                                                                                                                                                                                                                                                                                                                          

  2.  spring的配置文件                                                                                                                                       applicationContext.xml如下:                                                                                                                    

    <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"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/context          http://www.springframework.org/schema/context/spring-context-3.0.xsd          http://www.springframework.org/schema/data/mongo          http://www.springframework.org/schema/data/mongo/spring-mongo-1.0.xsd          http://www.springframework.org/schema/beans          http://www.springframework.org/schema/beans/spring-beans-3.0.xsd"><context:property-placeholder location="classpath:mongodb.properties" /><!-- 对mongodb的一些参数连接的设置 --><mongo:mongo  host="${mongo.host}" port="${mongo.port}"><mongo:options     connections-per-host="${mongo.connectionsPerHost}"threads-allowed-to-block-for-connection-multiplier="${mongo.threadsAllowedToBlockForConnectionMultiplier}"connect-timeout="${mongo.connectTimeout}" max-wait-time="${mongo.maxWaitTime}"auto-connect-retry="${mongo.autoConnectRetry}" socket-keep-alive="${mongo.socketKeepAlive}"socket-timeout="${mongo.socketTimeout}" slave-ok="${mongo.slaveOk}"write-number="1" write-timeout="0" write-fsync="true" /></mongo:mongo><!-- 配置一个MongoDbFactory,u数据库名叫mydb,mongo-ref指向mongo --><!-- 这个东西就是上面那个mongo:mongo配置的bean --><mongo:db-factory dbname="${mongo.database}" mongo-ref="mongo" /><!-- 这个bean的默认名称是mongoTemplate --><!-- 类似于spring里的jdbcTemplate,需要注入MongoDbFactory --><!-- mongoConverter不是必须的 --><bean id="mongoTemplate" class="org.springframework.data.mongodb.core.MongoTemplate"> <constructor-arg name="mongoDbFactory" ref="mongoDbFactory"/></bean><!-- 注入spring提供的模板 --><bean id="personRepository" class="com.spring.mongo.dao.impl.PersonRepository">           <property name="mongoTemplate" ref="mongoTemplate"></property>       </bean> <context:annotation-config /><!-- Scan components for annotations within the configured package --><context:component-scan base-package="com"><context:exclude-filter type="annotation"expression="org.springframework.context.annotation.Configuration" /></context:component-scan></beans>
     mongodb.properties文件如下:                                                                                                                                
    mongo.host=192.168.0.19mongo.port=27017mongo.database=xlkmongo.connectionsPerHost=8mongo.threadsAllowedToBlockForConnectionMultiplier=4mongo.connectTimeout=1500mongo.maxWaitTime=1500mongo.autoConnectRetry=truemongo.socketKeepAlive=truemongo.socketTimeout=1500mongo.slaveOk=true
    注:
     <mongo:mongo  host="${mongo.host}" port="${mongo.port}"  replica-set="">
     在mongo里也可以配置副本集分片模式,引用官方的解释:

     
  3. 使用@Document注解指明一个领域对象将被持久化到MongoDB中。@Id注解identifies。                  
    package com.spring.mongo.domain;import java.io.Serializable;import org.springframework.data.annotation.Id;import org.springframework.data.mongodb.core.index.Indexed;import org.springframework.data.mongodb.core.mapping.Document;/** *  * person实体类 </br>  * 类名: Person </br>  * 日期: 2014-4-15 上午11:43:08 </br>  * @author 许立亢  * @version 1.0 */@Documentpublic class Person implements Serializable {/** *  */private static final long serialVersionUID = 3617931430808763429L;@Idprivate String id;@Indexed     private String name;       private int age;public Person() {super();}public Person(String id, String name, int age) {super();this.id = id;this.name = name;this.age = age;}/** * @return the id */public String getId() {return id;}/** * @param id the id to set */public void setId(String id) {this.id = id;}/** * @return the name */public String getName() {return name;}/** * @param name the name to set */public void setName(String name) {this.name = name;}/** * @return the age */public int getAge() {return age;}/** * @param age the age to set */public void setAge(int age) {this.age = age;}/** *  * @param name * @param age */public Person(String name, int age) {super();this.name = name;this.age = age;}    public String toString() {           return "Person[id="+id+",name="+name+",age="+age+"]";       }   }
     
  4. 创建一个简单的接口,这个接口带有CRUD方法。
    package com.spring.mongo.dao;import java.util.List;import org.springframework.data.mongodb.core.mapreduce.MapReduceOptions;import com.mongodb.CommandResult;import com.spring.mongo.domain.Person;/** *  * person接口操作 </br>  * 类名: AbstractRepository </br>  * 日期: 2014-4-16 上午09:24:17 </br>  * @author 许立亢  * @version 1.0 */public interface AbstractRepository {public void insert(Person person);     public Person findOne(String id);       public List<Person> findAll();           public List<Person> findByRegex(String regex);    public void removeOne(String id);       public void removeAll();       public void updateInc(String id,String key,Number inc);    public void updateSet(String id,String key,Object value);        public void mapReduce(String inputCollectionName, String mapFunction,String reduceFunction,MapReduceOptions mapReduceOptions,Class<Person> entityClass);    public CommandResult executeCommand(String jsonCommand);public void createCollection();public void dropCollection();}
    实现刚才的接口, 程序启动时,注入MongoTemplate模板  ,        
    /** *  */package com.spring.mongo.dao.impl;import java.util.List;import java.util.regex.Pattern;import org.springframework.data.mongodb.core.MongoTemplate;import org.springframework.data.mongodb.core.mapreduce.MapReduceOptions;import org.springframework.data.mongodb.core.mapreduce.MapReduceResults;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 com.mongodb.BasicDBList;import com.mongodb.BasicDBObject;import com.mongodb.CommandResult;import com.spring.mongo.dao.AbstractRepository;import com.spring.mongo.domain.Person;/** *  * person持久化类 </br>  * 类名: PersonRepository </br>  * 日期: 2014-4-16 上午09:50:14 </br>  * @author 许立亢  * @version 1.0 */public class PersonRepository implements AbstractRepository {private MongoTemplate mongoTemplate;/** *  * 查找所以的document</br>  * 日期:2014-4-16 上午09:49:47 * @return  * @see com.spring.mongo.dao.AbstractRepository#findAll() */@Overridepublic List<Person> findAll() {return getMongoTemplate().find(new Query(), Person.class);}/** *  * 根据id查找一个document</br>  * 日期:2014-4-16 上午09:48:38 * @param id * @return  * @see com.spring.mongo.dao.AbstractRepository#findOne(java.lang.String) */@Overridepublic Person findOne(String id) {return getMongoTemplate().findOne(new Query(Criteria.where("id").is(id)), Person.class);}/** *  * 修改一条记录,增量修改</br>  * 日期:2014-4-16 上午09:49:22 * @param id  * @see com.spring.mongo.dao.AbstractRepository#findAndModify(java.lang.String) */@Overridepublic void updateInc(String id,String key,Number inc) {getMongoTemplate().updateFirst(new Query(Criteria.where("id").is(id)),new Update().inc(key, inc), Person.class);}/** *  * 修改一条记录</br>  * 日期:2014-4-16 上午09:59:01 * @param id * @param key * @param value  * @see com.spring.mongo.dao.AbstractRepository#updateSet(java.lang.String, java.lang.String, java.lang.Object) */@Overridepublic void updateSet(String id,String key,Object value) {getMongoTemplate().updateFirst(new Query(Criteria.where("id").is(id)),new Update().set(key, value), Person.class);}/* * (non-Javadoc) *  * @see com.mongo.dao.AbstractRepository#findByRegex(java.lang.String) */@Overridepublic List<Person> findByRegex(String regex) {Pattern pattern = Pattern.compile(regex, Pattern.CASE_INSENSITIVE);Criteria criteria = new Criteria("name").regex(pattern.toString());return getMongoTemplate().find(new Query(criteria), Person.class);}/** *  * 插入一个记录document</br>  * 日期:2014-4-16 上午09:48:17 * @param person  * @see com.spring.mongo.dao.AbstractRepository#insert(com.spring.mongo.domain.Person) */@Overridepublic void insert(Person person) {getMongoTemplate().insert(person);}/** *  * 删除全部的记录</br>  * 日期:2014-4-16 上午09:47:57  * @see com.spring.mongo.dao.AbstractRepository#removeAll() */@Overridepublic void removeAll() {List<Person> list = this.findAll();if (list != null) {for (Person person : list) {getMongoTemplate().remove(person);}}}/** *  * 根据id删除一个document</br>  * 日期:2014-4-16 上午09:47:26 * @param id  * @see com.spring.mongo.dao.AbstractRepository#removeOne(java.lang.String) */@Overridepublic void removeOne(String id) {Criteria criteria = Criteria.where("id").in(id);if (criteria != null) {Query query = new Query(criteria);if (query != null&& getMongoTemplate().findOne(query, Person.class) != null)getMongoTemplate().remove(getMongoTemplate().findOne(query, Person.class));}}/** * 创建一个表collection * </br>  * 日期:2014-4-16 上午10:01:28  * @see com.spring.mongo.dao.AbstractRepository#createCollection() */public void createCollection() {if (!mongoTemplate.collectionExists(Process.class)) {mongoTemplate.createCollection(Process.class);}}/** *  * 删除一个表collection</br>  * 日期:2014-4-16 上午10:03:16  * @see com.spring.mongo.dao.AbstractRepository#dropCollection() */public void dropCollection() {if (mongoTemplate.collectionExists(Process.class)) {mongoTemplate.dropCollection(Process.class);}}public void mapReduce(String inputCollectionName, String mapFunction,String reduceFunction,MapReduceOptions mapReduceOptions,Class<Person> entityClass){ MapReduceResults<Person> mapReduce = mongoTemplate.mapReduce(inputCollectionName, mapFunction, reduceFunction,mapReduceOptions, Person.class); BasicDBList list = (BasicDBList)mapReduce.getRawResults().get("results");       for (int i = 0; i < list.size(); i ++) {           BasicDBObject obj = (BasicDBObject)list.get(i);          System.out.println(obj.toString());      } }public CommandResult executeCommand(String jsonCommand){//String CommandResult commandResult = mongoTemplate.executeCommand(jsonCommand);return commandResult;}/** * @return the mongoTemplate */public MongoTemplate getMongoTemplate() {return mongoTemplate;}/** * @param mongoTemplate *            the mongoTemplate to set */public void setMongoTemplate(MongoTemplate mongoTemplate) {this.mongoTemplate = mongoTemplate;}}
                                
  5. 后写出我们的测试类开始进行测试                                                
  6. package spring;import java.util.List;import org.apache.commons.logging.Log;import org.apache.commons.logging.LogFactory;import org.junit.Before;import org.junit.Test;import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;import org.springframework.data.mongodb.core.mapreduce.MapReduceOptions;import com.spring.mongo.dao.AbstractRepository;import com.spring.mongo.dao.impl.PersonRepository;import com.spring.mongo.domain.Person;/** *  * 创建测试用例 </br>  * 类名: MongoUserTest </br>  * 日期: 2014-4-16 上午10:24:56 </br>  * @author 许立亢  * @version 1.0 */public class MongoUserTest {private static Log log = LogFactory.getLog(MongoUserTest.class.getName());private  AbstractRepository pr=null;@Beforepublic void init(){ log.debug("开始启动"); ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");  pr= (PersonRepository)ctx.getBean("personRepository"); }@Testpublic void insert(){Person p=new Person("许立亢",24); pr.insert(p); log.debug("添加成功");}@Testpublic void findOne(){String id="534ddf83aa17bf2cdb40af4a";Person p= pr.findOne(id);log.info(p);}@Testpublic void listAll(){List<Person> list=pr.findAll();log.debug("查询结果如下:");for (Person p:list){log.debug(p.toString());System.out.println(p.toString());}}@Testpublic void removeOne(){pr.removeOne("534ddf83aa17bf2cdb40af4a");log.info("删除成功");}@Testpublic void removeAll(){pr.removeAll();log.info("全部删除.....");}@Test public void updateInc(){pr.updateInc("534ddfe4aa174e725653fd11","age",12);log.info("修改........");}@Test public void updateSet(){pr.updateSet("534ddfe4aa174e725653fd11","age",12);log.info("修改........");}@Testpublic void mapReduce(){String inputCollectionName = "person";String mapFunction = "function(){" +"emit({age:this.age},{count:1,name:this.name});" +"};";String reduceFunction = "function( key , values ){" + " var reduced = {count:0,name:''};" + "values.forEach(function(val) {" +"reduced.count += 1;" +"reduced.name = val.name;" +"});" +"return reduced;" +"};";MapReduceOptions mapReduceOptions = new MapReduceOptions().outputCollection("ddd");pr.mapReduce(inputCollectionName, mapFunction,reduceFunction, mapReduceOptions, Person.class);}public static void main(String[] args) {}}
          

 

附加:

一下是spring官方提供的使用java代码注解实现:



 

 

 

 

 

  

 

 

 

原创粉丝点击