亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 課堂 > 基礎知識 > 正文

Session重疊問題教學

2024-09-12 20:29:53
字體:
來源:轉載
供稿:網友
        周五晚上終于把這個算法初步實現了.
       連續加班忙碌了一個星期,終于有點曙光了.
       從這個問題的緣起,到目前應該已經優化了快100倍了
       但是周末的時候,想想還是不對.
       小花貍Session合并算法(對,以后這個算法就叫這個名稱了)實現的合并速度應該是非常快的.代價僅僅是掃描一遍記錄.
       這1.6秒到底用在哪里了?
 
后來經過反復調試.發現還有兩塊可以優化改進的地方.
改進后的過程如下:
 
drop procedure p;  
DELIMITER $$    
    
CREATE DEFINER=`root`@`localhost` PROCEDURE `p`()    
BEGIN      
    declare done int default 0;          
    declare v_roomid bigint;      
    declare v_time timestamp(6);      
    declare v_cur_type smallint;    
    
    declare v_before_roomid bigint default -1;    
    declare v_before_type smallint default -1;    
    declare v_before_time timestamp(6) ;    
    
    declare v_num bigint default 0;    
    
    
    declare cur_test CURSOR for select roomid,type,timepoint from tmp_time_point order by roomid,timepoint,type ;    
    DECLARE  CONTINUE HANDLER FOR NOT FOUND  SET done = 1;          
    
          
    drop table if exists t1;      
    drop table if exists t2;    
    drop table if exists tmp_time_point;      
    drop table if exists tmp_result;    
    drop table if exists tmp_min_range;    
    drop table if exists tmp_s;  
    CREATE temporary TABLE `t1` (      
      `roomid` int(11) NOT NULL DEFAULT '0',      
      `userid` bigint(20) NOT NULL DEFAULT '0',      
      `s` timestamp(6),      
      `e` timestamp(6),  
       primary key(roomid,userid,s,e)  
    ) ENGINE=memory;      
    
   CREATE temporary TABLE `t2` (      
      `roomid` int(11) NOT NULL DEFAULT '0',      
      `s` timestamp(6),      
      `e` timestamp(6)  
    ) ENGINE=memory;      
    
    CREATE temporary TABLE `tmp_min_range` (      
      `roomid` int(11) NOT NULL DEFAULT '0',      
      `s` timestamp(6),      
      `e` timestamp(6),      
      primary key(roomid,s,e),  
      key(roomid,e)  
    ) ENGINE=memory;      
    
    create temporary table tmp_time_point(      
            roomid bigint,      
            timepoint timestamp(6),      
            type smallint,    
            key(roomid,timepoint)      
    ) engine=memory;      
        
    create temporary table tmp_result(      
            roomid bigint,      
            timepoint timestamp(6),    
            c int    
    ) engine=memory;      
      
    create temporary table tmp_s(  
        roomid bigint,  
        userid bigint,  
        s timestamp,  
        e timestamp,  
        i int  
    ) engine=memory;  
      
SET @A=0;      
SET @B=0;      
  
insert into tmp_s  
    SELECT x.roomid,x.userid,s,e,datediff(e,s)+1 i   
    FROM     
    (    
        (    
            SELECT @B:=@B+1 AS id,roomid,userid,s      
            FROM (      
                SELECT DISTINCT roomid, userid, roomstart AS s          
                FROM u_room_log a          
                WHERE NOT EXISTS (SELECT *          
                    FROM u_room_log b          
                    WHERE a.roomid = b.roomid          
                        AND a.userid = b.userid          
                        AND a.roomstart > b.roomstart          
                        AND a.roomstart <= b.roomend)    
            ) AS p    
        ) AS x,      
        (    
            SELECT @A:=@A+1 AS id,roomid,userid,e      
            FROM     
            (      
                SELECT DISTINCT roomid, userid, roomend AS e          
                FROM u_room_log a          
                WHERE NOT EXISTS (SELECT *          
                    FROM u_room_log b          
                    WHERE a.roomid = b.roomid          
                        AND a.userid = b.userid          
                        AND a.roomend >= b.roomstart          
                        AND a.roomend < b.roomend)      
            ) AS o    
        ) AS y      
    )     
    WHERE x.id = y.id AND x.roomid = y.roomid AND x.userid = y.userid   ;     
  
