c語言 手寫平衡二叉樹(一)

定義結(jié)構(gòu)體

struct Person
{
//父節(jié)點(diǎn)
    struct Person* parent;
    //左子節(jié)點(diǎn)
    struct Person* lchild;
    //右子節(jié)點(diǎn)
    struct Person* rchild;
    int m;
};

struct ABLtree
{//保存根節(jié)點(diǎn)信息
    struct Person* root;
};

初始化根節(jié)點(diǎn)

struct ABLtree* init_tree()
{
//在堆區(qū)開辟一個ABLtree大小的空間
    struct ABLtree* tree = malloc(sizeof(struct ABLtree));
    //開辟失敗返回NULL
    if (!tree)
    {
        return NULL;
    }
    //初始化根節(jié)點(diǎn)為NULL并返回
    tree->root = NULL;
    return tree;
}

添加節(jié)點(diǎn)

//比較函數(shù)队腐,返回兩者的差值
int compare(struct Person* p1, struct Person* p2)
{
    return ((p1->m) - (p2->m));
}
//添加節(jié)點(diǎn)函數(shù)跋涣,內(nèi)部調(diào)用
void insertPerson(struct Person* tree, struct Person* data, int(*compare)(struct Person*, struct Person*))
{
//如果當(dāng)前節(jié)點(diǎn)大于插入的節(jié)點(diǎn)
    if (compare(tree, data) > 0)
    {//if當(dāng)前節(jié)點(diǎn)的左節(jié)點(diǎn)為空
        if (NULL==tree->lchild)
        {//把數(shù)據(jù)賦值給節(jié)點(diǎn)的左子節(jié)點(diǎn)
            tree->lchild = data;
            //把當(dāng)前節(jié)點(diǎn)賦值給數(shù)據(jù)的父節(jié)點(diǎn)
            data->parent = tree;
            //返回
            return;
        }
        //不為空逻锐,則把數(shù)據(jù)的左子節(jié)點(diǎn)傳入函數(shù)
        insertPerson(tree->lchild, data, compare);
    }
    else {
    //if當(dāng)前節(jié)點(diǎn)的右節(jié)點(diǎn)為空
        if (NULL==tree->rchild)
        {
        //把數(shù)據(jù)賦值給節(jié)點(diǎn)的右子節(jié)點(diǎn)
            tree->rchild = data;
        //把當(dāng)前節(jié)點(diǎn)賦值給數(shù)據(jù)的父節(jié)點(diǎn)
            data->parent = tree;
            //返回
            return;
        }
        //不為空,則把數(shù)據(jù)的右子節(jié)點(diǎn)傳入函數(shù)
        insertPerson(tree->rchild, data, compare);
    }

}
//添加節(jié)點(diǎn)函數(shù)边篮,外部調(diào)用
void initer(struct ABLtree* mytree, struct Person* data, int(*compare)(struct Person*, struct Person*))
{
//判斷ABLtree是否NULL
    if (!tree)
    {
        return;
    }
    //判斷插入的數(shù)據(jù)是否NULL
    if (!data)
    {
        return;
    }
    //判斷比較函數(shù)是否NULL
    if (!compare)
    {
        return;
    }
    //如果根節(jié)點(diǎn)為空,把新添加的數(shù)據(jù)賦值給根節(jié)點(diǎn)却特,并返回
    if (!mytree->root)
    {
        mytree->root = data;
        return;
    }
    //利用遞歸添加子節(jié)點(diǎn)
    insertPerson((mytree->root), data, compare);
    //調(diào)節(jié)平衡
    mytree->root=Adjtree(mytree->root);

}

調(diào)節(jié)平衡

