青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

The Fourth Dimension Space

枯葉北風寒,忽然年以殘,念往昔,語默心酸。二十光陰無一物,韶光賤,寐難安; 不畏形影單,道途阻且慢,哪曲折,如渡飛湍。斬浪劈波酬壯志,同把酒,共言歡! -如夢令

POJ 1177 Picture 經典線段樹+離散化+掃描線

 

        弄了一天,總算搞懂了掃描線是怎么回事,剛開始的時候在網上搜索,代碼基本沒有注釋,很難看懂,隨后搜索到了陳宏的論文,2頁紙能寫完的東西,他居然可以寫那么長,粗略的掃描了一下,感覺原線段和超元線段的定義很不錯,其他的實在講的有點羅嗦就跳過了。鑒于以后還會有同樣想要學習掃描線的同學,下面我來簡單的介紹一下掃描線的實現過程吧,希望對大家有所幫助。

       首先說離散化:因為有的時候題目中給出的數據范圍可能非常大,如果直接建立成線段樹的話,絕對超內存,怎么辦呢?其實這里所謂離散化就是把這些數排列在數組中,然后用數組的下標來代替這個數,這樣我們最多只有N*2個點

          如何求周長? 那么現在我假設讀者都明白離散化,以及線段樹了!

  具體做法有兩種形式  可以對x離散也可以對y 離散! 我這里以對y  進行離散(這樣可能更符和我們平常的思維方式)

1:首先將矩形的豎邊全部存起來(要用一個標記變量標記該邊是否為入邊),然后按照豎邊的X 的大小排好序;

2:在存儲矩形豎邊的同時要把舉行的Y坐標存起來,然后排序,最后去掉重復的點!

3:建樹了根據第二步中最后留下來點的個數num建立一個區間長度為[0,num-1]的線段樹;

4:這步很關鍵了,把排好序的豎邊從左到右開始掃描了,如果是矩形的左邊那么就插入,要是是矩形的右邊了那么就從線段樹里刪除了?。ㄔ谶@里每次插入和刪除線段樹要維護好兩個重要的值,一個是當前線段樹的被覆蓋的區間長度的總和第二個是當前線段樹中被覆蓋的區間有多少個)!
PS:其中這兩句代碼非常重要,讀者可以畫個簡單的圖進行理解,起始的時候我沒明白要記錄線段段數的作用,仔細研究了這部分代碼發現算線段的段數是為了求得橫邊的長度,還有一點要注意的是,這棵線段樹要建成節點為單元線段的形式,即如果區間為[0,3]
線段樹要建成 [0,3]
                         /        \
                      [0,1]    [1,3]
                                   /    \
                               [1,2] [2,3]

這樣,我剛開始的時候嘗試了一下建成節點的方式,即節點是[1,1] [2,2]這樣,結果發現統計區間段數根本沒法進行,后來參考過網上的代碼發現要這樣建樹,改了一下就過了,可能平時對這種建樹方式還是不太熟悉吧。下面可以開始想想如何才能用掃描線求面積并了,感覺基本思想應該差不多。
//POJ 1177
//N個矩形求總周長
//線段樹+離散化+掃描線
//2010年7月21日19:35:45
//Coded By abilitytao

#include
<iostream>
#include
<cmath>
#include
<algorithm>
using namespace std;

#define MAXN 10010
struct STnode //線段樹的節點
{
    
int l,r;
    
int len;//區間內代表的長度
    int segnum;//區間內被分成的段數,不懂的話再結合代碼看看
    int cover;//區間被覆蓋的次數
    int sum;//區間中被覆蓋的總長度
    bool lcover,rcover;//標記左右端點是否被覆蓋,用于合并區間時候統計區間內的離散線段數
    STnode()//初始化
    {
        l 
= r = 0;
        len 
= segnum = cover = sum = 0
        lcover 
= rcover = false;
    }

}
;
STnode ST[MAXN
*4];//整棵線段樹

struct Line
{

    
int st,ed;//豎邊的兩個y值
    int x;//此條邊的x值
    bool InOut;//是否為入邊
    bool operator<(Line o) const//重載小于符號 
    {
        
return x<o.x;
    }

}
;


Line Yline[MAXN];
//存儲豎邊
int Index[MAXN];//存儲離散后的y值
int cnt=0;
int n;//存儲矩形的數目


