mongodb的唯一ID

时间:2009-08-29 23:55:03

标签: mongodb uniqueidentifier nosql

如果我正在构建博客,我可以使用博客标题作为唯一标识符,并通过URL解析它。但是,如果我想使用数字怎么办?你知道twitter有www.twitter.com/username/statuses/9834542吗?有没有人想出一个很好的方法来使这项工作?使用“_id”是不可能的,因为它太长了。

5 个答案:

答案 0 :(得分:22)

只要您能保证唯一性,您就不会受限于使用默认的“_id”MongoDB耗材。

因此,由您决定如何生成此数字。如果您想将此数字存储在MongoDB中,则可以将其存储在单独的集合中,并为每个所需的新URL递增它。

通过使用the $inc verb来增加字段,或者您可能想要了解MongoDB如何atomically update或增加值。

答案 1 :(得分:15)

可以使用findandmodify命令完成。

我们假设我们有一个名为sequences的特殊集合,我们希望有一个邮政编号序列(名为postid),您可以使用与此类似的代码:

> db.runCommand( { "findandmodify" : "sequences",
                   "query" : { "name" : "postid"},
                   "update" : { $inc : { "id" : 1 }},
                   "new" : true } );

此命令将以原子方式返回更新的(new)文档以及状态。如果命令成功完成,value字段包含返回的文档。

答案 2 :(得分:7)

如果要在MongoDB中为自己的字段添加唯一性约束,请使用索引。然后,您可以使用任何要生成数字的散列算法并测试它的唯一性。 MongoDB文档中的示例是

db.things.ensureIndex({firstname: 1, lastname: 1}, {unique: true});

将阻止您插入具有相同名字和姓氏的文档作为另一个文档。

documentation中提供了更多信息。

答案 3 :(得分:4)

我通过使用数据创建集合'序列'解决了这个问题:

  • 名称
  • currurt value

我正在使用Morhpia,因此请使用DAO。但是你也可以在没有Morhpia的情况下做到这一点。 想法是使用$ atomic(可能会因为仅更新1个实例而省略)和$inc修饰符运算符。

序列

@Entity(value = "sys_sequence", noClassnameStored = true)
public class SequenceM {

    /**
     * Names of entity
     */
    public static enum Entity {
        USER,
        CAPABILITY_HISTORY;

        public String getEntityName() {
            return this.name().toLowerCase();
        }
    }

    @Id
    private ObjectId uid;

    @Property
    @Indexed(unique = true)
    private String name;

    @Property
    private Long value;

 //..getters/setters/etc
 }

SequenceDAO上的方法:

@NotNull
public Long nextValue(final @NotNull SequenceM.Entity entity) {
    final DB db = this.ds.getDB();
    final WriteConcern writeConcern = getWriteConcern();

    //optimization for JVM instance
    synchronized(entity) {
        do {
            SequenceM sequence = findOne("name", entity.getEntityName());

            final DBObject q = BasicDBObjectBuilder.start().add("name", entity.getEntityName()).add("value", sequence.getValue()).add("$atomic", 1).get();
            final DBObject o = BasicDBObjectBuilder.start().add("$inc", BasicDBObjectBuilder.start().add("value", 1).get()).get();

            WriteResult writeResult = db.getCollection("sys_sequence").update(q, o, false, true, writeConcern);

            if(writeResult.getN() == 1) {
                return sequence.getValue() + 1;
            }
        } while(true);
    }
}

/**
 * Determining writing concern basing on configuration
 */
private WriteConcern getWriteConcern() {
    return isOneNodeOnly ? WriteConcern.SAFE : REPLICATION_SAFE;
}

根据MongoDB配置(仅一个节点或主/从或副本集),您必须使用正确的WriteConcern。在一个具有一个实例的环境中使用REPLICATION_SAFE只会导致无限循环。

答案 4 :(得分:1)

从技术上讲,ID号太大而无法缩短它。但是,可以填补一个策略。这是从Hex传递到字母数字,从而减少了tulizar的字符数,并在Url中看起来更漂亮。我真的服务得很好......这是

function encode(hex) {
    return new Buffer(hex, 'hex').toString('base64').replace('+', '-').replace('/', '_');
};

function decode(NoHex) {
    return new Buffer( NoHex.replace('-','+').replace('_','/'), 'base64').toString('hex');
};

IdString= MyDoc._id.toString(); 
Idencode = encode( IdString ) // 16 Caracters a-Z and 0-9 
console.log( IdEncode ); //You see That 'aqswedasdfdsadsf'
IdDecode = decode( IdEncode );
IdDecode === IdString // Is true!!!

当然这种技术使用相同的id,mongo。