2021-12-7 Leetcode 1128. 等价多米诺骨牌对的数量

该博客讨论了在一个C++程序中如何使用unordered_map有效地计算等价多米诺骨牌对的数量,对比了unordered_map和unordered_multimap在处理键值对去重上的差异,并提供了错误的multimap实现示例,强调了unordered_map在这类问题中的优势。

在这里插入图片描述

class Solution {
public:
    int numEquivDominoPairs(vector<vector<int>>& dominoes) {
        unordered_map<int,int> hash_map;
        int ret = 0;
        for(auto& a : dominoes){
            int minNum = min(a[0],a[1]);
            int maxNum = max(a[0],a[1]);
            if(hash_map.find(minNum*10 + maxNum) != hash_map.end()){
                ret += hash_map[minNum*10 + maxNum];
            }
            hash_map[minNum*10 + maxNum] += 1;
        }
        return ret;
    }
};

疑惑:
为什么 unordered_map < vector< int >,int > 这个容器构建不了 ?
如果这道题使用 multimap 怎么做?
以下为错误的解法,不建议用 multimap 做题,multimap 没有对 key -value 去重的功能,只能插入一遍,如果要做的话,就使用先全部插入,再按照 max 检索一遍,至少要检索两遍,时间复杂度极高。

class Solution {
public:
    int numEquivDominoPairs(vector<vector<int>>& dominoes) {
        unordered_multimap<int,int> multi;
        int ret = 0;
         for(auto& a : dominoes){
            int minNum = min(a[0],a[1]);
            int maxNum = max(a[0],a[1]);
            // unordered_multimap<int,int> :: iterator it;
            // it->first = minNum;
            // it->second = maxNum;
            // if(multi.find(it) != multi.end()){
            //     ret ++;
            // }
            //上面这种方法是错误的因为,it->first 是一个 const 
            if(multi.find(minNum) != multi.end()){
                auto found = multi.find(minNum);
                if(found->second == maxNum)
                ret++;
            }
            //上面这种查找方式是有问题的,当存在一个 key 对应多个 value 的时候,只能够查找到第一个 value。
           
            //else multi[minNum] = maxNum;
            // error: type 'unordered_multimap<int, int>' does not provide a subscript operator
            //写程序的意义不在于每天写对一道,而在于不断尝试新的用法,明白新的用法为什么不行。
            else multi.insert(make_pair(minNum,maxNum));
         }
         return ret;
    }
};

今天学到的与 multimap 有关的一切:
查找方式:

#include <iostream>
#include <map>
#include <vector>
#include <algorithm>
using namespace std;
 
int main()
{
    multimap<int, int> testMap;
    testMap.insert(make_pair(5,1));
    testMap.insert(make_pair(5,2));
    testMap.insert(make_pair(5,3));
    testMap.insert(make_pair(5,4));
    int searchItem = 5;
     
    /*第一种方法*/
    //使用 count 函数查找 key 出现的次数
    multimap<int, int>::size_type  cnt = testMap.count(searchItem);
    //使用 find 找到 key 开始的地方,如果有多个重复出现的 key ,返回第一个 key,可以在后面的检索当中通过 ++ 来移动。
    multimap<int, int>::iterator  iter = testMap.find(searchItem);
    for(;cnt > 0; cnt--, iter++)
    {
          cout<<iter->first<<"->"<<iter->second<<endl;
    }
    cout<<endl;
     
    /*第二种方法*/
    //如果有多个 key ,lower_bound()是找到第一个不小于 key  的元素。
    //如果有多个 key,upper_bound()是找到第一个不大于 key 的元素。
    multimap<int, int>::iterator iterBeg = testMap.lower_bound(searchItem);
    multimap<int, int>::iterator iterEnd = testMap.upper_bound(searchItem);
    for(;iterBeg != iterEnd;iterBeg++)
    {
        cout<<iterBeg->first<<"->"<<iterBeg->second<<endl;   
    }
    cout<<endl;
     
    /*第三种方法*/
    auto ret = testMap.equal_range(searchItem);
    auto it = ret.first;
    while(it!=ret.second)
    {
        cout<<it->first<<"->"<<it->second<<endl;
         ++it;
    }
    return 0;
}

参考文章:https://www.cnblogs.com/ladawn/p/8203789.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值