🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

16312閲覧

【Django】model の主キーにAutoFieldを設定しても自動採番してくれない

azuapricot

総合スコア2341

Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

1グッド

1クリップ

投稿2019/11/06 08:30

編集2019/11/07 02:18

いつもお世話になっております。

早速本題に入らせていただきます。

現在、DjangoとPythonを使用してWebアプリケーションを制作しているのですが、

データが既に登録されているデータベースに対してレコードを新規登録する際、
AutoField にしたIDが 自動採番されずキー重複エラー が出てしまいます。

以下が現在のコードとなります。


models.py

Python

1class TestMaster(models.Model): 2 class Meta: 3 db_table = 'test_master' 4 5 test_id = models.AutoField(verbose_name="ID", primary_key=True) 6 title = models.CharField(verbose_name="タイトル", null=False) 7 content = models.CharField(verbose_name="本文", null=False) 8 creator = models.CharField(verbose_name="作成者" null=False)

views.py やったことその1

Python

1TestMaster.objects.create(title="title", 2 content="content", 3 creator="ADMIN" 4 )

views.py やったことその2

Python

1b2 = TestMaster(title="title", 2 content="content", 3 creator="ADMIN" 4 ) 5b2.save()

** 出力されたエラー文**

重複キーが一意性制約"test_master_pkey"に違反しています DETAIL: キー (test_id)=(1) はすでに存在します。

参考にしたサイト1
参考にしたサイト2

DBに直接レコードを追加してしまったがためにAutoFieldが機能していないのか、
それともただ単に書き方が悪いのか・・・

お時間ある方でAutoField設定項目について知識をお持ちの方いらっしゃいましたらご助力頂けますと幸いです。


調査した結果の追記
Django ってDBを直接編集することを想定してないのか・・・?
どうやったら同期とれるんだ・・・
登録する前にidの最大値とってくるしかないのか・・・?


かなしみの暫定対応

Python

1entry = TestMaster.objects.all().aggregate(Max('test_id')) 2 3 TestMaster.objects.create(test_id=entry["test_id__max"] + 1, 4 title=title, 5 content=content, 6 creator=user 7 )

排他制御入れないとだめそう?だけど一応これなら自動採番もどきはできる・・・
けど自動採番したい・・・


「Djangoでは、モデルのプライマリーキーがシーケンスによって決まります。」

Django-モデルのプライマリキーをUUIDにする

シーケンス・・・?と思ってDBを見てみれば確かに自動採番するときにシーケンスの次の番号を入れようとしてるらしい。

じゃあ手運用後にシーケンスの値合わせてあげれば解決するのでは?
シーケンスの値を変更をするには

SELECT setval('test_master_test_id_seq', (SELECT MAX(test_id) FROM test_master));

Python

1TestMaster.objects.create(title="title", 2 content="content", 3 creator="ADMIN" 4 )

DBに ID=101 で登録された!


結論

  • pgAdminで手動でレコード追加する場合は シーケンスを更新してあげないといけない
  • ↑が嫌なら、自動採番じゃなくてUUIDなどに変更すべき

個人的にはシーケンス更新が楽な気がするのでとりあえず解決

同じ事象で困っている方の役に立ちますように。

Nama29😄を押しています

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

僕もDjango勉強中で、ちょうど同じ問題に遭遇しました。。。

確かに、DBに直接書き込んだらserial(?)がインクリメントされないと思うので、
同様のエラーが出るような気がします。

ちょっと趣旨から逸れるかもですが、↓の一番下の方法でも出来そうですね。
https://torina.top/detail/230/

投稿2019/11/06 14:08

reekun

総合スコア10

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

azuapricot

2019/11/07 01:54

回答ありがとうございます、質問本文に参考になるかはわかりませんが解決方法記載しましたので良ければご覧ください。
guest

0

自己解決

「Djangoでは、モデルのプライマリーキーがシーケンスによって決まります。」

解決方法は質問本文下部に記載しました。

同じ事象にぶち当たった方のお役に立ちますように。

投稿2019/11/07 01:55

azuapricot

総合スコア2341

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問