mysql:error_1071
Differences
This shows you the differences between two versions of the page.
Next revision | Previous revision | ||
mysql:error_1071 [2016/07/01 22:23] – created peter | mysql:error_1071 [2020/07/15 09:30] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== MySQL - Error #1071 - Specified key was too long; max key length is 767 bytes ====== | ====== MySQL - Error #1071 - Specified key was too long; max key length is 767 bytes ====== | ||
- | 767 bytes is the stated prefix limitation for InnoDB tables - its 1,000 bytes long for MyISAM tables. | + | The max key length depends on the character set you use. If you use **latin1** then the largest column you can index is **varchar(767)** with 1 byte equalling 1 character. |
- | When using UTF-8 encoding, the maximum length of the columns in the key needs to factor in the additional | + | * MySQL ' |
+ | * MySQL ' | ||
- | MySQL assumes worst case for the number of bytes per character in the string. | + | Thus, you need to multiply your key field' |
+ | * For **uft8** a maximum of 255 characters can therefore be used for an InnoDB key field because 767/ | ||
+ | * For **uft8mb4** a maximum of 191 characters can therefore be used for an InnoDB key field because 767/ | ||
- | You can get the key to apply by specifying | + | There is also a separate 3072 byte limit per index. The 767 byte limit is per column, |
+ | |||
+ | |||
+ | One workaround for these limits is to only index part of a field, for example, here only 191 characters | ||
<code mysql> | <code mysql> | ||
- | ALTER TABLE `mytable` ADD UNIQUE ( column1(255)); | + | ALTER TABLE `mytable` ADD UNIQUE ( column1(191)); |
</ | </ | ||
- | to | + | |
+ | But what if you want to index more than 767 bytes of a column in InnoDB? | ||
+ | |||
+ | In that case you should consider using **innodb_large_prefix**, | ||
+ | |||
+ | < | ||
+ | The InnoDB internal maximum key length is 3500 bytes, but MySQL itself restricts this to 3072 bytes. | ||
+ | This limit applies to the length of the combined index key in a multi-column index. | ||
+ | </ | ||
+ | |||
+ | |||
+ | ===== Using innodb_large_prefix ===== | ||
+ | |||
+ | Here are a few pre-requisites for using **innodb_large_prefix**: | ||
+ | |||
+ | * At the database level you have to use **innodb_file_format=BARRACUDA** | ||
+ | * At the table level you have to use **ROW_FORMAT=DYNAMIC** or **ROW_FORMAT=COMPRESSED** | ||
+ | |||
+ | The default file format is still Antelope for backwards compatibility, | ||
+ | |||
+ | You can set both **innodb_file_format** and **innodb_large_prefix** dynamically, | ||
+ | |||
+ | Here’s an example. | ||
<code mysql> | <code mysql> | ||
- | ALTER TABLE `mytable` ADD UNIQUE | + | mysql> create table if not exists utf8_test |
+ | -> day date not null, | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> ) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ||
+ | ERROR 1071 (42000): Specified key was too long; max key length is 767 bytes | ||
+ | </ | ||
+ | |||
+ | If I enable **innodb_large_prefix** I can create the table successfully: | ||
+ | |||
+ | <code mysql> | ||
+ | mysql> set global innodb_file_format = BARRACUDA; | ||
+ | Query OK, 0 rows affected (0.00 sec) | ||
+ | |||
+ | mysql> set global innodb_large_prefix = ON; | ||
+ | Query OK, 0 rows affected (0.00 sec) | ||
+ | |||
+ | mysql> create table if not exists utf8_test ( | ||
+ | -> day date not null, | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> ) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC; | ||
+ | Query OK, 0 rows affected (0.02 sec) | ||
+ | </ | ||
+ | |||
+ | The examples are similar for **latin1**, but I can use columns three times as long since it’s a single-byte character set. | ||
+ | |||
+ | |||
+ | <code mysql> | ||
+ | mysql> create table if not exists latin1_test ( | ||
+ | -> day date not null, | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> ) ENGINE=InnoDB DEFAULT CHARSET=latin1; | ||
+ | ERROR 1071 (42000): Specified key was too long; max key length is 767 bytes | ||
+ | |||
+ | mysql> set global innodb_file_format = BARRACUDA; | ||
+ | Query OK, 0 rows affected (0.00 sec) | ||
+ | |||
+ | mysql> set global innodb_large_prefix = ON; | ||
+ | Query OK, 0 rows affected (0.00 sec) | ||
+ | |||
+ | mysql> create table if not exists latin1_test ( | ||
+ | -> day date not null, | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> ) ENGINE=InnoDB DEFAULT CHARSET=latin1 ROW_FORMAT=DYNAMIC; | ||
+ | Query OK, 0 rows affected (0.02 sec) | ||
+ | </ | ||
+ | |||
+ | And here’s what happens if I try to create an index longer than 3072 bytes: | ||
+ | |||
+ | <code mysql> | ||
+ | mysql> create table if not exists long_index_test ( | ||
+ | -> day date not null, | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> | ||
+ | -> ) ENGINE=InnoDB DEFAULT CHARSET=latin1 ROW_FORMAT=DYNAMIC; | ||
+ | ERROR 1071 (42000): Specified key was too long; max key length is 3072 bytes | ||
</ | </ | ||
mysql/error_1071.1467411827.txt.gz · Last modified: 2020/07/15 09:30 (external edit)