void Build(int l,int r,int i)//創建線段樹
{

    ST[i].l
=l;
    ST[i].r
=r;
    ST[i].cover
=0;
    ST[i].len
=Index[r]-Index[l];
    ST[i].segnum
=0;
    ST[i].sum
=0;
    ST[i].lcover
=ST[i].rcover=false;
    
//建立線段的時候進行初始化
    if(r-l>1)
    
{
        
int mid=(l+r)>>1;
        Build(l,mid,i
*2);
        Build(mid,r,i
*2+1);
    }

}


void GetLen(int i)//求節點包含的線段總長度
{
    
if(ST[i].cover>0)
        ST[i].sum
=ST[i].len;
    
else if(ST[i].r-ST[i].l>1)
        ST[i].sum
=ST[i*2].sum+ST[i*2+1].sum;
    
else
        ST[i].sum
=0;
}


void GetSegNum(int i)//求該區間所包含的線段數總量(就是含有不想交的線段的條數)
{

    
if(ST[i].cover>0)
    
{
        ST[i].lcover
=ST[i].rcover=true;
        ST[i].segnum
=1;
    }

    
else if(ST[i].r-ST[i].l>1)
    
{
        ST[i].lcover
=ST[i*2].lcover;
        ST[i].rcover
=ST[i*2+1].rcover;
        ST[i].segnum
=ST[i*2].segnum+ST[i*2+1].segnum-ST[i*2].rcover*ST[i*2+1].lcover;
    }

    
else
    
{
        ST[i].lcover
=ST[i].rcover=false;
        ST[i].segnum
=0;//特殊處理下葉子節點
    }

}



void Insert(int l,int r,int i)//插入一條線段
{
    
if(ST[i].l==l&&ST[i].r==r)
        ST[i].cover
++;
    
else
    
{
        
int mid=(ST[i].l+ST[i].r)>>1;
        
if(r<=mid)
            Insert(l,r,i
*2);
        
else if(l>=mid)
            Insert(l,r,i
*2+1);
        
else
        
{
            Insert(l,mid,i
*2);
            Insert(mid,r,i
*2+1);
        }

    }

    GetLen(i);
    GetSegNum(i);
}


void Delete(int l,int r,int i)//刪除矩形的右邊

    
if(ST[i].l==l&&ST[i].r==r)
        ST[i].cover
--;
    
else
    
{
        
int mid=(ST[i].l+ST[i].r)>>1;
        
if(r<=mid)
            Delete(l,r,i
*2);
        
else if(l>=mid)
            Delete(l,r,i
*2+1);
        
else
        
{
            Delete(l,mid,i
*2);
            Delete(mid,r,i
*2+1);
        }

    }

    GetLen(i);
    GetSegNum(i);
    
//這個后序操作非常精彩!
}



int GetIndex(int x)// 返回x的下標
{
    
return lower_bound(Index,Index + cnt,x) - Index;//lower_bound函數返回一個元素在容器中的迭代器,數組可以看成特殊的容器,所以這里返回的迭代器就是指針
}



int main()
{
    
while(scanf("%d",&n)!=EOF)
    
{
        cnt
=0;
        
int i,j,k;
        
int x1,x2,y1,y2;
        
for(i=0;i<n;i++)
        
{
            scanf(
"%d%d%d%d",&x1,&y1,&x2,&y2);
            Yline[i
*2].x=x1;
            Yline[i
*2+1].x=x2;
            Yline[
2*i].st=Yline[2*i+1].st=y1;
            Yline[
2*i].ed=Yline[2*i+1].ed=y2;
            Yline[
2*i].InOut=true;//標記入邊
            Yline[2*i+1].InOut=false;
            Index[
2*i]=y1;
            Index[
2*i+1]=y2;
        }

        sort(Index,Index
+n*2);
        sort(Yline,Yline
+n*2);
        
for(int i=1;i<n*2;i++)//Y數組去重
        {
            
if(Index[i]!=Index[i-1])
                Index[cnt
++]=Index[i-1];
        }

        Index[cnt
++]=Index[2*n-1];//這里很容易錯!
        Build(0,cnt-1,1);
        
int Ans=0;
        
int Lsum=0;//上一次記錄的長度,畫個圖很好理解
        for(int i=0;i<2*n-1;i++)
        
{
            
if(Yline[i].InOut)
                Insert(GetIndex(Yline[i].st),GetIndex(Yline[i].ed),
1);
            
else
                Delete(GetIndex(Yline[i].st),GetIndex(Yline[i].ed),
1);
            
//畫個圖還是很好理解下面這兩行的
            Ans+=ST[1].segnum*(Yline[i+1].x-Yline[i].x)*2;
            Ans
+=abs(ST[1].sum-Lsum);
            Lsum
=ST[1].sum;
        }

        
//特殊處理最后一條出邊,因為沒有下一條豎邊了
        Delete(GetIndex(Yline[2*n-1].st),GetIndex(Yline[2*n-1].ed),1);
        Ans
+=abs(ST[1].sum-Lsum);
        printf(
"%d\n",Ans);

    }

    
return 0;

}



