PAT 1066 Root of AVL Tree (25)

題意

給出N個正整數(shù)你虹,將他們依次插入一課初始為空的AVL樹上贯底,求插入后的根節(jié)點(diǎn)的值

思路

無論是LL,LR,RL,RR型休建,我們都可以看成把第二大的當(dāng)作子樹的根袍镀,最小的為它的左子樹岗屏,最大的為右子樹辆琅,這樣就可以方便的計算啦(具體過程可以參考鄧俊輝的《數(shù)據(jù)結(jié)構(gòu)》相關(guān)內(nèi)容漱办,強(qiáng)推!M裱獭C渚)

代碼


#include<iostream>
#include<cmath>
#include<algorithm>
#include<stack>
#define getFater( a)  (a->father)
#define getHeight(a) (a==NULL?0:a->height)
#define isRoot(a) (a->father == NULL)
using namespace std;

typedef struct NODE
{
    int value;
    NODE * father;
    int height;
    NODE * lchild, *rchild;
}NODE ,*TREE;
TREE t;
NODE * opt34(NODE *a, NODE* b, NODE *c, NODE *t1, NODE* t2, NODE*t3, NODE*t4);


//判斷是不是父節(jié)點(diǎn)的左子樹
bool isLchild(NODE *a) { return (!isRoot(a) && (a->father->lchild == a)); }
//獲取決定node的高度的那個子節(jié)點(diǎn)
NODE * getTallerChild(NODE * node) { return (((getHeight(node->lchild)) > (getHeight(node->rchild))) ? (node->lchild) : (node->rchild)); }


//更新高度
void updateHeight(NODE * node)
{
    node->height =  max(getHeight(node->lchild), getHeight(node->rchild))+1;
    
}

//插入
NODE * insert(NODE * &node,NODE * &father,int a)
{
    if (node == NULL)
    {
        node = new NODE;
        node->lchild = node->rchild = NULL;
        if (father != node)
            node->father = father;
        else
            node->father = NULL;
        node->height = 1;
        node->value = a;
        return node;
    }
    else
    {
        if (node->value < a)
             return insert(node->rchild, node, a);
        else
            return insert(node->lchild,node,a);
    }
    
}

//旋轉(zhuǎn)
NODE * rotateAT(NODE * a)
{
    NODE *b = a->father;
    NODE *c = b->father;
    if (isLchild(a))
    {
        if (isLchild(b))
        {
            b->father = c->father;
            return opt34(a, b, c, a->lchild, a->rchild, b->rchild, c->rchild);
        }
        else
        {
            a->father = c->father;
            return opt34(c, a, b, c->lchild, a->lchild, a->rchild, b->rchild);
        }
    }
    else
    {
        if (!(isLchild(b)))
        {
            b->father = c->father;
            return opt34(c, b, a, c->lchild, b->lchild, a->lchild, a->rchild);
        }
        else
        {
            a->father = c->father;
            return opt34(b, a, c, b->lchild, a->lchild, a->rchild, c->rchild);
        }
    }
}

//具體操作3+4
NODE * opt34(NODE *a,NODE* b,NODE *c,NODE *t1,NODE* t2,NODE*t3,NODE*t4)
{
    a->lchild = t1; if (t1 != NULL) t1->father = a;
    a->rchild = t2; if (t2 != NULL) t2->father = a;
    updateHeight(a);
    c->lchild = t3; if (t3 != NULL) t3->father = c;
    c->rchild = t4; if (t4 != NULL) t4->father = c;
    updateHeight(c);
    b->lchild = a; a->father = b;
    b->rchild = c; c->father = b;
    updateHeight(b);
    return b;
}

bool AvlBalanced(NODE * g)
{
    int height;
    height = getHeight(g->lchild)- getHeight(g->rchild);
    return (-2 < height)&&(height < 2);
}


//插入過程
void insertAVL(int x)
{
    NODE *temp = insert(t,t,x);
    for (NODE * g = getFater(temp);g!=NULL;g = getFater(g))
    {
               /*
              原來代碼這么寫,然后一直只有最后一個測試點(diǎn)過了隅很,并且出現(xiàn)段錯誤
              (isRoot(g) ? t : (isLchild(g) ? (g)->father->lchild : (g)->father->rchild)) =               rotateAT(getTallerChild(getTallerChild(g)));
              然后改為下文這么一串就對了
              */

        if (!AvlBalanced(g))//情愿代碼寫長一點(diǎn)撞牢,嗚嗚嗚
        {
            if (g == t)
            {
                t = rotateAT(getTallerChild(getTallerChild(g)));
            }
            else
            {
                if (g == g->father->lchild)
                {
                    g->father->lchild = rotateAT(getTallerChild(getTallerChild(g)));
                }
                else
                    g->father->rchild = rotateAT(getTallerChild(getTallerChild(g)));
            }
            
            break;
        }
        else
            updateHeight(g);
    }
    return ;
}

