设矩阵的行数和列数分别为 m 和 n。
由于循环左移 n 次等价于循环左移 0 次,循环左移 n+1 次等价于循环左移 1 次,…,可以让 k = k%n,结果一样。
如果此时 k=0,那么操作不会影响矩阵,直接返回 true。
否则,直接按题意模拟即可。
/*
* @lc app=leetcode.cn id=2946 lang=cpp
*
* [2946] 循环移位后的矩阵相似检查
*/
// @lc code=start
class Solution
{
public:
bool areSimilar(vector<vector<int>> &mat, int k)
{
if (mat.empty())
return false;
int m = mat.size(), n = m ? mat[0].size() : 0;
k = k % n;
if (k == 0)
return true;
for (int i = 0; i < m; i++)
{
if (i % 2)
{
for (int j = 0; j < n; j++)
if (mat[i][(j + k) % n] != mat[i][j])
return false;
}
else
{
for (int j = 0; j < n; j++)
if (mat[i][(j + n - k) % n] != mat[i][j])
return false;
}
}
return true;
}
};
// @lc code=end
时间复杂度:O(m*n)其中 m 和 n 分别是矩阵 mat 的行数和列数。
空间复杂度:O(1)。
暴力枚举每个子字符串的开头 i,令 j=i 并一路直达字符串 s 的末尾。
在这个过程中,设当前字符为 c = s[i],设置 vowels 和 consonants 统计元音字符和非元音字符的个数。
每当满足 vowels == consonants && (vowels * consonants) % k == 0
条件,说明 s[i…j] 是一个非空美丽子字符串,计数器 count 自增 1。
最后返回 count 即为答案。
/*
* @lc app=leetcode.cn id=2947 lang=cpp
*
* [2947] 统计美丽子字符串 I
*/
// @lc code=start
class Solution
{
public:
int beautifulSubstrings(string s, int k)
{
int len = s.length(), count = 0;
for (int i = 0; i < len; i++)
{
int vowels = 0, consonants = 0;
for (int j = i; j < len; j++)
{
if (isVowels(s[j]))
vowels++;
else
consonants++;
if (vowels == consonants && (vowels * consonants) % k == 0)
count++;
}
}
return count;
}
// 辅函数 - 判断字符 c 是否是元音字母
bool isVowels(const char &c)
{
if (c == 'a' || c == 'e' || c == 'i' || c == 'o' || c == 'u')
return true;
return false;
}
};
// @lc code=end
时间复杂度:O(len2),其中 len 是字符串 s 的长度。
空间复杂度:O(1)。
分组排序。
把 nums[i] 及其下标 i 绑在一起排序(也可以单独排序下标),然后把 nums 分成若干段,每一段都是递增的且相邻元素之差不超过 limit,那么这一段可以随意排序。
/*
* @lc app=leetcode.cn id=2948 lang=cpp
*
* [2948] 交换得到字典序最小的数组
*/
// @lc code=start
class Solution
{
public:
vector<int> lexicographicallySmallestArray(vector<int> &nums, int limit)
{
// 特判
if (nums.empty())
return {};
if (limit <= 0)
return nums;
int n = nums.size();
vector<int> index(n, 0);
iota(index.begin(), index.end(), 0);
// for (int i = 0; i < n; i++)
// index[i] = i;
sort(index.begin(), index.end(), [&](int i, int j)
{ return nums[i] < nums[j]; });
vector<int> ans(n, 0);
// 分组循环
int i = 0;
// 外层循环:准备工作 + 更新答案
while (i < n)
{
int start = i;
i++;
// 找出分组
while (i < n && nums[index[i]] - nums[index[i - 1]] <= limit)
i++;
// [start, i) 属于一个分组
// 提取出 nums[start, i) 中的下标
vector<int> subIndex(index.begin() + start, index.begin() + i);
sort(subIndex.begin(), subIndex.end());
// 插入到答案数组
for (int j = 0; j < subIndex.size(); j++)
ans[subIndex[j]] = nums[index[start + j]];
}
return ans;
}
};
// @lc code=end
时间复杂度:O(nlogn),其中 n 是数组 nums 的长度。
空间复杂度:O(n),其中 n 是数组 nums 的长度。
题解:分解质因子+前缀和+哈希表(Python/Java/C++/Go)
/*
* @lc app=leetcode.cn id=2949 lang=cpp
*
* [2949] 统计美丽子字符串 II
*/
// @lc code=start
class Solution
{
public:
long long beautifulSubstrings(string s, int k)
{
int n = s.length();
k = mySqrt(4 * k);
// , 出现次数>
map<pair<int, int>, int> cnt;
cnt[make_pair(k - 1, 0)]++;
int preSum = 0;
long long ans = 0;
for (int i = 0; i < n; i++)
{
int bit = isVowels(s[i]) ? 1 : -1;
preSum += bit;
cnt[make_pair(i % k, preSum)];
ans += cnt[make_pair(i % k, preSum)];
cnt[make_pair(i % k, preSum)]++;
}
return ans;
}
int mySqrt(int n)
{
int res = 1;
for (int i = 2; i * i <= n; i++)
{
int i2 = i * i;
while (n % i2 == 0)
{
res *= i;
n /= i2;
}
if (n % i == 0)
{
res *= i;
n /= i;
}
}
if (n > 1)
res *= n;
return res;
}
// 辅函数 - 判断字符 c 是否是元音字母
bool isVowels(const char &c)
{
if (c == 'a' || c == 'e' || c == 'i' || c == 'o' || c == 'u')
return true;
return false;
}
};
// @lc code=end
时间复杂度:O(n+sqrt(k)),其中 n 是字符串 s 的长度。
空间复杂度:O(n),其中 n 是字符串 s 的长度。