菜鳥記錄:c語言實現PAT甲級1004--Counting Leaves

来源:https://www.cnblogs.com/whf10000010/archive/2023/04/17/17325936.html
-Advertisement-
Play Games

好消息:與上題的Emergency是同樣的方法。壞消息:又錯了&&c++真的比c方便太多太多。 A family hierarchy is usually presented by a pedigree tree. Your job is to count those family members ...


    好消息:與上題的Emergency是同樣的方法。壞消息:又錯了&&c++真的比c方便太多太多。

A family hierarchy is usually presented by a pedigree tree. Your job is to count those family members who have no child.

Input Specification:

Each input file contains one test case. Each case starts with a line containing 0<N<100, the number of nodes in a tree, and M (<N), the number of non-leaf nodes. Then M lines follow, each in the format:

ID K ID[1] ID[2] ... ID[K]
 

where ID is a two-digit number representing a given non-leaf node, K is the number of its children, followed by a sequence of two-digit ID's of its children. For the sake of simplicity, let us fix the root ID to be 01.

The input ends with N being 0. That case must NOT be processed.

Output Specification:

For each test case, you are supposed to count those family members who have no child for every seniority level starting from the root. The numbers must be printed in a line, separated by a space, and there must be no extra space at the end of each line.

The sample case represents a tree with only 2 nodes, where 01 is the root and 02 is its only child. Hence on the root 01 level, there is 0 leaf node; and on the next level, there is 1 leaf node. Then we should output 0 1 in a line.

Sample Input:

2 1
01 1 02
 

Sample Output:

0 1

題目分析

    分析家族族譜,算出非葉子節點,就是非孩子數量。,所有的節點用兩位數字表示,例如:01,02,06等,其中為了方便,令01為根節點。

    輸入: N(100個以內的的節點) M(非葉子節點數)

       【接下的M行內】 ID(非葉子節點)K(所連的葉子 / 孩子節點個數)ID[0] ID[1]...ID[K](K個葉子節點)

    輸出:(每層的葉子節點個數)中間用空格隔開————>註意,pat對輸出有著固定且嚴格的格式,所以最後結果的第一個數字前是沒有空格的。

    就是用廣度優先或深度優先遍歷每一層,同時標記每層葉子節點並記錄,最後數組輸出

個人想法

    其實一開始,我覺得很簡單,既然只看葉子節點個數,那每次輸入ID[0-K]時,我進行記錄不就行了嗎,每次輸入ID都是不同的層級,最後得到的的不就是每層孩子數,所以我在寫輸入的時候加了一點點變數,最後輸出。

 1 #include<stdio.h>
 2 #include<stdlib.h>
 3 #include<string.h>
 4 
 5 int ID[101];
 6 int book[101];//記錄輸出每層葉節點個數    
 7 int b;        //每層的葉節點數
 8 
 9 int main() {
10     int N, M;
11     scanf("%d%d", &N, &M);
12     int K;
13     int a = 1;
14     for (int i = 1; i <= N; i+=K+1) {
15         scanf("%d%d", &ID[i], &K);     
16         if( K != 0)
17         for (int j = i+1; j <= i+K; j++) {
18             scanf("%d", &ID[j]);
19             b++;    //沒輸入一次葉節點數就增加一個
20         }
21         book[a] = b;//記錄
22         b = 0;//歸零進行下次計算    
23         a++;//記錄數組下標移動
24     }
25     printf("%d", book[0]);//輸出第一層
26     for (int i = 1; i <= M; i++) {
27         printf(" %d", book[i]);//輸出接下來的每層,並於上一個數字隔開
28 
29     }
30     return 0;
31 }            
View Code

    最後得分13分。。。意料之中,舉例思考的時候都只考慮了二叉樹,編寫的過程中便覺的那如果一個節點有有多個非葉節點甚至沒有葉節點應該怎麼填寫。回過頭來再看這段,更是覺得搞笑,如果這樣可行,那直接記錄K就好了。於是嘗試著寫了第二種,先構建一顆完整的樹,然後深度遍歷。

    首先,變數的設置

1 int N, M;//同題目N,M
2 int K;
3 typedef struct NODE{
4     int nodes;//記錄每個節點擁有的葉子節點數
5     int ID[101];    //記錄各個葉子節點
6 };
7 struct NODE child[101];//
8 int book[101];//保存、輸出每層葉子節點數
9 int max;    //比較和更新每層的葉子節點數,因為左邊的遍歷完,右邊的還要遍歷並記錄

    這裡使用了結構體而不是簡單的數組或者二維數組記錄,是因為退出遞歸的條件需要知道此時節點後續有無別的節點。對比之前的深度遍歷我們可以知道,在每次遞歸的結束條件語句中,會判斷後續“無路可走”後才停止並return,1003中是

 if (curlen > mindis[curcity])return;   

  ,即如果所求的路徑走這時變長就停止這條路。在此題則是你所談的節點後面無節點就說明是葉子節點,那麼就停止遍歷,並記錄數量。但是我開始在這使用二維數組,判定條件:

if(child[curnode]==0) return;

