Spring Boot Mongo - Дубликат ключа E11000
Я создаю простой REST API с помощью Spring-Boot-Starter-Data-Mongodb, и я всегда получаю E11000 duplicate key error
при попытке вставить мой 2-й ряд.
Руководство Spring для начинающих имеет довольно простую конфигурацию, которой я следовал, но я, должно быть, что-то упустил.
Я удалил коллекцию и начал заново, первый документ сохраняется нормально, а второй пытается сохранить как id=0. Как заставить Spring/Mongo правильно увеличиваться?
Вот ошибка, которую я получаю:
org.springframework.dao.DuplicateKeyException: { "serverUsed" : "localhost:27017" , "ok" : 1 , "n" : 0 , "err" : "E11000 duplicate key error index: test.game.$_id_ dup key: { : 0 }" , "code" : 11000}; nested exception is com.mongodb.MongoException$DuplicateKey: { "serverUsed" : "localhost:27017" , "ok" : 1 , "n" : 0 , "err" : "E11000 duplicate key error index: test.game.$_id_ dup key: { : 0 }" , "code" : 11000}
Игра
package com.recursivechaos.boredgames.domain;
import org.springframework.data.annotation.Id;
import org.springframework.data.mongodb.core.mapping.Document;
@Document
public class Game {
@Id
private long id;
private String title;
private String description;
public Game() {
}
public long getId() {
return id;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
}
Игровой репозиторий
package com.recursivechaos.boredgames.repository;
import com.recursivechaos.boredgames.domain.Game;
import org.springframework.data.mongodb.repository.MongoRepository;
import org.springframework.data.repository.query.Param;
import java.util.List;
public interface GameRepository extends MongoRepository<Game, Long> {
List<Game> findByTitle(@Param("title") String title);
}
AppConfig
package com.recursivechaos.boredgames.configuration;
import com.mongodb.Mongo;
import org.springframework.context.annotation.Bean;
import org.springframework.data.authentication.UserCredentials;
import org.springframework.data.mongodb.MongoDbFactory;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.SimpleMongoDbFactory;
public class AppConfig {
public
@Bean
MongoDbFactory mongoDbFactory() throws Exception {
UserCredentials userCredentials = new UserCredentials("username", "password");
SimpleMongoDbFactory boredgamesdb = new SimpleMongoDbFactory(new Mongo(), "boredgamesdb", userCredentials);
return boredgamesdb;
}
public
@Bean
MongoTemplate mongoTemplate() throws Exception {
return new MongoTemplate(mongoDbFactory());
}
}
Спасибо за внимание!
2 ответа
Вы используете примитив long
который имеет неявное, предварительно назначенное значение. Следовательно, это значение передается MongoDB, и оно сохраняется как есть, так как предполагается, что вы хотите определить идентификаторы вручную.
Хитрость заключается в том, чтобы просто использовать тип оболочки, так как это может быть null
MongoDB обнаруживает отсутствие значения и автоматически заполняет ObjectID
для тебя. Тем не мение, Long
не будет работать как ObjectID
s не вписываются в числовое пространство Long
, Типы идентификаторов, поддерживаемые для автоматической генерации: ObjectId
, String
а также BigInteger
,
Все это документировано в справочной документации.
Я также понял, что если вы попытаетесь вставить один и тот же объект более одного раза, он выдает аналогичную ошибку. По-видимому, это не потому, что значение объекта одинаково, а потому, что ссылка на объект одинакова, например.
// This would throw the exception
SomeObject object = new SomeObject()
something.insert(object)
something.insert(object)
// But this will not because the references are different although it is
// the same object
SomeObject object1 = new SomeObject()
SomeObject object2 = new SomeObject()
something.insert(object)
something.insert(object)