select max(i) into @c from tmp_s;  
      
insert ignore into t1(roomid,userid,s,e)    
select         
roomid,  userid,        
if(date(s)!=date(e) and id>1,date(s+interval id-1 date(s+interval id-1 date(e) ,e,date_format(s+interval id-1 '%Y-%m-%d 23:59:59')) e        
from tmp_s t1 STRAIGHT_JOIN      
nums on(nums.id<=t1.i)  
where nums.id<=@c  
     
;        
    
insert into t2 (roomid,s,e)    
select roomid,    
s+interval startnum/1000000 second s,    
e-interval endnum/1000000 second e    
 from (    
    select     
    roomid,    
    s,e,    
    startnum,    
    when @eflag=eflag then @rn:=@rn+1 when @eflag:=eflag then @rn else @rn end endnum    
    from (    
        select * from (    
            select when @sflag=sflag then @rn:=@rn+1 when @sflag:=sflag then @rn else @rn end startnum,roomid,s,e,sflag,eflag from    
            (    
                select * from     
                (    
                    select t1.*,concat('[',roomid,'],',s) sflag,concat('[',roomid,'],',e) eflag from t1 order by roomid ,sflag    
                )a,(select @sflag:='',@rn:=0,@eflag:='') vars    
            ) b      
        ) bb order by roomid,eflag    
    ) c    
) d ;    
     
    insert into tmp_time_point(roomid,timepoint,type) select roomid,s,1 from t2;    
    insert into tmp_time_point(roomid,timepoint,type) select roomid,e,0 from t2;    
       
    insert ignore into tmp_min_range(roomid,s,e)    
                select   roomid,starttime  starttime, endtime  endtime from (      
                    select       
                    if(@roomid=roomid,@d,'')  as starttime,@d:=str_to_date(timepoint,'%Y-%m-%d %H:%i:%s.%f'),@roomid:=roomid,p.roomid,str_to_date(timepoint,'%Y-%m-%d %H:%i:%s.%f') endtime      
                    from tmp_time_point p,(select @d:='',@roomid:=-1) vars      
                    order by roomid,timepoint      
                ) v4 where starttime!='' and date(starttime)=date(endtime);    
    
    open cur_test;          
    repeat          
        fetch cur_test into v_roomid,v_cur_type,v_time;          
        if done !=1 then        
            -- 第一行或者每個房間的第一行    
            if v_before_roomid=-1 or v_roomid!=v_before_roomid  then    
                set v_before_roomid:=v_roomid;    
                set v_before_type:=1;    
                set v_before_time:='0000-00-00 00:00:00';    
                set v_num:=0;    
            end if;    
                
                
            if v_before_type=1  then    
             
                set v_num:=v_num+1;    
          
                insert into tmp_result(roomid,timepoint,c) values(v_roomid,v_time,v_num);    
            end if;    
                
            if v_before_type=0 then    
                   
                set v_num:=v_num-1;    
    
                insert into tmp_result(roomid,timepoint,c) values(v_roomid,v_time,v_num);    
            end if;    
    
            set v_before_roomid:=v_roomid;    
            set v_before_type:=v_cur_type;    
            set v_before_time:=v_time;    
        end if;        
    until done end repeat;          
    close cur_test;       
      
    select roomid,date(s) dt,round(second,date_format(s,'%Y-%m-%d %H:%i:%s'),date_format(e,'%Y-%m-%d %H:%i:%s')))/60) ts,max(c)-1 c from (         
        select a.roomid,a.s,a.e,r.c,r.timepoint from tmp_result r     
        inner join     
        tmp_min_range a on( r.timepoint=a.e and r.roomid=a.roomid)    
        where     c>2    
    ) a group by roomid,date(s);      
    
END    
 
第一處改進
    原來同一房間同一用戶重疊時間合并,然后再拆分跨天數據,用的是一條SQL
    現在改進如下
create temporary table tmp_s(  
        roomid bigint,  
        userid bigint,  
        s timestamp,  
        e timestamp,  
        i int  
    ) engine=memory;  
      
SET @A=0;      
SET @B=0;      
  
