如果我正在构建博客,我可以使用博客标题作为唯一标识符,并通过URL解析它。但是,如果我想使用数字怎么办?你知道twitter有www.twitter.com/username/statuses/9834542吗?有没有人想出一个很好的方法来使这项工作?使用“_id”是不可能的,因为它太长了。
答案 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)
我通过使用数据创建集合'序列'解决了这个问题:
我正在使用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。