成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

MySQL去除“關聯表”重復數據,以建立聯合唯一索引

數據庫 MySQL
昨天遇到一個問題,需要對一張關系表進行重構和優化。然而這張關系表由于已有代碼沒有注重并發導致了很多的臟數據,即重復數據。解決辦法:使用insert where not exists語句和清理臟數據并建立聯合唯一索引。

前言

昨天遇到一個問題,需要對一張關系表進行重構和優化。然而這張關系表由于已有代碼沒有注重并發導致了很多的臟數據,即重復數據。

表名thread_recommend,帖子推薦表,為兩個實體user_id和thread_id的(推薦)關系表,表結構很簡單如下: 

  1. /*用戶推薦帖子記錄表*/ 
  2.  
  3. CREATE TABLE `thread_recommend` ( 
  4.  
  5. `id` int(11) NOT NULL AUTO_INCREMENT, 
  6.  
  7. `thread_id` int(11) DEFAULT NULL COMMENT '被用戶推薦的帖子編號'
  8.  
  9. `user_id` int(11) DEFAULT NULL COMMENT '推薦該帖子的用戶編號'
  10.  
  11. `status` int(11) DEFAULT '1' COMMENT '狀態0 取消推薦,1推薦'
  12.  
  13. `created` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '推薦時間'
  14.  
  15. PRIMARY KEY (`id`), 
  16.  
  17. KEY `userid` (`user_id`) USING BTREE 
  18.  
  19. ) ENGINE=InnoDB; 

 問題在于,由于代碼不規范,在高并發時(或數據庫壓力大時造成的延時積壓時)會出現多個(相同thread_id和user_id的)組合,如下:

 

之后你們懂的,各種和原想不一致的神奇bug噴涌而出,比如:

我剛剛取消了推薦,怎么還顯示我推薦著!!

顯示的總推薦數怎么和實際推薦用戶加起來不一樣!!

解決方案一:使用insert where not exists語句

聲明:此方案并不是***方案,不推薦使用。

先上代碼:(這里拿另一個關系表的真實query舉例,原理一樣) 

  1. INSERT INTO `user_topic` (`user_id`, `topic_id`) 
  2.  
  3. SELECT :userId, :topicid FROM `user_topic` 
  4.  
  5. WHERE NOT EXISTS (SELECT * FROM `user_topic` 
  6.  
  7. WHERE `user_topic`.`user_id` = :userId 
  8.  
  9. AND `user_topic`.`topic_id` = :topicid) 
  10.  
  11. LIMIT 1;  

