- 15.1.20.1 Files Created by CREATE TABLE
- 15.1.20.2 CREATE TEMPORARY TABLE Statement
- 15.1.20.3 CREATE TABLE ... LIKE Statement
- 15.1.20.4 CREATE TABLE ... SELECT Statement
- 15.1.20.5 FOREIGN KEY Constraints
- 15.1.20.6 CHECK Constraints
- 15.1.20.7 Silent Column Specification Changes
- 15.1.20.8 CREATE TABLE and Generated Columns
- 15.1.20.9 Secondary Indexes and Generated Columns
- 15.1.20.10 Invisible Columns
- 15.1.20.11 Generated Invisible Primary Keys
- 15.1.20.12 Setting NDB Comment Options
CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tbl_name
    (create_definition,...)
    [table_options]
    [partition_options]
CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tbl_name
    [(create_definition,...)]
    [table_options]
    [partition_options]
    [IGNORE | REPLACE]
    [AS] query_expression
CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tbl_name
    { LIKE old_tbl_name | (LIKE old_tbl_name) }
create_definition: {
    col_name column_definition
  | {INDEX | KEY} [index_name] [index_type] (key_part,...)
      [index_option] ...
  | {FULLTEXT | SPATIAL} [INDEX | KEY] [index_name] (key_part,...)
      [index_option] ...
  | [CONSTRAINT [symbol]] PRIMARY KEY
      [index_type] (key_part,...)
      [index_option] ...
  | [CONSTRAINT [symbol]] UNIQUE [INDEX | KEY]
      [index_name] [index_type] (key_part,...)
      [index_option] ...
  | [CONSTRAINT [symbol]] FOREIGN KEY
      [index_name] (col_name,...)
      reference_definition
  | check_constraint_definition
}
column_definition: {
    data_type [NOT NULL | NULL] [DEFAULT {literal | (expr)} ]
      [VISIBLE | INVISIBLE]
      [AUTO_INCREMENT] [UNIQUE [KEY]] [[PRIMARY] KEY]
      [COMMENT 'string']
      [COLLATE collation_name]
      [COLUMN_FORMAT {FIXED | DYNAMIC | DEFAULT}]
      [ENGINE_ATTRIBUTE [=] 'string']
      [SECONDARY_ENGINE_ATTRIBUTE [=] 'string']
      [STORAGE {DISK | MEMORY}]
      [reference_definition]
      [check_constraint_definition]
  | data_type
      [COLLATE collation_name]
      [GENERATED ALWAYS] AS (expr)
      [VIRTUAL | STORED] [NOT NULL | NULL]
      [VISIBLE | INVISIBLE]
      [UNIQUE [KEY]] [[PRIMARY] KEY]
      [COMMENT 'string']
      [reference_definition]
      [check_constraint_definition]
}
data_type:
    (see Chapter 13, Data Types)
key_part: {col_name [(length)] | (expr)} [ASC | DESC]
index_type:
    USING {BTREE | HASH}
index_option: {
    KEY_BLOCK_SIZE [=] value
  | index_type
  | WITH PARSER parser_name
  | COMMENT 'string'
  | {VISIBLE | INVISIBLE}
  |ENGINE_ATTRIBUTE [=] 'string'
  |SECONDARY_ENGINE_ATTRIBUTE [=] 'string'
}
check_constraint_definition:
    [CONSTRAINT [symbol]] CHECK (expr) [[NOT] ENFORCED]
reference_definition:
    REFERENCES tbl_name (key_part,...)
      [MATCH FULL | MATCH PARTIAL | MATCH SIMPLE]
      [ON DELETE reference_option]
      [ON UPDATE reference_option]
reference_option:
    RESTRICT | CASCADE | SET NULL | NO ACTION | SET DEFAULT
table_options:
    table_option [[,] table_option] ...
table_option: {
    AUTOEXTEND_SIZE [=] value
  | AUTO_INCREMENT [=] value
  | AVG_ROW_LENGTH [=] value
  | [DEFAULT] CHARACTER SET [=] charset_name
  | CHECKSUM [=] {0 | 1}
  | [DEFAULT] COLLATE [=] collation_name
  | COMMENT [=] 'string'
  | COMPRESSION [=] {'ZLIB' | 'LZ4' | 'NONE'}
  | CONNECTION [=] 'connect_string'
  | {DATA | INDEX} DIRECTORY [=] 'absolute path to directory'
  | DELAY_KEY_WRITE [=] {0 | 1}
  | ENCRYPTION [=] {'Y' | 'N'}
  | ENGINE [=] engine_name
  | ENGINE_ATTRIBUTE [=] 'string'
  | INSERT_METHOD [=] { NO | FIRST | LAST }
  | KEY_BLOCK_SIZE [=] value
  | MAX_ROWS [=] value
  | MIN_ROWS [=] value
  | PACK_KEYS [=] {0 | 1 | DEFAULT}
  | PASSWORD [=] 'string'
  | ROW_FORMAT [=] {DEFAULT | DYNAMIC | FIXED | COMPRESSED | REDUNDANT | COMPACT}
  | START TRANSACTION 
  | SECONDARY_ENGINE_ATTRIBUTE [=] 'string'
  | STATS_AUTO_RECALC [=] {DEFAULT | 0 | 1}
  | STATS_PERSISTENT [=] {DEFAULT | 0 | 1}
  | STATS_SAMPLE_PAGES [=] value
  | tablespace_option
  | UNION [=] (tbl_name[,tbl_name]...)
}
partition_options:
    PARTITION BY
        { [LINEAR] HASH(expr)
        | [LINEAR] KEY [ALGORITHM={1 | 2}] (column_list)
        | RANGE{(expr) | COLUMNS(column_list)}
        | LIST{(expr) | COLUMNS(column_list)} }
    [PARTITIONS num]
    [SUBPARTITION BY
        { [LINEAR] HASH(expr)
        | [LINEAR] KEY [ALGORITHM={1 | 2}] (column_list) }
      [SUBPARTITIONS num]
    ]
    [(partition_definition [, partition_definition] ...)]
partition_definition:
    PARTITION partition_name
        [VALUES
            {LESS THAN {(expr | value_list) | MAXVALUE}
            |
            IN (value_list)}]
        [[STORAGE] ENGINE [=] engine_name]
        [COMMENT [=] 'string' ]
        [DATA DIRECTORY [=] 'data_dir']
        [INDEX DIRECTORY [=] 'index_dir']
        [MAX_ROWS [=] max_number_of_rows]
        [MIN_ROWS [=] min_number_of_rows]
        [TABLESPACE [=] tablespace_name]
        [(subpartition_definition [, subpartition_definition] ...)]
subpartition_definition:
    SUBPARTITION logical_name
        [[STORAGE] ENGINE [=] engine_name]
        [COMMENT [=] 'string' ]
        [DATA DIRECTORY [=] 'data_dir']
        [INDEX DIRECTORY [=] 'index_dir']
        [MAX_ROWS [=] max_number_of_rows]
        [MIN_ROWS [=] min_number_of_rows]
        [TABLESPACE [=] tablespace_name]
tablespace_option:
    TABLESPACE tablespace_name [STORAGE DISK]
  | [TABLESPACE tablespace_name] STORAGE MEMORY
query_expression:
    SELECT ...   (Some valid select or union statement)
      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.
    
      MySQL has no limit on the number of tables. The underlying file
      system may have a limit on the number of files that represent
      tables. Individual storage engines may impose engine-specific
      constraints. InnoDB permits up to 4 billion
      tables.
    
For information about the physical representation of a table, see Section 15.1.20.1, “Files Created by CREATE TABLE”.
      There are several aspects to the CREATE
      TABLE statement, described under the following topics in
      this section:
Table Name
- tbl_name- The table name can be specified as - db_name.tbl_nameto create the table in a specific database. This works regardless of whether there is a default database, assuming that the database exists. If you use quoted identifiers, quote the database and table names separately. For example, write- `mydb`.`mytbl`, not- `mydb.mytbl`.- Rules for permissible table names are given in Section 11.2, “Schema Object Names”. 
- IF NOT EXISTS- Prevents an error from occurring if the table exists. However, there is no verification that the existing table has a structure identical to that indicated by the - CREATE TABLEstatement.
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. For more information, see
      Section 15.1.20.2, “CREATE TEMPORARY TABLE Statement”.