special thank to this URL:http://angels1.0.blog.163.com/blog/static/84580504200893171819117/

posted on 2010-07-21 08:53 abilitytao 閱讀(5043) 評論(4)  編輯 收藏 引用

評論

# re: POJ 1177 Picture 經典線段樹+離散化+掃描線 2010-10-27 21:09 の屋

我也覺得太羅嗦了,沒幾句有用的  回復  更多評論   

# re: POJ 1177 Picture 經典線段樹+離散化+掃描線 2012-06-11 09:13 匿名

能具體說說MAXN是怎么來的嗎?因為我有看到Index[]的大小為MAXN,題意說輸入矩形的范圍是0~5000,若輸入矩形為5000,則每個矩形有兩個豎邊,每個豎邊要存兩個值,則10010在大小上是不夠的  回復  更多評論   

# re: POJ 1177 Picture 經典線段樹+離散化+掃描線 2012-08-01 15:56 forget~

你推薦的網址的代碼,樣例都沒過呢  回復  更多評論   

# re: POJ 1177 Picture 經典線段樹+離散化+掃描線[未登錄] 2013-08-20 23:35 路人甲

沒考慮重邊的情況~~  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美福利一区二区| 久久青草久久| 国产日韩亚洲欧美精品| 欧美人妖另类| 亚洲乱码国产乱码精品精可以看 | 亚洲国产成人在线播放| 精品白丝av| 夜夜精品视频| 亚洲欧美日本国产专区一区| 亚洲小说欧美另类婷婷| 国产精品v亚洲精品v日韩精品| 欧美激情在线免费观看| 欧美日韩视频免费播放| 国产精品永久免费观看| 国产一级揄自揄精品视频| 亚洲国产高清在线观看视频| 亚洲乱码国产乱码精品精 | 国产精品午夜国产小视频| 国产视频综合在线| 亚洲精品视频在线播放| 亚洲欧美网站| 欧美激情国产日韩| 亚洲天堂激情| 老司机免费视频一区二区| 欧美精品自拍偷拍动漫精品| 久久成人国产| 欧美日韩大片一区二区三区| 国内成+人亚洲+欧美+综合在线| 99精品国产热久久91蜜凸| 久久av在线| 夜夜嗨网站十八久久| 老妇喷水一区二区三区| 欧美香蕉视频| 亚洲国产精品小视频| 欧美一区二区精品久久911| 欧美黑人国产人伦爽爽爽| 亚洲一区二区黄| 欧美激情视频给我| 狠狠色综合色区| 午夜精品av| 亚洲日本精品国产第一区| 亚洲视频日本| 蜜臀99久久精品久久久久久软件| 国产精品高清免费在线观看| 亚洲国产美女| 麻豆成人在线观看| 欧美一级专区免费大片| 欧美日韩午夜剧场| 日韩一级黄色大片| 欧美激情偷拍| 久久男人av资源网站| 国内精品模特av私拍在线观看| 亚洲欧美日韩区| 一本色道久久88综合日韩精品| 久久尤物电影视频在线观看| 国产又爽又黄的激情精品视频| 亚洲性感美女99在线| 亚洲精品综合在线| 欧美女同在线视频| 一本色道久久99精品综合| 亚洲风情在线资源站| 狼人天天伊人久久| 亚洲国产va精品久久久不卡综合| 美女脱光内衣内裤视频久久影院 | 最新亚洲激情| 亚洲国产精品久久91精品| 午夜在线一区二区| 国产精品99久久久久久久vr| 欧美三级特黄| 亚洲尤物在线视频观看| 亚洲私拍自拍| 国产精品久久久久久户外露出 | 好吊视频一区二区三区四区| 欧美一区二区在线| 性欧美xxxx视频在线观看| 国产伦理一区| 久久躁日日躁aaaaxxxx| 另类成人小视频在线| 亚洲黄色免费电影| 亚洲人成在线影院| 国产精品久久一级| 欧美在线亚洲在线| 久久亚洲美女| 一区二区三区国产在线观看| 亚洲一区二区3| 91久久夜色精品国产九色| 亚洲精品久久久蜜桃| 香蕉国产精品偷在线观看不卡| 日韩一二三区视频| 国产精品成人免费精品自在线观看| 午夜在线一区二区| 久久五月婷婷丁香社区| 一区二区三区日韩| 亚洲欧美中文日韩v在线观看| 国产亚洲一级| 亚洲欧洲在线观看| 国产一区二区精品| 亚洲日本视频| 国产视频在线观看一区| 欧美黄色aaaa| 国产欧美视频在线观看| 欧美国产精品久久| 国产精品久在线观看| 欧美激情在线免费观看| 国产欧美欧美| 亚洲国产岛国毛片在线| 国产午夜精品一区理论片飘花| 欧美高清在线一区| 国产美女扒开尿口久久久| 欧美激情bt| 国产午夜精品理论片a级探花| 亚洲国产裸拍裸体视频在线观看乱了中文| 国产精品久久久久久久7电影| 亚洲第一综合天堂另类专| 国产欧美日韩免费| 亚洲理伦在线| 亚洲人成高清| 久久精品国产亚洲精品| 一本色道久久99精品综合| 久久在线免费观看| 欧美伊人久久久久久午夜久久久久| 欧美激情一级片一区二区| 男男成人高潮片免费网站| 国产一区二区三区高清在线观看| aⅴ色国产欧美| 亚洲免费成人| 欧美成人在线网站| 欧美激情一区二区三区四区| 在线观看91精品国产麻豆| 久久精品一区二区三区不卡牛牛| 性一交一乱一区二区洋洋av| 欧美日韩亚洲高清一区二区| 亚洲欧洲一区| 一区二区三区免费看| 欧美精品一区二| 亚洲片在线观看| 亚洲伦理在线| 欧美激情视频一区二区三区免费 | 国产精品你懂的在线| 日韩午夜在线电影| 一区二区成人精品| 欧美日韩另类国产亚洲欧美一级| 欧美激情精品久久久久久黑人 | 亚洲国产va精品久久久不卡综合| 欧美一级日韩一级| 久久精品99国产精品| 国内精品久久久久久久果冻传媒| 亚洲欧美日韩一区二区三区在线观看| 亚洲永久精品大片| 国产精品久久久久久久久婷婷 | 亚洲一区二区在线观看视频| 欧美顶级少妇做爰| 欧美另类专区| 亚洲欧美国产另类| 欧美色偷偷大香| 在线一区亚洲| 欧美影院一区| 国产午夜精品在线| 久久琪琪电影院| 欧美韩日视频| 在线视频日本亚洲性| 国产精品v日韩精品v欧美精品网站| 中文在线资源观看视频网站免费不卡| 午夜亚洲性色福利视频| 国产麻豆成人精品| 久久欧美中文字幕| 亚洲精品欧美日韩| 久久精品综合一区| 欧美一级大片在线观看| 香蕉成人久久| 一区在线影院| 欧美精品国产精品| 亚洲午夜精品久久久久久app| 欧美一区二区三区久久精品茉莉花| 国产精品网站视频| 另类酷文…触手系列精品集v1小说| 亚洲国语精品自产拍在线观看| 在线观看的日韩av| 欧美精品一区二区久久婷婷| 中日韩午夜理伦电影免费| 久久资源av| 亚洲午夜91| 在线观看一区二区视频| 欧美婷婷六月丁香综合色| 久久国产天堂福利天堂| 日韩一区二区精品视频| 美国十次成人| 欧美亚洲三区| 日韩一二在线观看| 在线观看福利一区| 国产欧美大片| 欧美日韩在线精品一区二区三区| 久久精品国产成人| 亚洲一区二区三区视频| 亚洲国产精品久久人人爱蜜臀| 欧美一区二区精品在线| 艳妇臀荡乳欲伦亚洲一区| 国内精品写真在线观看| 国产精品人人爽人人做我的可爱| 欧美激情1区2区|