DATABASE
ALTER DATABASE Syntax
1 | ALTER {DATABASE | SCHEMA} [db_name] |
ALTER DATABASE
enables you to change the overall
characteristics of a database. These characteristics are stored in the
data dictionary. To use ALTER DATABASE
, you need the
ALTER
privilege on the database. ALTER SCHEMA
is a synonym for ALTER DATABASE
.
The database name can be omitted from the first syntax, in which case the statement applies to the default database.
If you change the default character set or collation for a database, stored routines that use the database defaults must be dropped and recreated so that they use the new defaults. (In a stored routine, variables with character data types use the database defaults if the character set or collation are not specified explicitly.
CREATE DATABASE Syntax
1 | CREATE {DATABASE | SCHEMA} [IF NOT EXISTS] db_name |
CREATE DATABASE
creates a database with the given name.
To use this statement, you need the CREATE
privilege for
the database. CREATE SCHEMA
is a synonym for
CREATE DATABASE
.
An error occurs if the database exists and you did not specify
IF NOT EXISTS
.
CREATE DATABASE
is not permitted within a session that
has an active LOCK TABLES
statement.
create_specification
options specify database
characteristics. Database characteristics are stored in the data
dictionary.
- The
CHARACTER SET
clause specifies the default database character set. TheCOLLATE
clause specifies the default database collation.
DROP DATABASE Syntax
1 | DROP {DATABASE | SCHEMA} [IF EXISTS] db_name |
DROP DATABASE
drops all tables in the database and
deletes the database. Be very careful with this statement! To
use DROP DATABASE
, you need the DROP
privilege
on the database. DROP SCHEMA
is a synonym for
DROP DATABASE
.
Important
When a database is dropped, privileges granted specifically for the database are not automatically dropped. They must be dropped manually.
IF EXISTS
is used to prevent an error from occurring if
the database does not exist.
If the default database is dropped, the default database is unset
(the DATABASE()
function returns NULL
).
If you use DROP DATABASE
on a symbolically linked
database, both the link and the original database are deleted.
DROP DATABASE
returns the number of tables that were
removed.
The DROP DATABASE
statement removes from the given
database directory those files and directories that MySQL itself may
create during normal operation. This includes all files with the
extensions shown in the following list:
.BAK
.DAT
.HSH
.MRG
.MYD
.MYI
.cfg
.db
.ibd
.ndb
If other files or directories remain in the database directory after
MySQL removes those just listed, the database directory cannot be
removed. In this case, you must remove any remaining files or
directories manually and issue the DROP DATABASE
statement
again.
Dropping a database does not remove any TEMPORARY
tables
that were created in that database. TEMPORARY
tables are
automatically removed when the session that created them ends.
You can also drop databases with mysqladmin.
TABLE
ALTER TABLE Syntax
1 | ALTER TABLE tbl_name |
ALTER TABLE
changes the structure of a table. For
example, you can add or delete columns, create or destroy indexes,
change the type of existing columns, or rename columns or the table
itself. You can also change characteristics such as the storage engine
used for the table or the table comment.
To use
ALTER TABLE
, you needALTER
,CREATE
, andINSERT
privileges for the table. Renaming a table requiresALTER
andDROP
on the old table,ALTER
,CREATE
, andINSERT
on the new table.Following the table name, specify the alterations to be made. If none are given,
ALTER TABLE
does nothing.The syntax for many of the permissible alterations is similar to clauses of the
CREATE TABLE
statement.column_definition
clauses use the same syntax forADD
andCHANGE
as forCREATE TABLE
. For more information.The word
COLUMN
is optional and can be omitted, except forRENAME COLUMN
(to distinguish a column-renaming operation from theRENAME
table-renaming operation).Multiple
ADD
,ALTER
,DROP
, andCHANGE
clauses are permitted in a singleALTER TABLE
statement, separated by commas. This is a MySQL extension to standard SQL, which permits only one of each clause perALTER TABLE
statement. For example, to drop multiple columns in a single statement, do this:1
ALTER TABLE t2 DROP COLUMN c, DROP COLUMN d;
If a storage engine does not support an attempted
ALTER TABLE
operation, a warning may result. Such warnings can be displayed withSHOW WARNINGS
.
CREATE TABLE Syntax
1 | CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tbl_name |
CREATE TABLE
creates a table with the given name. You
must have the CREATE
privilege for the table.
By default, tables are created in the default database, using the
InnoDB
storage engine. An error occurs if the table exists,
if there is no default database, or if the database does not exist.
Temporary Tables
You can use the TEMPORARY
keyword when creating a table.
A TEMPORARY
table is visible only within the current
session, and is dropped automatically when the session is closed.
PRIMARY KEY
A unique index where all key columns must be defined as
NOT NULL
. If they are not explicitly declared as
NOT NULL
, MySQL declares them so implicitly (and silently).
A table can have only one PRIMARY KEY
. The name of a
PRIMARY KEY
is always PRIMARY
, which thus
cannot be used as the name for any other kind of index.
If you do not have a PRIMARY KEY
and an application asks
for the PRIMARY KEY
in your tables, MySQL returns the first
UNIQUE
index that has no NULL
columns as the
PRIMARY KEY
.
In InnoDB
tables, keep the PRIMARY KEY
short to minimize storage overhead for secondary indexes. Each secondary
index entry contains a copy of the primary key columns for the
corresponding row.
In the created table, a PRIMARY KEY
is placed first,
followed by all UNIQUE
indexes, and then the nonunique
indexes. This helps the MySQL optimizer to prioritize which index to use
and also more quickly to detect duplicated UNIQUE
keys.
A PRIMARY KEY
can be a multiple-column index. However,
you cannot create a multiple-column index using the
PRIMARY KEY
key attribute in a column specification. Doing
so only marks that single column as primary. You must use a separate
PRIMARY KEY(*
key_part*, ...)
clause.
If a table has a PRIMARY KEY
or
UNIQUE NOT NULL
index that consists of a single column that
has an integer type, you can use _rowid
to refer to the
indexed column in SELECT
statements, as described in Unique
Indexes.
In MySQL, the name of a PRIMARY KEY
is
PRIMARY
. For other indexes, if you do not assign a name,
the index is assigned the same name as the first indexed column, with an
optional suffix (_2
, _3
, ...
) to
make it unique. You can see index names for a table using
SHOW INDEX FROM *
tbl_name*
.
## Important
For users familiar with the ANSI/ISO SQL Standard, please note that no storage engine, including
InnoDB
, recognizes or enforces theMATCH
clause used in referential integrity constraint definitions. Use of an explicitMATCH
clause will not have the specified effect, and also causesON DELETE
andON UPDATE
clauses to be ignored. For these reasons, specifyingMATCH
should be avoided.The
MATCH
clause in the SQL standard controls howNULL
values in a composite (multiple-column) foreign key are handled when comparing to a primary key.InnoDB
essentially implements the semantics defined byMATCH SIMPLE
, which permit a foreign key to be all or partiallyNULL
. In that case, the (child table) row containing such a foreign key is permitted to be inserted, and does not match any row in the referenced (parent) table. It is possible to implement other semantics using triggers.Additionally, MySQL requires that the referenced columns be indexed for performance. However,
InnoDB
does not enforce any requirement that the referenced columns be declaredUNIQUE
orNOT NULL
. The handling of foreign key references to nonunique keys or keys that containNULL
values is not well defined for operations such asUPDATE
orDELETE CASCADE
. You are advised to use foreign keys that reference only keys that are bothUNIQUE
(orPRIMARY
) andNOT NULL
.MySQL parses but ignores “inline
REFERENCES
specifications” (as defined in the SQL standard) where the references are defined as part of the column specification. MySQL acceptsREFERENCES
clauses only when specified as part of a separateFOREIGN KEY
specification.
Table Options
Table options are used to optimize the behavior of the table. In most
cases, you do not have to specify any of them. These options apply to
all storage engines unless otherwise indicated. Options that do not
apply to a given storage engine may be accepted and remembered as part
of the table definition. Such options then apply if you later use
ALTER TABLE
to convert the table to use a different storage
engine.
ENGINE
Specifies the storage engine for the table, using one of the names shown in the following table. The engine name can be unquoted or quoted. The quoted name
'DEFAULT'
is recognized but ignored.Storage Engine Description InnoDB
Transaction-safe tables with row locking and foreign keys. The default storage engine for new tables. MyISAM
The binary portable storage engine that is primarily used for read-only or read-mostly workloads. MEMORY
The data for this storage engine is stored only in memory. CSV
Tables that store rows in comma-separated values format. ARCHIVE
The archiving storage engine. EXAMPLE
An example engine. FEDERATED
Storage engine that accesses remote tables. HEAP
This is a synonym for MEMORY
.MERGE
A collection of MyISAM
tables used as one table. Also known asMRG_MyISAM
.NDB
Clustered, fault-tolerant, memory-based tables, supporting transactions and foreign keys. Also known as NDBCLUSTER
.By default, if a storage engine is specified that is not available, the statement fails with an error. You can override this behavior by removing
NO_ENGINE_SUBSTITUTION
from the server SQL mode so that MySQL allows substitution of the specified engine with the default storage engine instead. Normally in such cases, this isInnoDB
, which is the default value for thedefault_storage_engine
system variable. WhenNO_ENGINE_SUBSTITUTION
is disabled, a warning occurs if the storage engine specification is not honored.AUTO_INCREMENT
The initial
AUTO_INCREMENT
value for the table. In MySQL 8.0, this works forMyISAM
,MEMORY
,InnoDB
, andARCHIVE
tables. To set the first auto-increment value for engines that do not support theAUTO_INCREMENT
table option, insert a “dummy” row with a value one less than the desired value after creating the table, and then delete the dummy row.For engines that support the
AUTO_INCREMENT
table option inCREATE TABLE
statements, you can also useALTER TABLE
tbl_nameAUTO_INCREMENT = N
to reset theAUTO_INCREMENT
value. The value cannot be set lower than the maximum value currently in the column.
DROP TABLE Syntax
1 | DROP [TEMPORARY] TABLE [IF EXISTS] |
DROP TABLE
removes one or more tables. You must have the
DROP
privilege for each table.
Be careful with this statement! For each table, it removes the table definition and all table data. If the table is partitioned, the statement removes the table definition, all its partitions, all data stored in those partitions, and all partition definitions associated with the dropped table.
Dropping a table also drops any triggers for the table.
RENAME TABLE Syntax
1 | RENAME TABLE |
RENAME TABLE
renames one or more tables. You must have
ALTER
and DROP
privileges for the original
table, and CREATE
and INSERT
privileges for
the new table.
TRUNCATE TABLE Syntax
1 | TRUNCATE [TABLE] tbl_name |
TRUNCATE TABLE
empties a table completely. It requires
the DROP
privilege. Logically, TRUNCATE TABLE
is similar to a DELETE
statement that deletes all rows, or
a sequence of DROP TABLE
and CREATE TABLE
statements.
To achieve high performance, TRUNCATE TABLE
bypasses the
DML method of deleting data. Thus, it does not cause
ON DELETE
triggers to fire, it cannot be performed for
InnoDB
tables with parent-child foreign key relationships,
and it cannot be rolled back like a DML operation. However,
TRUNCATE TABLE
operations on tables that use an atomic
DDL-supported storage engine are either fully committed or rolled back
if the server halts during their operation.
Although TRUNCATE TABLE
is similar to
DELETE
, it is classified as a DDL statement rather than a
DML statement. It differs from DELETE
in the following
ways:
- Truncate operations drop and re-create the table, which is much faster than deleting rows one by one, particularly for large tables.
- Truncate operations cause an implicit commit, and so cannot be rolled back.
- Truncation operations cannot be performed if the session holds an active table lock.
TRUNCATE TABLE
fails for anInnoDB
table orNDB
table if there are anyFOREIGN KEY
constraints from other tables that reference the table. Foreign key constraints between columns of the same table are permitted.- Truncation operations do not return a meaningful value for the number of deleted rows. The usual result is “0 rows affected,” which should be interpreted as “no information.”
- As long as the table definition is valid, the table can be
re-created as an empty table with
TRUNCATE TABLE
, even if the data or index files have become corrupted. - Any
AUTO_INCREMENT
value is reset to its start value. This is true even forMyISAM
andInnoDB
, which normally do not reuse sequence values. - When used with partitioned tables,
TRUNCATE TABLE
preserves the partitioning; that is, the data and index files are dropped and re-created, while the partition definitions are unaffected. - The
TRUNCATE TABLE
statement does not invokeON DELETE
triggers. - Truncating a corrupted
InnoDB
table is supported.
TRUNCATE TABLE
for a table closes all handlers for the
table that were opened with HANDLER OPEN
.
TRUNCATE TABLE
is treated for purposes of binary logging
and replication as DROP TABLE
followed by
CREATE TABLE
—that is, as DDL rather than DML. This is due
to the fact that, when using InnoDB
and other transactional
storage engines where the transaction isolation level does not permit
statement-based logging (READ COMMITTED
or
READ UNCOMMITTED
), the statement was not logged and
replicated when using STATEMENT
or MIXED
logging mode. (Bug #36763) However, it is still applied on replication
slaves using InnoDB
in the manner described previously.
In MySQL 5.7 and earlier, on a system with a large buffer pool and
innodb_adaptive_hash_index
enabled, a
TRUNCATE TABLE
operation could cause a temporary drop in
system performance due to an LRU scan that occurred when removing the
table's adaptive hash index entries (Bug #68184). The remapping of
TRUNCATE TABLE
to DROP TABLE
and
CREATE TABLE
in MySQL 8.0 avoids the problematic LRU
scan.
TRUNCATE TABLE
can be used with Performance Schema
summary tables, but the effect is to reset the summary columns to 0 or
NULL
, not to remove rows.
CREATE INDEX Syntax
1 | CREATE [UNIQUE | FULLTEXT | SPATIAL] INDEX index_name |
Normally, you create all indexes on a table at the time the table
itself is created with CREATE TABLE
. This guideline is
especially important for InnoDB
tables, where the primary
key determines the physical layout of rows in the data file.
CREATE INDEX
enables you to add indexes to existing
tables.
CREATE INDEX
is mapped to an ALTER TABLE
statement to create indexes. CREATE INDEX
cannot be used to
create a PRIMARY KEY
; use ALTER TABLE
instead.
An index specification of the form
(key_part1, key_part2, ...)
creates an index with multiple
key parts. Index key values are formed by concatenating the values of
the given key parts. For example
(col1, col2, col3)
specifies a multiple-column index with
index keys consisting of values from col1
,
col2
, and col3
.
A key_part
specification can end with ASC
or DESC
to specify whether index values are stored in
ascending or descending order. The default is ascending if no order
specifier is given. ASC
and DESC
are not
permitted for HASH
indexes. ASC
and
DESC
are also not supported for multi-valued indexes. As of
MySQL 8.0.12, ASC
and DESC
are not permitted
for SPATIAL
indexes.
InnoDB Storage Engine Index Characteristics**
Index Class | Index Type | Stores NULL VALUES | Permits Multiple NULL Values | IS NULL Scan Type | IS NOT NULL Scan Type |
---|---|---|---|---|---|
Primary key | BTREE |
No | No | N/A | N/A |
Unique | BTREE |
Yes | Yes | Index | Index |
Key | BTREE |
Yes | Yes | Index | Index |
FULLTEXT |
N/A | Yes | Yes | Table | Table |
SPATIAL |
N/A | No | No | N/A | N/A |
MyISAM Storage Engine Index Characteristics
Index Class | Index Type | Stores NULL VALUES | Permits Multiple NULL Values | IS NULL Scan Type | IS NOT NULL Scan Type |
---|---|---|---|---|---|
Primary key | BTREE |
No | No | N/A | N/A |
Unique | BTREE |
Yes | Yes | Index | Index |
Key | BTREE |
Yes | Yes | Index | Index |
FULLTEXT |
N/A | Yes | Yes | Table | Table |
SPATIAL |
N/A | No | No | N/A | N/A |
MEMORY Storage Engine Index Characteristics
Index Class | Index Type | Stores NULL VALUES | Permits Multiple NULL Values | IS NULL Scan Type | IS NOT NULL Scan Type |
---|---|---|---|---|---|
Primary key | BTREE |
No | No | N/A | N/A |
Unique | BTREE |
Yes | Yes | Index | Index |
Key | BTREE |
Yes | Yes | Index | Index |
Primary key | HASH |
No | No | N/A | N/A |
Unique | HASH |
Yes | Yes | Index | Index |
Key | HASH |
Yes | Yes | Index | Index |
NDB Storage Engine Index Characteristics
Index Class | Index Type | Stores NULL VALUES | Permits Multiple NULL Values | IS NULL Scan Type | IS NOT NULL Scan Type |
---|---|---|---|---|---|
Primary key | BTREE |
No | No | Index | Index |
Unique | BTREE |
Yes | Yes | Index | Index |
Key | BTREE |
Yes | Yes | Index | Index |
Primary key | HASH |
No | No | Table (see note 1) | Table (see note 1) |
Unique | HASH |
Yes | Yes | Table (see note 1) | Table (see note 1) |
Key | HASH |
Yes | Yes | Table (see note 1) | Table (see note 1) |
Table note:
USING HASH
prevents creation of an implicit ordered index.
DROP INDEX Syntax
1 | DROP INDEX index_name ON tbl_name |
DROP INDEX
drops the index named
index_name
from the table
tbl_name
. This statement is mapped to an
ALTER TABLE
statement to drop the index.
To drop a primary key, the index name is always PRIMARY
,
which must be specified as a quoted identifier because
PRIMARY
is a reserved word:
1 | DROP INDEX `PRIMARY` ON t; |
DELETE Syntax
DELETE
is a DML statement that removes rows from a
table.
A DELETE
statement can start with a WITH
clause to define common table expressions accessible within the
DELETE
.
Single-Table Syntax
1 | DELETE [LOW_PRIORITY] [QUICK] [IGNORE] FROM tbl_name [[AS] tbl_alias] |
The DELETE
statement deletes rows from
tbl_name
and returns the number of deleted rows.
To check the number of deleted rows, call the ROW_COUNT()
function.
Main Clauses
The conditions in the optional WHERE
clause identify
which rows to delete. With no WHERE
clause, all rows are
deleted.
where_condition
is an expression that evaluates
to true for each row to be deleted.
If the ORDER BY
clause is specified, the rows are
deleted in the order that is specified. The LIMIT
clause
places a limit on the number of rows that can be deleted. These clauses
apply to single-table deletes, but not multi-table deletes.
Multiple-Table Syntax
1 | DELETE [LOW_PRIORITY] [QUICK] [IGNORE] |
Privileges
You need the DELETE
privilege on a table to delete rows
from it. You need only the SELECT
privilege for any columns
that are only read, such as those named in the WHERE
clause.
Performance
When you do not need to know the number of deleted rows, the
TRUNCATE TABLE
statement is a faster way to empty a table
than a DELETE
statement with no WHERE
clause.
Unlike DELETE
, TRUNCATE TABLE
cannot be used
within a transaction or if you have a lock on the table.
To ensure that a given DELETE
statement does not take
too much time, the MySQL-specific
LIMIT *
row_count*
clause for
DELETE
specifies the maximum number of rows to be deleted.
If the number of rows to delete is larger than the limit, repeat the
DELETE
statement until the number of affected rows is less
than the LIMIT
value.
Subqueries
You cannot delete from a table and select from the same table in a subquery.
Partitioned Table Support
DELETE
supports explicit partition selection using the
PARTITION
option, which takes a list of the comma-separated
names of one or more partitions or subpartitions (or both) from which to
select rows to be dropped. Partitions not included in the list are
ignored. Given a partitioned table t
with a partition named
p0
, executing the statement
DELETE FROM t PARTITION (p0)
has the same effect on the
table as executing ALTER TABLE t TRUNCATE PARTITION (p0)
;
in both cases, all rows in partition p0
are dropped.
PARTITION
can be used along with a WHERE
condition, in which case the condition is tested only on rows in the
listed partitions. For example,
DELETE FROM t PARTITION (p0) WHERE c < 5
deletes rows
only from partition p0
for which the condition
c < 5
is true; rows in any other partitions are not
checked and thus not affected by theDELETE
.
The PARTITION
option can also be used in multiple-table
DELETE
statements. You can use up to one such option per
table named in the FROM
option.
Auto-Increment Columns
If you delete the row containing the maximum value for an
AUTO_INCREMENT
column, the value is not reused for a
MyISAM
or InnoDB
table. If you delete all rows
in the table with DELETE FROM *
tbl_name*
(without a WHERE
clause) inautocommit
mode,
the sequence starts over for all storage engines except
InnoDB
and MyISAM
.
For MyISAM
tables, you can specify an
AUTO_INCREMENT
secondary column in a multiple-column key.
In this case, reuse of values deleted from the top of the sequence
occurs even for MyISAM
tables.
Modifiers
The DELETE
statement supports the following
modifiers:
If you specify the
LOW_PRIORITY
modifier, the server delays execution of theDELETE
until no other clients are reading from the table. This affects only storage engines that use only table-level locking (such asMyISAM
,MEMORY
, andMERGE
).For
MyISAM
tables, if you use theQUICK
modifier, the storage engine does not merge index leaves during delete, which may speed up some kinds of delete operations.The
IGNORE
modifier causes MySQL to ignore errors during the process of deleting rows. (Errors encountered during the parsing stage are processed in the usual manner.) Errors that are ignored due to the use ofIGNORE
are returned as warnings. For more information, see Comparison of the IGNORE Keyword and Strict SQL Mode.
Order of Deletion
If the DELETE
statement includes an
ORDER BY
clause, rows are deleted in the order specified by
the clause. This is useful primarily in conjunction with
LIMIT
. For example, the following statement finds rows
matching the WHERE
clause, sorts them by
timestamp_column
, and deletes the first (oldest) one:
1 | DELETE FROM somelog WHERE user = 'jcole' |
ORDER BY
also helps to delete rows in an order required
to avoid referential integrity violations.
InnoDB Tables
If you are deleting many rows from a large table, you may exceed the
lock table size for an InnoDB
table. To avoid this problem,
or simply to minimize the time that the table remains locked, the
following strategy (which does not useDELETE
at all) might
be helpful:
Select the rows not to be deleted into an empty table that has the same structure as the original table:
1
INSERT INTO t_copy SELECT * FROM t WHERE ... ;
Use
RENAME TABLE
to atomically move the original table out of the way and rename the copy to the original name:1
RENAME TABLE t TO t_old, t_copy TO t;
Drop the original table:
1
DROP TABLE t_old;
No other sessions can access the tables involved while
RENAME TABLE
executes, so the rename operation is not
subject to concurrency problems.
MyISAM Tables
In MyISAM
tables, deleted rows are maintained in a
linked list and subsequent INSERT
operations reuse old row
positions. To reclaim unused space and reduce file sizes, use the
OPTIMIZE TABLE
statement or the
myisamchkutility to reorganize tables.
OPTIMIZE TABLE
is easier to use, but
myisamchk is faster.
The QUICK
modifier affects whether index leaves are
merged for delete operations. DELETE QUICK
is most useful
for applications where index values for deleted rows are replaced by
similar index values from rows inserted later. In this case, the holes
left by deleted values are reused.
DELETE QUICK
is not useful when deleted values lead to
underfilled index blocks spanning a range of index values for which new
inserts occur again. In this case, use of QUICK
can lead to
wasted space in the index that remains unreclaimed. Here is an example
of such a scenario:
- Create a table that contains an indexed
AUTO_INCREMENT
column. - Insert many rows into the table. Each insert results in an index value that is added to the high end of the index.
- Delete a block of rows at the low end of the column range using
DELETE QUICK
.
In this scenario, the index blocks associated with the deleted index
values become underfilled but are not merged with other index blocks due
to the use of QUICK
. They remain underfilled when new
inserts occur, because new rows do not have index values in the deleted
range. Furthermore, they remain underfilled even if you later use
DELETE
without QUICK
, unless some of the
deleted index values happen to lie in index blocks within or adjacent to
the underfilled blocks. To reclaim unused index space under these
circumstances, use OPTIMIZE TABLE
.
If you are going to delete many rows from a table, it might be faster
to use DELETE QUICK
followed by
OPTIMIZE TABLE
. This rebuilds the index rather than
performing many index block merge operations.
Multi-Table Deletes
You can specify multiple tables in a DELETE
statement to
delete rows from one or more tables depending on the condition in the
WHERE
clause. You cannot use ORDER BY
or
LIMIT
in a multiple-table DELETE
.
Thetable_references
clause lists the tables
involved in the join.
For the first multiple-table syntax, only matching rows from the
tables listed before the FROM
clause are deleted. For the
second multiple-table syntax, only matching rows from the tables listed
in the FROM
clause (before the USING
clause)
are deleted. The effect is that you can delete rows from many tables at
the same time and have additional tables that are used only for
searching:
1 | DELETE t1, t2 FROM t1 INNER JOIN t2 INNER JOIN t3 |
Or:
1 | DELETE FROM t1, t2 USING t1 INNER JOIN t2 INNER JOIN t3 |
These statements use all three tables when searching for rows to
delete, but delete matching rows only from tablest1
and
t2
.
The preceding examples use INNER JOIN
, but
multiple-table DELETE
statements can use other types of
join permitted in SELECT
statements, such as
LEFT JOIN
. For example, to delete rows that exist in
t1
that have no match in t2
, use a
LEFT JOIN
:
1 | DELETE t1 FROM t1 LEFT JOIN t2 ON t1.id=t2.id WHERE t2.id IS NULL; |
The syntax permits .*
after each
tbl_name
for compatibility with
Access.
If you use a multiple-table DELETE
statement involving
InnoDB
tables for which there are foreign key constraints,
the MySQL optimizer might process tables in an order that differs from
that of their parent/child relationship. In this case, the statement
fails and rolls back. Instead, you should delete from a single table and
rely on the ON DELETE
capabilities that InnoDB
provides to cause the other tables to be modified accordingly.
Note
If you declare an alias for a table, you must use the alias when referring to the table:
1 | DELETE t1 FROM test AS t1, test2 WHERE ... |
Table aliases in a multiple-table DELETE
should be
declared only in the table_references
part of the
statement. Elsewhere, alias references are permitted but not alias
declarations.
Correct:
1 | DELETE a1, a2 FROM t1 AS a1 INNER JOIN t2 AS a2 |
Incorrect:
1 | DELETE t1 AS a1, t2 AS a2 FROM t1 INNER JOIN t2 |
Table aliases are also supported for single-table DELETE
statements beginning with MySQL 8.0.16.
INSERT Syntax
1 | INSERT [LOW_PRIORITY | DELAYED | HIGH_PRIORITY] [IGNORE] |
INSERT
inserts new rows into an existing table. The
INSERT ... VALUES
and INSERT ... SET
forms of
the statement insert rows based on explicitly specified values. The
INSERT ... SELECT
form inserts rows selected from another
table or tables. INSERT
with an
ON DUPLICATE KEY UPDATE
clause enables existing rows to be
updated if a row to be inserted would cause a duplicate value in a
UNIQUE
index or PRIMARY KEY
.
Inserting into a table requires the INSERT
privilege for
the table. If the ON DUPLICATE KEY UPDATE
clause is used
and a duplicate key causes an UPDATE
to be performed
instead, the statement requires the UPDATE
privilege for
the columns to be updated. For columns that are read but not modified
you need only the SELECT
privilege (such as for a column
referenced only on the right hand side of an
col_name
=expr
assignment in
an ON DUPLICATE KEY UPDATE
clause).
When inserting into a partitioned table, you can control which
partitions and subpartitions accept new rows. ThePARTITION
option takes a list of the comma-separated names of one or more
partitions or subpartitions (or both) of the table. If any of the rows
to be inserted by a given INSERT
statement do not match one
of the partitions listed, theINSERT
statement fails with
the error Found a row not matching the given partition set.
INSERT ... SELECT Syntax
1 | INSERT [LOW_PRIORITY | HIGH_PRIORITY] [IGNORE] |
With INSERT ... SELECT
, you can quickly insert many rows
into a table from the result of a SELECT
statement, which
can select from one or many tables. For example:
1 | INSERT INTO tbl_temp2 (fld_id) |
The following conditions hold for INSERT ... SELECT
statements:
Specify
IGNORE
to ignore rows that would cause duplicate-key violations.The target table of the
INSERT
statement may appear in theFROM
clause of theSELECT
part of the query. However, you cannot insert into a table and select from the same table in a subquery.When selecting from and inserting into the same table, MySQL creates an internal temporary table to hold the rows from the
SELECT
and then inserts those rows into the target table. However, you cannot useINSERT INTO t ... SELECT ... FROM t
whent
is aTEMPORARY
table, becauseTEMPORARY
tables cannot be referred to twice in the same statement.AUTO_INCREMENT
columns work as usual.To ensure that the binary log can be used to re-create the original tables, MySQL does not permit concurrent inserts for
INSERT ... SELECT
statements (see Section 8.11.3, “Concurrent Inserts”).To avoid ambiguous column reference problems when the
SELECT
and theINSERT
refer to the same table, provide a unique alias for each table used in theSELECT
part, and qualify column names in that part with the appropriate alias.
You can explicitly select which partitions or subpartitions (or both)
of the source or target table (or both) are to be used with a
PARTITION
option following the name of the table. When
PARTITION
is used with the name of the source table in the
SELECT
portion of the statement, rows are selected only
from the partitions or subpartitions named in its partition list. When
PARTITION
is used with the name of the target table for the
INSERT
portion of the statement, it must be possible to
insert all rows selected into the partitions or subpartitions named in
the partition list following the option. Otherwise, the
INSERT ... SELECT
statement fails. For more information and
examples.
The order in which a SELECT
statement with no
ORDER BY
clause returns rows is nondeterministic. This
means that, when using replication, there is no guarantee that such a
SELECT
returns rows in the same order on the master and the
slave, which can lead to inconsistencies between them. To prevent this
from occurring, always write INSERT ... SELECT
statements
that are to be replicated using an ORDER BY
clause that
produces the same row order on the master and the slave.
INSERT ... ON DUPLICATE KEY UPDATE Syntax
If you specify an ON DUPLICATE KEY UPDATE
clause and a
row to be inserted would cause a duplicate value in a
UNIQUE
index or PRIMARY KEY
, an
UPDATE
of the old row occurs. For example, if column
a
is declared as UNIQUE
and contains the value
1
, the following two statements have similar effect:
1 | INSERT INTO t1 (a,b,c) VALUES (1,2,3) |
(The effects are not identical for an InnoDB
table where
a
is an auto-increment column. With an auto-increment
column, an INSERT
statement increases the auto-increment
value but UPDATE
does not.)
If column b
is also unique, the INSERT
is
equivalent to this UPDATE
statement instead:
1 | UPDATE t1 SET c=c+1 WHERE a=1 OR b=2 LIMIT 1; |
LOAD DATA Syntax
1 | LOAD DATA |
The LOAD DATA
statement reads rows from a text file into
a table at a very high speed. LOAD DATA
is the complement
of SELECT ... INTO OUTFILE
. To write data from a table to a
file, use SELECT ... INTO OUTFILE
. To read the file back
into a table, use LOAD DATA
. The syntax of the
FIELDS
and LINES
clauses is the same for both
statements.
REPLACE Syntax
1 | REPLACE [LOW_PRIORITY | DELAYED] |
REPLACE
works exactly like INSERT
, except
that if an old row in the table has the same value as a new row for
aPRIMARY KEY
or a UNIQUE
index, the old row is
deleted before the new row is inserted.
You can also load data files by using the
mysqlimport utility; mysqlimport
operates by sending a LOAD DATA
statement to the
server.
SELECT Syntax
1 | SELECT |
SELECT
is used to retrieve rows selected from one or
more tables, and can include UNION
statements and
subqueries. A SELECT
statement can start with a
WITH
clause to define common table expressions accessible
within the SELECT
.
The most commonly used clauses of SELECT
statements are
these:
Each
select_expr
indicates a column that you want to retrieve. There must be at least oneselect_expr
.table_references
indicates the table or tables from which to retrieve rows.SELECT
supports explicit partition selection using thePARTITION
with a list of partitions or subpartitions (or both) following the name of the table in atable_reference
. In this case, rows are selected only from the partitions listed, and any other partitions of the table are ignored.SELECT ... PARTITION
from tables using storage engines such asMyISAM
that perform table-level locks (and thus partition locks) lock only the partitions or subpartitions named by thePARTITION
option.For more information, see Partitioning and Locking.
The
WHERE
clause, if given, indicates the condition or conditions that rows must satisfy to be selected.where_condition
is an expression that evaluates to true for each row to be selected. The statement selects all rows if there is noWHERE
clause.In the
WHERE
expression, you can use any of the functions and operators that MySQL supports, except for aggregate (summary) functions.
SELECT
can also be used to retrieve rows computed
without reference to any table.
For example:
1 | mysql> SELECT 1 + 1; |
You are permitted to specify DUAL
as a dummy table name
in situations where no tables are referenced:
1 | mysql> SELECT 1 + 1 FROM DUAL; |
DUAL
is purely for the convenience of people who require
that all SELECT
statements should have FROM
and possibly other clauses. MySQL may ignore the clauses. MySQL does not
require FROM DUAL
if no tables are referenced.
In general, clauses used must be given in exactly the order shown in
the syntax description. For example, a HAVING
clause must
come after any GROUP BY
clause and before any
ORDER BY
clause. The exception is that the
INTO
clause can appear either as shown in the syntax
description or immediately following the
select_expr
list.
JOIN Syntax
MySQL supports the following JOIN
syntax for the
table_references
part of SELECT
statements and multiple-table DELETE
and
UPDATE
statements:
1 | table_references: |
A table reference is also known as a join expression.
A table reference (when it refers to a partitioned table) may contain
a PARTITION
option, including a list of comma-separated
partitions, subpartitions, or both. This option follows the name of the
table and precedes any alias declaration. The effect of this option is
that rows are selected only from the listed partitions or subpartitions.
Any partitions or subpartitions not named in the list are ignored.
UNION Syntax
1 | SELECT ... |
UNION
is used to combine the result from multiple
SELECT
statements into a single result set.
The column names from the first SELECT
statement are
used as the column names for the results returned. Selected columns
listed in corresponding positions of each SELECT
statement
should have the same data type. (For example, the first column selected
by the first statement should have the same type as the first column
selected by the other statements.)
UPDATE Syntax
UPDATE
is a DML statement that modifies rows in a
table.
An UPDATE
statement can start with a WITH
clause to define common table expressions accessible within
theUPDATE
. See Section 13.2.13, “WITH Syntax (Common Table
Expressions)”.
Single-table syntax:
1 | UPDATE [LOW_PRIORITY] [IGNORE] table_reference |
Multiple-table syntax:
1 | UPDATE [LOW_PRIORITY] [IGNORE] table_references |
For the single-table syntax, the UPDATE
statement
updates columns of existing rows in the named table with new values. The
SET
clause indicates which columns to modify and the values
they should be given. Each value can be given as an expression, or the
keyword DEFAULT
to set a column explicitly to its default
value. The WHERE
clause, if given, specifies the conditions
that identify which rows to update. With no WHERE
clause,
all rows are updated. If theORDER BY
clause is specified,
the rows are updated in the order that is specified. The
LIMIT
clause places a limit on the number of rows that can
be updated.
For the multiple-table syntax, UPDATE
updates rows in
each table named in table_references
that satisfy
the conditions. Each matching row is updated once, even if it matches
the conditions multiple times. For multiple-table syntax,
ORDER BY
and LIMIT
cannot be used.
For partitioned tables, both the single-single and multiple-table
forms of this statement support the use of aPARTITION
option as part of a table reference. This option takes a list of one or
more partitions or subpartitions (or both). Only the partitions (or
subpartitions) listed are checked for matches, and a row that is not in
any of these partitions or subpartitions is not updated, whether it
satisfies the where_condition
or not.
Note
Unlike the case when using PARTITION
with an
INSERT
or REPLACE
statement, an otherwise
valid UPDATE ... PARTITION
statement is considered
successful even if no rows in the listed partitions (or subpartitions)
match thewhere_condition
.
Transactional and Locking Statements
MySQL supports local transactions (within a given client session)
through statements such as
SET autocommit
,START TRANSACTION
,
COMMIT
, and ROLLBACK
.
START TRANSACTION, COMMIT, and ROLLBACK Syntax
1 | START TRANSACTION |
These statements provide control over use of transactions:
START TRANSACTION
orBEGIN
start a new transaction.COMMIT
commits the current transaction, making its changes permanent.ROLLBACK
rolls back the current transaction, canceling its changes.SET autocommit
disables or enables the default autocommit mode for the current session.
By default, MySQL runs with autocommit mode enabled. This means that,
when not otherwise inside a transaction, each statement is atomic, as if
it were surrounded by START TRANSACTION
and
COMMIT
. You cannot use ROLLBACK
to undo the
effect; however, if an error occurs during statement execution, the
statement is rolled back.
To disable autocommit mode implicitly for a single series of
statements, use the START TRANSACTION
statement:
1 | START TRANSACTION; |
With START TRANSACTION
, autocommit remains disabled
until you end the transaction with COMMIT
or
ROLLBACK
. The autocommit mode then reverts to its previous
state.
Statements That Cannot Be Rolled Back
Some statements cannot be rolled back. In general, these include data definition language (DDL) statements, such as those that create or drop databases, those that create, drop, or alter tables or stored routines.
You should design your transactions not to include such statements.
If you issue a statement early in a transaction that cannot be rolled
back, and then another statement later fails, the full effect of the
transaction cannot be rolled back in such cases by issuing a
ROLLBACK
statement.
Statements That Cause an Implicit Commit
The statements listed in this section (and any synonyms for them)
implicitly end any transaction active in the current session, as if you
had done a COMMIT
before executing the statement.
Most of these statements also cause an implicit commit after executing. The intent is to handle each such statement in its own special transaction. Transaction-control and locking statements are exceptions: If an implicit commit occurs before execution, another does not occur after.
Data definition language (DDL) statements that define or modify database objects.
ALTER EVENT
,ALTER FUNCTION
,ALTER PROCEDURE
,ALTER SERVER
,ALTER TABLE
,ALTER VIEW
,CREATE DATABASE
,CREATE EVENT
,CREATE FUNCTION
,CREATE INDEX
,CREATE PROCEDURE
,CREATE ROLE
,CREATE SERVER
,CREATE SPATIAL REFERENCE SYSTEM
,CREATE TABLE
,CREATE TRIGGER
,CREATE VIEW
,DROP DATABASE
,DROP EVENT
,DROP FUNCTION
,DROP INDEX
,DROP PROCEDURE
,DROP ROLE
,DROP SERVER
,DROP SPATIAL REFERENCE SYSTEM
,DROP TABLE
,DROP TRIGGER
,DROP VIEW
,INSTALL PLUGIN
,RENAME TABLE
,TRUNCATE TABLE
,UNINSTALL PLUGIN
.CREATE TABLE
andDROP TABLE
statements do not commit a transaction if theTEMPORARY
keyword is used. (This does not apply to other operations on temporary tables such asALTER TABLE
andCREATE INDEX
, which do cause a commit.) However, although no implicit commit occurs, neither can the statement be rolled back, which means that the use of such statements causes transactional atomicity to be violated. For example, if you useCREATE TEMPORARY TABLE
and then roll back the transaction, the table remains in existence.The
CREATE TABLE
statement inInnoDB
is processed as a single transaction. This means that aROLLBACK
from the user does not undoCREATE TABLE
statements the user made during that transaction.CREATE TABLE ... SELECT
causes an implicit commit before and after the statement is executed when you are creating nontemporary tables. (No commit occurs forCREATE TEMPORARY TABLE ... SELECT
.)Statements that implicitly use or modify tables in the
mysql
database.ALTER USER
,CREATE USER
,DROP USER
,GRANT
,RENAME USER
,REVOKE
,SET PASSWORD
.Transaction-control and locking statements.
BEGIN
,LOCK TABLES
,SET autocommit = 1
(if the value is not already 1),START TRANSACTION
,UNLOCK TABLES
.UNLOCK TABLES
commits a transaction only if any tables currently have been locked withLOCK TABLES
to acquire nontransactional table locks. A commit does not occur forUNLOCK TABLES
followingFLUSH TABLES WITH READ LOCK
because the latter statement does not acquire table-level locks.Transactions cannot be nested. This is a consequence of the implicit commit performed for any current transaction when you issue a
START TRANSACTION
statement or one of its synonyms.Statements that cause an implicit commit cannot be used in an XA transaction while the transaction is in an
ACTIVE
state.The
BEGIN
statement differs from the use of theBEGIN
keyword that starts aBEGIN ... END
compound statement. The latter does not cause an implicit commit. See Section 13.6.1, “BEGIN ... END Compound-Statement Syntax”.Data loading statements.
LOAD DATA
.LOAD DATA
causes an implicit commit only for tables using theNDB
storage engine.Administrative statements.
ANALYZE TABLE
,CACHE INDEX
,CHECK TABLE
,FLUSH
,LOAD INDEX INTO CACHE
,OPTIMIZE TABLE
,REPAIR TABLE
,RESET
(but notRESET PERSIST
).Replication control statements.
START SLAVE
,STOP SLAVE
,RESET SLAVE
,CHANGE MASTER TO
.
USER
CREATE USER Syntax
1 | CREATE USER [IF NOT EXISTS] |
The CREATE USER
statement creates new MySQL accounts. It
enables authentication, role, SSL/TLS, resource-limit, and
password-management properties to be established for new accounts. It
also controls whether accounts are initially locked or unlocked.
To use CREATE USER
, you must have the global
CREATE USER
privilege, or the INSERT
privilege
for the mysql
system database. When the
read_only
system variable is enabled,
CREATE USER
additionally requires the
CONNECTION_ADMIN
or SUPER
privilege.
CREATE USER
either succeeds for all named users or rolls
back and has no effect if any error occurs. By default, an error occurs
if you try to create a user that already exists. If the
IF NOT EXISTS
clause is given, the statement produces a
warning for each named user that already exists, rather than an
error.
Important
Under some circumstances, CREATE USER
may be recorded in
server logs or on the client side in a history file such as
~/.mysql_history
, which means that cleartext passwords may
be read by anyone having read access to that information. For
information about the conditions under which this occurs for the server
logs and how to control it. For similar information about client-side
logging.
DROP USER Syntax
1 | DROP USER [IF EXISTS] user [, user] ... |
The DROP USER
statement removes one or more MySQL
accounts and their privileges. It removes privilege rows for the account
from all grant tables.
GRANT Syntax
1 | GRANT |
The GRANT
statement assigns privileges and roles to
MySQL user accounts and roles. There are several aspects to the
GRANT
statement, described under the following topics:
- GRANT General Overview
- Object Quoting Guidelines
- Account Names
- Privileges Supported by MySQL
- Global Privileges
- Database Privileges
- Table Privileges
- Column Privileges
- Stored Routine Privileges
- Proxy User Privileges
- Granting Roles
- The
AS
Clause and Privilege Restrictions - Other Account Characteristics
- MySQL and Standard SQL Versions of GRANT