int main()
{
    t = NULL;
    int n;
    int value;
    int ore[25];
    scanf("%d",&n);
    for (int i = 0; i < n; i++)
    {
        scanf("%d",&value);
        insertAVL(value);
    }
    printf("%d\n",t->value);
    system("pause");
    return 0;
}


后記

這一題的通過率超級高,然而我一開始還是只對了一個測試點(diǎn)叔营,并且測試點(diǎn)4和5出現(xiàn)段錯誤屋彪,現(xiàn)在還是不知道為什么,希望能夠靈光一現(xiàn)绒尊,然后解決畜挥,嘻嘻

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市婴谱,隨后出現(xiàn)的幾起案子蟹但,更是在濱河造成了極大的恐慌,老刑警劉巖谭羔,帶你破解...
    沈念sama閱讀 219,366評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件华糖,死亡現(xiàn)場離奇詭異,居然都是意外死亡瘟裸,警方通過查閱死者的電腦和手機(jī)客叉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,521評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來话告,“玉大人兼搏,你說我怎么就攤上這事∩彻” “怎么了佛呻?”我有些...
    開封第一講書人閱讀 165,689評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長病线。 經(jīng)常有香客問我吓著,道長,這世上最難降的妖魔是什么送挑? 我笑而不...
    開封第一講書人閱讀 58,925評論 1 295
  • 正文 為了忘掉前任夜矗,我火速辦了婚禮,結(jié)果婚禮上让虐,老公的妹妹穿的比我還像新娘紊撕。我一直安慰自己,他們只是感情好赡突,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,942評論 6 392
  • 文/花漫 我一把揭開白布对扶。 她就那樣靜靜地躺著区赵,像睡著了一般。 火紅的嫁衣襯著肌膚如雪浪南。 梳的紋絲不亂的頭發(fā)上笼才,一...
    開封第一講書人閱讀 51,727評論 1 305
  • 那天,我揣著相機(jī)與錄音络凿,去河邊找鬼骡送。 笑死,一個胖子當(dāng)著我的面吹牛絮记,可吹牛的內(nèi)容都是我干的摔踱。 我是一名探鬼主播,決...
    沈念sama閱讀 40,447評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼怨愤,長吁一口氣:“原來是場噩夢啊……” “哼派敷!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起撰洗,我...
    開封第一講書人閱讀 39,349評論 0 276
  • 序言:老撾萬榮一對情侶失蹤篮愉,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后差导,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體试躏,經(jīng)...
    沈念sama閱讀 45,820評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,990評論 3 337
  • 正文 我和宋清朗相戀三年设褐,在試婚紗的時候發(fā)現(xiàn)自己被綠了颠蕴。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,127評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡络断,死狀恐怖裁替,靈堂內(nèi)的尸體忽然破棺而出项玛,到底是詐尸還是另有隱情貌笨,我是刑警寧澤,帶...
    沈念sama閱讀 35,812評論 5 346
  • 正文 年R本政府宣布襟沮,位于F島的核電站锥惋,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏开伏。R本人自食惡果不足惜膀跌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,471評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望固灵。 院中可真熱鬧捅伤,春花似錦、人聲如沸巫玻。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,017評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至熄诡,卻和暖如春可很,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背凰浮。 一陣腳步聲響...
    開封第一講書人閱讀 33,142評論 1 272
  • 我被黑心中介騙來泰國打工我抠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人袜茧。 一個月前我還...
    沈念sama閱讀 48,388評論 3 373
  • 正文 我出身青樓菜拓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親惫周。 傳聞我的和親對象是個殘疾皇子尘惧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,066評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 定義 平衡二叉樹,是對二叉搜索樹的一種優(yōu)化递递。 向二叉搜索樹中插入元素時喷橙,不同的插入次序,將構(gòu)造出不同結(jié)構(gòu)的樹登舞。通俗...
    IAM四十二閱讀 4,014評論 0 2
  • 一直以來贰逾,我都很少使用也避免使用到樹和圖,總覺得它們神秘而又復(fù)雜,但是樹在一些運(yùn)算和查找中也不可避免的要使用到菠秒,那...
    24K男閱讀 6,753評論 5 14
  • 樹的高度和深度 From CSDN 暴走抹茶MOUKOY 1.高度 對于高度的理解疙剑,我們不管他數(shù)據(jù)結(jié)構(gòu)什么什么知識...
    _Nullptr閱讀 1,304評論 0 2
  • 1. AVL樹 AVL樹簡單來說是帶有平衡條件的二叉查找樹.傳統(tǒng)來說是其每個節(jié)點(diǎn)的左子樹和右子樹的高度最多差1(注...
    fredal閱讀 1,834評論 0 4
  • 樹 樹 樹是一種重要的數(shù)據(jù)結(jié)構(gòu),通過鏈表建立一棵樹: 用鏈表的結(jié)構(gòu)構(gòu)建樹践叠,就是使樹的每一條分支都建立一個鏈表來存儲...
    JnJnLee閱讀 228評論 0 0