日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

信息学奥赛一本通 1981:【18NOIP普及组】对称二叉树 | 洛谷 P5018【NOIP2018 普及组】 对称二叉树

發(fā)布時間:2025/3/17 编程问答 14 豆豆
生活随笔 收集整理的這篇文章主要介紹了 信息学奥赛一本通 1981:【18NOIP普及组】对称二叉树 | 洛谷 P5018【NOIP2018 普及组】 对称二叉树 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

【題目鏈接】

ybt 1981:【18NOIP普及組】對稱二叉樹
洛谷 P5018【NOIP2018 普及組】 對稱二叉樹

【題目考點】

  • 二叉樹
  • 【解題思路】

  • 先求出二叉樹中各子樹的結(jié)點數(shù)
  • 遍歷二叉樹,判斷每個子樹是否是對稱二叉樹
    • 判斷一個二叉樹是否是對稱二叉樹:
    • 對稱變換:將該樹中所有結(jié)點的左右子樹交換。顯然,一棵樹經(jīng)過兩次對稱變換后,會變?yōu)樵瓉淼臉洹?/li>
    • 兩樹對稱:如果樹1經(jīng)過對稱變換可以變?yōu)闃?,那么稱樹1與樹2對稱。顯然,樹1與樹2對稱時,樹2與樹1對稱。
    • 如果一個二叉樹是對稱二叉樹,那么其左子樹應(yīng)該與右子樹對稱。
      • 證明:二叉樹是對稱的,那么其經(jīng)過對稱變換后的左子樹,是由原右子樹變換過來的。已知對稱變換后的樹與原樹相同,那么變換后的左子樹與原左子樹相同,那么原左子樹與原右子樹是對稱的。
    • 設(shè)函數(shù)check(),參數(shù)為樹1、樹2兩棵樹的根,該函數(shù)判斷樹1、樹2是否對稱。
    • 判斷兩棵樹是否對稱:
      • 如果兩棵樹都是空樹(根結(jié)點地址都為-1),那么它們是對稱的。
      • 如果兩棵樹都不是空樹,同時滿足如下條件時,二者才是對稱的
        • 兩棵樹根結(jié)點的權(quán)值應(yīng)該相同
        • 兩棵樹的結(jié)點數(shù)應(yīng)該相同
        • 樹1的左子樹應(yīng)該與樹2右子樹對稱。
        • 樹1的右子樹應(yīng)該與樹2的左子樹對稱

    【題解代碼】

    解法1:

    #include<bits/stdc++.h> using namespace std; #define N 1000005 typedef struct Node {int val, left, right;//val:權(quán)值, left:左孩子地址 right:右孩子地址 }Node; Node tree[N];//tree[i],為id為i的結(jié)點 int treeNodeNum[N];//treeNodeNum[i]保存id為i的結(jié)點的子樹的結(jié)點數(shù)。 int n; int mxNodeNum;//結(jié)點數(shù)量最大的對稱子樹的結(jié)點數(shù)//判斷以root1為根的子樹和以root2為根的子樹是否對稱 bool check(int root1, int root2) {if(root1 == -1 && root2 == -1)//空結(jié)點,相同return true;else if(root1 != -1 && root2 != -1 && //左右子樹都有結(jié)點tree[root1].val == tree[root2].val && //樹根結(jié)點權(quán)相同,treeNodeNum[root1] == treeNodeNum[root2] && //樹結(jié)點數(shù)相同check(tree[root1].left, tree[root2].right) && //經(jīng)過對稱變換后,root2樹的右子樹應(yīng)該與root1樹的左子樹對稱check(tree[root1].right, tree[root2].left))//經(jīng)過對稱變換后,root2樹的左子樹應(yīng)該與root1樹的右子樹對稱return true;elsereturn false; }//獲取樹的結(jié)點數(shù) int getNodeNum(int root) {if(root == -1)return 0;else{int nodeNum = getNodeNum(tree[root].left) + getNodeNum(tree[root].right) + 1;treeNodeNum[root] = nodeNum;return nodeNum;} }//確定結(jié)點最多的對稱的子樹的結(jié)點數(shù) void Search(int root) {if(root != -1){if(treeNodeNum[root] <= mxNodeNum)//剪枝,如果當(dāng)前子樹的結(jié)點數(shù)已經(jīng)比獲得的最大對稱子樹結(jié)點數(shù)要小,那么沒必要再搜索下去。return;if(check(tree[root].left, tree[root].right))//如果樹root是對稱的,那么其左右子樹對稱mxNodeNum = treeNodeNum[root];//如果對稱,記錄結(jié)點數(shù)。由于已有剪枝操作,此時treeNodeNum[root]一定大于mxNodeNumelse{Search(tree[root].left);Search(tree[root].right);}} }int main() {scanf("%d\n", &n);for(int i = 1; i <= n; ++i)scanf("%d", &tree[i].val);for(int i = 1; i <= n; ++i)scanf("%d %d", &tree[i].left, &tree[i].right);getNodeNum(1);//初始化樹的各子樹結(jié)點數(shù)Search(1);//從根結(jié)點開始,遍歷所有子樹,找出對稱子樹,并看其結(jié)點數(shù),保存最大結(jié)點數(shù)printf("%d", mxNodeNum);return 0; }

    總結(jié)

    以上是生活随笔為你收集整理的信息学奥赛一本通 1981:【18NOIP普及组】对称二叉树 | 洛谷 P5018【NOIP2018 普及组】 对称二叉树的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。