欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

MySQL?中?Varchar(50)?和?varchar(500)?區(qū)別介紹

 更新時間:2024年08月01日 09:29:38   作者:數(shù)據(jù)派  
網(wǎng)上說Varchar(50)和varchar(500)存儲空間上是一樣的,真的是這樣嗎,基于性能考慮,是因為過長的字段會影響到查詢性能,本文我將帶著這兩個問題探討驗證一下,需要的朋友可以參考下

問題

我們在設(shè)計表結(jié)構(gòu)的時候,設(shè)計規(guī)范里面有一條如下規(guī)則:對于可變長度的字段,在滿足條件的前提下,盡可能使用較短的變長字段長度。為什么這么規(guī)定,主要基于兩個方面

  • 基于存儲空間的考慮

  • 基于性能的考慮

網(wǎng)上說Varchar(50)和varchar(500)存儲空間上是一樣的,真的是這樣嗎?基于性能考慮,是因為過長的字段會影響到查詢性能?
本文我將帶著這兩個問題探討驗證一下:

驗證存儲空間的區(qū)別

1、準(zhǔn)備兩張表

CREATE TABLE `category_info_varchar_50` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主鍵',
  `name` varchar(50) NOT NULL COMMENT '分類名稱',
  `is_show` tinyint(4) NOT NULL DEFAULT '0' COMMENT '是否展示:0 禁用,1啟用',
  `sort` int(11) NOT NULL DEFAULT '0' COMMENT '序號',
  `deleted` tinyint(1) DEFAULT '0' COMMENT '是否刪除',
  `create_time` datetime NOT NULL COMMENT '創(chuàng)建時間',
  `update_time` datetime NOT NULL COMMENT '更新時間',
  PRIMARY KEY (`id`) USING BTREE,
  KEY `idx_name` (`name`) USING BTREE COMMENT '名稱索引'
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='分類';
CREATE TABLE `category_info_varchar_500` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主鍵',
  `name` varchar(500) NOT NULL COMMENT '分類名稱',
  `is_show` tinyint(4) NOT NULL DEFAULT '0' COMMENT '是否展示:0 禁用,1啟用',
  `sort` int(11) NOT NULL DEFAULT '0' COMMENT '序號',
  `deleted` tinyint(1) DEFAULT '0' COMMENT '是否刪除',
  `create_time` datetime NOT NULL COMMENT '創(chuàng)建時間',
  `update_time` datetime NOT NULL COMMENT '更新時間',
  PRIMARY KEY (`id`) USING BTREE,
  KEY `idx_name` (`name`) USING BTREE COMMENT '名稱索引'
) ENGINE=InnoDB AUTO_INCREMENT=288135 DEFAULT CHARSET=utf8mb4 COMMENT='分類';

2、準(zhǔn)備數(shù)據(jù)

給每張表插入相同的數(shù)據(jù),為了凸顯不同,插入100萬條數(shù)據(jù)

DELIMITER $$
CREATE PROCEDURE batchInsertData(IN total INT)
BEGIN
    DECLARE start_idx INT DEFAULT 1;
    DECLARE end_idx INT;
    DECLARE batch_size INT DEFAULT 500;
    DECLARE insert_values TEXT;
    SET end_idx = LEAST(total, start_idx + batch_size - 1);
    WHILE start_idx <= total DO
        SET insert_values = '';
        WHILE start_idx <= end_idx DO
            SET insert_values = CONCAT(insert_values, CONCAT('(\'name', start_idx, '\', 0, 0, 0, NOW(), NOW()),'));
            SET start_idx = start_idx + 1;
        END WHILE;
        SET insert_values = LEFT(insert_values, LENGTH(insert_values) - 1); -- Remove the trailing comma
        SET @sql = CONCAT('INSERT INTO category_info_varchar_50 (name, is_show, sort, deleted, create_time, update_time) VALUES ', insert_values, ';');
        PREPARE stmt FROM @sql;
        EXECUTE stmt;
      SET @sql = CONCAT('INSERT INTO category_info_varchar_500 (name, is_show, sort, deleted, create_time, update_time) VALUES ', insert_values, ';'); 
      PREPARE stmt FROM @sql;
        EXECUTE stmt;
        SET end_idx = LEAST(total, start_idx + batch_size - 1);
    END WHILE;
END$$
DELIMITER ;
CALL batchInsertData(1000000);

