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

隨筆-162  評論-223  文章-30  trackbacks-0
需求分析
   在數據結構中,樹有兩種存儲方式,一種是鏈式存儲,另一種是順序存儲。前者就是使用指針來記錄樹結點間的關系,在新增結點或刪除結點時,只需改變與父結點或兄弟結點的指針值即可,實現較為簡單;后者就是使用數組來存儲,可以用相對偏移量來記錄樹結點間的關系,在新增結點或刪除結點時,則不僅是改變與父結點或兄弟結點的相對偏移量,還需要改變其它結點的相對偏移量,實現較為復雜。近來在項目中,對一個普通文本文件進行分析提取數據,而這個文件內的數據從內容看,具有層次嵌套關系,要將這樣的數據發送到服務器去處理,我考慮了兩種如下方法:
 (1)自定義XML格式,在本地使用XML庫,如libxml2、tinyxml等,將數據寫到XML臨時文件或內存中,再將這個XML臨時文件或內存發過去,在服務器那邊使用XML庫來解析。這種方法比較通用而且跨平臺,如果XML庫不支持將其所存儲的數據轉儲到一塊連續內存,那么就只能先存到XML臨時文件,再將這個文件發過去,這樣一來,就存在磁盤IO操作,效率較低。否則,就可以先將數據轉儲到一塊連續內存,再將這塊內存發過去,這樣一來,這塊連續內存就需要另外開辟,因此多了一套內存管理操作,但是比用臨時文件方式,沒有磁盤IO,效率要高些。
 (2)實現基于順序存儲的樹,而且還是多叉樹,因為實際數據具有多層次嵌套關系,將數據放進這顆樹中,再直接將這顆樹發過去,在服務器那邊直接解析這顆樹,這樣一來,不用臨時文件,沒有磁盤IO,無須另外開辟內存,充分利有現有空間,效率較高。

設計開發
   從服務器效率至上的觀點考慮,我選擇了第2種方法,并實現了基于順序存儲的多叉樹,關于順序存儲,又有兩種如下方式:
  (1)深度優先存儲,按照自上而下從左到右存儲樹的所有結點,先存儲結點及它的孩子,再存儲它的兄弟。因此結點的孩子和兄弟都不一定是連續的,當一個結點的所有孩子都是葉子結點時,則所有孩子是連續存放的。結點和它的第一個孩子(若有)是連續的,如下圖所示                            
                      
 
  (2)廣度優先存儲,
按照從左到右自上而下存儲樹的所有結點,先存儲結點及它的兄弟,再存儲它的孩子,因此結點的孩子和兄弟都是連續存放的,孩子與其父親之間不一定是連續的,如下圖所示 
                      

   本文描述第1種存儲方式實現的多叉樹,介紹三種主要操作:設置根結點、增加結點和刪除結點,為簡單起見,使用vector作為內部動態數組,使用索引而非迭代器作為外部接口,來訪問結點,索引0表示空索引,有效索引從1開始。關于迭代器的設計,有諸多考慮,如前序、后序、廣度優先、指定深度、葉子結點等各種遍歷方法,因時間和篇幅原因,不能一一講述,待后面有時間會陸續補充完善。
   1)樹結點定義,由5個偏移量域和1個數據域組成,C++代碼描述如下    
 1template<typename T>
 2struct order_tree_node
 3{
 4    size_t parent_;      
 5    size_t first_child_;  
 6    size_t last_child_;  
 7    size_t prev_sibling_; 
 8    size_t next_sibling_; 
 9    T      data_;         
10
11    order_tree_node();        
12    order_tree_node(const T& data);    
13}
;
  為了方便,定義了order_tree_node的兩個構造函數,其實現如下
 1    template<typename T>
 2    order_tree_node<T>::order_tree_node()
 3        :parent_(0)
 4        ,first_child_(0)
 5        ,last_child_(0)
 6        ,prev_sibling_(0)
 7        ,next_sibling_(0)
 8    {
 9    }

