貪心算法Day6學習心得

第一道:738. 單調遞增的數字 - 力扣(LeetCode)

這道題目暴力算法肯定是最容易想到的,先附上暴力的代碼:

class Solution {
private:// 判斷一個數字的各位上是否是遞增bool checkNum(int num) {int max = 10;while (num) {int t = num % 10;if (max >= t) max = t;else return false;num = num / 10;}return true;}
public:int monotoneIncreasingDigits(int N) {for (int i = N; i > 0; i--) { // 從大到小遍歷if (checkNum(i)) return i;}return 0;}
};

當然,還是要用貪心的思路來想一下。

題目要求小于等于N的最大單調遞增的整數,那么拿一個兩位的數字來舉例。

例如:98,一旦出現strNum[i - 1] > strNum[i]的情況(非單調遞增),首先想讓strNum[i - 1]--,然后strNum[i]給為9,這樣這個整數就是89,即小于98的最大的單調遞增整數。

這一點如果想清楚了,這道題就好辦了。

此時是從前向后遍歷還是從后向前遍歷呢?

從前向后遍歷的話,遇到strNum[i - 1] > strNum[i]的情況,讓strNum[i - 1]減一,但此時如果strNum[i - 1]減一了,可能又小于strNum[i - 2]。

這么說有點抽象,舉個例子,數字:332,從前向后遍歷的話,那么就把變成了329,此時2又小于了第一位的3了,真正的結果應該是299。

那么從后向前遍歷,就可以重復利用上次比較得出的結果了,從后向前遍歷332的數值變化為:332 -> 329 -> 299

確定了遍歷順序之后,那么此時局部最優就可以推出全局,找不出反例,試試貪心。

C++代碼如下:

class Solution {
public:int monotoneIncreasingDigits(int N) {string strNum = to_string(N);// flag用來標記賦值9從哪里開始// 設置為這個默認值,為了防止第二個for循環在flag沒有被賦值的情況下執行int flag = strNum.size();for (int i = strNum.size() - 1; i > 0; i--) {if (strNum[i - 1] > strNum[i] ) {flag = i;strNum[i - 1]--;}}for (int i = flag; i < strNum.size(); i++) {strNum[i] = '9';}return stoi(strNum);}
};

然后看第二道,也是貪心的最后一道:968. 監控二叉樹 - 力扣(LeetCode)

這道題也是挺有意思的,首先要想,如何放置,才能讓攝像頭最小的呢?

從題目中示例,其實可以得到啟發,發現題目示例中的攝像頭都沒有放在葉子節點上!

這是很重要的一個線索,攝像頭可以覆蓋上中下三層,如果把攝像頭放在葉子節點上,就浪費的一層的覆蓋。

所以把攝像頭放在葉子節點的父節點位置,才能充分利用攝像頭的覆蓋面積。

那么為什么不從頭結點開始看起呢,為啥要從葉子節點看呢?

因為頭結點放不放攝像頭也就省下一個攝像頭, 葉子節點放不放攝像頭省下了的攝像頭數量是指數階別的。

所以我們要從下往上看,局部最優:讓葉子節點的父節點安攝像頭,所用攝像頭最少,整體最優:全部攝像頭數量所用最少!

局部最優推出全局最優,找不出反例,那么就按照貪心來!

此時,大體思路就是從低到上,先給葉子節點父節點放個攝像頭,然后隔兩個節點放一個攝像頭,直至到二叉樹頭結點。

此時這道題目還有兩個難點:

  1. 二叉樹的遍歷
  2. 如何隔兩個節點放一個攝像頭

#確定遍歷順序

在二叉樹中如何從低向上推導呢?

可以使用后序遍歷也就是左右中的順序,這樣就可以在回溯的過程中從下到上進行推導了。

后序遍歷代碼如下:

int traversal(TreeNode* cur) {// 空節點,該節點有覆蓋if (終止條件) return ;int left = traversal(cur->left);    // 左int right = traversal(cur->right);  // 右邏輯處理                            // 中return ;
}

注意在以上代碼中取了左孩子的返回值,右孩子的返回值,即left 和 right, 以后推導中間節點的狀態

#如何隔兩個節點放一個攝像頭

此時需要狀態轉移的公式,大家不要和動態的狀態轉移公式混到一起,本題狀態轉移沒有擇優的過程,就是單純的狀態轉移!

來看看這個狀態應該如何轉移,先來看看每個節點可能有幾種狀態:

有如下三種:

  • 該節點無覆蓋
  • 本節點有攝像頭
  • 本節點有覆蓋

分別有三個數字來表示:

  • 0:該節點無覆蓋
  • 1:本節點有攝像頭
  • 2:本節點有覆蓋

可能會想有沒有第四種狀態:本節點無攝像頭,其實無攝像頭就是 無覆蓋 或者 有覆蓋的狀態,所以一共還是三個狀態。

因為在遍歷樹的過程中,就會遇到空節點,那么問題來了,空節點究竟是哪一種狀態呢? 空節點表示無覆蓋? 表示有攝像頭?還是有覆蓋呢?

回歸本質,為了讓攝像頭數量最少,我們要盡量讓葉子節點的父節點安裝攝像頭,這樣才能攝像頭的數量最少。

那么空節點不能是無覆蓋的狀態,這樣葉子節點就要放攝像頭了,空節點也不能是有攝像頭的狀態,這樣葉子節點的父節點就沒有必要放攝像頭了,而是可以把攝像頭放在葉子節點的爺爺節點上。

所以空節點的狀態只能是有覆蓋,這樣就可以在葉子節點的父節點放攝像頭了

接下來就是遞推關系。

那么遞歸的終止條件應該是遇到了空節點,此時應該返回2(有覆蓋),原因上面已經解釋過了。

代碼如下:

// 空節點,該節點有覆蓋
if (cur == NULL) return 2;

遞歸的函數,以及終止條件已經確定了,再來看單層邏輯處理。

主要有如下四類情況:

  • 情況1:左右節點都有覆蓋

左孩子有覆蓋,右孩子有覆蓋,那么此時中間節點應該就是無覆蓋的狀態了。

如圖:

968.監控二叉樹2

代碼如下:

// 左右節點都有覆蓋
if (left == 2 && right == 2) return 0;
  • 情況2:左右節點至少有一個無覆蓋的情況

如果是以下情況,則中間節點(父節點)應該放攝像頭:

  • left == 0 && right == 0 左右節點無覆蓋
  • left == 1 && right == 0 左節點有攝像頭,右節點無覆蓋
  • left == 0 && right == 1 左節點有無覆蓋,右節點攝像頭
  • left == 0 && right == 2 左節點無覆蓋,右節點覆蓋
  • left == 2 && right == 0 左節點覆蓋,右節點無覆蓋

這個不難理解,畢竟有一個孩子沒有覆蓋,父節點就應該放攝像頭。

此時攝像頭的數量要加一,并且return 1,代表中間節點放攝像頭。

代碼如下:

if (left == 0 || right == 0) {result++;return 1;
}
  • 情況3:左右節點至少有一個有攝像頭

如果是以下情況,其實就是 左右孩子節點有一個有攝像頭了,那么其父節點就應該是2(覆蓋的狀態)

  • left == 1 && right == 2 左節點有攝像頭,右節點有覆蓋
  • left == 2 && right == 1 左節點有覆蓋,右節點有攝像頭
  • left == 1 && right == 1 左右節點都有攝像頭

代碼如下:

if (left == 1 || right == 1) return 2;

從這個代碼中,可以看出,如果left == 1, right == 0 怎么辦?其實這種條件在情況2中已經判斷過了,如圖:

968.監控二叉樹1

  1. 情況4:頭結點沒有覆蓋

以上都處理完了,遞歸結束之后,可能頭結點 還有一個無覆蓋的情況,如圖:

968.監控二叉樹3

所以遞歸結束之后,還要判斷根節點,如果沒有覆蓋,result++,代碼如下:

int minCameraCover(TreeNode* root) {result = 0;if (traversal(root) == 0) { // root 無覆蓋result++;}return result;
}

C++代碼如下:

// 版本一
class Solution {
private:int result;int traversal(TreeNode* cur) {// 空節點,該節點有覆蓋if (cur == NULL) return 2;int left = traversal(cur->left);    // 左int right = traversal(cur->right);  // 右// 情況1// 左右節點都有覆蓋if (left == 2 && right == 2) return 0;// 情況2// left == 0 && right == 0 左右節點無覆蓋// left == 1 && right == 0 左節點有攝像頭,右節點無覆蓋// left == 0 && right == 1 左節點有無覆蓋,右節點攝像頭// left == 0 && right == 2 左節點無覆蓋,右節點覆蓋// left == 2 && right == 0 左節點覆蓋,右節點無覆蓋if (left == 0 || right == 0) {result++;return 1;}// 情況3// left == 1 && right == 2 左節點有攝像頭,右節點有覆蓋// left == 2 && right == 1 左節點有覆蓋,右節點有攝像頭// left == 1 && right == 1 左右節點都有攝像頭// 其他情況前段代碼均已覆蓋if (left == 1 || right == 1) return 2;// 以上代碼我沒有使用else,主要是為了把各個分支條件展現出來,這樣代碼有助于讀者理解// 這個 return -1 邏輯不會走到這里。return -1;}public:int minCameraCover(TreeNode* root) {result = 0;// 情況4if (traversal(root) == 0) { // root 無覆蓋result++;}return result;}
};

在以上代碼的基礎上,再進行精簡,代碼如下:

// 版本二
class Solution {
private:int result;int traversal(TreeNode* cur) {if (cur == NULL) return 2;int left = traversal(cur->left);    // 左int right = traversal(cur->right);  // 右if (left == 2 && right == 2) return 0;else if (left == 0 || right == 0) {result++;return 1;} else return 2;}
public:int minCameraCover(TreeNode* root) {result = 0;if (traversal(root) == 0) { // root 無覆蓋result++;}return result;}
};

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/916004.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/916004.shtml
英文地址,請注明出處:http://en.pswp.cn/news/916004.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

數據的評估與清洗篇---上手清理索引和列名

重命名索引和列名 在讀取數據時,如果我們發現數據的索引或者列名亂七八糟的,可以使用DataFrame的rename方法對它們進行重新命名。 df1.rename(index={...})df1.rename(columns={...}) 重命名索引 如果想改索引就把可選參數index指定為一個字典,針對索引,把要修改…

【ICML2025】時間序列|TimePro:炸裂!線性復雜度實現高效長程多元時間序列預測!

論文地址&#xff1a;https://arxiv.org/pdf/2505.20774 代碼地址&#xff1a;https://github.com/xwmaxwma/TimePro 為了更好地理解時間序列模型的理論與實現&#xff0c;推薦參考UP “ThePPP時間序列” 的教學視頻。該系列內容系統介紹了時間序列相關知識&#xff0c;并提供配…

2025真實面試試題分析-iOS客戶端開發

以下是對iOS客戶端開發工程師面試問題的分類整理、領域占比分析及高頻問題精選&#xff08;基于??85道問題&#xff0c;總出現次數118次??&#xff09;。按技術領域整合為??7大核心類別??&#xff0c;按占比排序并精選高頻問題標注優先級&#xff08;1-5&#x1f31f;&…

計算機網絡簡答題(大雪圣期末參考資料)

1、網絡性能指標/計算機網絡有哪些常用的性能指標&#xff1f;答&#xff1a;速率&#xff0c;帶寬&#xff0c;吞吐量&#xff0c;時延&#xff08;發送時延、傳播時延、處理時延、排隊時延&#xff09;&#xff0c;時延帶寬積&#xff0c;往返時間RTT和信道&#xff08;或網絡…

紅寶書單詞學習筆記 list 76-100

list 76NO.WordMeaning1staleadj. 不新鮮的&#xff1b;陳腐的2stalln. 小隔間&#xff1b;攤位&#xff1b;牲畜棚&#xff1b;v. 停頓&#xff1b;(使) 熄火&#xff1b;故意拖延3staplen. 訂書釘&#xff1b;主要產品&#xff1b;主要部分&#xff1b;主食&#xff1b;v. 用…

Vue3 學習教程,從入門到精通,Vue 3 計算屬性(Computed Properties)知識點詳解與案例代碼(15)

Vue 3 計算屬性&#xff08;Computed Properties&#xff09;知識點詳解與案例代碼 在 Vue 3 中&#xff0c;計算屬性&#xff08;Computed Properties&#xff09; 是用于基于響應式數據派生新數據的一種方式。計算屬性具有以下特點&#xff1a; 緩存性&#xff1a;只有在依賴…

2.5 PN-PTCP

Pro?net Precision Transparent Clock Protocol (PN-PTCP) PN-PTCP&#xff08;精確透明時鐘協議&#xff09;是一種專用于 Profinet 的 二層協議&#xff0c;其作用是為網絡中的設備提供高精度的時間同步。用于實現網絡設備的高精度時間同步。

WordPress與Typecho站點CloudFlare緩存優化實戰指南

文章目錄 WordPress與Typecho站點CloudFlare緩存加速全攻略 引言 一、CloudFlare緩存基礎原理 1.1 CloudFlare工作流程 1.2 緩存類型 二、基礎配置指南 2.1 CloudFlare賬戶設置 2.2 緩存配置 2.3 頁面規則設置 三、高級緩存策略 3.1 動態內容緩存 WordPress方案: Typecho方案:…

【OpenCV實現多圖像拼接】

文章目錄1 OpenCV 圖像拼接核心原理2 OpenCV 圖像拼接實現代碼方法一&#xff1a;使用 OpenCV 內置 Stitcher 類&#xff08;推薦&#xff09;方法二&#xff1a;手動實現核心步驟關鍵參數說明3 常見問題處理4 增量式圖像拼接&#xff08;Incremental Image Stitching&#xff…

haproxy 算法

一、靜態算法按照事先定義好的規則輪詢公平調度&#xff0c;不關心后端服務器的當前負載、連接數和響應速度 等&#xff0c;且無法實時修改權重(只能為0和1,不支持其它值)&#xff0c;只能靠重啟HAProxy生效。(不管后端死活&#xff09;1.1、static-rr&#xff1a;基于權重的輪…

Go 的第一類對象與閉包

1. Go 的第一類對象&#xff08;First-Class Citizens&#xff09; 什么是第一類對象&#xff1f; 第一類對象是指能夠像 普通值 一樣使用的對象&#xff0c;通常可以賦值給變量、傳遞給函數、作為函數返回值等。在很多編程語言中&#xff0c;函數本身不被視為第一類對象&#…

深度分析Android多線程編程

理解并正確運用多線程是構建高性能、流暢、響應迅速的 Android 應用的關鍵&#xff0c;但也充滿挑戰和陷阱。 核心挑戰&#xff1a;UI 線程&#xff08;主線程&#xff09;的限制 唯一性&#xff1a; Android 應用只有一個主線程&#xff0c;負責處理所有用戶交互&#xff08;觸…

uniapp在app中關于解決輸入框鍵盤彈出后遮住輸入框問題

問題描述&#xff1a; uniapp的app中&#xff0c;當表單頁面過長時&#xff0c;點擊下方的輸入框時&#xff0c;彈出鍵盤后會把輸入框給擋住&#xff0c;導致看不到輸入內容。 解決方案&#xff1a; 在page.json中&#xff0c;找到此頁面的配置&#xff0c;加上style中的softin…

二分查找----5.尋找旋轉排序數組中的最小值

題目鏈接 /** 數組在某處進行旋轉,分割為兩個獨立的遞增區間,找出數組的最小值;特殊情況:若旋轉次數是數組長度的倍數,則數組不變 特點: 常規情況: 數組被分割為兩個獨立的子區間,左半區的最小值大于右半區的最大值 依據數組長度,mid可能落在左半區也有可能落在右半區,最小值在…

Eureka-服務注冊,服務發現

在遠程調用的時候&#xff0c;我們寫的url是寫死的。 String url "<http://127.0.0.1:9090/product/>" orderInfo.getProductId();當換個機器&#xff0c;或者新增個機器&#xff0c;導致ip變換&#xff0c;從而使得 url 發生了變化&#xff0c;接著就需要去…

ubuntu24的一些小問題

截圖Keyboard -> Keyboard Shortcus -> View and customize Shortcus如上&#xff0c;可以修改默認的快捷按鍵。比如截圖按鍵可以修改。 ibus輸入法無法&#xff0c;輸入V異常問題 也是困擾了很久&#xff0c;發現是這樣的&#xff1a;https://github.com/libpinyin/ibus…

Python Locust庫詳解:從入門到分布式壓力測試實戰

一、Locust核心優勢 作為一款基于Python的開源負載測試工具&#xff0c;Locust通過協程架構實現了高效資源利用。其獨特優勢體現在&#xff1a; 純Python腳本&#xff1a;用熟悉的語言定義用戶行為&#xff0c;支持條件判斷和復雜邏輯分布式擴展&#xff1a;單節點支持數千并發…

Redis數據類型與內部編碼

在Redis中通常普遍認為&#xff0c;使用redis的能進行查詢&#xff0c;插入&#xff0c;刪除&#xff0c;修改操作都是O(1)是因為他是利用hash表實現的&#xff0c;但是&#xff0c;背后的實現不一定是一個標準的hash表&#xff0c;它內部的數據類型還會有變數&#xff0c;不過…

03-netty基礎-多路復用select、poll、epoll

1 什么是多路復用多路復用&#xff08;Multiplexing&#xff09; 是一種讓單個線程同時處理多個 I/O 通道的技術&#xff0c;核心是通過系統調用將 I/O 狀態查詢的工作交給操作系統內核&#xff0c;應用程序只需等待內核通知哪些通道就緒。多路&#xff1a;指的是多個socket網絡…

網易大模型算法面經總結第一篇

網友一 MHA的原理&#xff0c;是如何進行加速的&#xff0c;用的什么框架推理。 回答&#xff1a; ①先答一下什么是MHA&#xff1a;Multi-Head Attention&#xff08;MHA&#xff09;是 Transformer 的核心機制&#xff0c;并行地關注輸入序列中不同位置的多種信息 ②回答MHA的…