【ICPC】2022 昆明站 D题 题解

题目大意

定义一种序列的合法划分:从左往右依次选择,可以把该数归到 S A SA SA中,或者 S B SB SB中,假如随意划分,有 2 n 2^n 2n中划分方法,但需要满足:

  • S A SA SA 是不严格递增序列
  • S B SB SB 是不严格递减序列

一个序列有很多种合法的划分,现在给定 k k k,请构造一种序列,它的合法划分数刚好是 k k k

题目链接

思路

我们需要找到一种方便计算其结果的构造
考虑一个不严格递增序列 1 , 1 , 1 , 2 , 2 , 2 , 2 , 2 , 3 , 3... x , x , x 1,1,1,2,2,2,2,2,3,3...x,x,x 1,1,1,2,2,2,2,2,3,3...x,x,x

上述序列的合法构造就是,任选一个数,任选一些和它相同的数放到 S B SB SB中,剩下的放 S A SA SA中。

那么对于每一个中数 i i i, 合法情况就是 i ∗ c n t [ i ] i * cnt[i] icnt[i], c n t [ i ] cnt[i] cnt[i] i i i的数。但是需要注意的是,所有情况中,有一种是所有数都不放入 S B SB SB中,那么这些情况会重复。除了第一次计算不用减去,剩余的 i i i的情况都需要减去 1 1 1

所有总的方案数就是 1 + ∑ 2 c n t [ i ] − 1 , i ∈ [ 1 , n ] 1 + \sum 2^{cnt[i]}-1,i \in [1,n] 1+2cnt[i]1,i[1,n]
而这种构造方式,是可以构造出所有的自然数的。

我们直接从最高位枚举 c n t [ i ] cnt[i] cnt[i],如果 k > ( 1 < < c n t [ i ) ] k > (1 << cnt[i)] k>(1<<cnt[i)],那就减去它,同时 i + + i++ i++
(剩下 1 1 1时就可以结束了)

代码

#include <cstdio>
#include <iostream>
#include <vector>
using namespace std;

int k, t = 1;
vector<int> ans;

int main()
{
    scanf("%d", &k);
    if(k == 0)
        printf("12\n1 1 4 5 1 4 1 1 4 5 1 4\n");
    else if(k == 1)
        printf("6\n1 1 4 5 1 4\n");
    else {
        for(int i = 29; i >= 0; --i) {
            while(k > (1 << i) - 1) {
                k -= (1 << i) - 1;
                for(int j = 1; j <= i; ++j)
                    ans.push_back(t);
                ++t;
            }
            if(k == 1)
                break;
        }
        printf("%d\n", ans.size());
        for(int i = 0; i < ans.size(); ++i)
            printf("%d ", ans[i]);
    }
    return 0;
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值