【数论】P9034 「KDOI-04」Again Counting Set|普及+
本文研究了集合计数问题,给出了满足特定条件的整数集合数量的计算方法。通过分析集合的最小值、最大值和mex等性质,推导出当k=2时有1种解,当k≥4时解的数量与n和k的关系式。代码实现了基于数学推导的快速计算,能够处理大规模输入数据(n,k≤10^18)。测试样例验证了算法的正确性,展示了不同n和k组合下的计算结果。该算法通过分类讨论和数学公式直接计算答案,避免了暴力枚举,具有较高的时间效率。
本文涉及知识点
「KDOI-04」Again Counting Set
题目背景

题目描述
小 S 不喜欢集合,不喜欢自然数,不喜欢求和,不喜欢求积,不喜欢最小值,不喜欢最大值,不喜欢 mex \operatorname{mex} mex,所以有了这题。
给出 n , k n,k n,k,求有多少个可重整数集合 S S S 满足:
- ∣ S ∣ = k |S|=k ∣S∣=k;
- 对于任意 x ∈ S x\in S x∈S, 0 ≤ x ≤ n 0\le x\le n 0≤x≤n;
- ∏ x ∈ S x = min x ∈ S x \displaystyle{\prod_{x\in S} x=\min_{x\in S} x} x∈S∏x=x∈Sminx;
- ∑ x ∈ S x = min x ∈ S x + max x ∈ S x + mex ( S ) \displaystyle{\sum_{x\in S} x=\min_{x\in S} x+\max_{x\in S}x+{\operatorname{mex}}(S)} x∈S∑x=x∈Sminx+x∈Smaxx+mex(S)。
注: m e x \bf{mex} mex 指集合中没有出现过的最小的自然数。
输入格式
本题包含多组测试数据。
输入的第一行包含一个正整数 T T T,表示测试数据组数。
对于每组测试数据,输入包含一行两个正整数 n , k n,k n,k。
输出格式
对于每组测试数据,输出一行一个整数表示答案。
样例 #1
样例输入 #1
7
1 4
2 4
5 3
2 100
3 8
20 50
499122178 4
样例输出 #1
1
2
0
3
5
39
998244353
提示
【补充说明】
为了更好的让选手理解题面,给出若干合法/不合法集合例子:
- { 0 , 1 , 2 , 2 } \{0,1,2,2\} {0,1,2,2}。
该集合是一个符合要求的集合,因为 0 × 1 × 2 × 2 = 0 = min { 0 , 1 , 2 , 2 } 0\times 1\times 2\times 2=0=\min\{0,1,2,2\} 0×1×2×2=0=min{0,1,2,2}, 0 + 1 + 2 + 2 = 5 , min { 0 , 1 , 2 , 2 } + max { 0 , 1 , 2 , 2 } + mex { 0 , 1 , 2 , 2 } = 0 + 2 + 3 = 5 0+1+2+2=5,\min\{0,1,2,2\}+\max\{0,1,2,2\}+\operatorname{mex}\{0,1,2,2\}=0+2+3=5 0+1+2+2=5,min{0,1,2,2}+max{0,1,2,2}+mex{0,1,2,2}=0+2+3=5。
- { 3 , 5 } \{3,5\} {3,5}
该集合不是一个符合要求的集合,因为虽然 3 + 5 = 8 , min { 3 , 5 } + max { 3 , 5 } + mex { 3 , 5 } = 3 + 5 + 0 = 8 3+5=8,\min\{3,5\}+\max\{3,5\}+\operatorname{mex}\{3,5\}=3+5+0=8 3+5=8,min{3,5}+max{3,5}+mex{3,5}=3+5+0=8,但是 3 × 5 ≠ min { 3 , 5 } 3\times 5\not=\min\{3,5\} 3×5=min{3,5}。
- { 1 , 9 , 1 , 9 , 8 , 1 , 0 } \{1,9,1,9,8,1,0\} {1,9,1,9,8,1,0}。
该集合不是一个符合要求的集合,因为虽然 1 × 9 × 1 × 9 × 8 × 1 × 0 = 0 = min { 1 , 9 , 1 , 9 , 8 , 1 , 0 } 1\times 9\times 1\times 9\times 8\times 1\times 0=0=\min\{1,9,1,9,8,1,0\} 1×9×1×9×8×1×0=0=min{1,9,1,9,8,1,0},但是其和为 29 29 29 而并非 min + max + mex = 0 + 9 + 2 = 11 \min+\max+\operatorname{mex}=0+9+2=11 min+max+mex=0+9+2=11。
【数据范围】
对于 100 % 100\% 100% 的数据,保证 1 ≤ T ≤ 10 6 1\le T\le10^6 1≤T≤106, 1 ≤ n , k ≤ 10 18 1\le n,k\le10^{18} 1≤n,k≤1018。
| 测试点编号 | 分值 | T ≤ T\le T≤ | k ≤ k\le k≤ | n n n |
|---|---|---|---|---|
| 1 1 1 | 10 10 10 | 5 5 5 | 5 5 5 | ≤ 5 \le5 ≤5 |
| 2 2 2 | 10 10 10 | 10 5 10^5 105 | 10 18 10^{18} 1018 | = 1 =1 =1 |
| 3 3 3 | 10 10 10 | 10 5 10^5 105 | 10 18 10^{18} 1018 | = 2 =2 =2 |
| 4 4 4 | 10 10 10 | 10 5 10^5 105 | 10 18 10^{18} 1018 | = 3 =3 =3 |
| 5 5 5 | 10 10 10 | 10 5 10^5 105 | 10 18 10^{18} 1018 | = 4 =4 =4 |
| 6 6 6 | 10 10 10 | 10 5 10^5 105 | 10 18 10^{18} 1018 | = 5 =5 =5 |
| 7 7 7 | 10 10 10 | 10 5 10^5 105 | 10 10 10 | ≤ 10 \le10 ≤10 |
| 8 8 8 | 10 10 10 | 10 5 10^5 105 | 10 3 10^3 103 | ≤ 10 3 \le10^3 ≤103 |
| 9 9 9 | 10 10 10 | 10 6 10^6 106 | 10 18 10^{18} 1018 | ≤ 10 8 \le10^{8} ≤108 |
| 10 10 10 | 10 10 10 | 10 6 10^6 106 | 10 18 10^{18} 1018 | ≤ 10 18 \le10^{18} ≤1018 |
数论
令x1是最小值,x2是最大值。
一,k等于1。
x1 = x1 +mes(x)+x1,如果x1等于0,mes(S)等于1。不成立。
如果x不等于0,则x =2x。1等于2,矛盾。
故k等于1,方案数为0。
二,k等于2。
x1*x2 =x1 如果x1等于0,式子二恒成立。0+x2= 0+mes(s)+x2,式子三无法成立。
如果0 < x1<=x2 。x2 = 1。即x1=x2=1。式子三也成立。
故 k =2 ,方案数1。
三,k >=3。
根据式子二,要么x1是0,要么全部是1。
全部是1,判断 k= 1+0+1,与k >=3矛盾。故只需要考虑x1是0。
如果x1是0,式子三 ⟺ \iff ⟺ 除掉最小值,最大值之和 =mes(S)
如果mse(S)是1,中间必定有一个数是1,和mes(1)矛盾。
如果mes(S)是2,中间必定两2个1,
a,其它全部是0,x2 可以是1也可以>2。 k >=4,方案数: 1 + max(n-2,0)
如果mes(S)是3, 两种情况:a,1,1,1,x2是2,其它全部是0。b,1,2,x2是2或>3。
a, k >=5,方案数一,01112,其它全部为0。
b,k >=4 ,方案是数:012x ,1+ max(n-3)
mes(4) 0121 ,x2是3,其它0。 k >=5,n >=3,方案1。
mes(5)是5 或更多不成立。 因为1+2+3 >5。
总结:
k等于2,返回1。
如果k >=5,n>=2, ans++。
k >=5,n >=3 ans++;
如果 k>=4, ans += 1 + max(n-2,0)
k>=4,n>=2 ans+= 1+ max(n-3,0)
代码
核心代码
#include <iostream>
#include <sstream>
#include <vector>
#include<map>
#include<unordered_map>
#include<set>
#include<unordered_set>
#include<string>
#include<algorithm>
#include<functional>
#include<queue>
#include <stack>
#include<iomanip>
#include<numeric>
#include <math.h>
#include <climits>
#include<assert.h>
#include<cstring>
#include<list>
#include <bitset>
using namespace std;
template<class T1, class T2>
std::istream& operator >> (std::istream& in, pair<T1, T2>& pr) {
in >> pr.first >> pr.second;
return in;
}
template<class T1, class T2, class T3 >
std::istream& operator >> (std::istream& in, tuple<T1, T2, T3>& t) {
in >> get<0>(t) >> get<1>(t) >> get<2>(t) ;
return in;
}
template<class T1, class T2, class T3, class T4 >
std::istream& operator >> (std::istream& in, tuple<T1, T2, T3, T4>& t) {
in >> get<0>(t) >> get<1>(t) >> get<2>(t) >> get<3>(t);
return in;
}
template<class T = int>
vector<T> Read() {
int n;
scanf("%d", &n);
vector<T> ret(n);
for(int i=0;i < n ;i++) {
cin >> ret[i];
}
return ret;
}
template<class T = int>
vector<T> Read(int n) {
vector<T> ret(n);
for (int i = 0; i < n; i++) {
cin >> ret[i];
}
return ret;
}
class Solution {
public:
long long Ans(const long long N, const long long K) {
if (2 == K) { return 1; }
long long ans = 0;
if (K >= 5) {
if (N >= 2) ans++;
if (N >= 3) ans++;
}
if (K >= 4) {
ans += 1 + max(N - 2, 0LL);
if (N >= 2) {
ans += 1 + max(N - 3, 0LL);
}
}
return ans;
}
};
int main() {
#ifdef _DEBUG
freopen("a.in", "r", stdin);
#endif // DEBUG
int T;
cin >> T;
long long N, K;
for (int i= 0; i < T; i++) {
cin >> N >> K;
auto res = Solution().Ans(N, K);
printf("%lld\r\n", res);
}
#ifdef _DEBUG
//printf("K=%d", K);
//Out(b, "b=");
//Out(a, ",a=");
#endif // DEBUG
return 0;
}
单元测试
long long N, K;
TEST_METHOD(TestMethod11)
{
auto res = Solution().Ans(1,4);
AssertEx(1LL, res);
}
TEST_METHOD(TestMethod12)
{
auto res = Solution().Ans(2, 4);
AssertEx(2LL, res);
}
TEST_METHOD(TestMethod13)
{
auto res = Solution().Ans(5, 3);
AssertEx(0LL, res);
}
TEST_METHOD(TestMethod14)
{
auto res = Solution().Ans(2, 100);
AssertEx(3LL, res);
}
TEST_METHOD(TestMethod15)
{
auto res = Solution().Ans(3, 8);
AssertEx(5LL, res);
}
TEST_METHOD(TestMethod16)
{
auto res = Solution().Ans(20, 50);
AssertEx(39LL, res);
}
TEST_METHOD(TestMethod17)
{
auto res = Solution().Ans(499122178, 4);
AssertEx(998244353LL, res);
}
扩展阅读
| 我想对大家说的话 |
|---|
| 工作中遇到的问题,可以按类别查阅鄙人的算法文章,请点击《算法与数据汇总》。 |
| 学习算法:按章节学习《喜缺全书算法册》,大量的题目和测试用例,打包下载。重视操作 |
| 有效学习:明确的目标 及时的反馈 拉伸区(难度合适) 专注 |
| 闻缺陷则喜(喜缺)是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。 |
| 子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。 |
| 如果程序是一条龙,那算法就是他的是睛 |
| 失败+反思=成功 成功+反思=成功 |
视频课程
先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771
如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176
测试环境
操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。
更多推荐



所有评论(0)