1、WriteConcern

当Java程序往MongoDB中写数据的时候,可以通过设置WriteConcern参数,来控制写操作策略。

WriteConcern是MongoDB Java Driver提供的一个类,它提供了若干常量,其中最常用的2个如下:

ACKNOWLEDGED——Java Driver会等待MongoDB Server确认写操作完成。

UNACKNOWLEDGED——Java Driver把写操作请求发往MongoDB Server,然后立刻返回,不管MongoDB Server是否执行了该写操作。

显然,通过ACKNOWLEDGED能知道写操作是否成功,但执行写入操作会比较耗时;UNACKNOWLEDGED写入速度会非常快,但无法知道写入是否成功。

2、全局默认配置

接下来,我们结合BuguMongo中的代码,看看WriteConcern在实际中是如何使用的。

MongoDB Java Driver默认使用ACKNOWLEDGED,这个默认值是在MongoClientOptions中定义的:

public class MongoClientOptions {
    …
    private WriteConcern writeConcern = WriteConcern.ACKNOWLEDGED;
    …
}

也就是说,MongoDB Java Driver默认所有的写操作都是需要确认再返回的。如果你要改变这个全局的配置,可以在建立BuguConnection连接的时候,赋值一个自定义的MongoClientOptions。例如:

MongoClientOptions myOptions = MongoClientOptions.builder().writeConcern(WriteConcern.UNACKNOWLEDGED).build();
BuguConnection conn = BuguConnection.getInstance();
conn.setOptions(myOptions);
conn.connect("192.168.0.100", 27017, "mydb", "username", "password");

3、BuguDao中的WriteConcern

然而,在一个项目中,使用同一个写入策略,是不太现实的。我们往往需要经常变换写入策略。别担心,BuguDao提供了setWriteConcern()方法,你可以针对不同的Collection设定不同的写入策略,而且可以随时改变。

如果不进行设置的话,BuguDao默认使用的是ACKNOWLEDGED。

我们可以进行一个比较实验,分别使用ACKNOWLEDGED和UNACKNOWLEDGED,往MongoDB中写入1万条数据,看看耗时相差多少。

@Entity
public class Foo extends SimpleEntity {

    private String content;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

}

实验1:

BuguDao dao = new BuguDao(Foo.class);
long begin = System.currentTimeMillis();
for(int i=0; i<10000; i++){
    Foo foo = new Foo();
    foo.setContent("abcdefg" + i);
    dao.save(foo);
}
long end = System.currentTimeMillis();
System.out.println(end - begin);

这里,我们不对WriteConcern进行设置,而是使用默认的ACKNOWLEDGED。代码执行3次,在我的笔记本电脑上,输出的结果分别是:

55007, 55503, 54066

实验2:

BuguDao dao = new BuguDao(Foo.class);
//设置WriteConcern为UNACKNOWLEDGED,而不是默认的ACKNOWLEDGED
dao.setWriteConcern(WriteConcern.UNACKNOWLEDGED);  
long begin = System.currentTimeMillis();
for(int i=0; i<10000; i++){
    Foo foo = new Foo();
    foo.setContent("abcdefg" + i);
    dao.save(foo);
}
long end = System.currentTimeMillis();
System.out.println(end - begin);

代码执行3次,在我的笔记本电脑上,输出的结果分别是:

1736, 2636, 1470

可以看出,写入的速度快了很多。

4、写操作的返回值

上面的实验中,我们忽略了dao.save(foo)的返回值。实际上,BuguDao中的所有写操作,返回值都是WriteResult,而不是void。例如:

public WriteResult save(T t)

如果你使用的WriteConcern是UNACKNOWLEDGED,那么WriteResult是没有任何用处的。但如果你使用的是ACKNOWLEDGED,那么可以从WriteResult中获得写操作的结果,例如:操作是否成功,总共写了多少条数据,等等。