0128-1996. 游戏中弱角色的数量

120 阅读3分钟

「这是我参与2022首次更文挑战的第6天,活动详情查看:2022首次更文挑战」。

你正在参加一个多角色游戏,每个角色都有两个主要属性:攻击防御 。给你一个二维整数数组 properties,其中 properties[i] = [attacki, defensei] 表示游戏中第 i 个角色的属性。

如果存在一个其他角色的攻击和防御等级 都严格高于 该角色的攻击和防御等级,则认为该角色为 弱角色 。更正式地,如果认为角色 i 弱于 存在的另一个角色 j ,那么 attackj > attackidefensej > defensei

返回 弱角色 的数量。

示例 1:

输入:properties = [[5,5],[6,3],[3,6]]
输出:0
解释:不存在攻击和防御都严格高于其他角色的角色。

示例 2:

输入:properties = [[2,2],[3,3]]
输出:1
解释:第一个角色是弱角色,因为第二个角色的攻击和防御严格大于该角色。

示例 3:

输入:properties = [[1,5],[10,4],[4,3]]
输出:1
解释:第三个角色是弱角色,因为第二个角色的攻击和防御严格大于该角色。

排序

对于每个角色,我们需要判断是否存在一个攻击值和防御值都高于它的角色,从而确定该角色是否为弱角色。我们可以按角色的某个属性(比如攻击值)从大到小的顺序遍历,同时记录已经遍历过的角色防御值的最大值 maxDefmaxDef。对于当前角色 pp,如果 pp 的防御值严格小于 maxDefmaxDef,那么说明存在防御值比 pp 高的角色(记作 qq),如果此时 qq 的攻击值也严格大于 pp,则可以确定 pp 为弱角色。

如何保证当 qq 的防御值 maxDefmaxDef 严格大于 pp 的防御值时,qq 的攻击值一定大于 pp 的攻击值,这是比较难处理的一点,因为可能存在攻击值相同的角色。

  • 最简单的处理办法就是将攻击值相同的角色进行单独分组,严格保证攻击值相同的角色被分到同一组,遍历时记录攻击值严格大于当前分组且防御值最大的角色 qq。遍历当前分组时,如果发现角色 pp 的防御值严格小于 qq 的防御值,则此时可以肯定角色 pp 属于弱角色。
  • 实际处理时,对于攻击值相同的角色,我们按照其防御值从小到大进行排序且按照攻击值从大到小开始遍历,这样就可以保证当前已经遍历过的最大防御值角色 qq 的防御值 maxDefmaxDef 严格大于当前角色 pp 的防御值时,则此时 qq 的攻击值一定严格大于 pp 的攻击值。因为相同的攻击值按照防御值从大到小进行排列,如果出现已经遍历过的角色 qq 的防御值大于 pp 的防御值,则此时我们可以肯定可以推理出角色 qq 与角色 pp 攻击值一定不相同。
var numberOfWeakCharacters = function(properties) {
    properties.sort((o1, o2) => {
        return o1[0] === o2[0] ? (o1[1] - o2[1]) : (o2[0] - o1[0]);
    });
    let maxDef = 0;
    let ans = 0;
    for (const p of properties) {
        if (p[1] < maxDef) {
            ans++;
        } else {
            maxDef = p[1];
        }
    }
    return ans;
};

复杂度分析

  • 时间复杂度:O(nlogn)O(nlog⁡n),其中 nn 为数组的长度。排序的时间复杂度为 O(nlogn)O(nlog⁡n),遍历数组的时间为 O(n)O(n),总的时间复杂度为 O(nlogn+n)=O(nlogn)O(nlog⁡n+n)=O(nlog⁡n)
  • 空间复杂度:O(logn)O(log⁡n),其中 nn 为数组的长度。排序时使用的栈空间为 O(logn)O(log⁡n)

image.png

单调栈

对于角色 pp,如果我们找到一个角色 qq 的防御值与攻击值都严格高于 pp 的攻击值和防御值,则我们认为角色 pp 为弱角色。

  • 我们联想到使用单调递增栈的解法,单调递增栈中保证栈内所有的元素都按照从小到大进行排列。按照角色攻击值的大小从低到高依次遍历每个元素,使用单调递增栈保存所有角色的防御值,遍历时如果发现栈顶的角色 p 的防御值小于当前的角色 q 的防御值,则可以认为找到攻击值和防御值都严格大于 p 的角色 q。
  • 如果所有角色的攻击值都不相同,则上述的单调递增栈的解法比较简单,难点在于如何处理攻击值相同但防御值不同的角色比较问题。我们按照攻击值相同时防御值从大到小进行排序,这样即可保证攻击值相同但防御值不同时的角色在进行比较时不会产生计数。
var numberOfWeakCharacters = function(properties) {
    properties.sort((o1, o2) => {
        return o1[0] === o2[0] ? (o2[1] - o1[1]) : (o1[0] - o2[0]);
    });
    let ans = 0;
    const st = [];
    for (const p of properties) {
        while (st.length && st[st.length - 1] < p[1]) {
            st.pop();
            ans++;
        }
        st.push(p[1]);
    }
    return ans;
};

复杂度分析

  • 时间复杂度:O(nlogn)O(nlog⁡n),其中 n 为数组的长度。排序的时间复杂度为 O(nlogn)O(nlog⁡n),然后需要一次遍历的时间为 O(n)O(n),总的时间复杂度 O(nlogn+n)=O(nlogn)O(nlog⁡n+n)=O(nlog⁡n)
  • 空间复杂度:O(n)O(n),其中 nn 为数组的长度。需要栈来保存中间变量。

image.png