3、驗證存儲空間

查詢第一張表SQL

SELECT
    table_schema AS "數(shù)據(jù)庫",
    table_name AS "表名",
    table_rows AS "記錄數(shù)",
    TRUNCATE ( data_length / 1024 / 1024, 2 )  AS "數(shù)據(jù)容量(MB)",
    TRUNCATE ( index_length / 1024 / 1024, 2 )  AS "索引容量(MB)" 
FROM
    information_schema.TABLES 
WHERE
    table_schema = 'test_mysql_field' 
and TABLE_NAME = 'category_info_varchar_50'
ORDER BY
    data_length DESC,
    index_length DESC;

查詢結(jié)果

查詢第二張表SQL

SELECT
    table_schema AS "數(shù)據(jù)庫",
    table_name AS "表名",
    table_rows AS "記錄數(shù)",
    TRUNCATE ( data_length / 1024 / 1024, 2 )  AS "數(shù)據(jù)容量(MB)",
    TRUNCATE ( index_length / 1024 / 1024, 2 )  AS "索引容量(MB)" 
FROM
    information_schema.TABLES 
WHERE
    table_schema = 'test_mysql_field' 
and TABLE_NAME = 'category_info_varchar_500'
ORDER BY
    data_length DESC,
    index_length DESC;

查詢結(jié)果

4、結(jié)論

兩張表在占用空間上確實是一樣的,并無差別。

驗證性能區(qū)別

1、驗證索引覆蓋查詢

select name from category_info_varchar_50 where name = 'name100000'
-- 耗時0.012s
select name from category_info_varchar_500 where name = 'name100000'
-- 耗時0.012s
select name from category_info_varchar_50 order by name;
-- 耗時0.370s
select name from category_info_varchar_500 order by name;
-- 耗時0.379s

通過索引覆蓋查詢性能差別不大

2、驗證索引查詢

select * from category_info_varchar_50 where name = 'name100000'
--耗時 0.012s
select * from category_info_varchar_500 where name = 'name100000'
--耗時 0.012s
select * from category_info_varchar_50 where name in('name100','name1000','name100000','name10000','name1100000',
'name200','name2000','name200000','name20000','name2200000','name300','name3000','name300000','name30000','name3300000',
'name400','name4000','name400000','name40000','name4400000','name500','name5000','name500000','name50000','name5500000',
'name600','name6000','name600000','name60000','name6600000','name700','name7000','name700000','name70000','name7700000','name800',
'name8000','name800000','name80000','name6600000','name900','name9000','name900000','name90000','name9900000') 
-- 耗時 0.011s -0.014s 
-- 增加 order by name 耗時 0.012s - 0.015s
select * from category_info_varchar_50 where name in('name100','name1000','name100000','name10000','name1100000',
'name200','name2000','name200000','name20000','name2200000','name300','name3000','name300000','name30000','name3300000',
'name400','name4000','name400000','name40000','name4400000','name500','name5000','name500000','name50000','name5500000',
'name600','name6000','name600000','name60000','name6600000','name700','name7000','name700000','name70000','name7700000','name800',
'name8000','name800000','name80000','name6600000','name900','name9000','name900000','name90000','name9900000') 
-- 耗時  0.012s -0.014s 
-- 增加 order by name 耗時 0.014s - 0.017s

索引范圍查詢性能基本相同, 增加了order By后開始有一定性能差別;

3、驗證全表查詢和排序

全表無排序

全表有排序

select * from category_info_varchar_50 order by  name ;
--耗時 1.498s
select * from category_info_varchar_500 order by  name  ;
--耗時 4.875s

結(jié)論:

全表掃描無排序情況下,兩者性能無差異,在全表有排序的情況下, 兩種性能差異巨大;

分析原因

varchar50 全表執(zhí)行sql分析

我發(fā)現(xiàn)86%的時花在數(shù)據(jù)傳輸上,接下來我們看狀態(tài)部分,關(guān)注Created_tmp_files和sort_merge_passes

Created_tmp_files為3

sort_merge_passes為95

varchar500 全表執(zhí)行sql分析

增加了臨時表排序

Created_tmp_files 為 4

sort_merge_passes為645

關(guān)于sort_merge_passes, Mysql給出了如下描述:

Number of merge passes that the sort algorithm has had to do. If this value is large, you may want to increase the value of the sort_buffer_size.