insert into tmp_s  
    SELECT x.roomid,x.userid,s,e,datediff(e,s)+1 i   
    FROM     
    (    
        (    
            SELECT @B:=@B+1 AS id,roomid,userid,s      
            FROM (      
                SELECT DISTINCT roomid, userid, roomstart AS s          
                FROM u_room_log a          
                WHERE NOT EXISTS (SELECT *          
                    FROM u_room_log b          
                    WHERE a.roomid = b.roomid          
                        AND a.userid = b.userid          
                        AND a.roomstart > b.roomstart          
                        AND a.roomstart <= b.roomend)    
            ) AS p    
        ) AS x,      
        (    
            SELECT @A:=@A+1 AS id,roomid,userid,e      
            FROM     
            (      
                SELECT DISTINCT roomid, userid, roomend AS e          
                FROM u_room_log a          
                WHERE NOT EXISTS (SELECT *          
                    FROM u_room_log b          
                    WHERE a.roomid = b.roomid          
                        AND a.userid = b.userid          
                        AND a.roomend >= b.roomstart          
                        AND a.roomend < b.roomend)      
            ) AS o    
        ) AS y      
    )     
    WHERE x.id = y.id AND x.roomid = y.roomid AND x.userid = y.userid   ;     
  
select max(i) into @c from tmp_s;  
      