10    template<typename T>
11    order_tree_node<T>::order_tree_node(const T& data)
12        :parent_(0)
13        ,first_child_(0)
14        ,last_child_(0)
15        ,prev_sibling_(0)
16        ,next_sibling_(0)
17        ,data_(data)
18    {
19    }
   2)設置根結點,為方便實現,根結點固定存放在數組中第1個位置,對應下標為0,C++代碼描述如下  
 1    template<typename T>
 2    inline typename mtree<T,false>::iterator_base mtree<T,false>::set_root(const T& val)
 3    {
 4        if (!base_type::empty())
 5        {
 6            *(get_root()) = val;
 7        }

 8        else
 9        {
10            tree_node node(val);
11            push_back(node);
12        }

13        return iterator_base(this,0);
14    }
   這里用到了get_root函數來獲取根結點,其實現如下
 1    template<typename T>
 2    inline typename mtree<T,false>::iterator_base mtree<T,false>::get_root() 
 3    {
 4        return iterator_base(this,0);
 5    }

 6    template<typename T>
 7    inline typename mtree<T,false>::const_iterator_base mtree<T,false>::get_root() const
 8    {
 9        return const_iterator_base(this,0);
10    }
   3)增加結點,這里要分為三步,第一步要找到插入位置,第二步插入結點,第三步改變相關結點的相對偏移量,這里相關結點包括當前所插結點、所插結點兄弟結點、父結點、祖先結點及其右兄弟結點;注意,這里可以作一些異常安全考慮,即如果第二步操作失敗了,則可直接返回,這樣就可保證整顆樹不受影響。為了簡單起見,以下C++代碼對異常安全沒有作處理,描述如下
 1    template<typename T>
 2    template<typename tree_iterator>
 3    inline tree_iterator mtree<T,false>::append_child(tree_iterator iter,const T& val)
 4    {
 5        assert(!iter.is_null());
 6        size_t off = append_child(iter.off_,val);    
 7        tree_iterator it(iter);
 8        it.off_ = off;
 9        return it;
10    }

11    template<typename T>
12    inline typename mtree<T,false>::fd_iterator mtree<T,false>::append_child(fd_iterator iter,const T& val)
13    {
14        assert(!iter.is_null());
15        size_t off = append_child(iter.off_,val);    
16        fd_iterator it(iter);
17        it.off_ = off; ++it.depth_;
18        return it;
19    }
   以上模板成員函數及其深度迭代器的特化版本都調用了內部append_child(size_t)函數,該函數實現如下:
 1    template<typename T>
 2    inline size_t mtree<T,false>::append_child(size_t index,const T& val)
 3    {
 4        size_t parent = index, pos;
 5        tree_node *p_parent = &(*this)[parent],*p_node, *p_child;
 6
 7        //找到插入位置
 8        pos = parent; p_node = p_parent;
 9        while (p_node->last_child_)
10        {
11            pos += p_node->last_child_;
12            p_node = &(*this)[pos];
13        }

14        size_t child = ++pos; 
15        //插入結點
16        tree_node node(val);
17        if (child >= this->size())
18            push_back(node);
19        else
20            base_type::insert(begin()+child,node);
21
22        //更新當前結點的prev_sibling值和其左兄弟結點的next_sibling值
23        p_parent = &(*this)[parent];
24        p_child = &(*this)[child]; 
25        if (p_parent->last_child_)
26        {
27            pos = parent+p_parent->last_child_;
28            (*this)[pos].next_sibling_ = p_child->prev_sibling_ = child-pos;
29        }

30        //從父結點開始,向上更新當前結點所有右邊結點的偏移量
31        size_t next; 
32        tree_node* p_next;
33        pos = parent;
34        do 
35        {
36            p_node = &(*this)[pos];
37            if (p_node->next_sibling_)
38            {
39                if (p_node->parent_)
40                    ++(*this)[pos-p_node->parent_].last_child_;
41                //更新其祖先結點的next_sibling值
42                ++p_node->next_sibling_;
43                next = pos + p_node->next_sibling_;
44                p_next = &(*this)[next];
45                //更新其祖先結點的第一個右兄弟結點的prev_sibling值
46                ++p_next->prev_sibling_;
47                //更新其祖先結點的所有右兄弟結點的parent值
48                do 
49                {
50                    p_next = &(*this)[next];
51                    ++p_next->parent_;
52                    next += p_next->next_sibling_;
53                }
 while(p_next->next_sibling_);
54            }
    
55            pos -= p_node->parent_;
56        }
 while(p_node->parent_);
57
58        //更新當前結點的parent值和其父結點的firsh_child和last_child值
59        p_parent->last_child_ = p_child->parent_ = child-parent;
60        if (!p_parent->first_child_)
61            p_parent->first_child_ = p_child->parent_;
62        return child;
63    }
   4)刪除結點,分為兩步,第一步先刪除結點及其所有后代結點,也就是刪除以該結點為根的子樹,由于這顆子樹所有結點是連續存放的,因此可以批量一起刪除,第二步更新所有相關結點的偏移量,這里相關結點包括所刪除結點的兄弟結點、父結點、祖先結點及其右兄弟結點。注意,這里可以作一些異常安全考慮,即如果第二步操作失敗了,則可直接返回,這樣就可保證整顆樹不受影響。為了簡單起見,以下C++代碼對異常安全沒有作處理,描述如下
 1    template<typename T>
 2    template<typename tree_iterator>
 3    tree_iterator mtree<T,false>::erase(tree_iterator iter)
 4    {
 5        assert(!iter.is_null());
 6
 7        tree_iterator it(iter);
 8        it.skip_progeny(true); 
 9        ++it;
10        size_t num = erase(iter.off_);
11        if (!it.is_null() && it.off_>iter.off_)
12            it.off_ -= num;
13        return it;
14    }
   當刪除一個結點時,實質也就是刪除以該結點為根的子樹時,要注意迭代器的行為,這里應該要跳過所有后代結點,直接進到下一個結點進行后續遍歷,由于具有多種迭代器,因此使用了模板成員函數,其內
