6

标题可能不是很清楚,问题在这里

我正在以这种形式执行更新:

db.poi.update({
  _id: ObjectId("50f40cd052187a491707053b"),
  "votes.userid": {
    "$ne": "50f5460d5218fe9d1e2c7b4f"
  }
},
{
  $push: {
    votes: {
      "userid": "50f5460d5218fe9d1e2c7b4f", 
      "value": 1
    }
  },
  $inc: { "score":1 }
})

仅当没有具有相同用户 ID 的文档时才在数组中插入文档(解决方法,因为唯一索引不适用于数组)。该代码在 mongo 控制台上运行良好。在我的应用程序中,我正在使用这个:

@Override
public void vote(String id, Vote vote) {
    Query query = new Query(Criteria.where("_id").is(id).and("votes.userid").ne(vote.getUserid()));
    Update update = new Update().inc("score", vote.getValue()).push("votes", vote);
    mongoOperations.updateFirst(query, update, Poi.class);
}

如果作为“userid”我使用不能是 mongo ObjectId 的字符串,这很好用,但是如果我在示例中使用字符串,则执行的查询将像这样翻译(来自 mongosniff):

update  flags:0 q:{ _id: ObjectId('50f40cd052187a491707053b'), votes.userid: { $ne: ObjectId('50f5460d5218fe9d1e2c7b4f') } } o:{ $inc: { score: 1 }, $push: { votes: { userid: "50f5460d5218fe9d1e2c7b4f", value: 1 } } }

该字符串现在是一个 Objectid。这是一个错误吗?BasicQuery 做同样的事情。我看到的唯一其他解决方案是对所有类 id 使用 ObjectId 而不是 String。

有什么想法吗?

更新:

这是投票类

public class Vote {
  private String userid;
  private int value;
}

这是用户类

@Document
public class User {

  @Id
  private String id;
  private String username;
}

这是我正在执行此更新的类和 mongo 文档

@Document
public class MyClass {

  @Id
  private String id;
  @Indexed
  private String name;
  int score
  private Set<Vote>votes = new HashSet<Vote>();
}

作为杰森

{
  "_id" : ObjectId("50f40cd052187a491707053b"),
  "name" : "Test",
  "score" : 12,
  "votes" : [
    {
      "userid" : "50f5460d5218fe9d1e2c7b4f",
      "value" : 1
    }
  ]
}

votes.userid 中的 Userid 被推送为 String,但相同的 String 被比较为 $ne 中的 ObjectId

4

1 回答 1

2

在我看来,问题可以这样描述:如果您在类中使用 String 代替 ObjectId,如果您想在其他文档(和嵌入式文档)中使用这些 id 作为引用(没有 dbrefs),它们会被推送作为字符串(没关系,因为它们是字符串)。这很好,因为 spring data 可以再次将它们映射到 objectid,但是如果您像我提到的那样进行查询,那就不好了;该字段在比较中转换为 objectid(在本例中为 $ne 运算符),但在嵌入文档中被视为字符串。所以,总结一下,在我看来,在这种情况下 $ne 运算符应该将字段视为字符串。

我的解决方案是编写一个自定义转换器,将字符串作为 objectid 存储在 id 是参考的文档中

public class VoteWriteConverter implements Converter<Vote, DBObject> {

  @Override
  public DBObject convert(Vote vote) {
    DBObject dbo = new BasicDBObject();
    dbo.put("userid", new ObjectId(vote.getUserid()));
    dbo.put("value", vote.getValue());
    return dbo;
  }
}
于 2013-01-16T11:22:52.047 回答