insert ignore into t1(roomid,userid,s,e)    
select         
roomid,  userid,        
if(date(s)!=date(e) and id>1,date(s+interval id-1 date(s+interval id-1 date(e) ,e,date_format(s+interval id-1 '%Y-%m-%d 23:59:59')) e        
from tmp_s t1 STRAIGHT_JOIN      
nums on(nums.id<=t1.i)  
where nums.id<=@c  
     
;        
 
先把同一房間同一用戶的重疊部分合并,然后暫存臨時表
記錄最大的間隔時間,然后再拆分數據
 
拆分數據的時候 使用STRAIGHT_JOIN 強制連接順序.
這樣避免因為數字輔助表過大,而導致性能陡然變差.
 
 
第二處改進
    原來使用distinct的查詢, 都改為在臨時表上增加主鍵.
    然后使用insert ignore into 代替 insert into
    這樣大概優化了300毫秒
 
經過反復優化之后,執行時間大致穩定在1250毫秒 至 1300 毫秒
 
各個部分耗時分析如下
填充tmp_s,合并同一房間同一用戶的重疊部分,耗時655毫秒
填充t1,拆分跨天的用戶數據,耗時62毫秒
填充t2,用戶時間段首尾相交或者首尾全部重合的數據拆分,耗時140毫秒
填充tmp_min_range,計算最小間隔范圍,耗時156毫秒
小花貍Session合并算法,耗時219毫秒
結果統計展示,耗時47毫秒

(編輯:武林網)

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美孕妇与黑人孕交| 777国产偷窥盗摄精品视频| 欧美国产乱视频| 亚洲自拍欧美色图| 日韩在线观看精品| 亚洲视频在线观看免费| 日韩有码视频在线| 成人激情视频免费在线| 国产精品白丝av嫩草影院| 在线日韩欧美视频| 中文字幕亚洲欧美一区二区三区| 成人情趣片在线观看免费| 亚洲理论在线a中文字幕| 欧美视频在线视频| 91免费在线视频网站| 97视频在线观看播放| 欧美性猛交99久久久久99按摩| 国产午夜精品视频免费不卡69堂| 69**夜色精品国产69乱| 日本精品久久久| 国产日韩换脸av一区在线观看| 亚洲精品有码在线| 亚洲国产成人精品一区二区| 国产精品久久久久久久一区探花| 亚洲激情视频在线| 一区二区三欧美| 日韩麻豆第一页| 欧美一级在线播放| 成人黄色短视频在线观看| 国产成+人+综合+亚洲欧洲| 亚洲欧美日韩精品久久奇米色影视| 国产精品久久久精品| 夜夜嗨av一区二区三区免费区| 中日韩美女免费视频网站在线观看| 亚洲精品国产综合久久| 国产精品一区二区三区久久久| 在线视频国产日韩| 亚洲第一男人av| 久久这里只有精品99| 91久久精品国产91久久| 国产激情综合五月久久| 九九精品在线播放| 色综合老司机第九色激情| 国产一区二区黄| 91精品国产高清久久久久久| 成人激情春色网| 91精品视频免费观看| 精品人伦一区二区三区蜜桃免费| 精品中文字幕乱| 亚洲国产成人爱av在线播放| 中文字幕自拍vr一区二区三区| 国产精品久久久久久婷婷天堂| 欧美电影在线免费观看网站| 国产一区二区黄| 国产精品户外野外| 欧美黄网免费在线观看| 久久精品国产视频| 亚洲精品国精品久久99热| 国产精品一区二区三区久久| 一色桃子一区二区| 庆余年2免费日韩剧观看大牛| 亚洲成人免费网站| 91精品一区二区| 亚洲国产精品va在线| 亚洲成色777777女色窝| 97精品国产97久久久久久免费| 国产欧美日韩中文字幕在线| 亚洲精品suv精品一区二区| 欧美高清在线观看| 国产一区二区三区直播精品电影| 日韩免费观看网站| 91久久精品国产91久久| 欧美日韩亚洲网| 一区二区av在线| 欧美日韩中文字幕| 欧美精品一区三区| 日韩在线中文字| 国产精自产拍久久久久久| 国产精品久久久久久五月尺| 精品亚洲va在线va天堂资源站| 自拍偷拍亚洲精品| 播播国产欧美激情| 午夜精品99久久免费| 欧美激情一级二级| 亚洲欧洲日本专区| 亚洲午夜国产成人av电影男同| 欧美在线观看www| 亚洲色在线视频| 国产精品毛片a∨一区二区三区|国| 伊人久久男人天堂| 欧美亚洲国产成人精品| 欧美乱妇40p| 国产福利精品av综合导导航| 亚洲欧美在线免费| 亚洲高清不卡av| 亚洲天堂av电影| 日韩av日韩在线观看| 国产一区二区三区视频在线观看| 国产欧美在线看| 精品久久香蕉国产线看观看亚洲| 麻豆国产va免费精品高清在线| 精品国产乱码久久久久久婷婷| 欧美夜福利tv在线| 精品福利在线视频| 美女av一区二区三区| 在线精品91av| 欧美日韩一区二区免费在线观看| 欧美猛少妇色xxxxx| 国产欧美日韩中文字幕在线| 国产成人小视频在线观看| 亚洲欧美色婷婷| 日本成熟性欧美| www.久久久久久.com| 国产精品成人av在线| 国产精品91久久久久久| 亚洲色图25p| 日韩色av导航| 懂色av中文一区二区三区天美| 精品国产依人香蕉在线精品| 一区二区国产精品视频| 萌白酱国产一区二区| 日韩黄色av网站| 欧美丝袜第一区| 国产精品极品美女粉嫩高清在线| 欧美多人乱p欧美4p久久| 欧美大学生性色视频| 欧美激情在线观看| 久久精品国产亚洲7777| 亚洲欧美一区二区精品久久久| 久久精品久久久久电影| 亚洲精品在线看| 国产精品视频播放| 亚洲人午夜色婷婷| 亚洲乱码av中文一区二区| 国产精品亚洲第一区| 欧美在线视频一二三| 欧美性猛交xxxx偷拍洗澡| 欧美视频国产精品| 亚洲精品国偷自产在线99热| 欧美极品美女视频网站在线观看免费| 欧美大全免费观看电视剧大泉洋| 欧美激情a在线| 欧美最近摘花xxxx摘花| 欧美大学生性色视频| 91嫩草在线视频| 亚洲二区在线播放视频| 色综久久综合桃花网| 亚洲第一精品夜夜躁人人躁| 亚洲欧美国产精品| 亚洲日韩第一页| 91性高湖久久久久久久久_久久99| 国产精品美乳一区二区免费| 亚洲欧美中文在线视频| 久久精品国产免费观看| …久久精品99久久香蕉国产| 日韩中文字幕视频在线| 成人激情在线观看| 最近2019中文免费高清视频观看www99| 亚洲色图激情小说| 91精品久久久久久综合乱菊| 欧美黑人性视频| 国产专区精品视频| 欧美日韩亚洲网| 亚洲奶大毛多的老太婆|