Table Cloning and Copying
- LIKE- Use - CREATE TABLE ... LIKEto create an empty table based on the definition of another table, including any column attributes and indexes defined in the original table:- CREATE TABLE new_tbl LIKE orig_tbl;- For more information, see Section 15.1.20.3, “CREATE TABLE ... LIKE Statement”. 
- [AS]- query_expression- To create one table from another, add a - SELECTstatement at the end of the- CREATE TABLEstatement:- CREATE TABLE new_tbl AS SELECT * FROM orig_tbl;- For more information, see Section 15.1.20.4, “CREATE TABLE ... SELECT Statement”. 
- IGNORE | REPLACE- The - IGNOREand- REPLACEoptions indicate how to handle rows that duplicate unique key values when copying a table using a- SELECTstatement.- For more information, see Section 15.1.20.4, “CREATE TABLE ... SELECT Statement”. 
Column Data Types and Attributes
There is a hard limit of 4096 columns per table, but the effective maximum may be less for a given table and depends on the factors discussed in Section 10.4.7, “Limits on Table Column Count and Row Size”.
- data_type- data_typerepresents the data type in a column definition. For a full description of the syntax available for specifying column data types, as well as information about the properties of each type, see Chapter 13, Data Types.- AUTO_INCREMENTapplies only to integer types.
- Character data types ( - CHAR,- VARCHAR, the- TEXTtypes,- ENUM,- SET, and any synonyms) can include- CHARACTER SETto specify the character set for the column.- CHARSETis a synonym for- CHARACTER SET. A collation for the character set can be specified with the- COLLATEattribute, along with any other attributes. For details, see Chapter 12, Character Sets, Collations, Unicode. Example:- CREATE TABLE t (c CHAR(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_bin);- MySQL 8.4 interprets length specifications in character column definitions in characters. Lengths for - BINARYand- VARBINARYare in bytes.
- For - CHAR,- VARCHAR,- BINARY, and- VARBINARYcolumns, indexes can be created that use only the leading part of column values, using- col_name(- length)- BLOBand- TEXTcolumns also can be indexed, but a prefix length must be given. Prefix lengths are given in characters for nonbinary string types and in bytes for binary string types. That is, index entries consist of the first- lengthcharacters of each column value for- CHAR,- VARCHAR, and- TEXTcolumns, and the first- lengthbytes of each column value for- BINARY,- VARBINARY, and- BLOBcolumns. Indexing only a prefix of column values like this can make the index file much smaller. For additional information about index prefixes, see Section 15.1.15, “CREATE INDEX Statement”.- Only the - InnoDBand- MyISAMstorage engines support indexing on- BLOBand- TEXTcolumns. For example:- CREATE TABLE test (blob_col BLOB, INDEX(blob_col(10)));- If a specified index prefix exceeds the maximum column data type size, - CREATE TABLEhandles the index as follows:- For a nonunique index, either an error occurs (if strict SQL mode is enabled), or the index length is reduced to lie within the maximum column data type size and a warning is produced (if strict SQL mode is not enabled). 
- For a unique index, an error occurs regardless of SQL mode because reducing the index length might enable insertion of nonunique entries that do not meet the specified uniqueness requirement. 
 
- JSONcolumns cannot be indexed. You can work around this restriction by creating an index on a generated column that extracts a scalar value from the- JSONcolumn. See Indexing a Generated Column to Provide a JSON Column Index, for a detailed example.
 
- NOT NULL | NULL- If neither - NULLnor- NOT NULLis specified, the column is treated as though- NULLhad been specified.- In MySQL 8.4, only the - InnoDB,- MyISAM, and- MEMORYstorage engines support indexes on columns that can have- NULLvalues. In other cases, you must declare indexed columns as- NOT NULLor an error results.
- DEFAULT- Specifies a default value for a column. For more information about default value handling, including the case that a column definition includes no explicit - DEFAULTvalue, see Section 13.6, “Data Type Default Values”.- If the - NO_ZERO_DATEor- NO_ZERO_IN_DATESQL mode is enabled and a date-valued default is not correct according to that mode,- CREATE TABLEproduces a warning if strict SQL mode is not enabled and an error if strict mode is enabled. For example, with- NO_ZERO_IN_DATEenabled,- c1 DATE DEFAULT '2010-00-00'produces a warning.
- VISIBLE,- INVISIBLE- Specify column visibility. The default is - VISIBLEif neither keyword is present. A table must have at least one visible column. Attempting to make all columns invisible produces an error. For more information, see Section 15.1.20.10, “Invisible Columns”.
- AUTO_INCREMENT- An integer column can have the additional attribute - AUTO_INCREMENT. When you insert a value of- NULL(recommended) or- 0into an indexed- AUTO_INCREMENTcolumn, the column is set to the next sequence value. Typically this is- value+1- valueis the largest value for the column currently in the table.- AUTO_INCREMENTsequences begin with- 1.- To retrieve an - AUTO_INCREMENTvalue after inserting a row, use the- LAST_INSERT_ID()SQL function or the- mysql_insert_id()C API function. See Section 14.15, “Information Functions”, and mysql_insert_id().- If the - NO_AUTO_VALUE_ON_ZEROSQL mode is enabled, you can store- 0in- AUTO_INCREMENTcolumns as- 0without generating a new sequence value. See Section 7.1.11, “Server SQL Modes”.- There can be only one - AUTO_INCREMENTcolumn per table, it must be indexed, and it cannot have a- DEFAULTvalue. An- AUTO_INCREMENTcolumn works properly only if it contains only positive values. Inserting a negative number is regarded as inserting a very large positive number. This is done to avoid precision problems when numbers “wrap” over from positive to negative and also to ensure that you do not accidentally get an- AUTO_INCREMENTcolumn that contains- 0.- For - MyISAMtables, you can specify an- AUTO_INCREMENTsecondary column in a multiple-column key. See Section 5.6.9, “Using AUTO_INCREMENT”.- To make MySQL compatible with some ODBC applications, you can find the - AUTO_INCREMENTvalue for the last inserted row with the following query:- SELECT * FROM tbl_name WHERE auto_col IS NULL- This method requires that - sql_auto_is_nullvariable is not set to 0. See Section 7.1.8, “Server System Variables”.- For information about - InnoDBand- AUTO_INCREMENT, see Section 17.6.1.6, “AUTO_INCREMENT Handling in InnoDB”. For information about- AUTO_INCREMENTand MySQL Replication, see Section 19.5.1.1, “Replication and AUTO_INCREMENT”.
- COMMENT- A comment for a column can be specified with the - COMMENToption, up to 1024 characters long. The comment is displayed by the- SHOW CREATE TABLEand- SHOW FULL COLUMNSstatements. It is also shown in the- COLUMN_COMMENTcolumn of the Information Schema- COLUMNStable.
- COLUMN_FORMAT- In NDB Cluster, it is also possible to specify a data storage format for individual columns of - NDBtables using- COLUMN_FORMAT. Permissible column formats are- FIXED,- DYNAMIC, and- DEFAULT.- FIXEDis used to specify fixed-width storage,- DYNAMICpermits the column to be variable-width, and- DEFAULTcauses the column to use fixed-width or variable-width storage as determined by the column's data type (possibly overridden by a- ROW_FORMATspecifier).- For - NDBtables, the default value for- COLUMN_FORMATis- FIXED.- In NDB Cluster, the maximum possible offset for a column defined with - COLUMN_FORMAT=FIXEDis 8188 bytes. For more information and possible workarounds, see Section 25.2.7.5, “Limits Associated with Database Objects in NDB Cluster”.- COLUMN_FORMATcurrently has no effect on columns of tables using storage engines other than- NDB. MySQL 8.4 silently ignores- COLUMN_FORMAT.
- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEoptions are used to specify column attributes for primary and secondary storage engines. The options are reserved for future use.- The value assigned to this option is a string literal containing a valid JSON document or an empty string (''). Invalid JSON is rejected. - CREATE TABLE t1 (c1 INT ENGINE_ATTRIBUTE='{"key":"value"}');- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEvalues can be repeated without error. In this case, the last specified value is used.- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEvalues are not checked by the server, nor are they cleared when the table's storage engine is changed.
- STORAGE- For - NDBtables, it is possible to specify whether the column is stored on disk or in memory by using a- STORAGEclause.- STORAGE DISKcauses the column to be stored on disk, and- STORAGE MEMORYcauses in-memory storage to be used. The- CREATE TABLEstatement used must still include a- TABLESPACEclause:- mysql> CREATE TABLE t1 ( -> c1 INT STORAGE DISK, -> c2 INT STORAGE MEMORY -> ) ENGINE NDB; ERROR 1005 (HY000): Can't create table 'c.t1' (errno: 140) mysql> CREATE TABLE t1 ( -> c1 INT STORAGE DISK, -> c2 INT STORAGE MEMORY -> ) TABLESPACE ts_1 ENGINE NDB; Query OK, 0 rows affected (1.06 sec)- For - NDBtables,- STORAGE DEFAULTis equivalent to- STORAGE MEMORY.- The - STORAGEclause has no effect on tables using storage engines other than- NDB. The- STORAGEkeyword is supported only in the build of mysqld that is supplied with NDB Cluster; it is not recognized in any other version of MySQL, where any attempt to use the- STORAGEkeyword causes a syntax error.
- GENERATED ALWAYS- Used to specify a generated column expression. For information about generated columns, see Section 15.1.20.8, “CREATE TABLE and Generated Columns”. - Stored generated columns can be indexed. - InnoDBsupports secondary indexes on virtual generated columns. See Section 15.1.20.9, “Secondary Indexes and Generated Columns”.
Indexes, Foreign Keys, and CHECK Constraints
      Several keywords apply to creation of indexes, foreign keys, and
      CHECK constraints. For general background in
      addition to the following descriptions, see
      Section 15.1.15, “CREATE INDEX Statement”,
      Section 15.1.20.5, “FOREIGN KEY Constraints”, and
      Section 15.1.20.6, “CHECK Constraints”.
- CONSTRAINT- symbol- The - CONSTRAINTclause may be given to name a constraint. If the clause is not given, or a- symbol- symbolis not included following the- CONSTRAINTkeyword, MySQL automatically generates a constraint name, with the exception noted below. The- symbolvalue, if used, must be unique per schema (database), per constraint type. A duplicate- symbolresults in an error. See also the discussion about length limits of generated constraint identifiers at Section 11.2.1, “Identifier Length Limits”.Note- If the - CONSTRAINTclause is not given in a foreign key definition, or a- symbol- symbolis not included following the- CONSTRAINTkeyword, MySQL automatically generates a constraint name.- The SQL standard specifies that all types of constraints (primary key, unique index, foreign key, check) belong to the same namespace. In MySQL, each constraint type has its own namespace per schema. Consequently, names for each type of constraint must be unique per schema, but constraints of different types can have the same name. 
- 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 KEYis always- PRIMARY, which thus cannot be used as the name for any other kind of index.- If you do not have a - PRIMARY KEYand an application asks for the- PRIMARY KEYin your tables, MySQL returns the first- UNIQUEindex that has no- NULLcolumns as the- PRIMARY KEY.- In - InnoDBtables, keep the- PRIMARY KEYshort to minimize storage overhead for secondary indexes. Each secondary index entry contains a copy of the primary key columns for the corresponding row. (See Section 17.6.2.1, “Clustered and Secondary Indexes”.)- In the created table, a - PRIMARY KEYis placed first, followed by all- UNIQUEindexes, and then the nonunique indexes. This helps the MySQL optimizer to prioritize which index to use and also more quickly to detect duplicated- UNIQUEkeys.- A - PRIMARY KEYcan be a multiple-column index. However, you cannot create a multiple-column index using the- PRIMARY KEYkey attribute in a column specification. Doing so only marks that single column as primary. You must use a separate- PRIMARY KEY(clause.- key_part, ...)- If a table has a - PRIMARY KEYor- UNIQUE NOT NULLindex that consists of a single column that has an integer type, you can use- _rowidto refer to the indexed column in- SELECTstatements, as described in Unique Indexes.- In MySQL, the name of a - PRIMARY KEYis- 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. See Section 15.7.7.23, “SHOW INDEX Statement”.- tbl_name
- KEY | INDEX- KEYis normally a synonym for- INDEX. The key attribute- PRIMARY KEYcan also be specified as just- KEYwhen given in a column definition. This was implemented for compatibility with other database systems.
- UNIQUE- A - UNIQUEindex creates a constraint such that all values in the index must be distinct. An error occurs if you try to add a new row with a key value that matches an existing row. For all engines, a- UNIQUEindex permits multiple- NULLvalues for columns that can contain- NULL. If you specify a prefix value for a column in a- UNIQUEindex, the column values must be unique within the prefix length.- If a table has a - PRIMARY KEYor- UNIQUE NOT NULLindex that consists of a single column that has an integer type, you can use- _rowidto refer to the indexed column in- SELECTstatements, as described in Unique Indexes.
- FULLTEXT- A - FULLTEXTindex is a special type of index used for full-text searches. Only the- InnoDBand- MyISAMstorage engines support- FULLTEXTindexes. They can be created only from- CHAR,- VARCHAR, and- TEXTcolumns. Indexing always happens over the entire column; column prefix indexing is not supported and any prefix length is ignored if specified. See Section 14.9, “Full-Text Search Functions”, for details of operation. A- WITH PARSERclause can be specified as an- index_optionvalue to associate a parser plugin with the index if full-text indexing and searching operations need special handling. This clause is valid only for- FULLTEXTindexes.- InnoDBand- MyISAMsupport full-text parser plugins. See Full-Text Parser Plugins and Writing Full-Text Parser Plugins for more information.
- SPATIAL- You can create - SPATIALindexes on spatial data types. Spatial types are supported only for- InnoDBand- MyISAMtables, and indexed columns must be declared as- NOT NULL. See Section 13.4, “Spatial Data Types”.
- FOREIGN KEY- MySQL supports foreign keys, which let you cross-reference related data across tables, and foreign key constraints, which help keep this spread-out data consistent. For definition and option information, see - reference_definition, and- reference_option.- Partitioned tables employing the - InnoDBstorage engine do not support foreign keys. See Section 26.6, “Restrictions and Limitations on Partitioning”, for more information.
- CHECK- The - CHECKclause enables the creation of constraints to be checked for data values in table rows. See Section 15.1.20.6, “CHECK Constraints”.
- key_part- A - key_partspecification can end with- ASCor- DESCto specify whether index values are stored in ascending or descending order. The default is ascending if no order specifier is given.
- Prefixes, defined by the - lengthattribute, can be up to 767 bytes long for- InnoDBtables that use the- REDUNDANTor- COMPACTrow format. The prefix length limit is 3072 bytes for- InnoDBtables that use the- DYNAMICor- COMPRESSEDrow format. For- MyISAMtables, the prefix length limit is 1000 bytes.- Prefix limits are measured in bytes. However, prefix lengths for index specifications in - CREATE TABLE,- ALTER TABLE, and- CREATE INDEXstatements are interpreted as number of characters for nonbinary string types (- CHAR,- VARCHAR,- TEXT) and number of bytes for binary string types (- BINARY,- VARBINARY,- BLOB). Take this into account when specifying a prefix length for a nonbinary string column that uses a multibyte character set.
- The - exprfor a- key_partspecification can take the form- (CASTto create a multi-valued index on a- json_pathAS- typeARRAY)- JSONcolumn. Multi-Valued Indexes, provides detailed information regarding creation of, usage of, and restrictions and limitations on multi-valued indexes.
 
- index_type- Some storage engines permit you to specify an index type when creating an index. The syntax for the - index_typespecifier is- USING.- type_name- Example: - CREATE TABLE lookup (id INT, INDEX USING BTREE (id) ) ENGINE = MEMORY;- The preferred position for - USINGis after the index column list. It can be given before the column list, but support for use of the option in that position is deprecated and you should expect it to be removed in a future MySQL release.
- index_option- index_optionvalues specify additional options for an index.- KEY_BLOCK_SIZE- For - MyISAMtables,- KEY_BLOCK_SIZEoptionally specifies the size in bytes to use for index key blocks. The value is treated as a hint; a different size could be used if necessary. A- KEY_BLOCK_SIZEvalue specified for an individual index definition overrides the table-level- KEY_BLOCK_SIZEvalue.- For information about the table-level - KEY_BLOCK_SIZEattribute, see Table Options.
- WITH PARSER- The - WITH PARSERoption can be used only with- FULLTEXTindexes. It associates a parser plugin with the index if full-text indexing and searching operations need special handling.- InnoDBand- MyISAMsupport full-text parser plugins. If you have a- MyISAMtable with an associated full-text parser plugin, you can convert the table to- InnoDBusing- ALTER TABLE.
- COMMENT- Index definitions can include an optional comment of up to 1024 characters. - You can set the - InnoDB- MERGE_THRESHOLDvalue for an individual index using the- index_option- COMMENTclause. See Section 17.8.11, “Configuring the Merge Threshold for Index Pages”.
- VISIBLE,- INVISIBLE- Specify index visibility. Indexes are visible by default. An invisible index is not used by the optimizer. Specification of index visibility applies to indexes other than primary keys (either explicit or implicit). For more information, see Section 10.3.12, “Invisible Indexes”. 
- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEoptions are used to specify index attributes for primary and secondary storage engines. The options are reserved for future use.
 - For more information about permissible - index_optionvalues, see Section 15.1.15, “CREATE INDEX Statement”. For more information about indexes, see Section 10.3.1, “How MySQL Uses Indexes”.
- For - reference_definitionsyntax details and examples, see Section 15.1.20.5, “FOREIGN KEY Constraints”.- InnoDBand- NDBtables support checking of foreign key constraints. The columns of the referenced table must always be explicitly named. Both- ON DELETEand- ON UPDATEactions on foreign keys are supported. For more detailed information and examples, see Section 15.1.20.5, “FOREIGN KEY Constraints”.- For other storage engines, MySQL Server parses and ignores the - FOREIGN KEYsyntax in- CREATE TABLEstatements.Important- For users familiar with the ANSI/ISO SQL Standard, please note that no storage engine, including - InnoDB, recognizes or enforces the- MATCHclause used in referential integrity constraint definitions. Use of an explicit- MATCHclause does not have the specified effect, and also causes- ON DELETEand- ON UPDATEclauses to be ignored. For these reasons, specifying- MATCHshould be avoided.- The - MATCHclause in the SQL standard controls how- NULLvalues in a composite (multiple-column) foreign key are handled when comparing to a primary key.- InnoDBessentially implements the semantics defined by- MATCH SIMPLE, which permit a foreign key to be all or partially- NULL. 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, - InnoDBdoes not enforce any requirement that the referenced columns be declared- UNIQUEor- NOT NULL. The handling of foreign key references to nonunique keys or keys that contain- NULLvalues is not well defined for operations such as- UPDATEor- DELETE CASCADE. You are advised to use foreign keys that reference only keys that are both- UNIQUE(or- PRIMARY) and- NOT NULL.- MySQL parses but ignores “inline - REFERENCESspecifications” (as defined in the SQL standard) where the references are defined as part of the column specification. MySQL accepts- REFERENCESclauses only when specified as part of a separate- FOREIGN KEYspecification. For more information, see Section 1.7.2.3, “FOREIGN KEY Constraint Differences”.
- For information about the - RESTRICT,- CASCADE,- SET NULL,- NO ACTION, and- SET DEFAULToptions, see Section 15.1.20.5, “FOREIGN KEY Constraints”.
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. See Chapter 17, The InnoDB Storage Engine, and in particular Section 17.1, “Introduction to InnoDB” if you have MySQL experience but are new to - InnoDB.- MyISAM- The binary portable storage engine that is primarily used for read-only or read-mostly workloads. See Section 18.2, “The MyISAM Storage Engine”. - MEMORY- The data for this storage engine is stored only in memory. See Section 18.3, “The MEMORY Storage Engine”. - CSV- Tables that store rows in comma-separated values format. See Section 18.4, “The CSV Storage Engine”. - ARCHIVE- The archiving storage engine. See Section 18.5, “The ARCHIVE Storage Engine”. - EXAMPLE- An example engine. See Section 18.9, “The EXAMPLE Storage Engine”. - FEDERATED- Storage engine that accesses remote tables. See Section 18.8, “The FEDERATED Storage Engine”. - HEAP- This is a synonym for - MEMORY.- MERGE- A collection of - MyISAMtables used as one table. Also known as- MRG_MyISAM. See Section 18.7, “The MERGE Storage Engine”.- NDB- Clustered, fault-tolerant, memory-based tables, supporting transactions and foreign keys. Also known as - NDBCLUSTER. See Chapter 25, MySQL NDB Cluster 8.4.- 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_SUBSTITUTIONfrom the server SQL mode (see Section 7.1.11, “Server SQL Modes”) so that MySQL allows substitution of the specified engine with the default storage engine instead. Normally in such cases, this is- InnoDB, which is the default value for the- default_storage_enginesystem variable. When- NO_ENGINE_SUBSTITUTIONis disabled, a warning occurs if the storage engine specification is not honored.
- AUTOEXTEND_SIZE- Defines the amount by which - InnoDBextends the size of the tablespace when it becomes full. The setting must be a multiple of 4MB. The default setting is 0, which causes the tablespace to be extended according to the implicit default behavior. For more information, see Section 17.6.3.9, “Tablespace AUTOEXTEND_SIZE Configuration”.
- AUTO_INCREMENT- The initial - AUTO_INCREMENTvalue for the table. In MySQL 8.4, this works for- MyISAM,- MEMORY,- InnoDB, and- ARCHIVEtables. To set the first auto-increment value for engines that do not support the- AUTO_INCREMENTtable 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_INCREMENTtable option in- CREATE TABLEstatements, you can also use- ALTER TABLEto reset the- tbl_nameAUTO_INCREMENT =- N- AUTO_INCREMENTvalue. The value cannot be set lower than the maximum value currently in the column.
- AVG_ROW_LENGTH- An approximation of the average row length for your table. You need to set this only for large tables with variable-size rows. - When you create a - MyISAMtable, MySQL uses the product of the- MAX_ROWSand- AVG_ROW_LENGTHoptions to decide how big the resulting table is. If you don't specify either option, the maximum size for- MyISAMdata and index files is 256TB by default. (If your operating system does not support files that large, table sizes are constrained by the file size limit.) If you want to keep down the pointer sizes to make the index smaller and faster and you don't really need big files, you can decrease the default pointer size by setting the- myisam_data_pointer_sizesystem variable. (See Section 7.1.8, “Server System Variables”.) If you want all your tables to be able to grow above the default limit and are willing to have your tables slightly slower and larger than necessary, you can increase the default pointer size by setting this variable. Setting the value to 7 permits table sizes up to 65,536TB.
- [DEFAULT] CHARACTER SET- Specifies a default character set for the table. - CHARSETis a synonym for- CHARACTER SET. If the character set name is- DEFAULT, the database character set is used.
- CHECKSUM- Set this to 1 if you want MySQL to maintain a live checksum for all rows (that is, a checksum that MySQL updates automatically as the table changes). This makes the table a little slower to update, but also makes it easier to find corrupted tables. The - CHECKSUM TABLEstatement reports the checksum. (- MyISAMonly.)
- [DEFAULT] COLLATE- Specifies a default collation for the table. 
- COMMENT- A comment for the table, up to 2048 characters long. - You can set the - InnoDB- MERGE_THRESHOLDvalue for a table using the- table_option- COMMENTclause. See Section 17.8.11, “Configuring the Merge Threshold for Index Pages”.- Setting NDB_TABLE options. The table comment in a - CREATE TABLEthat creates an- NDBtable or an- ALTER TABLEstatement which alters one can also be used to specify one to four of the- NDB_TABLEoptions- NOLOGGING,- READ_BACKUP,- PARTITION_BALANCE, or- FULLY_REPLICATEDas a set of name-value pairs, separated by commas if need be, immediately following the string- NDB_TABLE=that begins the quoted comment text. An example statement using this syntax is shown here (emphasized text):- CREATE TABLE t1 ( c1 INT NOT NULL AUTO_INCREMENT PRIMARY KEY, c2 VARCHAR(100), c3 VARCHAR(100) ) ENGINE=NDB COMMENT="NDB_TABLE=READ_BACKUP=0,PARTITION_BALANCE=FOR_RP_BY_NODE";- Spaces are not permitted within the quoted string. The string is case-insensitive. - The comment is displayed as part of the output of - SHOW CREATE TABLE. The text of the comment is also available as the TABLE_COMMENT column of the MySQL Information Schema- TABLEStable.- This comment syntax is also supported with - ALTER TABLEstatements for- NDBtables. Keep in mind that a table comment used with- ALTER TABLEreplaces any existing comment which the table might have had previously.- Setting the - MERGE_THRESHOLDoption in table comments is not supported for- NDBtables (it is ignored).- For complete syntax information and examples, see Section 15.1.20.12, “Setting NDB Comment Options”. 
- COMPRESSION- The compression algorithm used for page level compression for - InnoDBtables. Supported values include- Zlib,- LZ4, and- None. The- COMPRESSIONattribute was introduced with the transparent page compression feature. Page compression is only supported with- InnoDBtables that reside in file-per-table tablespaces, and is only available on Linux and Windows platforms that support sparse files and hole punching. For more information, see Section 17.9.2, “InnoDB Page Compression”.
- CONNECTION- The connection string for a - FEDERATEDtable.Note- Older versions of MySQL used a - COMMENToption for the connection string.
- DATA DIRECTORY,- INDEX DIRECTORY- For - InnoDB, the- DATA DIRECTORY='clause permits creating tables outside of the data directory. The- directory'- innodb_file_per_tablevariable must be enabled to use the- DATA DIRECTORYclause. The full directory path must be specified, and known to- InnoDB. For more information, see Section 17.6.1.2, “Creating Tables Externally”.- When creating - MyISAMtables, you can use the- DATA DIRECTORY='clause, the- directory'- INDEX DIRECTORY='clause, or both. They specify where to put a- directory'- MyISAMtable's data file and index file, respectively. Unlike- InnoDBtables, MySQL does not create subdirectories that correspond to the database name when creating a- MyISAMtable with a- DATA DIRECTORYor- INDEX DIRECTORYoption. Files are created in the directory that is specified.- You must have the - FILEprivilege to use the- DATA DIRECTORYor- INDEX DIRECTORYtable option.Important- Table-level - DATA DIRECTORYand- INDEX DIRECTORYoptions are ignored for partitioned tables. (Bug #32091)- These options work only when you are not using the - --skip-symbolic-linksoption. Your operating system must also have a working, thread-safe- realpath()call. See Section 10.12.2.2, “Using Symbolic Links for MyISAM Tables on Unix”, for more complete information.- If a - MyISAMtable is created with no- DATA DIRECTORYoption, the- .MYDfile is created in the database directory. By default, if- MyISAMfinds an existing- .MYDfile in this case, it overwrites it. The same applies to- .MYIfiles for tables created with no- INDEX DIRECTORYoption. To suppress this behavior, start the server with the- --keep_files_on_createoption, in which case- MyISAMdoes not overwrite existing files and returns an error instead.- If a - MyISAMtable is created with a- DATA DIRECTORYor- INDEX DIRECTORYoption and an existing- .MYDor- .MYIfile is found,- MyISAMalways returns an error, and does not overwrite a file in the specified directory.Important- You cannot use path names that contain the MySQL data directory with - DATA DIRECTORYor- INDEX DIRECTORY. This includes partitioned tables and individual table partitions. (See Bug #32167.)
- DELAY_KEY_WRITE- Set this to 1 if you want to delay key updates for the table until the table is closed. See the description of the - delay_key_writesystem variable in Section 7.1.8, “Server System Variables”. (- MyISAMonly.)
- ENCRYPTION- The - ENCRYPTIONclause enables or disables page-level data encryption for an- InnoDBtable. A keyring plugin must be installed and configured before encryption can be enabled. The- ENCRYPTIONclause can be specified when creating a table in an a file-per-table tablespace, or when creating a table in a general tablespace.- The - ENCRYPTIONoption is supported only by the- InnoDBstorage engine; thus it works only if the default storage engine is- InnoDB, or if the- CREATE TABLEstatement also specifies- ENGINE=InnoDB. Otherwise the statement is rejected with- ER_CHECK_NOT_IMPLEMENTED.- A table inherits the default schema encryption if an - ENCRYPTIONclause is not specified. If the- table_encryption_privilege_checkvariable is enabled, the- TABLE_ENCRYPTION_ADMINprivilege is required to create a table with an- ENCRYPTIONclause setting that differs from the default schema encryption. When creating a table in a general tablespace, table and tablespace encryption must match.- Specifying an - ENCRYPTIONclause with a value other than- 'N'or- ''is not permitted when using a storage engine that does not support encryption.- For more information, see Section 17.13, “InnoDB Data-at-Rest Encryption”. 
- The - ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEoptions are used to specify table attributes for primary and secondary storage engines. The options are reserved for future use.- The value assigned to either of these options must be a string literal containing a valid JSON document or an empty string (''). Invalid JSON is rejected. - CREATE TABLE t1 (c1 INT) ENGINE_ATTRIBUTE='{"key":"value"}';- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEvalues can be repeated without error. In this case, the last specified value is used.- ENGINE_ATTRIBUTEand- SECONDARY_ENGINE_ATTRIBUTEvalues are not checked by the server, nor are they cleared when the table's storage engine is changed.
- INSERT_METHOD- If you want to insert data into a - MERGEtable, you must specify with- INSERT_METHODthe table into which the row should be inserted.- INSERT_METHODis an option useful for- MERGEtables only. Use a value of- FIRSTor- LASTto have inserts go to the first or last table, or a value of- NOto prevent inserts. See Section 18.7, “The MERGE Storage Engine”.
- KEY_BLOCK_SIZE- For - MyISAMtables,- KEY_BLOCK_SIZEoptionally specifies the size in bytes to use for index key blocks. The value is treated as a hint; a different size could be used if necessary. A- KEY_BLOCK_SIZEvalue specified for an individual index definition overrides the table-level- KEY_BLOCK_SIZEvalue.- For - InnoDBtables,- KEY_BLOCK_SIZEspecifies the page size in kilobytes to use for compressed- InnoDBtables. The- KEY_BLOCK_SIZEvalue is treated as a hint; a different size could be used by- InnoDBif necessary.- KEY_BLOCK_SIZEcan only be less than or equal to the- innodb_page_sizevalue. A value of 0 represents the default compressed page size, which is half of the- innodb_page_sizevalue. Depending on- innodb_page_size, possible- KEY_BLOCK_SIZEvalues include 0, 1, 2, 4, 8, and 16. See Section 17.9.1, “InnoDB Table Compression” for more information.- Oracle recommends enabling - innodb_strict_modewhen specifying- KEY_BLOCK_SIZEfor- InnoDBtables. When- innodb_strict_modeis enabled, specifying an invalid- KEY_BLOCK_SIZEvalue returns an error. If- innodb_strict_modeis disabled, an invalid- KEY_BLOCK_SIZEvalue results in a warning, and the- KEY_BLOCK_SIZEoption is ignored.- The - Create_optionscolumn in response to- SHOW TABLE STATUSreports the actual- KEY_BLOCK_SIZEused by the table, as does- SHOW CREATE TABLE.- InnoDBonly supports- KEY_BLOCK_SIZEat the table level.- KEY_BLOCK_SIZEis not supported with 32KB and 64KB- innodb_page_sizevalues.- InnoDBtable compression does not support these pages sizes.- InnoDBdoes not support the- KEY_BLOCK_SIZEoption when creating temporary tables.
- MAX_ROWS- The maximum number of rows you plan to store in the table. This is not a hard limit, but rather a hint to the storage engine that the table must be able to store at least this many rows. Important- The use of - MAX_ROWSwith- NDBtables to control the number of table partitions is deprecated. It remains supported in later versions for backward compatibility, but is subject to removal in a future release. Use PARTITION_BALANCE instead; see Setting NDB_TABLE options.- The - NDBstorage engine treats this value as a maximum. If you plan to create very large NDB Cluster tables (containing millions of rows), you should use this option to insure that- NDBallocates sufficient number of index slots in the hash table used for storing hashes of the table's primary keys by setting- MAX_ROWS = 2 *, where- rows- rowsis the number of rows that you expect to insert into the table.- The maximum - MAX_ROWSvalue is 4294967295; larger values are truncated to this limit.
- MIN_ROWS- The minimum number of rows you plan to store in the table. The - MEMORYstorage engine uses this option as a hint about memory use.
- PACK_KEYS- Takes effect only with - MyISAMtables. Set this option to 1 if you want to have smaller indexes. This usually makes updates slower and reads faster. Setting the option to 0 disables all packing of keys. Setting it to- DEFAULTtells the storage engine to pack only long- CHAR,- VARCHAR,- BINARY, or- VARBINARYcolumns.- If you do not use - PACK_KEYS, the default is to pack strings, but not numbers. If you use- PACK_KEYS=1, numbers are packed as well.- When packing binary number keys, MySQL uses prefix compression: - Every key needs one extra byte to indicate how many bytes of the previous key are the same for the next key. 
- The pointer to the row is stored in high-byte-first order directly after the key, to improve compression. 
 - This means that if you have many equal keys on two consecutive rows, all following “same” keys usually only take two bytes (including the pointer to the row). Compare this to the ordinary case where the following keys takes - storage_size_for_key + pointer_size(where the pointer size is usually 4). Conversely, you get a significant benefit from prefix compression only if you have many numbers that are the same. If all keys are totally different, you use one byte more per key, if the key is not a key that can have- NULLvalues. (In this case, the packed key length is stored in the same byte that is used to mark if a key is- NULL.)
- PASSWORD- This option is unused. 
- ROW_FORMAT- Defines the physical format in which the rows are stored. - When creating a table with strict mode disabled, the storage engine's default row format is used if the specified row format is not supported. The actual row format of the table is reported in the - Row_formatcolumn in response to- SHOW TABLE STATUS. The- Create_optionscolumn shows the row format that was specified in the- CREATE TABLEstatement, as does- SHOW CREATE TABLE.- Row format choices differ depending on the storage engine used for the table. - For - InnoDBtables:- The default row format is defined by - innodb_default_row_format, which has a default setting of- DYNAMIC. The default row format is used when the- ROW_FORMAToption is not defined or when- ROW_FORMAT=DEFAULTis used.- If the - ROW_FORMAToption is not defined, or if- ROW_FORMAT=DEFAULTis used, operations that rebuild a table also silently change the row format of the table to the default defined by- innodb_default_row_format. For more information, see Defining the Row Format of a Table.
- For more efficient - InnoDBstorage of data types, especially- BLOBtypes, use the- DYNAMIC. See DYNAMIC Row Format for requirements associated with the- DYNAMICrow format.
- To enable compression for - InnoDBtables, specify- ROW_FORMAT=COMPRESSED. The- ROW_FORMAT=COMPRESSEDoption is not supported when creating temporary tables. See Section 17.9, “InnoDB Table and Page Compression” for requirements associated with the- COMPRESSEDrow format.
- The row format used in older versions of MySQL can still be requested by specifying the - REDUNDANTrow format.
- When you specify a non-default - ROW_FORMATclause, consider also enabling the- innodb_strict_modeconfiguration option.
- ROW_FORMAT=FIXEDis not supported. If- ROW_FORMAT=FIXEDis specified while- innodb_strict_modeis disabled,- InnoDBissues a warning and assumes- ROW_FORMAT=DYNAMIC. If- ROW_FORMAT=FIXEDis specified while- innodb_strict_modeis enabled, which is the default,- InnoDBreturns an error.
- For additional information about - InnoDBrow formats, see Section 17.10, “InnoDB Row Formats”.
 - For - MyISAMtables, the option value can be- FIXEDor- DYNAMICfor static or variable-length row format. myisampack sets the type to- COMPRESSED. See Section 18.2.3, “MyISAM Table Storage Formats”.- For - NDBtables, the default- ROW_FORMATis- DYNAMIC.
- START TRANSACTION- This is an internal-use table option, used to permit - CREATE TABLE ... SELECTto be logged as a single, atomic transaction in the binary log when using row-based replication with a storage engine that supports atomic DDL. Only- BINLOG,- COMMIT, and- ROLLBACKstatements are permitted after- CREATE TABLE ... START TRANSACTION. For related information, see Section 15.1.1, “Atomic Data Definition Statement Support”.
- STATS_AUTO_RECALC- Specifies whether to automatically recalculate persistent statistics for an - InnoDBtable. The value- DEFAULTcauses the persistent statistics setting for the table to be determined by the- innodb_stats_auto_recalcconfiguration option. The value- 1causes statistics to be recalculated when 10% of the data in the table has changed. The value- 0prevents automatic recalculation for this table; with this setting, issue an- ANALYZE TABLEstatement to recalculate the statistics after making substantial changes to the table. For more information about the persistent statistics feature, see Section 17.8.10.1, “Configuring Persistent Optimizer Statistics Parameters”.
- STATS_PERSISTENT- Specifies whether to enable persistent statistics for an - InnoDBtable. The value- DEFAULTcauses the persistent statistics setting for the table to be determined by the- innodb_stats_persistentconfiguration option. The value- 1enables persistent statistics for the table, while the value- 0turns off this feature. After enabling persistent statistics through a- CREATE TABLEor- ALTER TABLEstatement, issue an- ANALYZE TABLEstatement to calculate the statistics, after loading representative data into the table. For more information about the persistent statistics feature, see Section 17.8.10.1, “Configuring Persistent Optimizer Statistics Parameters”.
- STATS_SAMPLE_PAGES- The number of index pages to sample when estimating cardinality and other statistics for an indexed column, such as those calculated by - ANALYZE TABLE. For more information, see Section 17.8.10.1, “Configuring Persistent Optimizer Statistics Parameters”.
- TABLESPACE- The - TABLESPACEclause can be used to create an- InnoDBtable in an existing general tablespace, a file-per-table tablespace, or the system tablespace.- CREATE TABLE tbl_name ... TABLESPACE [=] tablespace_name- The general tablespace that you specify must exist prior to using the - TABLESPACEclause. For information about general tablespaces, see Section 17.6.3.3, “General Tablespaces”.- The - tablespace_name- To create a table in the system tablespace, specify - innodb_systemas the tablespace name.- CREATE TABLE tbl_name ... TABLESPACE [=] innodb_system- Using - TABLESPACE [=] innodb_system, you can place a table of any uncompressed row format in the system tablespace regardless of the- innodb_file_per_tablesetting. For example, you can add a table with- ROW_FORMAT=DYNAMICto the system tablespace using- TABLESPACE [=] innodb_system.- To create a table in a file-per-table tablespace, specify - innodb_file_per_tableas the tablespace name.- CREATE TABLE tbl_name ... TABLESPACE [=] innodb_file_per_tableNote- If - innodb_file_per_tableis enabled, you need not specify- TABLESPACE=innodb_file_per_tableto create an- InnoDBfile-per-table tablespace.- InnoDBtables are created in file-per-table tablespaces by default when- innodb_file_per_tableis enabled.- The - DATA DIRECTORYclause is permitted with- CREATE TABLE ... TABLESPACE=innodb_file_per_tablebut is otherwise not supported for use in combination with the- TABLESPACEclause. The directory specified in a- DATA DIRECTORYclause must be known to- InnoDB. For more information, see Using the DATA DIRECTORY Clause.Note- Support for - TABLESPACE = innodb_file_per_tableand- TABLESPACE = innodb_temporaryclauses with- CREATE TEMPORARY TABLEis deprecated; expect it to be removed in a future version of MySQL.- The - STORAGEtable option is employed only with- NDBtables.- STORAGEdetermines the type of storage used, and can be either of- DISKor- MEMORY.- TABLESPACE ... STORAGE DISKassigns a table to an NDB Cluster Disk Data tablespace.- STORAGE DISKcannot be used in- CREATE TABLEunless preceded by- TABLESPACE- tablespace_name.- For - STORAGE MEMORY, the tablespace name is optional, thus, you can use- TABLESPACEor simply- tablespace_nameSTORAGE MEMORY- STORAGE MEMORYto specify explicitly that the table is in-memory.- See Section 25.6.11, “NDB Cluster Disk Data Tables”, for more information. 
- Used to access a collection of identical - MyISAMtables as one. This works only with- MERGEtables. See Section 18.7, “The MERGE Storage Engine”.- You must have - SELECT,- UPDATE, and- DELETEprivileges for the tables you map to a- MERGEtable.Note- Formerly, all tables used had to be in the same database as the - MERGEtable itself. This restriction no longer applies.
Table Partitioning
      partition_options can be used to
      control partitioning of the table created with
      CREATE TABLE.
    
      Not all options shown in the syntax for
      partition_options at the beginning of
      this section are available for all partitioning types. Please see
      the listings for the following individual types for information
      specific to each type, and see Chapter 26, Partitioning, for
      more complete information about the workings of and uses for
      partitioning in MySQL, as well as additional examples of table
      creation and other statements relating to MySQL partitioning.
    
Partitions can be modified, merged, added to tables, and dropped from tables. For basic information about the MySQL statements to accomplish these tasks, see Section 15.1.9, “ALTER TABLE Statement”. For more detailed descriptions and examples, see Section 26.3, “Partition Management”.
- PARTITION BY- If used, a - partition_optionsclause begins with- PARTITION BY. This clause contains the function that is used to determine the partition; the function returns an integer value ranging from 1 to- num, where- numis the number of partitions. (The maximum number of user-defined partitions which a table may contain is 1024; the number of subpartitions—discussed later in this section—is included in this maximum.)Note- The expression ( - expr) used in a- PARTITION BYclause cannot refer to any columns not in the table being created; such references are specifically not permitted and cause the statement to fail with an error. (Bug #29444)
- HASH(- expr)- Hashes one or more columns to create a key for placing and locating rows. - expris an expression using one or more table columns. This can be any valid MySQL expression (including MySQL functions) that yields a single integer value. For example, these are both valid- CREATE TABLEstatements using- PARTITION BY HASH:- CREATE TABLE t1 (col1 INT, col2 CHAR(5)) PARTITION BY HASH(col1); CREATE TABLE t1 (col1 INT, col2 CHAR(5), col3 DATETIME) PARTITION BY HASH ( YEAR(col3) );- You may not use either - VALUES LESS THANor- VALUES INclauses with- PARTITION BY HASH.- PARTITION BY HASHuses the remainder of- exprdivided by the number of partitions (that is, the modulus). For examples and additional information, see Section 26.2.4, “HASH Partitioning”.- The - LINEARkeyword entails a somewhat different algorithm. In this case, the number of the partition in which a row is stored is calculated as the result of one or more logical- ANDoperations. For discussion and examples of linear hashing, see Section 26.2.4.1, “LINEAR HASH Partitioning”.
- KEY(- column_list)- This is similar to - HASH, except that MySQL supplies the hashing function so as to guarantee an even data distribution. The- column_listargument is simply a list of 1 or more table columns (maximum: 16). This example shows a simple table partitioned by key, with 4 partitions:- CREATE TABLE tk (col1 INT, col2 CHAR(5), col3 DATE) PARTITION BY KEY(col3) PARTITIONS 4;- For tables that are partitioned by key, you can employ linear partitioning by using the - LINEARkeyword. This has the same effect as with tables that are partitioned by- HASH. That is, the partition number is found using the- &operator rather than the modulus (see Section 26.2.4.1, “LINEAR HASH Partitioning”, and Section 26.2.5, “KEY Partitioning”, for details). This example uses linear partitioning by key to distribute data between 5 partitions:- CREATE TABLE tk (col1 INT, col2 CHAR(5), col3 DATE) PARTITION BY LINEAR KEY(col3) PARTITIONS 5;- The - ALGORITHM={1 | 2}option is supported with- [SUB]PARTITION BY [LINEAR] KEY.- ALGORITHM=1causes the server to use the same key-hashing functions as MySQL 5.1;- ALGORITHM=2means that the server employs the key-hashing functions implemented and used by default for new- KEYpartitioned tables in MySQL 5.5 and later. (Partitioned tables created with the key-hashing functions employed in MySQL 5.5 and later cannot be used by a MySQL 5.1 server.) Not specifying the option has the same effect as using- ALGORITHM=2. This option is intended for use chiefly when upgrading or downgrading- [LINEAR] KEYpartitioned tables between MySQL 5.1 and later MySQL versions, or for creating tables partitioned by- KEYor- LINEAR KEYon a MySQL 5.5 or later server which can be used on a MySQL 5.1 server. For more information, see Section 15.1.9.1, “ALTER TABLE Partition Operations”.- mysqldump writes this option encased in versioned comments. - ALGORITHM=1is shown when necessary in the output of- SHOW CREATE TABLEusing versioned comments in the same manner as mysqldump.- ALGORITHM=2is always omitted from- SHOW CREATE TABLEoutput, even if this option was specified when creating the original table.- You may not use either - VALUES LESS THANor- VALUES INclauses with- PARTITION BY KEY.
- RANGE(- expr)- In this case, - exprshows a range of values using a set of- VALUES LESS THANoperators. When using range partitioning, you must define at least one partition using- VALUES LESS THAN. You cannot use- VALUES INwith range partitioning.Note- For tables partitioned by - RANGE,- VALUES LESS THANmust be used with either an integer literal value or an expression that evaluates to a single integer value. In MySQL 8.4, you can overcome this limitation in a table that is defined using- PARTITION BY RANGE COLUMNS, as described later in this section.- Suppose that you have a table that you wish to partition on a column containing year values, according to the following scheme. - Partition Number: - Years Range: - 0 - 1990 and earlier - 1 - 1991 to 1994 - 2 - 1995 to 1998 - 3 - 1999 to 2002 - 4 - 2003 to 2005 - 5 - 2006 and later - A table implementing such a partitioning scheme can be realized by the - CREATE TABLEstatement shown here:- CREATE TABLE t1 ( year_col INT, some_data INT ) PARTITION BY RANGE (year_col) ( PARTITION p0 VALUES LESS THAN (1991), PARTITION p1 VALUES LESS THAN (1995), PARTITION p2 VALUES LESS THAN (1999), PARTITION p3 VALUES LESS THAN (2002), PARTITION p4 VALUES LESS THAN (2006), PARTITION p5 VALUES LESS THAN MAXVALUE );- PARTITION ... VALUES LESS THAN ...statements work in a consecutive fashion.- VALUES LESS THAN MAXVALUEworks to specify “leftover” values that are greater than the maximum value otherwise specified.- VALUES LESS THANclauses work sequentially in a manner similar to that of the- caseportions of a- switch ... caseblock (as found in many programming languages such as C, Java, and PHP). That is, the clauses must be arranged in such a way that the upper limit specified in each successive- VALUES LESS THANis greater than that of the previous one, with the one referencing- MAXVALUEcoming last of all in the list.
- RANGE COLUMNS(- column_list)- This variant on - RANGEfacilitates partition pruning for queries using range conditions on multiple columns (that is, having conditions such as- WHERE a = 1 AND b < 10or- WHERE a = 1 AND b = 10 AND c < 10). It enables you to specify value ranges in multiple columns by using a list of columns in the- COLUMNSclause and a set of column values in each- PARTITION ... VALUES LESS THAN (partition definition clause. (In the simplest case, this set consists of a single column.) The maximum number of columns that can be referenced in the- value_list)- column_listand- value_listis 16.- The - column_listused in the- COLUMNSclause may contain only names of columns; each column in the list must be one of the following MySQL data types: the integer types; the string types; and time or date column types. Columns using- BLOB,- TEXT,- SET,- ENUM,- BIT, or spatial data types are not permitted; columns that use floating-point number types are also not permitted. You also may not use functions or arithmetic expressions in the- COLUMNSclause.- The - VALUES LESS THANclause used in a partition definition must specify a literal value for each column that appears in the- COLUMNS()clause; that is, the list of values used for each- VALUES LESS THANclause must contain the same number of values as there are columns listed in the- COLUMNSclause. An attempt to use more or fewer values in a- VALUES LESS THANclause than there are in the- COLUMNSclause causes the statement to fail with the error Inconsistency in usage of column lists for partitioning.... You cannot use- NULLfor any value appearing in- VALUES LESS THAN. It is possible to use- MAXVALUEmore than once for a given column other than the first, as shown in this example:- CREATE TABLE rc ( a INT NOT NULL, b INT NOT NULL ) PARTITION BY RANGE COLUMNS(a,b) ( PARTITION p0 VALUES LESS THAN (10,5), PARTITION p1 VALUES LESS THAN (20,10), PARTITION p2 VALUES LESS THAN (50,MAXVALUE), PARTITION p3 VALUES LESS THAN (65,MAXVALUE), PARTITION p4 VALUES LESS THAN (MAXVALUE,MAXVALUE) );- Each value used in a - VALUES LESS THANvalue list must match the type of the corresponding column exactly; no conversion is made. For example, you cannot use the string- '1'for a value that matches a column that uses an integer type (you must use the numeral- 1instead), nor can you use the numeral- 1for a value that matches a column that uses a string type (in such a case, you must use a quoted string:- '1').- For more information, see Section 26.2.1, “RANGE Partitioning”, and Section 26.4, “Partition Pruning”. 
- LIST(- expr)- This is useful when assigning partitions based on a table column with a restricted set of possible values, such as a state or country code. In such a case, all rows pertaining to a certain state or country can be assigned to a single partition, or a partition can be reserved for a certain set of states or countries. It is similar to - RANGE, except that only- VALUES INmay be used to specify permissible values for each partition.- VALUES INis used with a list of values to be matched. For instance, you could create a partitioning scheme such as the following:- CREATE TABLE client_firms ( id INT, name VARCHAR(35) ) PARTITION BY LIST (id) ( PARTITION r0 VALUES IN (1, 5, 9, 13, 17, 21), PARTITION r1 VALUES IN (2, 6, 10, 14, 18, 22), PARTITION r2 VALUES IN (3, 7, 11, 15, 19, 23), PARTITION r3 VALUES IN (4, 8, 12, 16, 20, 24) );- When using list partitioning, you must define at least one partition using - VALUES IN. You cannot use- VALUES LESS THANwith- PARTITION BY LIST.Note- For tables partitioned by - LIST, the value list used with- VALUES INmust consist of integer values only. In MySQL 8.4, you can overcome this limitation using partitioning by- LIST COLUMNS, which is described later in this section.
- LIST COLUMNS(- column_list)- This variant on - LISTfacilitates partition pruning for queries using comparison conditions on multiple columns (that is, having conditions such as- WHERE a = 5 AND b = 5or- WHERE a = 1 AND b = 10 AND c = 5). It enables you to specify values in multiple columns by using a list of columns in the- COLUMNSclause and a set of column values in each- PARTITION ... VALUES IN (partition definition clause.- value_list)- The rules governing regarding data types for the column list used in - LIST COLUMNS(and the value list used in- column_list)- VALUES IN(are the same as those for the column list used in- value_list)- RANGE COLUMNS(and the value list used in- column_list)- VALUES LESS THAN(, respectively, except that in the- value_list)- VALUES INclause,- MAXVALUEis not permitted, and you may use- NULL.- There is one important difference between the list of values used for - VALUES INwith- PARTITION BY LIST COLUMNSas opposed to when it is used with- PARTITION BY LIST. When used with- PARTITION BY LIST COLUMNS, each element in the- VALUES INclause must be a set of column values; the number of values in each set must be the same as the number of columns used in the- COLUMNSclause, and the data types of these values must match those of the columns (and occur in the same order). In the simplest case, the set consists of a single column. The maximum number of columns that can be used in the- column_listand in the elements making up the- value_listis 16.- The table defined by the following - CREATE TABLEstatement provides an example of a table using- LIST COLUMNSpartitioning:- CREATE TABLE lc ( a INT NULL, b INT NULL ) PARTITION BY LIST COLUMNS(a,b) ( PARTITION p0 VALUES IN( (0,0), (NULL,NULL) ), PARTITION p1 VALUES IN( (0,1), (0,2), (0,3), (1,1), (1,2) ), PARTITION p2 VALUES IN( (1,0), (2,0), (2,1), (3,0), (3,1) ), PARTITION p3 VALUES IN( (1,3), (2,2), (2,3), (3,2), (3,3) ) );
- PARTITIONS- num- The number of partitions may optionally be specified with a - PARTITIONSclause, where- num- numis the number of partitions. If both this clause and any- PARTITIONclauses are used,- nummust be equal to the total number of any partitions that are declared using- PARTITIONclauses.Note- Whether or not you use a - PARTITIONSclause in creating a table that is partitioned by- RANGEor- LIST, you must still include at least one- PARTITION VALUESclause in the table definition (see below).
- SUBPARTITION BY- A partition may optionally be divided into a number of subpartitions. This can be indicated by using the optional - SUBPARTITION BYclause. Subpartitioning may be done by- HASHor- KEY. Either of these may be- LINEAR. These work in the same way as previously described for the equivalent partitioning types. (It is not possible to subpartition by- LISTor- RANGE.)- The number of subpartitions can be indicated using the - SUBPARTITIONSkeyword followed by an integer value.
- Rigorous checking of the value used in - PARTITIONSor- SUBPARTITIONSclauses is applied and this value must adhere to the following rules:- The value must be a positive, nonzero integer. 
- No leading zeros are permitted. 
- The value must be an integer literal, and cannot not be an expression. For example, - PARTITIONS 0.2E+01is not permitted, even though- 0.2E+01evaluates to- 2. (Bug #15890)
 
- partition_definition- Each partition may be individually defined using a - partition_definitionclause. The individual parts making up this clause are as follows:- PARTITION- partition_name- Specifies a logical name for the partition. 
- VALUES- For range partitioning, each partition must include a - VALUES LESS THANclause; for list partitioning, you must specify a- VALUES INclause for each partition. This is used to determine which rows are to be stored in this partition. See the discussions of partitioning types in Chapter 26, Partitioning, for syntax examples.
- [STORAGE] ENGINE- MySQL accepts a - [STORAGE] ENGINEoption for both- PARTITIONand- SUBPARTITION. Currently, the only way in which this option can be used is to set all partitions or all subpartitions to the same storage engine, and an attempt to set different storage engines for partitions or subpartitions in the same table raises the error ERROR 1469 (HY000): The mix of handlers in the partitions is not permitted in this version of MySQL.
- COMMENT- An optional - COMMENTclause may be used to specify a string that describes the partition. Example:- COMMENT = 'Data for the years previous to 1999'- The maximum length for a partition comment is 1024 characters. 
- DATA DIRECTORYand- INDEX DIRECTORY- DATA DIRECTORYand- INDEX DIRECTORYmay be used to indicate the directory where, respectively, the data and indexes for this partition are to be stored. Both the- data_dir- index_dir- The directory specified in a - DATA DIRECTORYclause must be known to- InnoDB. For more information, see Using the DATA DIRECTORY Clause.- You must have the - FILEprivilege to use the- DATA DIRECTORYor- INDEX DIRECTORYpartition option.- Example: - CREATE TABLE th (id INT, name VARCHAR(30), adate DATE) PARTITION BY LIST(YEAR(adate)) ( PARTITION p1999 VALUES IN (1995, 1999, 2003) DATA DIRECTORY = '/var/appdata/95/data' INDEX DIRECTORY = '/var/appdata/95/idx', PARTITION p2000 VALUES IN (1996, 2000, 2004) DATA DIRECTORY = '/var/appdata/96/data' INDEX DIRECTORY = '/var/appdata/96/idx', PARTITION p2001 VALUES IN (1997, 2001, 2005) DATA DIRECTORY = '/var/appdata/97/data' INDEX DIRECTORY = '/var/appdata/97/idx', PARTITION p2002 VALUES IN (1998, 2002, 2006) DATA DIRECTORY = '/var/appdata/98/data' INDEX DIRECTORY = '/var/appdata/98/idx' );- DATA DIRECTORYand- INDEX DIRECTORYbehave in the same way as in the- CREATE TABLEstatement's- table_optionclause as used for- MyISAMtables.- One data directory and one index directory may be specified per partition. If left unspecified, the data and indexes are stored by default in the table's database directory. - The - DATA DIRECTORYand- INDEX DIRECTORYoptions are ignored for creating partitioned tables if- NO_DIR_IN_CREATEis in effect.
- MAX_ROWSand- MIN_ROWS- May be used to specify, respectively, the maximum and minimum number of rows to be stored in the partition. The values for - max_number_of_rowsand- min_number_of_rowsmust be positive integers. As with the table-level options with the same names, these act only as “suggestions” to the server and are not hard limits.
- TABLESPACE- May be used to designate an - InnoDBfile-per-table tablespace for the partition by specifying- TABLESPACE `innodb_file_per_table`. All partitions must belong to the same storage engine.- Placing - InnoDBtable partitions in shared- InnoDBtablespaces is not supported. Shared tablespaces include the- InnoDBsystem tablespace and general tablespaces.
 
- subpartition_definition- The partition definition may optionally contain one or more - subpartition_definitionclauses. Each of these consists at a minimum of the- SUBPARTITION, where- name- nameis an identifier for the subpartition. Except for the replacement of the- PARTITIONkeyword with- SUBPARTITION, the syntax for a subpartition definition is identical to that for a partition definition.- Subpartitioning must be done by - HASHor- KEY, and can be done only on- RANGEor- LISTpartitions. See Section 26.2.6, “Subpartitioning”.
Partitioning by Generated Columns
Partitioning by generated columns is permitted. For example:
CREATE TABLE t1 (
  s1 INT,
  s2 INT AS (EXP(s1)) STORED
)
PARTITION BY LIST (s2) (
  PARTITION p1 VALUES IN (1)
);
      Partitioning sees a generated column as a regular column, which
      enables workarounds for limitations on functions that are not
      permitted for partitioning (see
      Section 26.6.3, “Partitioning Limitations Relating to Functions”). The
      preceding example demonstrates this technique:
      EXP() cannot be used directly in
      the PARTITION BY clause, but a generated column
      defined using EXP() is permitted.