//計(jì)算樹的高度
int getHight(struct Person* p)
{
//如果節(jié)點(diǎn)不存在救崔,返回0
    if (!p)
    {
        return 0;
    }
    //計(jì)算左節(jié)點(diǎn)高度
    int m = getHight(p->lchild);
    //計(jì)算右節(jié)點(diǎn)高度
    int n = getHight(p->rchild);
    //左節(jié)點(diǎn)和右節(jié)點(diǎn)中的最大值+1
    return m > n ? ++m : ++n;
}
//向右旋轉(zhuǎn)
struct Person* rightHand(struct Person* data)
{
//記錄當(dāng)前節(jié)點(diǎn)
    struct Person* temp = data;
    //把當(dāng)前節(jié)點(diǎn)的左節(jié)點(diǎn)賦值給當(dāng)前節(jié)點(diǎn)
    data = data->lchild;
    //修改當(dāng)前節(jié)點(diǎn)的父指向
    data->parent = temp->parent;
//記錄修改后當(dāng)前節(jié)點(diǎn)的右指向
    struct Person* myrchild = data->rchild;
    //把當(dāng)前節(jié)點(diǎn)右指向賦值給以前節(jié)點(diǎn)的左指向
    temp->lchild = data->rchild;
    //如果不為空惶看,修改父指向
    if (data->rchild)
    {
        myrchild->parent = temp;
    }
        //把以前節(jié)點(diǎn)賦值給當(dāng)前節(jié)點(diǎn)的右節(jié)點(diǎn)
    data->rchild = temp;
        //修改以前節(jié)點(diǎn)的父指向
    temp->parent = data;
    //返回修改后的節(jié)點(diǎn)
    return data;
}
//向左旋轉(zhuǎn)
struct Person* leftHand(struct Person* data)
{
    //記錄當(dāng)前節(jié)點(diǎn)
    struct Person* temp = data;
    //把當(dāng)前節(jié)點(diǎn)的右節(jié)點(diǎn)賦值給當(dāng)前節(jié)點(diǎn)
    data = data->rchild;
    //修改當(dāng)前節(jié)點(diǎn)的父指向
    data->parent = temp->parent;
//記錄修改后當(dāng)前節(jié)點(diǎn)的左指向
    struct Person* mylchild = data->lchild;
    //把當(dāng)前節(jié)點(diǎn)左指向賦值給以前節(jié)點(diǎn)的右指向
    temp->rchild = data->lchild;
    //如果不為空捏顺,修改父指向
    if (mylchild)
    {
        mylchild->parent = temp;
    }
    //把以前節(jié)點(diǎn)賦值給當(dāng)前節(jié)點(diǎn)的左節(jié)點(diǎn)
    data->lchild = temp;
    //修改以前節(jié)點(diǎn)的父指向
    temp->parent = data;
    //返回修改后的節(jié)點(diǎn)
    return data;
}
struct Person* Adjtree(struct Person* tree)
{
//如果節(jié)點(diǎn)為空,返回
    if (NULL == tree)
    {
        return NULL;
    }
    //左節(jié)點(diǎn)或右節(jié)點(diǎn)有一個不為空才進(jìn)行判斷
    if (NULL != tree->lchild || NULL != tree->rchild)
    {
    //把左右節(jié)點(diǎn)傳入函數(shù)
    tree->lchild=Adjtree(tree->lchild);
        tree->rchild=Adjtree(tree->rchild);
        //計(jì)算左右節(jié)點(diǎn)的高度
        int l = getHight(tree->lchild);
        int r = getHight(tree->rchild);
        //判斷左右節(jié)點(diǎn)的高度纬黎,如果左節(jié)點(diǎn)比右節(jié)高度多于1草丧,進(jìn)行旋轉(zhuǎn)
        if ((l - r) > 1)
        {//比較子節(jié)點(diǎn)的左右節(jié)點(diǎn)高度
            struct Person* data = tree->lchild;
            int ll = getHight(data->lchild);
            int rr = getHight(data->rchild);
            //如果右節(jié)點(diǎn)大于左節(jié)點(diǎn)
            if (rr > ll)
            {//左旋一次
                tree->lchild=leftHand(data);
            }
            //右旋一次
            tree=rightHand(tree);
        }else//判斷左右節(jié)點(diǎn)的高度,如果右節(jié)點(diǎn)比左節(jié)高度多于1莹桅,進(jìn)行旋轉(zhuǎn)
        if ((r - l) > 1)
        {//比較子節(jié)點(diǎn)的左右節(jié)點(diǎn)高度
            struct Person* data = tree->rchild;
            int ll = getHight(data->lchild);
            int rr = getHight(data->rchild);
            //如果左節(jié)點(diǎn)大于右節(jié)點(diǎn)
            if (ll > rr)
            {
            //右旋一次
                tree->rchild=rightHand(data);
            }
            //左旋一次
            tree=leftHand(tree);
        }
    }
    //返回修改后的節(jié)點(diǎn)
    return tree;
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市烛亦,隨后出現(xiàn)的幾起案子诈泼,更是在濱河造成了極大的恐慌,老刑警劉巖煤禽,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件铐达,死亡現(xiàn)場離奇詭異,居然都是意外死亡檬果,警方通過查閱死者的電腦和手機(jī)瓮孙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來选脊,“玉大人杭抠,你說我怎么就攤上這事】疑叮” “怎么了偏灿?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長钝的。 經(jīng)常有香客問我翁垂,道長,這世上最難降的妖魔是什么硝桩? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任沿猜,我火速辦了婚禮,結(jié)果婚禮上碗脊,老公的妹妹穿的比我還像新娘啼肩。我一直安慰自己,他們只是感情好望薄,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布疟游。 她就那樣靜靜地躺著,像睡著了一般痕支。 火紅的嫁衣襯著肌膚如雪颁虐。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天卧须,我揣著相機(jī)與錄音另绩,去河邊找鬼儒陨。 笑死,一個胖子當(dāng)著我的面吹牛笋籽,可吹牛的內(nèi)容都是我干的蹦漠。 我是一名探鬼主播,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼车海,長吁一口氣:“原來是場噩夢啊……” “哼笛园!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起侍芝,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤研铆,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后州叠,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體棵红,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年咧栗,在試婚紗的時候發(fā)現(xiàn)自己被綠了逆甜。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡致板,死狀恐怖交煞,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情可岂,我是刑警寧澤错敢,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站缕粹,受9級特大地震影響稚茅,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜平斩,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一亚享、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧绘面,春花似錦欺税、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至瘦馍,卻和暖如春歼秽,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背情组。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工燥筷, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留箩祥,地道東北人。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓肆氓,卻偏偏與公主長得像袍祖,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子谢揪,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評論 2 354

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