部調用了erase(size_t)重載版本,實現如下
 1template<typename T>
 2    inline size_t mtree<T,false>::erase(size_t index)
 3    {
 4        tree_node* p_node = &(*this)[index];
 5        size_t prev=p_node->prev_sibling_,next=p_node->next_sibling_,parent=p_node->parent_;
 6
 7        //計算以該結點為根的子樹所有結點數
 8        size_t num = size(index), pos;
 9        //批量刪除該結點及其所有后代結點
10        size_t first = index, last = first+num;
11        base_type::erase(begin()+first,begin()+last);
12
13        //保存兄弟結點及其父結點的偏移量
14        tree_node *p_prev=NULL, *p_next=NULL,*p_parent=NULL;
15        if (prev) p_prev = &(*this)[index-prev];
16        if (next) p_next = &(*this)[index];
17        if (parent) p_parent = &(*this)[index-parent];
18
19        if (p_next)  //被刪除結點不是最后一個孩子結點時
20        {
21            //更新父結點的last_child值
22            p_parent->last_child_ -= num;
23            //更新第一個右兄弟結點的prev_sibling值
24            p_next->prev_sibling_ = prev;
25            //更新所有右兄弟結點的parent值
26            pos = index;
27            do 
28            {
29                p_node = &(*this)[pos];
30                p_node->parent_ -= num;
31                pos += p_node->next_sibling_;
32            }
 while(p_node->next_sibling_);
33        }

34        else   //被刪除結點是最后一個孩子結點時
35        {
36
37            if (p_prev)
38            {
39                //更新左兄弟結點的next_sibling值和父結點的parent值
40                p_prev->next_sibling_ = next;
41                p_parent->last_child_ -= prev;
42            }

43            else //父結點只有一個該孩子結點時
44            {
45                if (p_parent)
46                {
47                    //更新父結點的first_child和last_child值
48                    p_parent->first_child_ = p_parent->last_child_ = 0;
49                }

50            }

51        }

52        if (NULL==p_parent)  return num;
53
54        //從父結點開始,向上更新當被刪除結點的所有右邊結點的偏移量
55        pos = index-parent;
56        do 
57        {
58            p_node = &(*this)[pos];
59            if (p_node->next_sibling_)
60            {
61                //更新祖先結點的next_sibling值
62                p_node->next_sibling_ -= num;
63                //更新祖先結點的第一個右兄弟結點的prev_sibling值
64                next = pos + p_node->next_sibling_;
65                p_next = &(*this)[next];
66                p_next->prev_sibling_ -= num;
67                if (p_node->parent_)  //存在父結點
68                {
69                    //更新父結點的last_child值
70                    (*this)[pos-p_node->parent_].last_child_ -= num;
71                }

72                //更新所有祖先結點的右兄弟結點的parent值
73                do 
74                {
75                    p_next = &(*this)[next];
76                    p_next->parent_ -= num;
77                    next += p_next->next_sibling_;
78                }
 while(p_next->next_sibling_);
79            }

80            pos -= p_node->parent_;
81        }
 while(p_node->parent_);
82        return num;
83    }

擴展優化
   由于是使用vector容器來管理樹結點tree_node,因此,如果數據是非平凡的類對象,當插入結點或刪除結點時,就存在著移動時拷貝構造、析構的開銷,而實際上這種開銷完全可以避免,這就需要自己設計實現數據的內存管理了,當加入結點時,只需將數據拷貝到這塊內存對應的位置上,當刪除結點時,只需移動后面的內存數據即可,關于移動調用C函數memmove即可;另外,這個mtree是個模板類,只能管理同一種類型的數據,如果想管理多種不同類型的數據,可以通過把mtree變為普通類,append_child變為模板成員函數,在tree_node中加入長度域來表示數據的大小來實現,這樣一來,獲取數據的函數也應該是模板成員函數,而具體的數據類型由業務層來決定。
