私は取得しています
org.hibernate.exception.ConstraintViolationException: Duplicate entry X for key 'PRIMARY'
のせいで
Caused by: com.mysql.jdbc.exceptions.MySQLIntegrityConstraintViolationException: Duplicate entry X for key 'PRIMARY'
私はスレッド化された作業(4つのスレッドで)を行っています。ロジックは次のとおりです。データベースに何かがあるかどうかを(主キーで)確認し、それを取得してオブジェクトを変更するか、そうでない場合は新しいオブジェクトを作成します。問題は、さらに2つのスレッドが同じオブジェクトを使用し、それがデータベースにない場合、両方がそれを作成しようとし、保存すると違反が発生することだと思います。私を困惑させているのは、hashCode メソッドと equals メソッドをオーバーライドしたことです。例外をキャッチしてデータベースにクエリを実行すると、JVM のオブジェクトとデータベースのオブジェクトが等しくなります。例外が発生してはならないということではないですか?
さらに困惑しているのは、データベースにクエリを実行するとnull
返されることがあることです。
ロックを作成してから同期しようとしたので、1 つのスレッドが同じ ID のオブジェクトで作業を行っている場合、もう 1 つのスレッドが最初のスレッドを終了させますが、改善されませんでした。
また、次のコードでシナリオを再作成しようとしましたが、問題なく実行されます。
public static void main(String[] args) throws Exception {
ThreadPoolExecutor executionPool = new ThreadPoolExecutor(4, 4, 9999, TimeUnit.HOURS, new LinkedBlockingQueue<Runnable>());
final Dao dao = new Dao();
TestIdentity t1 = new TestIdentity(1, "aaa", false);
dao.saveObject(t1);
TestIdentity t2 = new TestIdentity(1, "aaa", true);
dao.saveObject(t2);
dao.saveObject(t1);
dao.saveObject(t2);
final Wrapper w1 = new Wrapper(t1);
final Wrapper w2 = new Wrapper(t2);
Runnable runnable1 = new Runnable() {
public void run() {
while(true) {
w1.work(dao);
}
}
};
Runnable runnable2 = new Runnable() {
public void run() {
while(true) {
w2.work(dao);
}
}
};
executionPool.submit(runnable1);
executionPool.submit(runnable2);
}
private static class Wrapper {
TestIdentity t;
public Wrapper(TestIdentity t) {
this.t = t;
}
public void work(Dao dao) {
TestIdentity t2 = new TestIdentity(t.getId(), t.getField(), t.isBit());
dao.saveObject(t2);
t.setBit(true);
dao.saveObject(t2);
t.setBit(false);
}
}
クラスを使用するもの
@Entity
@Table(name="cycling_scraped.test_identity")
public class TestIdentity {
int id;
String field;
boolean bit;
public TestIdentity() {
super();
}
public TestIdentity(int id, String field, boolean bit) {
super();
this.id = id;
this.field = field;
this.bit = bit;
}
@Id
@Column(name="pk_id")
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
@Column(name="field")
public String getField() {
return field;
}
public void setField(String field) {
this.field = field;
}
@Column(name="bit")
public boolean isBit() {
return bit;
}
public void setBit(boolean bit) {
this.bit = bit;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + (bit ? 1231 : 1237);
result = prime * result + ((field == null) ? 0 : field.hashCode());
result = prime * result + id;
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
TestIdentity other = (TestIdentity) obj;
if (bit != other.bit)
return false;
if (field == null) {
if (other.field != null)
return false;
} else if (!field.equals(other.field))
return false;
if (id != other.id)
return false;
return true;
}
@Override
public String toString() {
return "TestIdentity [id=" + id + ", field=" + field + ", bit=" + bit
+ "]";
}
}
そしてテーブル
CREATE TABLE test_database.test_identity (
pk_id INT NOT NULL PRIMARY KEY,
field VARCHAR(100),
bit BIT
);