C++ - map 和 set的 例题

 前言

本博客在 一下文章关于 map 和 set 讲解之下,对 map 当中的 operator[] ()函数的功能运用,感受 map 功能强大。

 

349. 两个数组的交集 - 力扣(LeetCode)

给定两个数组 nums1 和 nums2 ,返回 它们的交集 。输出结果中的每个元素一定是 唯一 的。我们可以 不考虑输出结果的顺序 。

示例 1:

输入:nums1 = [1,2,2,1], nums2 = [2,2]
输出:[2]
示例 2:

输入:nums1 = [4,9,5], nums2 = [9,4,9,8,4]
输出:[9,4]
解释:[4,9] 也是可通过的

法一: 

我们可以,两个集合当中的内容都放进 两个 set 当中,第一个是 去重,第二是排序,好查找。然后,去两个set 一个一个遍历比较,有相同的就是交集。

法二:

还有比 法一 更好的方法,开始也都把两个集合放到 两个 set 当中。然后中序遍历保存结果,比如是下面两个结果:

1 3 5 7 9
3 5 6 7 8

 然后,遍历不再想上述一样暴力查找,而是:

都从第一位置开始,值小的一方开始++,直到遍历到相同元素停止,找到交集:

 第一次找到 3  ,停止。然后两个指针同时++,都之下下一个元素位置。

然后重复上述过程就可以找出交集。

当然,还可以用上述方法寻找差集

 代码实现:

class Solution {
public:
    vector<int> intersection(vector<int>& nums1, vector<int>& nums2) {
        set<int> s1(nums1.begin(),nums1.end());
        set<int> s2(nums2.begin(),nums2.end());

        vector<int> v;
        auto it1 = s1.begin();
        auto it2 = s2.begin();
        while(it1 != s1.end() && it2 != s2.end())
        {
            if(*it1 < *it2)
            {
                it1++;
            }
            else if(*it2 < *it1)
            {
                it2++;
            }
            else{
                v.push_back(*it1);
                ++it1;
                ++it2;
            }
        }
        return v;
    }
};

 上述这种方法可以用在服务器当中:
 

 上述这种模型当中,手机等等终端经常和 服务器 进行数据的比对,比如判断交集以外的数据,当手机当中有 服务器当中没有的 数据的时候,就需要上传到 服务器当中。这些本质是都是一些数据比对。使用上述 用 set 的方式是非常好的方法。

 
20. 有效的括号 - 力扣(LeetCode)

给定一个只包括 '('')''{''}''['']' 的字符串 s ,判断字符串是否有效。

有效字符串需满足:

  1. 左括号必须用相同类型的右括号闭合。
  2. 左括号必须以正确的顺序闭合。
  3. 每个右括号都有一个对应的相同类型的左括号。

 在上图当中可以感受一些 ,C++有了 map 之后 相比于 C 语言的是实现有多么好用。

而且,对于 c 当中硬性的括号判断,C++当中map 只需要多增加几个结点就可以了。C中中硬性的括号判断很冗余。

 138. 复制带随机指针的链表 - 力扣(LeetCode)

 C语言当中的实现过程可以看下面这篇博客:
赋值带随机指针的链表-CSDN博客

 以下是C++当中利用 map 的key 和 value 来实现的映射关系。

class Solution {
public:
    Node* copyRandomList(Node* head) {
        map<Node*, Node*> CopyNodeMap;  // key 值存储 原链表当中的结点
                                        // value 值存储 新链表当中结点
                                        // 两边位置相对一样
        Node* copyhead = nullptr, *copytail = nullptr;

        // 先构建新链表,构建 next 指针
        Node* cur = head;
        while(cur)
        {
            Node* copyNode = new Node(cur->val);
            if(copyhead == nullptr)
            {
                copyhead = copytail = copyNode;
            }
            else
            {
                copytail->next = copyNode;
                copytail = copytail->next;
            }

            // 添加新表和旧表映射关系
            CopyNodeMap[cur] = copytail;
            cur = cur->next;
        }

        // 新表当中 random赋值
        cur = head;
        Node* copyptr = copyhead;
        while(cur)
        {
            if(cur->random == nullptr)
                copyptr->random = nullptr;
            else
                copyptr->random = CopyNodeMap[cur->random];

            cur = cur->next;
            copyptr = copyptr->next;
        }

        return copyhead;
    }
};

环形链表 II

 

给定一个链表的头节点  head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null

如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意:pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。

不允许修改 链表。

示例 1:

 

输入:head = [3,2,0,-4], pos = 1
输出:返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点。
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        set<ListNode*> s;

        ListNode* lptr = head;
        while(lptr)
        {
            if(s.find(lptr) == s.end())
            {
                s.insert(lptr);
            }
            else
            {
                return lptr;
            }

            lptr = lptr->next;
        }

        return NULL;
        
    }
};

  • 4
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

chihiro1122

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值