MySQL - ALTER TABLE
概要
ALTER TABLE 文は、既存のテーブル構造を変更するためのDDL (Data Definition Language) 文である。
この文を使用することで、カラムの追加や削除、データ型の変更、制約の追加や削除、テーブル名の変更、ストレージエンジンの変更等、様々なテーブル構造の変更が可能である。
MySQL 8.0以降では、INSTANT DDLという強力な機能が導入され、多くの操作がメタデータの変更のみで完了し、秒以下で実行できるようになった。
従来はテーブル全体の再構築 (COPY) が必要だった操作でも、INPLACE方式やINSTANT方式で実行できるようになり、大規模テーブルに対するスキーマ変更が大幅に高速化された。
ALTER TABLE 文は、INSTANT、INPLACE、COPYの3つのアルゴリズムを使用する。
INSTANT方式は最も高速でメタデータのみを変更し、INPLACE方式はテーブルを部分的に再構築してDMLを一部許可し、
COPY方式は完全なテーブル再構築を行いDMLを完全にブロックする。
操作の種類によって、どのアルゴリズムが使用されるかが異なり、パフォーマンスへの影響も大きく変わる。
大規模なテーブルに対してスキーマ変更を行う場合は、実行前にアルゴリズムとロック方式を理解しておくことが重要である。
基本構文
ALTER TABLEの基本構文
ALTER TABLE 文は、テーブル名の後に変更操作を指定する。
# 基本構文
ALTER TABLE table_name
operation1,
operation2,
...;
# 単一の操作
ALTER TABLE employees ADD COLUMN email VARCHAR(255);
# 複数の操作を1文で実行 (カンマ区切り)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ADD COLUMN phone VARCHAR(20),
MODIFY COLUMN salary DECIMAL(10, 2);
複数の操作を1文にまとめることで、テーブルの再構築が1回で済み、パフォーマンスが向上する場合がある。
ALGORITHM と LOCK句
MySQL 8.0以降では、ALGORITHM 句 と LOCK 句を使用して、実行方法を明示的に指定できる。
# ALGORITHMを指定
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ALGORITHM=INSTANT;
# LOCKを指定
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
LOCK=NONE;
# 両方を指定
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ALGORITHM=INSTANT,
LOCK=NONE;
| 指定値 | 説明 | 実行時間の目安 |
|---|---|---|
| INSTANT | メタデータのみ変更、ディスク変更最小 | 秒以下 |
| INPLACE | テーブルを部分的に再構築、DML部分許可 | 分〜数時間 |
| COPY | テーブル全体を完全に再構築、DMLブロック | 数時間〜数日 |
| DEFAULT | MySQLが自動選択 (INSTANT -> INPLACE -> COPYの順に試行) | 操作により異なる |
| 指定値 | 説明 |
|---|---|
| NONE | ロックなし、SELECT / INSERT / UPDATE / DELETE 全て許可 |
| SHARED | 共有ロック、SELECT許可、DML (INSERT / UPDATE / DELETE) はブロック |
| EXCLUSIVE | 排他ロック、全ての操作をブロック |
| DEFAULT | MySQLが自動選択 (可能な限り緩いロックを使用) |
ALGORITHM を指定しない場合、MySQLは自動的に最適なアルゴリズムを選択する。
指定したアルゴリズムで操作が実行できない場合は、エラーが発生する。
# ALGORITHM=INSTANTを要求するが、INSTANTで実行できない操作の場合はエラー
ALTER TABLE employees
DROP COLUMN email,
ALGORITHM=INSTANT;
# MySQL 8.0.29未満ではエラー: ALGORITHM=INSTANT is not supported.
カラム操作
ADD COLUMN : カラム追加
ADD COLUMN は、テーブルに新しいカラムを追加する。
MySQL 8.0.12以降では、デフォルトで ALGORITHM=INSTANT が使用され、秒以下で実行される。
# 基本的なカラム追加 (テーブル末尾に追加)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255);
# デフォルト値を指定
ALTER TABLE employees
ADD COLUMN status VARCHAR(20) DEFAULT 'active';
# NOT NULL制約を指定
ALTER TABLE employees
ADD COLUMN department_id INT NOT NULL;
# 複数のカラムを同時に追加
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ADD COLUMN phone VARCHAR(20),
ADD COLUMN address TEXT;
カラムの位置を指定することもできる。
# テーブルの先頭にカラムを追加
ALTER TABLE employees
ADD COLUMN employee_code VARCHAR(10) FIRST;
# 特定のカラムの後にカラムを追加
ALTER TABLE employees
ADD COLUMN middle_name VARCHAR(50) AFTER first_name;
# 複数のカラムを異なる位置に追加
ALTER TABLE employees
ADD COLUMN employee_code VARCHAR(10) FIRST,
ADD COLUMN email VARCHAR(255) AFTER last_name;
MySQL 8.0.12以降では、FIRST や AFTER を指定しても、ALGORITHM=INSTANT で実行される。
ただし、テーブルには以下に示す制限がある。
- 最大1022カラムまで
- この制限を超えると、INSTANT追加できなくなる。
- 最大64行バージョンまで
- INSTANTでカラムを追加するたびに行バージョンが増加する。
- 64回を超えるとテーブル再構築が必要
# 行バージョンの確認
SELECT TABLE_NAME, TOTAL_ROW_VERSIONS
FROM information_schema.INNODB_TABLES
WHERE NAME = 'database_name/table_name';
DROP COLUMN : カラム削除
DROP COLUMN は、テーブルからカラムを削除する。
MySQL 8.0.29以降では、ALGORITHM=INSTANT で実行される。
MySQL 8.0.28以前では、ALGORITHM=INPLACE が使用される。
# カラムを削除
ALTER TABLE employees
DROP COLUMN email;
# 複数のカラムを削除
ALTER TABLE employees
DROP COLUMN email,
DROP COLUMN phone;
# COLUMNキーワードを省略 (可能だが非推奨)
ALTER TABLE employees
DROP email;
カラム削除には以下の制限がある。
- テーブルの最後の1列は削除できない
- 少なくとも1つのカラムが必要。
- 外部キー制約で参照されているカラムは削除できない
- 先に外部キー制約を削除する必要がある。
- インデックスに含まれるカラムは削除できない
- 先にインデックスを削除する必要がある。
# エラー例: 最後の1列を削除
ALTER TABLE single_column_table
DROP COLUMN only_column;
# ERROR 1090 (42000): You can't delete all columns with ALTER TABLE
# 外部キー制約とインデックスを削除してからカラムを削除
ALTER TABLE orders
DROP FOREIGN KEY fk_user_id,
DROP INDEX idx_user_id,
DROP COLUMN user_id;
MODIFY COLUMN : データ型と制約の変更
MODIFY COLUMN は、カラムのデータ型や制約を変更する。
カラム名は変更しない。
カラム名を変更する場合は、RENAME COLUMN または CHANGE COLUMN を使用する。
# データ型を変更
ALTER TABLE employees
MODIFY COLUMN salary DECIMAL(12, 2);
# NOT NULL制約を追加
ALTER TABLE employees
MODIFY COLUMN email VARCHAR(255) NOT NULL;
# NOT NULL制約を削除 (NULLを許可)
ALTER TABLE employees
MODIFY COLUMN email VARCHAR(255);
# デフォルト値を変更
ALTER TABLE employees
MODIFY COLUMN status VARCHAR(20) DEFAULT 'inactive';
# 複数のカラムを変更
ALTER TABLE employees
MODIFY COLUMN salary DECIMAL(12, 2),
MODIFY COLUMN email VARCHAR(255) NOT NULL;
データ型の変更は、互換性のある型への変更であれば ALGORITHM=INSTANT で実行される場合がある。
非互換な変更 (例: VARCHAR(50)型からINT型への変更) は、テーブルの再構築が必要となり、ALGORITHM=COPY が使用される。
# 互換性のある変更 (INSTANT可能)
ALTER TABLE employees
MODIFY COLUMN name VARCHAR(255); # VARCHAR(100) から VARCHAR(255)
# 非互換な変更 (COPY必要)
ALTER TABLE employees
MODIFY COLUMN department_id VARCHAR(10); # INT から VARCHAR
MODIFY COLUMN では、変更しない属性も含めて完全な定義を記述する必要がある。
# 元の定義: email VARCHAR(255) NOT NULL DEFAULT ''
# デフォルト値のみを変更したい場合でも、完全な定義を記述
ALTER TABLE employees
MODIFY COLUMN email VARCHAR(255) NOT NULL DEFAULT 'unknown@example.com';
# NOT NULLを省略すると、NULLを許可する定義になる
ALTER TABLE employees
MODIFY COLUMN email VARCHAR(255) DEFAULT 'unknown@example.com';
# この場合、NOT NULL制約が削除される
CHANGE COLUMN : カラム名とデータ型の変更
CHANGE COLUMN は、カラム名とデータ型の両方を変更できる。
ただし、カラム名のみを変更する場合は、RENAME COLUMN を使用することが推奨される。
# カラム名とデータ型を変更
ALTER TABLE employees
CHANGE COLUMN old_name new_name VARCHAR(100);
# カラム名のみを変更 (データ型は同じ)
ALTER TABLE employees
CHANGE COLUMN email email_address VARCHAR(255);
# データ型も同時に変更
ALTER TABLE employees
CHANGE COLUMN salary monthly_salary DECIMAL(12, 2);
CHANGE COLUMN は、旧形式のコマンドである。
MySQL 8.0以降では、カラム名の変更には RENAME COLUMN を使用することが推奨される。
# 新形式 : RENAME COLUMN (推奨)
ALTER TABLE employees
RENAME COLUMN email TO email_address;
# 旧形式 : CHANGE COLUMN (非推奨)
ALTER TABLE employees
CHANGE COLUMN email email_address VARCHAR(255);
RENAME COLUMN : カラム名の変更 (MySQL 8.0+)
RENAME COLUMN は、MySQL 8.0以降で使用可能なカラム名変更の専用構文である。
ALGORITHM=INSTANT で実行され、データ型や制約は変更されない。
# カラム名を変更
ALTER TABLE employees
RENAME COLUMN email TO email_address;
# 複数のカラム名を変更
ALTER TABLE employees
RENAME COLUMN first_name TO given_name,
RENAME COLUMN last_name TO family_name;
RENAME COLUMN は、CHANGE COLUMN より簡潔で、意図が明確である。
また、データ型を再指定する必要がないため、エラーが発生しにくい。
ALTER COLUMN SET/DROP DEFAULT : デフォルト値の変更
ALTER COLUMN SET DEFAULT は、カラムのデフォルト値を設定する。
ALTER COLUMN DROP DEFAULT は、カラムのデフォルト値を削除する。
どちらも ALGORITHM=INSTANT で実行される。
# デフォルト値を設定
ALTER TABLE employees
ALTER COLUMN status SET DEFAULT 'active';
# デフォルト値を削除
ALTER TABLE employees
ALTER COLUMN status DROP DEFAULT;
# 複数のカラムのデフォルト値を変更
ALTER TABLE employees
ALTER COLUMN status SET DEFAULT 'active',
ALTER COLUMN salary SET DEFAULT 0;
ALTER COLUMN は、デフォルト値の変更のみを行う。
データ型やNOT NULL制約を変更する場合は、MODIFY COLUMN を使用する。
テーブル操作
RENAME TABLE / RENAME TO : テーブル名の変更
RENAME TO は、テーブル名を変更する。
ALGORITHM=INSTANT で実行され、メタデータのみが変更される。
# テーブル名を変更
ALTER TABLE old_table_name
RENAME TO new_table_name;
# RENAME TOの代わりにRENAME ASも使用可能
ALTER TABLE old_table_name
RENAME AS new_table_name;
RENAME TABLE 文を使用することもできる。
# RENAME TABLE文 (推奨)
RENAME TABLE old_table_name TO new_table_name;
# 複数のテーブルを同時に変更
RENAME TABLE
old_table1 TO new_table1,
old_table2 TO new_table2,
old_table3 TO new_table3;
RENAME TABLE 文の方が簡潔で複数のテーブルを同時に変更できるため、推奨される。
ENGINE : ストレージエンジンの変更
ENGINE 句は、テーブルのストレージエンジンを変更する。
この操作は、ALGORITHM=COPY で実行され、テーブル全体が再構築される。
大規模なテーブルでは、非常に時間が掛かる可能性がある。
# ストレージエンジンをInnoDBに変更
ALTER TABLE employees
ENGINE=InnoDB;
# MyISAMからInnoDBへの変換 (一般的な用途)
ALTER TABLE old_myisam_table
ENGINE=InnoDB;
同じストレージエンジンを指定した場合でも、テーブルの再構築が実行される。
これを利用して、テーブルの最適化やディスク領域の再利用が可能である。
# テーブルの再構築による最適化
ALTER TABLE employees
ENGINE=InnoDB;
# 既にInnoDBでも、再構築される
ただし、OPTIMIZE TABLE 文を使用する方が意図が明確である。
# 推奨: OPTIMIZE TABLE
OPTIMIZE TABLE employees;
CONVERT TO CHARACTER SET : 文字セットの変換
CONVERT TO CHARACTER SET は、テーブルの文字セットとコレーションを変換する。
この操作は、全ての文字列カラム (CHAR, VARCHAR, TEXT) のデータを変換し、ALGORITHM=COPY で実行される。
# テーブル全体の文字セットをutf8mb4に変換
ALTER TABLE employees
CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
# コレーションを省略 (デフォルトのコレーションが使用される)
ALTER TABLE employees
CONVERT TO CHARACTER SET utf8mb4;
この操作は、既存のデータを変換するため、時間がかかる可能性がある。
また、文字セット変換により、一部の文字が失われる可能性もある (例: latin1からutf8への変換)。
DEFAULT CHARACTER SET : デフォルト文字セットの変更
DEFAULT CHARACTER SET は、テーブルのデフォルト文字セットを変更する。
この操作は、既存のカラムの文字セットは変更せず、新しく追加されるカラムのデフォルト文字セットのみを変更する。
ALGORITHM=INSTANT で実行される。
# デフォルト文字セットを変更 (既存カラムは影響なし)
ALTER TABLE employees
DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
# 既存カラムの文字セットは変わらない
# 今後追加されるカラムはutf8mb4になる
既存のカラムも含めて文字セットを変換する場合は、CONVERT TO CHARACTER SET を使用する。
# 比較: CONVERT TO CHARACTER SET (既存カラムも変換)
ALTER TABLE employees
CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
AUTO_INCREMENT : 次のシーケンス値の設定
AUTO_INCREMENT 句は、次に挿入される行の自動インクリメント値を設定する。
# 次のAUTO_INCREMENT値を1000に設定
ALTER TABLE employees
AUTO_INCREMENT = 1000;
# 現在の値より小さい値は無視される
ALTER TABLE employees
AUTO_INCREMENT = 1;
# 現在のAUTO_INCREMENT値が既に100の場合、この操作は無視される
現在の AUTO_INCREMENT 値を確認するには、SHOW CREATE TABLE を使用する。
# 現在のAUTO_INCREMENT値を確認
SHOW CREATE TABLE employees;
制約操作
PRIMARY KEY : 主キーの追加と削除
ADD PRIMARY KEY は、テーブルに主キーを追加する。
DROP PRIMARY KEY は、テーブルから主キーを削除する。
# 主キーを追加
ALTER TABLE employees
ADD PRIMARY KEY (id);
# 複合主キーを追加
ALTER TABLE order_items
ADD PRIMARY KEY (order_id, product_id);
# 主キーを削除
ALTER TABLE employees
DROP PRIMARY KEY;
主キーの追加は、ALGORITHM=INPLACE で実行される。
主キーの削除のみは、ALGORITHM=COPY が必要となる。
ただし、主キーの削除と追加を同時に行う場合は、ALGORITHM=INPLACE で実行される。
# 主キーを削除してから追加 (INPLACE可能)
ALTER TABLE employees
DROP PRIMARY KEY,
ADD PRIMARY KEY (employee_code);
主キーを追加する場合、カラムは NOT NULL である必要がある。
# エラー例: NULLを許可するカラムに主キーを追加
ALTER TABLE employees
ADD PRIMARY KEY (email);
# ERROR 1171 (42000): All parts of a PRIMARY KEY must be NOT NULL
# 先にNOT NULL制約を追加
ALTER TABLE employees
MODIFY COLUMN email VARCHAR(255) NOT NULL,
ADD PRIMARY KEY (email);
UNIQUE / INDEX : 一意制約とインデックスの操作
ADD UNIQUE は、一意制約 (ユニークインデックス) を追加する。
ADD INDEX は、通常のインデックスを追加する。
# 一意制約を追加
ALTER TABLE employees
ADD UNIQUE (email);
# 名前を指定して一意制約を追加
ALTER TABLE employees
ADD UNIQUE INDEX idx_email (email);
# 複合一意制約を追加
ALTER TABLE employees
ADD UNIQUE (department_id, employee_code);
# 通常のインデックスを追加
ALTER TABLE employees
ADD INDEX idx_department (department_id);
# 複合インデックスを追加
ALTER TABLE employees
ADD INDEX idx_dept_salary (department_id, salary);
インデックスの削除は、DROP INDEX を使用する。
# インデックスを削除
ALTER TABLE employees
DROP INDEX idx_email;
# 複数のインデックスを削除
ALTER TABLE employees
DROP INDEX idx_email,
DROP INDEX idx_department;
MySQL 8.0以降では、RENAME INDEX でインデックス名を変更できる。
# インデックス名を変更 (MySQL 8.0+)
ALTER TABLE employees
RENAME INDEX old_index_name TO new_index_name;
また、不可視インデックス (Invisible Index) の設定も可能である。
# インデックスを不可視にする (MySQL 8.0+)
ALTER TABLE employees
ALTER INDEX idx_email INVISIBLE;
# インデックスを可視にする
ALTER TABLE employees
ALTER INDEX idx_email VISIBLE;
不可視インデックスは、オプティマイザーによって使用されなくなるが、インデックス自体は維持される。
インデックスの削除前に、パフォーマンスへの影響を確認するために使用できる。
FOREIGN KEY : 外部キー制約の操作
ADD FOREIGN KEY は、外部キー制約を追加する。
DROP FOREIGN KEY は、外部キー制約を削除する。
# 外部キー制約を追加
ALTER TABLE orders
ADD FOREIGN KEY (user_id) REFERENCES users(id);
# 名前を指定して外部キー制約を追加
ALTER TABLE orders
ADD CONSTRAINT fk_user_id
FOREIGN KEY (user_id) REFERENCES users(id);
# ON DELETE / ON UPDATE句を指定
ALTER TABLE orders
ADD CONSTRAINT fk_user_id
FOREIGN KEY (user_id) REFERENCES users(id)
ON DELETE CASCADE
ON UPDATE CASCADE;
# 複合外部キー制約を追加
ALTER TABLE order_items
ADD CONSTRAINT fk_order
FOREIGN KEY (order_id, product_id) REFERENCES orders(id, product_id);
外部キー制約の削除は、制約名を指定する。
# 外部キー制約を削除
ALTER TABLE orders
DROP FOREIGN KEY fk_user_id;
外部キー制約名を確認するには、SHOW CREATE TABLE を使用する。
# 外部キー制約名を確認
SHOW CREATE TABLE orders;
# または、INFORMATION_SCHEMAから確認
SELECT CONSTRAINT_NAME, TABLE_NAME, REFERENCED_TABLE_NAME
FROM information_schema.KEY_COLUMN_USAGE
WHERE TABLE_SCHEMA = 'database_name'
AND TABLE_NAME = 'orders'
AND REFERENCED_TABLE_NAME IS NOT NULL;
外部キー制約の参照動作 (ON DELETE / ON UPDATE) には、以下のオプションがある。
CASCADE- 親レコードの削除 / 更新時、子レコードも削除/更新する。
SET NULL- 親レコードの削除 / 更新時、子レコードの外部キーを
NULLに設定する。
- 親レコードの削除 / 更新時、子レコードの外部キーを
RESTRICT- 親レコードの削除 / 更新を禁止する。(デフォルト)
NO ACTIONRESTRICTと同じ。(SQL標準)
SET DEFAULT- MySQL 8.0ではサポートされていない。
詳細は、MySQL - 外部キーのページを参照すること。
CHECK制約 (MySQL 8.0.16+)
ADD CHECK は、チェック制約を追加する。
DROP CHECK は、チェック制約を削除する。
チェック制約は、MySQL 8.0.16以降で使用可能である。
# チェック制約を追加
ALTER TABLE employees
ADD CHECK (salary >= 0);
# 名前を指定してチェック制約を追加
ALTER TABLE employees
ADD CONSTRAINT chk_salary CHECK (salary >= 0);
# 複雑な条件のチェック制約
ALTER TABLE employees
ADD CONSTRAINT chk_email CHECK (email LIKE '%@%');
# 複数カラムを参照するチェック制約
ALTER TABLE employees
ADD CONSTRAINT chk_dates CHECK (hire_date <= termination_date);
チェック制約の削除は、制約名を指定する。
# チェック制約を削除
ALTER TABLE employees
DROP CHECK chk_salary;
チェック制約は、ENFORCED または NOT ENFORCED を指定できる。
# チェック制約を無効化
ALTER TABLE employees
ALTER CHECK chk_salary NOT ENFORCED;
# チェック制約を有効化
ALTER TABLE employees
ALTER CHECK chk_salary ENFORCED;
チェック制約名を確認するには、INFORMATION_SCHEMA を使用する。
# チェック制約を確認
SELECT CONSTRAINT_NAME, CHECK_CLAUSE
FROM information_schema.CHECK_CONSTRAINTS
WHERE CONSTRAINT_SCHEMA = 'database_name'
AND TABLE_NAME = 'employees';
MySQL 8.0以降の新機能
INSTANT DDL
MySQL 8.0以降では、INSTANT DDL という機能により、多くのスキーマ変更がメタデータのみの変更で完了する。
INSTANT DDL は、テーブルデータを物理的に変更せず、秒以下で完了する。
大規模なテーブルに対するスキーマ変更が劇的に高速化された。
| 操作 | MySQL 8.0.12+ | MySQL 8.0.29+ |
|---|---|---|
| ADD COLUMN | 対応 | 対応 |
| DROP COLUMN | 非対応 (INPLACE) | 対応 |
| RENAME COLUMN | 対応 | 対応 |
| ALTER COLUMN SET/DROP DEFAULT | 対応 | 対応 |
| ENUM / SET型への値追加 | 対応 | 対応 |
| RENAME TABLE | 対応 | 対応 |
| DEFAULT CHARACTER SET | 対応 | 対応 |
INSTANT DDL の実行例を以下に示す。
# カラム追加 (INSTANT)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ALGORITHM=INSTANT;
# 実行時間: 約0.1秒 (65万行のテーブル)
# カラム削除 (MySQL 8.0.29+でINSTANT)
ALTER TABLE employees
DROP COLUMN email,
ALGORITHM=INSTANT;
# 実行時間: 約0.1秒 (65万行のテーブル)
# デフォルト値変更 (INSTANT)
ALTER TABLE employees
ALTER COLUMN status SET DEFAULT 'active',
ALGORITHM=INSTANT;
# 実行時間: 秒以下
INSTANT DDL の制限事項を以下に示す。
- 最大1022カラムまで
- InnoDBの最大カラム数制限。
- 最大64行バージョンまで
- INSTANTでカラムを追加するたびに行バージョンが増加する。
- 64回を超えるとテーブル再構築が必要。
- 一部のデータ型変更はINSTANT非対応
- 非互換なデータ型変更はCOPYが必要。
行バージョンを確認して、64回の制限に近づいている場合は、テーブルを再構築する。
# 行バージョンの確認
SELECT TABLE_NAME, TOTAL_ROW_VERSIONS
FROM information_schema.INNODB_TABLES
WHERE NAME = 'database_name/employees';
# テーブル再構築により行バージョンをリセット
ALTER TABLE employees
ENGINE=InnoDB,
ALGORITHM=COPY;
不可視カラム (Invisible Column)
MySQL 8.0.23以降では、不可視カラム (Invisible Column) を作成できる。
不可視カラムは、SELECT * から除外されるが、明示的に指定すれば取得できる。
# 不可視カラムを追加
ALTER TABLE employees
ADD COLUMN internal_notes TEXT INVISIBLE;
# 既存カラムを不可視にする
ALTER TABLE employees
MODIFY COLUMN internal_id INT INVISIBLE;
# 不可視カラムを可視にする
ALTER TABLE employees
MODIFY COLUMN internal_notes TEXT VISIBLE;
不可視カラムの動作確認を以下に示す。
# SELECT * では不可視カラムは表示されない
SELECT * FROM employees;
# internal_notesカラムは表示されない
# 明示的に指定すれば取得できる
SELECT id, name, internal_notes FROM employees;
# internal_notesカラムが表示される
不可視カラムは、以下のような用途で使用される。
- 内部的なメタデータの保存
- アプリケーションから隠したい情報を保存。
- 段階的なスキーマ変更
- 新しいカラムを不可視で追加し、テスト後に可視にする。
- 後方互換性の維持
- SELECT * を使用する既存コードに影響を与えずにカラムを追加。
その他のMySQL 8.0新機能
MySQL 8.0では、ALTER TABLE に関連する多くの機能が追加された。
RENAME INDEX(8.0+)- インデックス名を変更。
ALTER INDEX INVISIBLE/VISIBLE(8.0+)- インデックスを不可視/可視に変更。
CHECK制約 (8.0.16+)- チェック制約の追加と管理。
INSTANT DDLの拡張 (8.0.12+, 8.0.29+)- より多くの操作が
INSTANTに対応。
- より多くの操作が
- 式インデックス (8.0.13+)
- 関数や式に対するインデックスの作成。
# 式インデックスの作成 (MySQL 8.0.13+)
ALTER TABLE employees
ADD INDEX idx_upper_name ((UPPER(name)));
# 関数インデックスの作成
ALTER TABLE orders
ADD INDEX idx_year ((YEAR(ordered_at)));
パフォーマンスとアルゴリズム
INSTANT / INPLACE / COPYの違い
ALTER TABLE 文は、3つのアルゴリズムを使用する。
| アルゴリズム | 実行時間 | ディスク使用量 | ロック | DML許可 |
|---|---|---|---|---|
| INSTANT | 秒以下 | 最小 (メタデータのみ) | なし | 完全許可 |
| INPLACE | 分〜数時間 | 中程度 (一時ファイル) | 短時間のみ | 部分的に許可 |
| COPY | 最長 (数時間〜数日) | 大 (テーブル2倍以上) | 完全ロック | 完全ブロック |
パフォーマンスの実測例 (65万行のテーブル) を以下に示す。
- INSTANT (カラム追加)
- 約0.1秒
- INPLACE (インデックス追加)
- 約30秒〜数分
- COPY (ストレージエンジン変更)
- 約4.5時間
# INSTANTで実行 (最速)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ALGORITHM=INSTANT;
# INPLACEで実行 (中速)
ALTER TABLE employees
ADD INDEX idx_email (email),
ALGORITHM=INPLACE;
# COPYで実行 (最遅)
ALTER TABLE employees
ENGINE=InnoDB,
ALGORITHM=COPY;
MySQLは、ALGORITHM を指定しない場合、自動的に最適なアルゴリズムを選択する。
優先順位は、INSTANT -> INPLACE -> COPY の順である。
# ALGORITHMを指定しない (自動選択)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255);
# INSTANTで実行される
ALTER TABLE employees
ADD INDEX idx_email (email);
# INPLACEで実行される
ALTER TABLE employees
ENGINE=InnoDB;
# COPYで実行される
特定のアルゴリズムを強制する場合は、ALGORITHM 句を指定する。
指定したアルゴリズムで実行できない場合は、エラーが発生する。
# INSTANT強制 (実行できない場合はエラー)
ALTER TABLE employees
DROP COLUMN email,
ALGORITHM=INSTANT;
# MySQL 8.0.28以前ではエラー
# INPLACE強制 (実行できない場合はエラー)
ALTER TABLE employees
ENGINE=InnoDB,
ALGORITHM=INPLACE;
# COPYが必要な操作のためエラー
LOCKの種類と影響
LOCK 句は、ALTER TABLE 実行中のテーブルロックレベルを指定する。
| LOCKレベル | SELECT | INSERT | UPDATE | DELETE |
|---|---|---|---|---|
| NONE | 許可 | 許可 | 許可 | 許可 |
| SHARED | 許可 | ブロック | ブロック | ブロック |
| EXCLUSIVE | ブロック | ブロック | ブロック | ブロック |
| DEFAULT | MySQLが自動選択 | - | - | - |
# LOCK=NONE (最も緩い、DML許可)
ALTER TABLE employees
ADD INDEX idx_email (email),
LOCK=NONE;
# SELECT/INSERT/UPDATE/DELETE全て許可
# LOCK=SHARED (SELECTのみ許可)
ALTER TABLE employees
ADD INDEX idx_email (email),
LOCK=SHARED;
# SELECTは許可、DMLはブロック
# LOCK=EXCLUSIVE (完全ロック)
ALTER TABLE employees
ENGINE=InnoDB,
LOCK=EXCLUSIVE;
# 全ての操作をブロック
LOCK=NONE を指定した場合、操作が LOCK=NONE で実行できなければエラーになる。
# LOCK=NONEを強制
ALTER TABLE employees
ENGINE=InnoDB,
LOCK=NONE;
# エラー: LOCK=NONE is not supported. Reason: Cannot change storage engine.
大規模テーブルへの対処
大規模なテーブルに対して ALTER TABLE を実行する場合は、以下の点に注意する。
- ディスク容量の確保
- COPY方式では、テーブルサイズの2倍以上のディスク容量が必要。
- 実行時間の見積もり
- 事前に小規模なテーブルで実行時間を測定。
- ロックとDMLのブロック時間
- アプリケーションへの影響を考慮。
- メタデータロックの確認
- 長時間実行中のトランザクションがあると、
ALTER TABLEが待機する。
- 長時間実行中のトランザクションがあると、
実行中のロックを確認する方法を以下に示す。
# 実行中のプロセスを確認
SHOW PROCESSLIST;
# InnoDBのトランザクション状態を確認
SELECT * FROM information_schema.INNODB_TRX;
# メタデータロックの待機を確認 (MySQL 8.0+)
SELECT * FROM performance_schema.metadata_locks;
大規模テーブルのスキーマ変更には、以下の戦略が有効である。
- INSTANT DDLの活用
- MySQL 8.0以降では、可能な限りINSTANTで実行する。
- オンラインDDLツールの使用
- pt-online-schema-change (Percona Toolkit) や gh-ost (GitHub) 等のツールを使用。
- レプリケーション環境での段階的実行
- スレーブから順に実行し、最後にマスターを切り替える。
- ダウンタイムの確保
- メンテナンス時間を設けて実行。
# pt-online-schema-changeの使用例
pt-online-schema-change --alter "ADD COLUMN email VARCHAR(255)" \
D=database_name,t=employees \
--execute
複合操作の最適化
複数の操作を1つの ALTER TABLE 文にまとめることで、パフォーマンスが向上する。
# 非推奨 : 複数のALTER TABLE文を実行
ALTER TABLE employees ADD COLUMN email VARCHAR(255);
ALTER TABLE employees ADD COLUMN phone VARCHAR(20);
ALTER TABLE employees ADD INDEX idx_email (email);
# テーブル再構築が3回発生する可能性
# 推奨 : 1つのALTER TABLE文にまとめる
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ADD COLUMN phone VARCHAR(20),
ADD INDEX idx_email (email);
# テーブル再構築が1回で済む
ただし、INSTANT 操作 と COPY 操作を混在させると、全体が COPY になる場合がある。
# INSTANT操作のみ (高速)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ADD COLUMN phone VARCHAR(20);
# 両方INSTANT、全体がINSTANT
# INSTANT操作とCOPY操作を混在 (低速)
ALTER TABLE employees
ADD COLUMN email VARCHAR(255), # INSTANT
ENGINE=InnoDB; # COPY
# 全体がCOPYになる
実行計画を確認するには、EXPLAIN は使用できないが、実行前にテストテーブルで試すことが推奨される。
実行例と使用パターン
基本的な使用例
実際の使用例を以下に示す。
# 例1 : 新しいカラムを追加してインデックスを作成
ALTER TABLE users
ADD COLUMN created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
ADD COLUMN updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
ADD INDEX idx_created (created_at);
# 例2 : カラムのデータ型を変更
ALTER TABLE products
MODIFY COLUMN price DECIMAL(12, 2) NOT NULL;
# 例3 : 外部キー制約を追加
ALTER TABLE orders
ADD CONSTRAINT fk_user_id
FOREIGN KEY (user_id) REFERENCES users(id)
ON DELETE CASCADE;
# 例4 : 主キーを変更
ALTER TABLE employees
DROP PRIMARY KEY,
ADD PRIMARY KEY (employee_code);
# 例5 : 文字セットを変換
ALTER TABLE articles
CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
スキーマ変更のベストプラクティス
スキーマ変更を安全に実行するためのベストプラクティスを以下に示す。
- 事前にバックアップを取得する
- スキーマ変更前に必ずバックアップを取得。
- テスト環境で事前に実行する
- 本番環境と同じデータ量で実行時間を測定。
- 実行時間を見積もる
- COPYが必要な場合は、十分な時間を確保。
- ディスク容量を確認する
- COPY方式では、テーブルサイズの2倍以上の空き容量が必要。
- メタデータロックに注意する
- 長時間実行中のトランザクションがあると、
ALTER TABLEが待機する。
- 長時間実行中のトランザクションがあると、
- アプリケーションへの影響を考慮する
- ロックによるダウンタイムや、レスポンス遅延の可能性。
# 実行前のチェックリスト
# 1. ディスク容量の確認
SHOW TABLE STATUS LIKE 'employees';
# 2. 実行中のトランザクションの確認
SELECT * FROM information_schema.INNODB_TRX;
# 3. テストテーブルで実行時間を測定
CREATE TABLE test_employees LIKE employees;
INSERT INTO test_employees SELECT * FROM employees LIMIT 10000;
ALTER TABLE test_employees ADD COLUMN email VARCHAR(255);
# 4. 本番実行
ALTER TABLE employees
ADD COLUMN email VARCHAR(255),
ALGORITHM=INSTANT,
LOCK=NONE;
関連ページ