其實sort_merge_passes對應(yīng)的就是MySQL做歸并排序的次數(shù),也就是說,如果sort_merge_passes值比較大,說明sort_buffer和要排序的數(shù)據(jù)差距越大,我們可以通過增大sort_buffer_size或者讓填入sort_buffer_size的鍵值對更小來緩解sort_merge_passes歸并排序的次數(shù)。

最終結(jié)論

至此,我們不難發(fā)現(xiàn),當(dāng)我們最該字段進(jìn)行排序操作的時候,Mysql會根據(jù)該字段的設(shè)計的長度進(jìn)行內(nèi)存預(yù)估,如果設(shè)計過大的可變長度,會導(dǎo)致內(nèi)存預(yù)估的值超出sort_buffer_size的大小,導(dǎo)致mysql采用磁盤臨時文件排序,最終影響查詢性能。

相關(guān)文章

  • mysql中的int(5)到底有是多長

    mysql中的int(5)到底有是多長

    這篇文章主要介紹了mysql中的int(5)到底有是多長,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • sql format()函數(shù)的用法及簡單實例

    sql format()函數(shù)的用法及簡單實例

    下面小編就為大家?guī)硪黄猻ql format函數(shù)()的用法及簡單實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-05-05
  • MySQL InnoDB存儲引擎的深入探秘

    MySQL InnoDB存儲引擎的深入探秘

    這篇文章主要給大家介紹了關(guān)于MySQL InnoDB存儲引擎的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-02-02
  • Mysql5.7定時備份的實現(xiàn)

    Mysql5.7定時備份的實現(xiàn)

    這篇文章主要介紹了Mysql5.7定時備份的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • 如何測試mysql觸發(fā)器和存儲過程

    如何測試mysql觸發(fā)器和存儲過程

    本文將詳細(xì)介紹怎樣mysql觸發(fā)器和存儲過程,需要了解的朋友可以詳細(xì)參考下
    2012-11-11
  • Mysql 5.7.18 解壓版下載安裝及啟動mysql服務(wù)的圖文詳解

    Mysql 5.7.18 解壓版下載安裝及啟動mysql服務(wù)的圖文詳解

    這篇文章主要介紹了Mysql 5.7.18 解壓版下載安裝及啟動mysql服務(wù)的圖文詳解,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2017-05-05
  • mysql5.7.42到mysql8.2.0的升級(rpm方式)

    mysql5.7.42到mysql8.2.0的升級(rpm方式)

    隨著數(shù)據(jù)量的增長和業(yè)務(wù)需求的變更,我們可能需要升級MySQL,本文主要介紹了mysql5.7.42到mysql8.2.0的升級(rpm方式),具有一定的參考價值,感興趣的可以了解一下
    2024-03-03
  • mysql報錯:MySQL server version for the right syntax to use near type=InnoDB的解決方法

    mysql報錯:MySQL server version for the right syntax to use nea

    這篇文章主要介紹了mysql報錯:MySQL server version for the right syntax to use near type=InnoDB的解決方法,涉及MySQL語句的使用技巧,需要的朋友可以參考下
    2016-01-01
  • 很全面的MySQL處理重復(fù)數(shù)據(jù)代碼

    很全面的MySQL處理重復(fù)數(shù)據(jù)代碼

    這篇文章主要為大家詳細(xì)介紹了MySQL處理重復(fù)數(shù)據(jù)的實現(xiàn)代碼,如何防止數(shù)據(jù)表出現(xiàn)重復(fù)數(shù)據(jù)及如何刪除數(shù)據(jù)表中的重復(fù)數(shù)據(jù),感興趣的小伙伴們可以參考一下
    2016-05-05
  • mysql觸發(fā)器實時檢測一條語句進(jìn)行備份刪除思路詳解

    mysql觸發(fā)器實時檢測一條語句進(jìn)行備份刪除思路詳解

    遇到過這樣一個需求,在一張表里會不時出現(xiàn) “違規(guī)” 字樣的字段,需要在出現(xiàn)這個字段的時候,把整行的數(shù)據(jù)刪掉,針對這個需求我們該如何操作呢,下面跟隨小編看下mysql觸發(fā)器實時檢測一條語句進(jìn)行備份刪除的解決思路,一起看看吧
    2021-09-09

最新評論