(相同方法見http://stackoverflow.com/a/31...)

通過這種“插入時判斷不存在才插入并返回行數為1,存在的話返回行數為0”的方法,可以做到:

  • 只有在返回行數為1的情況下才執行之后邏輯(如緩存內的統計數+1,緩存內帖子推薦人增加此userId等等)
  • 如果返回行數為0,則接口返回error

解決方案二:清理臟數據并建立聯合唯一索引

這個方案是本文的核心了,也是我們目前認為的***實踐。

***步:查找user_id, thread_id的聯合duplication 

  1. SELECT a.* FROM `thread_recommend` a 
  2.  
  3. INNER JOIN (SELECT * FROM `thread_recommend` GROUP BY `thread_id`, `user_id` HAVING COUNT(id) > 1) b ON a.`thread_id` = b.`thread_id` AND a.`user_id` = b.`user_id` 
  4.  
  5. ORDER BY a.`user_id` ASC, a.`thread_id` ASC, a.`id` DESC  

或簡單的版本 

  1. SELECT * FROM `thread_recommend` 
  2.  
  3. WHERE (`user_id`, `thread_id`) IN (SELECT `user_id`, `thread_id` FROM `thread_recommend` GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1);  

得到

 

哇!所有的重復項都在這里了,好想馬上把它們干掉!

現在需要將重復的條目中ID更大的所有條目都刪除,只留ID最小的那一個。

刪之前先獲得需要刪除項,比對一下,

  1. SELECT * FROM `thread_recommend` 
  2.  
  3. WHERE (`user_id`, `thread_id`) IN (SELECT `user_id`, `thread_id` FROM `thread_recommend` GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1) 
  4.  
  5. AND `id` NOT IN (SELECT MIN(`id`) FROM `thread_recommend` GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1);  

下一步,SELECT * FROM改成DELETE FROM,刪除!

  1. DELETE FROM `thread_recommend` 
  2.  
  3. WHERE (`user_id`, `thread_id`) IN (SELECT `user_id`, `thread_id` FROM `thread_recommend` GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1) 
  4.  
  5. AND `id` NOT IN (SELECT MIN(`id`) FROM `thread_recommend` GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1);  

Oops!報錯! You can't specify target table 'thread_recommend' for update in FROM clause

這是Mysql的一個小問題,我們參見解決方案 http://stackoverflow.com/a/14... 后修改一下SQL就好:

  1. DELETE FROM `thread_recommend` 
  2.  
  3. WHERE (`user_id`, `thread_id`) IN (SELECT `user_id`, `thread_id` FROM (SELECT * FROM `thread_recommend`) a GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1) 
  4.  
  5. AND `id` NOT IN (SELECT MIN(`id`) FROM (SELECT * FROM `thread_recommend`) b GROUP BY `user_id`, `thread_id` HAVING COUNT(1) > 1);  

***,加聯合唯一索引!

  1. ALTER TABLE `thread_recommend` 
  2.  
  3. ADD UNIQUE KEY `thread_id_user_id_unique`(`thread_id`,`user_id`) USING BTREE;  

Of course,如果上述清理工作沒有完成將會報錯!

完!

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2024-08-19 09:43:00

2022-08-04 08:22:49

MySQL索引

2023-01-03 07:44:53

MySQL查詢重復

2019-10-21 09:55:12

數據庫PostgreSQL Oracle

2023-02-26 00:00:06

MySQL索引故障

2024-03-26 12:16:13

MySQLInnodb數據庫

2024-03-11 05:00:00

Python集合開發

2021-04-08 10:55:53

MySQL數據庫代碼

2011-04-13 13:05:14

重復數據刪除

2011-08-02 12:46:46

Oracle數據表建立索引

2010-09-02 10:36:51

SQL刪除

2021-11-30 10:00:01

SQL數據重復

2024-05-24 09:29:28

2011-08-18 11:18:25

Oracle唯一約束唯一索引

2025-01-24 00:00:00

數據RoaringBitmap

2011-04-13 13:13:09

重復數據刪除

2024-10-16 17:04:13

2021-09-14 13:15:43

MySQL數據庫腳本

2021-06-28 10:25:47

MySQL數據庫重復數據

2023-11-14 14:41:01

數據庫清除
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美精品久久久久久 | 美女视频黄色片 | www国产亚洲精品久久网站 | 国产精品乱码一区二三区小蝌蚪 | 视频一区中文字幕 | 日韩一区二区三区在线观看 | 韩日精品一区 | 特黄特黄a级毛片免费专区 av网站免费在线观看 | 国产成人精品一区二三区在线观看 | 精品日韩一区二区三区 | 日本欧美黄色片 | 国产xxx在线观看 | 国产一区二区三区四区三区四 | 国产精品国产馆在线真实露脸 | 日韩在线欧美 | 日韩和的一区二区 | 日本一道本视频 | 成人av电影网 | 国产精品我不卡 | 欧美天堂在线观看 | 91麻豆精品国产91久久久更新资源速度超快 | 欧洲国产精品视频 | 午夜久草| 91精品国产综合久久久久久丝袜 | 黑人精品欧美一区二区蜜桃 | 亚洲国产欧美日韩 | 精品欧美视频 | 久久久久久国产 | 精精国产xxxx视频在线播放7 | 国产欧美日韩在线播放 | 日日摸日日碰夜夜爽亚洲精品蜜乳 | 黄色片大全在线观看 | 日日夜夜草 | 91se在线| 狠狠躁夜夜躁人人爽天天高潮 | 夜夜撸av | 久久精品日产第一区二区三区 | 国产福利视频 | 亚洲国产精品99久久久久久久久 | 中文字幕视频在线看5 | 日韩在线免费播放 |