網(wǎng)站首頁 編程語言 正文
一、練習(xí)題目
題目鏈接 | 難度 |
---|---|
1472. 設(shè)計(jì)瀏覽器歷史記錄 | ★★★☆☆ |
430. 扁平化多級(jí)雙向鏈表 | ★★★☆☆ |
劍指 Offer II 028. 展平多級(jí)雙向鏈表 | ★★★☆☆ |
劍指 Offer 36. 二叉搜索樹與雙向鏈表 | ★★★★☆ |
二、算法思路
1、設(shè)計(jì)瀏覽器歷史記錄
1.這是一個(gè)模擬題;
2.初始化生成一個(gè)頭結(jié)點(diǎn),記錄一個(gè)當(dāng)前結(jié)點(diǎn);
3.向前 和 向后 是兩個(gè)類似的過程,可以統(tǒng)一實(shí)現(xiàn),注意一些邊界條件。
struct Node {
string val;
Node* prev;
Node* next;
};
class BrowserHistory {
Node * List, *Current;
public:
BrowserHistory(string homepage) {
List = new Node();
List->prev = List->next = nullptr;
List->val = homepage;
Current = List;
}
void visit(string url) {
Node *Next = Current->next;
if(Next == nullptr) {
Current->next = new Node();
Current->next->next = nullptr;
Current->next->prev = Current;
}else {
Node *tmp = Next->next;
Next->next = nullptr;
// free
while(tmp) {
Node *node = tmp->next;
delete tmp;
tmp = node;
}
}
Current->next->val = url;
Current = Current->next;
}
string back(int steps) {
string str = Current->val;
Node *pre;
while(steps-- && Current) {
pre = Current;
Current = Current->prev;
if(Current) str = Current->val;
}
if(nullptr == Current) Current = pre;
return str;
}
string forward(int steps) {
string str = Current->val;
Node *pre;
while(steps-- && Current) {
pre = Current;
Current = Current->next;
if(Current) str = Current->val;
}
if(nullptr == Current) Current = pre;
return str;
}
};
2、扁平化多級(jí)雙向鏈表
1.利用一個(gè)遞歸函數(shù)last = dfs(now),一旦遇到child域非空的結(jié)點(diǎn),則遞歸計(jì)算clast = dfs(now->child),返回值是遞歸展平后的最后一個(gè)結(jié)點(diǎn),然后進(jìn)行雙向鏈表的鏈接操作。
2.例如,當(dāng)前有 child域的結(jié)點(diǎn)為now,它的下一個(gè)結(jié)點(diǎn)是next,遞歸計(jì)算以后得到展平的鏈表的最后一個(gè)結(jié)點(diǎn)為 clast,則有如下關(guān)系:
now <---> now->child ... clast <---> next
3.根據(jù)以上關(guān)系調(diào)整雙向鏈表,注意不要忘記將child域置空。
4.當(dāng)遍歷到這個(gè)雙向鏈表的最后一個(gè)結(jié)點(diǎn)的時(shí)候,如果它有child域,則當(dāng)前鏈表的最后一個(gè)結(jié)點(diǎn)就是clast,否則就是它自己now;
class Solution {
Node* dfs(Node* head) {
Node *now = head;
Node *last = nullptr;
while(now) {
Node *cLast;
if(now->child) {
cLast = dfs(now->child);
Node *next = now->next;
// now <--> cFirst ... cLast <---> next;
now->next = now->child;
now->child = nullptr;
now->next->prev = now;
if(next) {
next->prev = cLast;
}
cLast->next = next;
}
if(now->next == nullptr) {
if(now->child) {
last = cLast;
}else {
last = now;
}
}
now = now->next;
}
return last;
}
public:
Node* flatten(Node* head) {
if(head == nullptr) {
return nullptr;
}
Node *last = dfs(head);
last->next = nullptr;
return head;
}
};
3、展平多級(jí)雙向鏈表
(1)同上一題。
4、二叉搜索樹與雙向鏈表
(1)遇到這樣的題,首先需要設(shè)計(jì)好遞歸函數(shù);
(2)像這個(gè)問題,對(duì)于 左子樹 和 右子樹,需要知道雙向鏈表的 頭結(jié)點(diǎn) 和 尾結(jié)點(diǎn),所以遞歸的時(shí)候需要返回 兩個(gè)值,于是可以直接采用函數(shù)傳指針進(jìn)行返回,由于二叉樹的結(jié)點(diǎn)本身就是指針,所以需要傳 二級(jí)指針;
(3)遞歸計(jì)算左子樹變成雙向鏈表的情況;
(4)遞歸計(jì)算右子樹變成雙向鏈表的情況;
(5)將左子樹的雙向鏈表鏈接到root左邊,將右子樹的雙向鏈表鏈接到root右邊,然后根據(jù)遞歸函數(shù)的實(shí)際作用,返回 頭結(jié)點(diǎn) 和 尾結(jié)點(diǎn)。
class Solution {
void dfs(Node *root, Node **minNode, Node **maxNode) {
if(root == nullptr) {
*minNode = nullptr;
*maxNode = nullptr;
return ;
}
Node *lminNode, *lmaxNode, *rminNode, *rmaxNode;
if(root->left) {
dfs(root->left, &lminNode, &lmaxNode);
lmaxNode->right = root;
root->left = lmaxNode;
*minNode = lminNode;
}else {
*minNode = root;
}
if(root->right) {
dfs(root->right, &rminNode, &rmaxNode);
rminNode->left = root;
root->right = rminNode;
*maxNode = rmaxNode;
}else {
*maxNode = root;
}
}
public:
Node* treeToDoublyList(Node* root) {
if(root == nullptr) {
return nullptr;
}
Node *minNode, *maxNode;
dfs(root, &minNode, &maxNode);
maxNode->right = minNode;
minNode->left = maxNode;
return minNode;
}
};
原文鏈接:https://blog.csdn.net/WhereIsHeroFrom/article/details/124744280
相關(guān)推薦
- 2022-05-21 ?python?中的條件判斷語句的使用介紹_python
- 2022-04-10 為Xamarin.Forms的導(dǎo)航欄增加搜索功能_C#教程
- 2022-10-17 Kotlin編程循環(huán)控制示例詳解_Android
- 2023-03-20 C#中程序自刪除實(shí)現(xiàn)方法_C#教程
- 2022-01-17 git git版本回退 回滾 解決方案
- 2024-01-27 DO、DTO、BO、VO、POJO區(qū)別
- 2022-05-27 C++左值與右值,右值引用,移動(dòng)語義與完美轉(zhuǎn)發(fā)詳解_C 語言
- 2022-05-27 python繪制棉棒圖的方法詳解_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支