認為全局變數初始化為0,而這樣判斷說明此行均為0時就說明它是空的是葉子節點,但是實際運行出現了死迴圈,因為你在輸入的for語句中都會因為預設給每一行賦值,導致你輸入的值令每行都不是全為0(有點混亂)。而c++中用 child.[curnode]size()【child是vector型】判斷長度,如果等於0 ,則進入葉子節點的計數。所以對c語言使用結構體記錄每個節點的葉子節點數和葉子節點編號。

 1 void dfs(int curnode ,int curlevel);
 2 int main() {
 3     int a;
 4     scanf("%d%d", &N, &M);
 5     for (int i = 0; i < M; i++) {
 6         scanf("%d%d", &a, &K);
 7         child[a].node = K;//非葉節點ID,所有的葉子節點數
 8         for (int j = 0; j <K; j++) {
 9             scanf("%d", &child[a].ID[j]);  //葉子節點序號
10         }
11     }
12     dfs(1, 0);//從 1 開始是因為01為根節點,所有存放元素的是child[01]不是child[0]
13     printf("%d", book[0]);
14     for (int i = 1; i <= max; i++) {
15         printf(" %d", book[i]);
16 
17     }
18     return 0;
19 }
20 void dfs(int curnode, int curlevel) {
21     if (child[curnode].node == 0) {   //判斷有無葉節點,無則進入此記錄保存
22         if(max<curlevel)
23         max = curlevel;        //更新當前遍歷所在的葉子節點層數
24         book[curlevel]++;        //沒到該層,葉子節點數加 1 
25         return;
26     }
27     for (int i = 0; i < child[curnode].node; i++) {    //因為對該節點進行遍歷,所以在它有的葉子節點數範圍內迴圈
28         dfs(child[curnode].ID[i], curlevel + 1);    //把當前的某個葉子節點的數值帶入下一個節點的索引,層級加1.
29     }
30  }

 


<-----------------------------------完整代碼----------------------------------->

 1 #define _CRT_SECURE_NO_WARNINGS
 2 #include<stdio.h>
 3 #include<stdlib.h>
 4 #include<string.h>
 5 
 6 
 7 int N, M;
 8 int K;
 9 typedef struct NODE{
10     int node;
11     int ID[101];
12 };
13 struct NODE child[101];
14 int book[101];
15 int max;
16 
17 void dfs(int curnode ,int curlevel);
18 int main() {
19     int a;
20     scanf("%d%d", &N, &M);
21     for (int i = 0; i < M; i++) {
22         scanf("%d%d", &a, &K);
23         child[a].node = K;
24         for (int j = 0; j <K; j++) {
25             scanf("%d", &child[a].ID[j]);
26         }
27     }
28     dfs(1, 0);
29     printf("%d", book[0]);
30     for (int i = 1; i <= max; i++) {
31         printf(" %d", book[i]);
32 
33     }
34     return 0;
35 }
36 void dfs(int curnode, int curlevel) {
37     if (child[curnode].node == 0) {
38         if(max<curlevel)
39         max = curlevel;
40         book[curlevel]++;
41         return;
42     }
43     for (int i = 0; i < child[curnode].node; i++) {
44         dfs(child[curnode].ID[i], curlevel + 1);
45     }
46  }
View Code

 

總結

    其實掌握深度遍歷總體大概的流程,結合題目改變判定條件寫出總體的框架基本都能得分,然後調試以後進行修改就大差不差了。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 簡介 迭代器模式(Iterator Pattern),是一種結構型設計模式。給數據對象構建一套按順序訪問集合對象元素的方式,而不需要知道數據對象的底層表示。 迭代器模式是與集合共存的,我們只要實現一個集合,就需要同時提供這個集合的迭代器,就像Java中的Collection,List、Set、Map ...
  • 今天學習C語言學習了三個部分: 第一個部分是軟體環境的搭建,如何搭建一個項目 使用工具:visual studio 2010 搭建過程:新建項目、配置設置(主要是解決運行後一閃而過的問題) 第二部分是編寫一個簡單的C語言程式代碼 #include<stdio.h> //引入頭文件 io指的是輸入與輸 ...
  • 前言 在SpringBoot框架中,我們使用最多的是Tomcat,這是SpringBoot預設的容器技術,而且是內嵌式的Tomcat。同時,SpringBoot也支持Undertow容器,我們可以很方便的用Undertow替換Tomcat,而Undertow的性能和記憶體使用方面都優於Tomcat,那 ...
  • package.json 備忘清單 如果你以前用過 Node.js,則可能會遇到 package.json 文件。它是一個 JSON 文件,位於項目的根目錄中。你的 package.json 包含關於項目的重要信息。它包含關於項目的使人類可讀元數據(如項目名稱和說明)以及功能元數據(如程式包版本號和 ...
  • # 請先使用命令 pip install sxtwl 安裝依賴庫後,再執行以下腳本 import sxtwl ymc = ["正", "二", "三", "四", "五", "六", "七", "八", "九", "十" ,"冬", "臘"] rmc = ["初一", "初二", "初三", &qu ...
  • 源代碼下載鏈接: 一、賓館客房管理系統開發初衷 隨著互聯網技術的迅速發展,電腦技術的普及以及信息化時代的推波助瀾,賓館客房需求的逐漸增大,這也是挑戰了賓館客房管理方面的技術,以前的人工管理方式已經不再適應現在的環境,取而代之的是先進的賓館客房管理系統,提高了賓館的工作效率,為想要入住賓館的人提供更 ...
  • 代碼如下: import com.google.zxing.BarcodeFormat; import com.google.zxing.EncodeHintType; import com.google.zxing.MultiFormatWriter; import com.google.zxin ...
  • 併發工具類 通常我們所說的併發包也就是java.util.concurrent(JUC),集中了Java併發的各種工具類, 合理地使用它們能幫忙我們快速地完成功能 。 作者: 博學谷狂野架構師 GitHub:GitHub地址 (有我精心準備的130本電子書PDF) 只分享乾貨、不吹水,讓我們一起加油 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...