posted on 2011-07-13 15:10 春秋十二月 閱讀(4640) 評論(9)  編輯 收藏 引用 所屬分類: Algorithm

評論:
# re: 判斷整數的正負零特性 2011-07-13 17:08 | jejer
這不是擴展 叫刪減吧...  回復  更多評論
  
# re: 判斷整數的正負零特性[未登錄] 2011-07-13 17:10 | xiaok
uint32_t a = (uint32_t) val, b = (uint32_t)-val;
return (b>>31) - (a>>31);

這樣可以把  回復  更多評論
  
# re: 判斷整數的正負零特性 2011-07-13 20:52 | 草原神鷹
@xiaok
你這樣不對,舉個例子, 若int32_t val = 0x80000000,則check32(val)為0,而不是-1。  回復  更多評論
  
# re: 判斷整數的正負零特性 2011-07-13 23:49 | flyinghearts
位運算技巧 建議看 hacker's delight 這本書。

兩道題,可以直接用:
 return (value >= 0) - (value <= 0);
  return value == 0;  //判斷是否為0
雖然有條件判斷,但是編譯器優化后的代碼一般不存在條件跳轉。

非要用位運算的話:

bool is_zero(int value)
{
//return value == 0;
const int bits = CHAR_BIT * sizeof(value) - 1;
// return 1 + (((value - 1) & ~value) >> bits);
return 1 + (((value & -value) - 1) >> bits);
}

int get_sign(int value)
{
//return (value >= 0) - (value <= 0);
const int bits = CHAR_BIT * sizeof(value) - 1;
return (value >> bits) | -(-value >> bits);
}



  回復  更多評論
  
# re: 判斷整數的正負零特性[未登錄] 2011-07-14 10:02 | xiaok
@草原神鷹
沒明白=,=

0x80000000 不是=-2147483648<0么  回復  更多評論
  
# re: 判斷整數的正負零特性 2011-07-14 13:18 | crazy
(!!val)+2*(val>>31)  回復  更多評論
  
# re: 判斷整數的正負零特性[未登錄] 2011-07-14 21:06 | cexer
嵌入式?如果是一般應用開發面試出這種題有點不靠譜,寫產品的人哪能鉆到這樣的細節里去。  回復  更多評論
  
# re: 判斷整數的正負零特性 2011-07-15 18:02 | Khan's Notebook
負數不是補碼么  回復  更多評論
  
