LeetCode 1125. 最小的必要团队

题目描述

作为项目经理,你规划了一份需求的技能清单 req_skills,并打算从备选人员名单 people 中选出些人组成一个「必要团队」( 编号为 i 的备选人员 people[i] 含有一份该备选人员掌握的技能列表)。

所谓「必要团队」,就是在这个团队中,对于所需求的技能列表 req_skills 中列出的每项技能,团队中至少有一名成员已经掌握。

我们可以用每个人的编号来表示团队中的成员:例如,团队 team = [0, 1, 3] 表示掌握技能分别为 people[0],people[1],和 people[3] 的备选人员。

请你返回 任一 规模最小的必要团队,团队成员用人员编号表示。你可以按任意顺序返回答案,本题保证答案存在。

示例 1:
输入:req_skills = ["java","nodejs","reactjs"], people = [["java"],["nodejs"],["nodejs","reactjs"]]
输出:[0,2]

题目解析

  1. 集合覆盖问题,求最右解需使用暴力解法。
  2. 将技能和人员进行状态压缩
  3. 使用顺退过程尝试对每一个团队状态添加任务,添加后得到新的团队状态,若人数比之前少,则更新新状态团队任务。
    dp[newskill] = min(dp[newskill], dp[skill] | k), 其中k为添加人员的skill

C++代码

class Solution {
public:
    vector<int> smallestSufficientTeam(vector<string>& req_skills, vector<vector<string>>& people) {
        int skLen = (1 << req_skills.size()) - 1;
        int peLen = people.size();
        vector<int> vecA(skLen + 1, 0);
        vector<int> vecB(skLen + 1, 0);
        vector<int> vecLen(skLen + 1, peLen);
        vector<int> ans;
        vecLen[0] = 0;
        map<string, int> skMap;
        for(int i = 0 ; i < req_skills.size(); i++) {
            skMap[req_skills[i]] = i;
        }

        for(int i = 0; i < people.size();i++) {
            int skill = 0;
            for(int j = 0; j < people[i].size(); j++) {
                skill |= (1 << skMap[people[i][j]]);
            }
        
            for(int j = 0; j < skLen; j++) {
                int nskill = j | skill;                
                if(nskill != j && vecLen[nskill] > vecLen[j] + 1) {
                    vecLen[nskill]=vecLen[j] + 1;
                    if(i < 30) {                        
                        vecA[nskill] = vecA[j] | (1 << i);  
                        vecB[nskill] = vecB[j];
                    } else {
                        vecA[nskill] = vecA[j];
                        vecB[nskill] = vecB[j] | (1 << (i-30));                        
                    }
                    
                }
            }
        }
        int i = 0;
        
        while(vecA[skLen]) {
            if((vecA[skLen] & 1) != 0) {
                ans.push_back(i);
            }
            vecA[skLen] >>= 1;
            i++;
        }
        
        i = 0;
        
        while(vecB[skLen]) {
            if((vecB[skLen] & 1) != 0) {
                ans.push_back(i+30);
            }
            vecB[skLen] >>= 1;
            i++;
        }
        
        return ans;
    }
};
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 12,988评论 0 13
  • 项目管理术语英汉对照表2018-7-20 A Abstract Resource 抽象资源 Abstraction...
    007明_阳阅读 11,590评论 0 51
  • 因为预测较长,所以此为托福写作4月考题预测第一部分,第二部分参见下一篇文章。 综合写作: 1. Andean遗址是...
    耐撕world阅读 5,687评论 0 0
  • 您可能要么领导团队,要么在团队中工作,但您知道如何建立一个有效的团队吗? 很多时候,团队很弱,项目失败,因为每个人...
    怀念的猫咪阅读 4,728评论 0 0
  • 《我和孩子的约定》 如果你我不相遇,就不会有别离;如果你我没有约定,就不会失望;如果你我没有血溶于水的联系,就不会...
    香草芬芳阅读 988评论 0 1