leetcode之形成两个异或相等数组的三元组数目(C++)

参考链接

  1. https://leetcode-cn.com/problems/count-triplets-that-can-form-two-arrays-of-equal-xor/
  2. https://leetcode-cn.com/problems/count-triplets-that-can-form-two-arrays-of-equal-xor/solution/xing-cheng-liang-ge-yi-huo-xiang-deng-sh-jud0/

题目描述

给你一个整数数组 arr 。

现需要从数组中取三个下标 i、j 和 k ,其中 (0 <= i < j <= k < arr.length) 。

a 和 b 定义如下:

a = arr[i] ^ arr[i + 1] ^ … ^ arr[j - 1]
b = arr[j] ^ arr[j + 1] ^ … ^ arr[k]
注意:^ 表示 按位异或 操作。

请返回能够令 a == b 成立的三元组 (i, j, k) 的数目。
在这里插入图片描述

解题思路

首先前缀和求出0到数组所有位置的异或值,用 S [ k + 1 ] S[k+1] S[k+1]表示数组下标0到 k k k的异或值。所以 a = S [ i ] ⨁ S [ j ] a = S[i] \bigoplus S[j] a=S[i]S[j] b = S [ j ] ⨁ S [ k + 1 ] b = S[j] \bigoplus S[k + 1] b=S[j]S[k+1]。要使 a = b a = b a=b,只需S[i] = S[k + 1]。直接使用三重循环遍历所有情况是容易想到的,难点在于如何优化。由于 S [ i ] = S [ k + 1 ] S[i] = S[k + 1] S[i]=S[k+1]的条件与 j j j无关,所以只要 S [ i ] = S [ k + 1 ] S[i] = S[k + 1] S[i]=S[k+1] j j j可以是 i + 1 i+1 i+1 k k k间的任一值,共 k − i k - i ki种,于是优化到了二重循环。

进一步分析,若 i = i 1 , i 2 , ⋯ i m i=i_1,i_2,\cdots i_{m} i=i1,i2,im都满足 S [ i ] = S [ k + 1 ] S[i] = S[k + 1] S[i]=S[k+1],那么, ( k − i 1 ) + ( k − i 2 ) + ( k − i m ) = m × k − ( i 1 + i 2 + ⋯ i m ) \left ( k-i_{1}\right )+\left ( k-i_{2}\right )+\left ( k-i_{m}\right )=m\times k-\left ( i_{1}+i_{2}+\cdots i_{m}\right ) (ki1)+(ki2)+(kim)=m×k(i1+i2+im)。所以我们需要用两个哈希表分别存储S[k]出现的次数和值为S[k]的下标之和。甚至,我们可以将这一步与求前缀和合并,只需要一次遍历。

代码

三重循环

class Solution {
public:
    int countTriplets(vector<int>& arr) {
        vector<int> nums(arr.size() + 1);
        for (int i = 1; i <= arr.size(); i ++)
        {
            nums[i] = nums[i - 1] ^ arr[i - 1];
        }

        int res = 0;
        for(int i = 0, j = 1, k = 1; j < arr.size(); j ++)
        {
            for (i = j - 1; i >= 0; i --)
            {
                for (k = j; k < arr.size(); k ++)
                {
                    if (nums[i] == nums[k + 1])
                    {
                        res += 1;
                    }
                }
            }
        }
        return res;
    }
};

二重循环

class Solution {
public:
    int countTriplets(vector<int> &arr) {
        int n = arr.size();
        vector<int> s(n + 1);
        for (int i = 0; i < n; ++i) {
            s[i + 1] = s[i] ^ arr[i];
        }
        int ans = 0;
        for (int i = 0; i < n; ++i) {
            for (int k = i + 1; k < n; ++k) {
                if (s[i] == s[k + 1]) {
                    ans += k - i;
                }
            }
        }
        return ans;
    }
};

一重循环

class Solution {
public:
    int countTriplets(vector<int> &arr) {
        int n = arr.size();
        unordered_map<int, int> cnt, total;
        int ans = 0, s = 0;
        for (int k = 0; k < n; ++k) {
            int val = arr[k];
            if (cnt.count(s ^ val)) {
                ans += cnt[s ^ val] * k - total[s ^ val];
            }
            ++cnt[s];
            total[s] += k;
            s ^= val;
        }
        return ans;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值