# re: 判斷整數的正負零特性 2012-01-10 12:45 | 張龍琪
@xiaok
解析下  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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| 久久精品视频亚洲| 免费久久久一本精品久久区| 免费在线看一区| 欧美国产激情二区三区| 亚洲高清在线观看一区| 久久亚洲国产精品日日av夜夜| 日韩午夜在线| 欧美激情一区二区三区不卡| 亚洲国产成人午夜在线一区 | 久久国产精品亚洲77777| 亚洲美女中文字幕| 欧美粗暴jizz性欧美20| 久久福利一区| 欧美经典一区二区| 久久亚洲精品视频| 亚洲一二三区在线| 久久综合狠狠综合久久综青草| 亚洲欧美日本伦理| 国产精品久久久爽爽爽麻豆色哟哟| 亚洲在线1234| 亚洲综合第一| 午夜精品久久久久久久99热浪潮 | 午夜久久黄色| 国产精品久久久久久久久搜平片| 老司机一区二区三区| 免费的成人av| 一本色道久久综合亚洲精品婷婷 | 久久久久久久一区二区三区| 亚洲欧美日本国产专区一区| 狠狠色丁香久久婷婷综合丁香| 狂野欧美一区| 国产在线精品一区二区夜色| 久久一区二区三区四区| 久久福利精品| 亚洲国产高清在线| 国产精品magnet| 久久精品亚洲精品国产欧美kt∨| 久久最新视频| 亚洲欧美日韩一区| 午夜精品久久久久久久99热浪潮| 国产精品女主播| 午夜视频一区在线观看| 亚洲高清资源| 亚洲精品中文字幕在线| 国产精品久久久久久久久婷婷 | 亚洲久久一区| 欧美成人精品h版在线观看| 欧美高清一区二区| 亚洲自拍偷拍福利| 日韩视频在线观看免费| 久久久久国产一区二区三区四区| 性欧美videos另类喷潮| 国产乱码精品一区二区三区忘忧草| 亚洲精品一区二区三区福利| 国产欧美日韩视频一区二区三区| 欧美激情片在线观看| 午夜亚洲精品| 蜜臀久久99精品久久久久久9| 亚洲第一黄色网| 欧美影片第一页| 欧美成人国产一区二区| 欧美在线影院在线视频| 一区在线播放| 今天的高清视频免费播放成人 | 欧美激情按摩| 亚洲日本成人网| 亚洲麻豆国产自偷在线| 久久在精品线影院精品国产| 久久天堂精品| 欧美大片免费| 欧美视频精品一区| 欧美日韩国产欧| 国产精一区二区三区| 亚洲精品麻豆| 亚洲福利视频三区| 亚洲人成艺术| 欧美在线观看你懂的| 欧美成人午夜激情| 亚洲视频www| 欧美日韩精品一区二区| 国产又爽又黄的激情精品视频| 亚洲一区二区三区中文字幕在线 | 樱花yy私人影院亚洲| 亚洲麻豆av| 国产日韩欧美精品一区| 久久精品一本久久99精品| 欧美激情精品久久久久久久变态 | 亚洲一区二区三区中文字幕在线 | 另类天堂视频在线观看| 亚洲视频免费在线| 蘑菇福利视频一区播放| 国产精品国产三级欧美二区| 亚洲国产一区二区视频| 裸体素人女欧美日韩| 亚洲一区尤物| 欧美婷婷久久| 亚洲免费激情| 亚洲国产精品一区二区www| 欧美一区二区高清| 国内激情久久| 久久综合伊人77777麻豆| 亚洲在线免费观看| 国产精品国产精品国产专区不蜜| 欧美伊久线香蕉线新在线| 亚洲黑丝在线| 久久人91精品久久久久久不卡| 欧美国产精品日韩| 亚洲嫩草精品久久| 欧美一区二区三区在线免费观看| 国内激情久久| 亚洲国产日韩欧美在线图片| 国产午夜亚洲精品羞羞网站| 国产精品高潮呻吟| 亚洲第一久久影院| 久久天天躁狠狠躁夜夜av| 亚洲欧洲午夜| 亚洲欧美日韩国产综合在线| 国产伦精品一区| 久久久久欧美| 欧美日韩精品一区视频| 久久综合九色综合久99| 欧美日韩在线播| 亚洲午夜成aⅴ人片| 亚洲午夜在线观看视频在线| 久久免费黄色| 亚洲毛片播放| 久久久蜜臀国产一区二区| 国产精品不卡在线| 亚洲激情偷拍| 国产女人18毛片水18精品| 这里只有精品在线播放| 欧美一区二区三区免费观看视频 | 欧美在线啊v| 欧美伦理在线观看| 久久精品国内一区二区三区| 欧美国产三级| 亚洲经典在线看| 在线亚洲一区二区| 久久久久一区二区三区四区| 亚洲一区免费在线观看| 国外成人网址| 欧美日韩精品一区二区| 欧美国产一区视频在线观看| 欧美亚州韩日在线看免费版国语版| 亚洲精品美女免费| 亚洲六月丁香色婷婷综合久久| 性欧美暴力猛交另类hd| 国产精品热久久久久夜色精品三区| 亚洲婷婷在线| 亚洲福利在线观看| 欧美一区永久视频免费观看| 影音先锋亚洲电影| 欧美日韩国产一级片| 亚洲在线不卡| 久久国产主播精品| 亚洲综合精品四区| 在线看欧美视频| 欧美日韩国产精品专区| 亚洲精品韩国| 久久视频免费观看| 欧美一区二区三区在线播放| 在线成人欧美| 国产真实久久| 日韩视频一区二区三区| 久久国产精品久久久久久电车 | 欧美福利视频一区| 另类专区欧美制服同性| 久久超碰97人人做人人爱| 国产精品国产一区二区| 一区二区三区视频免费在线观看| 久久精品中文字幕一区| 亚洲国产视频一区| 欧美在线一二三| 午夜精品一区二区三区电影天堂| 性欧美18~19sex高清播放| 亚洲色无码播放| 一区二区三区高清视频在线观看| 亚洲人在线视频| 欧美日韩在线大尺度| 亚洲美女91| 午夜精品久久久久久99热软件| 9久re热视频在线精品| 一区二区高清视频在线观看| 久久精品91久久香蕉加勒比| 久久综合久久综合这里只有精品| 另类激情亚洲| 国产一区二区电影在线观看| 一本色道久久综合亚洲精品小说| 亚洲精品久久久久久久久久久| 性色一区二区| 欧美片第1页综合| 国产一区二区精品久久99| 亚洲香蕉在线观看| 一区二区三区欧美日韩| 欧美亚洲在线观看| 欧美三日本三级少妇三2023|