题解 | #在二叉树中找到两个节点的最近公共祖先#

在二叉树中找到两个节点的最近公共祖先

http://www.nowcoder.com/practice/e0cc33a83afe4530bcec46eba3325116

1 思路

  • 基础 不是搜索树,是普通树
  • 公共祖先 和 相交的链表差异,指针没有next,只有root
  • 找到bfs (queue)优化的痕迹

更新 不是 dfs,二十bfs,活用队列存树结构,用KV之map存关系;

1.1 实现错误思路

    int lowestCommonAncestor(TreeNode* root, int o1, int o2) {
        // write code here
        vector<int> o1V;
        vector<int> o2V;
        
        queue<TreeNode*> q;
         
        //题目肯定又1个
        q.push(root);
        //std::find(v.begin(), v.end(), key) != v.end()
        while(std::find(o1V.begin(), o1V.end(), o1) != o1V.end() &&
             std::find(o2V.begin(), o2V.end(), o2) != o2V.end()){
            
            TreeNode* top = q.front();
            //我怎么知道是更新哪个链表拉? 只有局部关系,所有map两边共享
            q.pop();
            
            
        }

alt

1.2 容易出错的位置

alt

1.3 数据结构容易出错的地方

map哪儿插入,哪儿读取(count) vector如何查找

alt

2 code

非递归 bfs变形


/**
 * struct TreeNode {
 *	int val;
 *	struct TreeNode *left;
 *	struct TreeNode *right;
 * };
 */
#include <map>
class Solution {
public:
    /**
     * 
     * @param root TreeNode类 
     * @param o1 int整型 
     * @param o2 int整型 
     * @return int整型
     */
    int lowestCommonAncestor(TreeNode* root, int o1, int o2) {
        // write code here
//         vector<int> o1V;
//         vector<int> o2V;
        //hashmap
        map<int,int> parentMap;
        queue<TreeNode*> q;
         
        //题目肯定又1个
        q.push(root);
        //parentMap.put(root,-1);//确认没有负数吧 java style, 注意键值
        parentMap[root->val]=-1;
        //java style containKeys ,c++ count
        while(parentMap.count(o1) ==0  || parentMap.count(o2) ==0  ) //true true ;  false false
        {
            TreeNode * top = q.front();  q.pop();
            if(top->left){
                q.push(top->left);
                parentMap[top->left->val]= top->val;
            }
            if(top->right){
                q.push(top->right);
                parentMap[top->right->val]=top->val;
            }
        }
        //get o1的父亲链条
        vector<int> vec4a;//ArrayList<int> list4node1;
        int curr = o1;
        do{
            //vec4a.push_back(parentMap[curr]);
            vec4a.push_back(curr);
            curr = parentMap[curr];
        //}while(parentMap[curr]!= -1);
        }while(curr!=-1);
        int index = o2;
        while(std::find(vec4a.begin(), vec4a.end(), index) == vec4a.end() ){//        //stl style
            index = parentMap[index];//find parent ,until find first in vector;
        }
        return index;
        
    }
};
//参考
/**
 * struct TreeNode {
 *	int val;
 *	struct TreeNode *left;
 *	struct TreeNode *right;
 * };
 */
#include <map>
class Solution {
public:
    /**
     * 
     * @param root TreeNode类 
     * @param o1 int整型 
     * @param o2 int整型 
     * @return int整型
     */
    int lowestCommonAncestor(TreeNode* root, int o1, int o2) {
        // write code here
//         vector<int> o1V;
//         vector<int> o2V;
        //hashmap
        map<int,int> parentMap;
        queue<TreeNode*> q;
         
        //题目肯定又1个
        q.push(root);
        //parentMap.put(root,-1);//确认没有负数吧 java style, 注意键值
        parentMap[root->val]=-1;
        
        //stl style
        //std::find(v.begin(), v.end(), key) != v.end()
        //java style containKeys ,c++ count
        while(parentMap.count(o1) ==0  || parentMap.count(o2) ==0  ) //true true ;  false false
        {
            TreeNode * top = q.front();
            q.pop();
            
            if(top->left){
                q.push(top->left);
                parentMap[top->left->val]= top->val;
            }
            if(top->right){
                q.push(top->right);
                parentMap[top->right->val]=top->val;
            }
            
        }
        //get o1的父亲链条
        //ArrayList<int> list4node1;
        vector<int> vec4a;
        int curr = o1;
        
        do{
            //vec4a.push_back(parentMap[curr]);
            vec4a.push_back(curr);
            curr = parentMap[curr];
        //}while(parentMap[curr]!= -1);
        }while(curr!=-1);
        
        int index = o2;
        while(std::find(vec4a.begin(), vec4a.end(), index) == vec4a.end() ){
            index = parentMap[index];//find parent ,until find first in vector;
        }
        
        return index;
        
    }
};
todo

3 数据结构使用对比

需求:根据键值key修改对应的实值value。 若键存在,则值加1;若键不存在,则创建该键,并将键对应的值赋为1。


1 Java HashMap修改实值

Java的HashMap使用put()方法插入键值对元素<key, value>时,若键key相同,则会使用新值value覆盖之前的数据。

实现代码: 先判断键是否存在: ①键存在:先使用get()方法获取原值,加1后使用put()方法插入键值对元素,会覆盖旧值; ②键不存在:使用put()方法插入键值对元素。

HashMap<Character, Integer> mp = new HashMap<>();

/* 判断键key是否存在 */
//键存在
if(mp.containsKey(key)){
	mp.put(key, map.get(key) + 1);
//键不存在,直接插入键值对元素
}else{
	mp.put(key, 1);
}

简化写法:JDK 1.8的getOrDefault()方法 mp.put(key, mp.getOrDefault(key, 0) + 1);


2 C++ map修改实值

C++的map使用insert()函数插入键值对元素<key, value>时,若键key相同,则键值对元素插入失败,即不会覆盖之前的数据。

实现代码: 先判断键是否存在: ①键存在:使用迭代器重载运算符[]更新值; ②键不存在:使用insert()函数插入键值对元素(即对组)。

map<char, int> mp;

/* 判断键key是否存在 */
map<char, int>::iterator pos = mp.find(key);
//键存在,根据迭代器或重载运算符[]赋值
if(pos != mp.end()){
	//方法1:使用迭代器更新值
	//pos->second = mp[key] + 1;

	//方法2:使用重载运算符[]更新值
	mp[key] += 1;
//键不存在,使用insert()函数,直接插入键值对元素(对组)
}else{
	mp.insert(make_pair(key, 1));
}

简化写法:直接使用重载运算符[]获取并修改值,键不存在时则默认创建<key, 0>mp[key] += 1;


测试代码

template<typename T1, typename T2>
void printMap(const map<T1, T2>& mp) {	//形参使用const,避免被修改
	//形参使用const后,遍历时需使用只读迭代器const_iterator
	//使用typename关键字,防止编译器报错:C2760(语法错误,意外标记“标识符”)
	for (typename map<T1, T2>::const_iterator it = mp.begin(); it != mp.end(); it++) {
		cout << "key = " << (*it).first << " , value = " << it->second << endl;
	}
	cout << endl;
}

int main() {
	map<char, int> mp;
	mp.insert(make_pair('b', 2));
	printMap<char, int>(mp);
	//key = b , value = 2

	/* 判断键key是否存在 */
	char key = 'a';

	/*
	map<char, int>::iterator pos = mp.find(key);
	//键存在,根据迭代器或重载运算符[]赋值
	if (pos != mp.end()) {
		//方法1:使用迭代器更新值
		pos->second = mp[key] + 1;

		//方法2:使用重载运算符[]更新值
		//mp[key] += 1;
		//键不存在,使用insert()函数,直接插入键值对元素(对组)
	}
	else {
		mp.insert(make_pair(key, 1));
	}
	*/

	//方法3:直接使用重载运算符[]获取并更新值
	mp[key] += 1;

	printMap<char, int>(mp);
	//key = a , value = 1
	//key = b , value = 2
}
全部评论
确定路径后,要么 从顶向下比较路径,如果从底向上比较的话,哪个路径长这个还不知道哈,有额外的开销,所以代码还得升级哈;
点赞 回复 分享
发布于 2022-05-18 16:51

相关推荐

头像
10-13 18:10
已编辑
东南大学 C++
。收拾收拾心情下一家吧————————————————10.12更新上面不知道怎么的,每次在手机上编辑都会只有最后一行才会显示。原本不想写凉经的,太伤感情了,但过了一天想了想,凉经的拿起来好好整理,就像象棋一样,你进步最快的时候不是你赢棋的时候,而是在输棋的时候。那废话不多说,就做个复盘吧。一面:1,经典自我介绍2,项目盘问,没啥好说的,感觉问的不是很多3,八股问的比较奇怪,他会深挖性地问一些,比如,我知道MMU,那你知不知道QMMU(记得是这个,总之就是MMU前面加一个字母)4,知不知道slab内存分配器-&gt;这个我清楚5,知不知道排序算法,排序算法一般怎么用6,写一道力扣的,最长回文子串反问:1,工作内容2,工作强度3,关于友商的问题-&gt;后面这个问题问HR去了,和中兴有关,数通这个行业和友商相关的不要提,这个行业和别的行业不同,别的行业干同一行的都是竞争关系,数通这个行业的不同企业的关系比较微妙。特别细节的问题我确实不知道,但一面没挂我。接下来是我被挂的二面,先说说我挂在哪里,技术性问题我应该没啥问题,主要是一些解决问题思路上的回答,一方面是这方面我准备的不多,另一方面是这个面试写的是“专业面试二面”,但是感觉问的问题都是一些主管面/综合面才会问的问题,就是不问技术问方法论。我以前形成的思维定式就是专业面会就是会,不会就直说不会,但事实上如果问到方法论性质的问题的话得扯一下皮,不能按照上面这个模式。刚到位置上就看到面试官叹了一口气,有一些不详的预感。我是下午1点45左右面的。1,经典自我介绍2,你是怎么完成这个项目的,分成几个步骤。我大致说了一下。你有没有觉得你的步骤里面缺了一些什么,(这里已经在引导我往他想的那个方向走了),比如你一个人的能力永远是不够的,,,我们平时会有一些组内的会议来沟通我们的所思所想。。。。3,你在项目中遇到的最困难的地方在什么方面4,说一下你知道的TCP/IP协议网络模型中的网络层有关的协议......5,接着4问,你觉得现在的socket有什么样的缺点,有什么样的优化方向?6,中间手撕了一道很简单的快慢指针的问题。大概是在链表的倒数第N个位置插入一个节点。————————————————————————————————————10.13晚更新补充一下一面说的一些奇怪的概念:1,提到了RPC2,提到了fu(第四声)拷贝,我当时说我只知道零拷贝,知道mmap,然后他说mmap是其中的一种方式,然后他问我知不知道DPDK,我说不知道,他说这个是一个高性能的拷贝方式3,MMU这个前面加了一个什么字母我这里没记,别问我了4,后面还提到了LTU,VFIO,孩子真的不会。
走呀走:华子二面可能会有场景题的,是有些开放性的问题了
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务