質問をすることでしか得られない、回答やアドバイスがある。

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

ただいまの
回答率

90.11%

MySQLで特定の行を削除できないようにするにはどうすれば良いですか?

解決済

回答 4

投稿

  • 評価
  • クリップ 0
  • VIEW 991

qwe001

score 83

以下のようなテーブル定義があります。

CREATE TABLE categories (
  id int(11) NOT NULL AUTO_INCREMENT,
  title varchar(128) NOT NULL UNIQUE,
  PRIMARY KEY (id)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

INSERT INTO categories (title) VALUES
  ('未選択'),
  ('たまご'),
  ('やさい'),
  ('おにく')
;

諸事情あって、あるマスタデータの初期値(「未選択」)をテーブルに格納することになりました。
この初期値を用いてアプリケーションの条件分岐を書きますので、初期値がDBから削除されると色々都合が悪いです。

ので、この最初の1行目(id 1)だけは削除されないようにロックしたいです。(更新はできてもいいし、できなくてもいいです)

どのようにすれば、行削除を禁止することが出来ますでしょうか?
あるいは、SQLではそのようなことはできませんか?(アプリケーション側の記述で対応するべき?)

皆様ご助言の程よろしくお願いいたします。

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 4

+3

自分も「未選択」をレコードとして持つのは非常にナンセンスだと思います。

  • フロントエンドではレコードの存在によらず <option value="">未選択</option> を表示
  • バックエンドでは空文字列が送られてきたら無かったものとみなす

が正しい実装だと思います。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/02/05 10:50

    「諸事情」がどういうものかは非常に気になりますが,「上からの指示」とかだったら問答無用で異議あり案件ですね…

    キャンセル

  • 2019/02/05 11:01

    私は勝手に推測で書いていますが「未選択も許可して保存しよう」となったとか・・。それでもわざわざ持つ必要はないと思います。

    キャンセル

checkベストアンサー

+2

どこからどのように利用する想定か具体的に書かれていないのでわかりませんが、「未選択」自体をDBに持つ必要はないと思います。
もし、その選択肢の保存先で「未選択」を許可しているのならnullなり空なりを入れればいいだけで、基本は利用するプログラム側で付加すればいいかなと。
諸事情がどんな事情かわかりませんが、「未選択」は「実際の選択肢ではない」という認識です。
どうしてもというのでしたら、もしマスタを操作(追加や編集、削除)する画面から利用するのでしたら、データ取得時にid!=1でそもそも画面から操作できないようにするしかないかと。
念のために削除機能にもid=1が指定されたときはエラーを返すように対処は必要ですね。「編集できても良い」とのことですが、編集できる=削除できる ですので。

あと、これは蛇足ですが、実際はその選択肢を利用するテーブルが別にあるのでしょうし、外部キーを貼るなどして既に別で利用されている選択肢が削除されないように、番号がずれないようにする配慮は必要です。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

+2

どのようにすれば、行削除を禁止することが出来ますでしょうか?

MySQLレベルでもトリガーを仕掛ければ可能ですが、やることに対して大げさすぎる気がしますし、MySQLだけで処置をした場合、失敗すればデータベースエラーとなってしまいますので、結局アプリケーション側での対応も必要です。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

+2

諸事情あって、あるマスタデータの初期値(「未選択」)をテーブルに格納することになりました。
この初期値を用いてアプリケーションの条件分岐を書きますので、初期値がDBから削除されると色々都合が悪いです。

問題なのはcategoriesではなくて、categoriesを外部参照しているテーブルに未選択に該当するIDを設定しないと駄目な作りになっている事です。

外部参照側ではその関係が確定しない(今回なら未選択)時にはNullとするのが得策でしょう。
その場合、categoriesには未選択自体の登録を止めるの必要がありますね。

未選択を持たせるのならIDAUTO_INCREMENTをやめて、外部参照側で初期値として設定可能なように0などの固定値にする必要があります。
↑これだと削除しないようにするという問題が残りますので、得策ではないですね。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

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

  • ただいまの回答率 90.11%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる