Skip to content

Zeroplex 生活隨筆

軟體開發、伺服器和生活瑣事

小 縮小字型大小。 中 重設字型大小。 大 放大字型大小。

標籤: MySQL

快速列出 table rows 的幾個方法

Posted on 2021 年 7 月 7 日2021 年 7 月 7 日 By 日落 在〈快速列出 table rows 的幾個方法〉中尚無留言

建立一個 table 並塞 fake data 來當作測試的資料表:

use `test`;

CREATE TABLE `people` (
  `id` bigint UNSIGNED NOT NULL,
  `name` varchar(100) COLLATE utf8mb4_unicode_ci NOT NULL,
  `email` varchar(200) COLLATE utf8mb4_unicode_ci NOT NULL,
  `birth` date NOT NULL,
  `phone` varchar(100) COLLATE utf8mb4_unicode_ci NOT NULL,
  `addr` varchar(250) COLLATE utf8mb4_unicode_ci NOT NULL,
  `city` varchar(200) COLLATE utf8mb4_unicode_ci NOT NULL,
  `created_at` timestamp NULL DEFAULT NULL,
  `updated_at` timestamp NULL DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;

ALTER TABLE `people`
  ADD PRIMARY KEY (`id`);

正常情況下要查詢 rows count 大概就是以下的 query 方法:

SELECT count(*) from prople ;

這個 query 會跑去 scan table 或是 scan index,當資料量到一定規模,scan 會花上好幾分鐘。

另一個方法是去 information_schema 裡面做查詢,可以找到每個 table 的大概的狀況 (不會是準確的數字):

SELECT * 
FROM `TABLES` 
WHERE     TABLE_SCHEMA = 'test' 
      and TABLE_NAME = 'people'

不過我在找問題時用了 explain,意外發現一個更快、更準確的方法:explain 會列出 index size,如果是使用 primary key + auto increment,這個 index size 其實就是 table size:

> explain select count(9) from `people` \G

id            | 1
select_type   | SIMPLE
table         | people
partitions    | <null>
type          | index
possible_keys | <null>
key           | PRIMARY
key_len       | 8
ref           | <null>
rows          | 11366812
filtered      | 100.0
Extra         | Using index
Tags:MySQL

MySQL trigger 介紹

Posted on 2021 年 5 月 28 日2022 年 1 月 17 日 By 日落 在〈MySQL trigger 介紹〉中尚無留言

最近在研究 DBMS 的 trigger,不同 DBMS 的 trigger 的功能上差異不小,這裡僅對 MySQL 的 trigger 做筆記。

trigger 的意思同詞一樣,設計目的是讓 DB 的特定事件發生以後,去執行指定的動作。例如某 table 新增了一筆資料以後,將該筆資料記錄到另一個 table。

這次先使用「當使用者註冊成功以後,在 audit log 留下使用者 ID 和註冊的時間。先來建立 users 和 user_audit_log 這二個 table:

CREATE TABLE `users` (
    `id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY,
    `user_name` VARCHAR(200) NOT NULL,
    `mail` varchar(255) NOT NULL
);
CREATE TABLE `user_audit_log` (
    `id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY,
    `user_name` VARCHAR(200) NOT NULL,
    `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP()
);

然後來看一下 trigger 的 語法,trigger 可以針對 insert / update / delete 這幾個動作作出反應,另外也可以將 trigger 的動作放在以上幾個動作的之前 BEFORE 與之後 AFTER 執行,也就是說總共有以下這 6 種 trigger:

More “MySQL trigger 介紹” »

Tags:MySQL

MySQL 中 UPSERT (update or insert) 的語法

Posted on 2021 年 5 月 24 日2021 年 5 月 24 日 By 日落 在〈MySQL 中 UPSERT (update or insert) 的語法〉中尚無留言

MySQL 中要使用 IF ... ELSE 挺麻煩的,如果要在 insert 失敗 (資料已存在) 以後改對資料做 update 的話,可以考慮使用 INSERT .... ON DUPLICATE KEY。

假設要在 user 的 table 中,新增 user_id 和 mail:

INSERT INTO `user` (
    `user_id`,
    `mail`
) VALUES (
    "zero",
    "zero@zeroplex.tw"
)

當 user_id 是 unique index 時,上面的語法會失敗:duplicated key。
這時可以使用 MySQL 中的 INSERT … ON DUPLICATE KEY 語法,來讓 MySQL 自動判斷要使用 INSERT 還是 UPDATE:

INSERT INTO `user` (
    `user_id`,
    `mail`
) VALUES (
    "zero",
    "zero@zeroplex.tw"
) ON DUPLICATE KEY UPDATE 
    `mail` = "zero@zeroplex.tw"

上述語法,會讓 MySQL 執行時,若發現 zero 這個使用者已經存在時,變自動去更新他的 mail。

Tags:MySQL

快速複製 table 資料 – insert into select

Posted on 2021 年 1 月 18 日2021 年 3 月 12 日 By 日落 在〈快速複製 table 資料 – insert into select〉中尚無留言

遇到一個奇怪的需求,要備份特定資料表的資料。因為很懶所以直接從 SQL 動手。

用 show create table 複製原有資料表的 schema:

> show create table users G                                                                                         
***************************[ 1. row ]***************************
Table        | users
Create Table | CREATE TABLE `users` (
  `id` bigint NOT NULL AUTO_INCREMENT,
  `name` varchar(100) DEFAULT NULL,
  `email` varchar(100) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `idx_name` (`name`),
  KEY `idx_email` (`email`)
) ENGINE=InnoDB

再來用 「insert into select」 的方法來從原有的資料表拉資料:

insert into backup (name, email)
select name, email from users where id in (1, 3, 5)

這個作法好像會比一次一筆資料拉還要快一些,不過要注意的是 id (或是 auto_increment) 等欄位要忽略,讓 DB 自己處理,不然之後新增資料可能會出現 duplicated keys 之類的問題。

Tags:Database, db, MySQL

MySQL 5.7 安裝以後無法用 root 登入

Posted on 2020 年 6 月 10 日2021 年 3 月 12 日 By 日落 在〈MySQL 5.7 安裝以後無法用 root 登入〉中尚無留言

以往都是在安裝的時候,直接請你先填寫 root 密碼,以便安裝完畢以後直接登入。
但 MySQL 5.7 認證方式有更改,即使是 root 也沒有給你預設密碼。

新的方法要使用 OS 的 root 來登入 MySQL:

$ sudo mysql -u root
....
....
....
mysql>

建立 super user 帳號:

create user zeroplex@localhost identified by 'secret';
flush privileges;

grant all privileges on *.* to zeroplex@localhost with grant option;
flush privileges;

到這邊告一段落,之後 zeroplex@localhost 就可以使用原有的方式登入了。

Tags:Linux, MySQL

文章分頁

上一頁 1 2 3 4 ... 6 下一頁

其他

關於我  (About me)

小額贊助

  文章 RSS Feed

  留言 RSS Feed

Apache AWS Bash C/C++ Docker FreeBSD GCP Git Google Java JavaScript Laravel Linux Microsoft MSSQL MySQL Nginx PHP PHPUnit PostgreSQL Python Qt Ubuntu Unix Vim Web Windows WordPress XD 作業系統 分享 好站推薦 專題 攝影 新奇搞笑 新聞 旅遊 生活雜記 程式設計 網路架站 網頁設計 資訊學習 資訊安全 遊戲 音樂


創用 CC 授權條款
本著作係採用創用 CC 姓名標示-相同方式分享 4.0 國際 授權條款授權.