sql >> Database >  >> RDS >> Mysql

Is er een formule om de indexgrootte in InnoDB te schatten?

In InnoDB, de PRIMARY KEY is ingebed in de gegevens, dus je kunt het beschouwen als ruimteloos.

Voor een secundaire sleutel... Neem de MyISAM-formule, maar gebruik de kolommen van beide de secundaire sleutel en de PRIMARY KEY . Vermenigvuldig vervolgens met 3. (Er is veel overhead.) Toch kan het antwoord in beide richtingen een factor 2 afwijken.

Merk op dat als je veel secundaire sleutels hebt, de grootte van de PK een groot verschil maakt in de totale ruimte voor de tabel+indexen.

Voorbeeld

SET @db = 'world', @tbl = 'cities';
    SELECT      n_rows AS 'Approx Rows',
                'Data & PK' AS 'Type',
                clustered_index_size * 16384 AS Bytes,
                ROUND(clustered_index_size * 16384 / n_rows) AS 'Bytes/row',
                clustered_index_size AS Pages,
                ROUND(n_rows / clustered_index_size) AS 'Rows/page'
        FROM mysql.innodb_table_stats
        WHERE database_name = @db
          AND table_name = @tbl
    UNION
        SELECT  n_rows,
                'Secondary Indexes' AS 'BTrees',
                sum_of_other_index_sizes * 16384 AS Bytes,
                ROUND(sum_of_other_index_sizes * 16384 / n_rows) AS 'Bytes/row',
                sum_of_other_index_sizes AS Pages,
                ROUND(n_rows / sum_of_other_index_sizes) AS 'Rows/page'
        FROM mysql.innodb_table_stats
        WHERE database_name = @db
          AND table_name = @tbl
          AND sum_of_other_index_sizes > 0
          ;
-- (Percona has a different way.)

Uitgang:

+-------------+-------------------+-----------+-----------+-------+-----------+
| Approx Rows | Type              | Bytes     | Bytes/row | Pages | Rows/page |
+-------------+-------------------+-----------+-----------+-------+-----------+
|     2637973 | Data & PK         | 179077120 |        68 | 10930 |       241 |
|     2637973 | Secondary Indexes | 232341504 |        88 | 14181 |       186 |
+-------------+-------------------+-----------+-----------+-------+-----------+

De tabel heeft twee indexen:

PRIMARY KEY(...)  -- 14 bytes
INDEX(state, population)
INDEX(state, city)
  state CHAR(2) CHARACTER SET ascii -- 2 bytes
  population INT UNSIGNED -- 4 bytes
  city  -- AVG(LENGTH(city)) = 1+9.07 bytes

COUNT(*): 2,699,354  (the InnoDB estimate was not too far from this)

First index:  20    bytes * 2.7M rows = 54MB
Second index: 26.07 bytes * 2.7M rows = 70MB
Total:  124MB
Actual: 232MB
Ratio: 1.9x  (note: I skipped the "/0.67")

Om nog een punt te bewijzen, probeerde ik OPTIMIZE TABLE . De statistieken daarna waren in wezen hetzelfde:

+-------------+-------------------+-----------+-----------+-------+-----------+
| Approx Rows | Type              | Bytes     | Bytes/row | Pages | Rows/page |
+-------------+-------------------+-----------+-----------+-------+-----------+
|     2685828 | Data & PK         | 179077120 |        67 | 10930 |       246 |
|     2685828 | Secondary Indexes | 232341504 |        87 | 14181 |       189 |
+-------------+-------------------+-----------+-----------+-------+-----------+


  1. php uitvoeren via mysql?

  2. GROUP_CONCAT met JOINLEFT in Zend Db Select

  3. MySQL Query Tuning - Waarom is het gebruik van een waarde van een variabele zo veel langzamer dan het gebruik van een letterlijke?

  4. Foutcode:1052. Kolom 'ATTRIBUTE' in veldlijst is dubbelzinnig