AT2062 ~K Perm Counting 构造+动态规划+容斥

本文介绍了一种解决特定问题的算法思路,通过观察题目特性,利用链状结构进行动态规划,并结合二项式反演完成最终求解。文章详细解释了状态定义、状态转移以及容斥原理的应用。

看到这道题第一个想法肯定是按照套路钦定一些地方不合法,然后其他地方随便选,最后来一个二项式反演.   

但是我们发现这个 DP 状态很难设置.           

然后你发现一个非常神的性质:由于题中给的是绝对值,所以说一个位置要是不合法的话,只有两种数值的可能(+k和-k)  

然后把位置 $i$ 和值 $i+/-k$ 相连,你发现每一个值/位置都只出现在一条链中,然后链和链之间互不影响.  

所以我们就可以直接在链上进行动态规划. 

设 $f[i][j][0/1]$ 表示当前到第 $i$ 个点,已经选了 $j$ 个不合法,$i$ 是否和链中前一个位置相连.    

这个就非常好转移了.     

转移完这个后用二项式反演容斥一下即可. 

code: 

#include <cstdio> 
#include <algorithm>  
#define ll long long   
#define N 2008  
#define mod 924844033 
#define setIO(s) freopen(s".in","r",stdin) 
using namespace std; 
int f[N<<1][N<<1][2],link[N<<1],fac[N<<1];  
int main() 
{ 
    // setIO("input");  
    int i,j,n,k,tot=0; 
    scanf("%d%d",&n,&k);     
    fac[0]=1,f[0][0][0]=1;    
    for(i=1;i<=n;++i) fac[i]=(ll)fac[i-1]*i%mod;     
    for(i=1;i<=k;++i) 
    {
        for(int t=1;t<=2;++t) 
            for(j=i;j<=n;j+=k) 
                if(j!=i) link[++tot]=1; else ++tot;     
    }    
    for(i=1;i<=tot;++i) 
    {
        for(j=0;j<=n;++j) 
        {   
            f[i][j][0]=(ll)(f[i-1][j][0]+f[i-1][j][1])%mod;    
            if(link[i]) f[i][j][1]=f[i-1][j-1][0];      
        }
    }    
    int ans=0; 
    for(i=0;i<=n;++i) 
    {
        int tmp=(ll)(f[n<<1][i][0]+f[n<<1][i][1])%mod;       
        tmp=(ll)tmp*fac[n-i]%mod;      
        if(i&1) ans=(ll)(ans-tmp+mod)%mod;        
        else ans=(ll)(ans+tmp)%mod;   
    }   
    printf("%d\n",ans); 
    return 0; 
}

  

# T617559 「TPOI-5A」Luminescence ## 题目背景 ![](https://cdn.luogu.com.cn/upload/image_hosting/ownsj515.png) (图片来自 Phigros 曲绘,侵删。) ## 题目描述 给定 $n$ 与两个长度为 $n$ 的序列 $a,b$。定义一个 $0\sim n-1$ 的排列 $q$ 是 **魔怔的**,当且仅当: - $\forall 1\le k\le n,\min^k_{i=1}q_i=a_k$。 - $\forall 1\le k\le n,\min^n_{i=k}q_i=b_k$。 定义一个排列 $q$ 的权值为 $\sum_{1\le l\le r\le n}\operatorname{mex}_{l\le i\le r}q_i$,求所有魔怔的排列的权值之和。答案对 $998244353$ 取模。 一个集合 $M$ 的 $\operatorname{mex}(M)$ 定义为最小的没有在 $M$ 中出现的自然数。如 $\text{mex}\{1,2,3,4\}=0,\text{mex}\{0,1,3,4\}=2$。 ## 输入格式 本题有多组测试数据。对于每个测试点,先输入一个正整数 $T$,表示数据组数。 对于每一组测试数据,第一行一个正整数 $n$, 第二行输入 $n$ 个整数 $a_1,a_2,\dots,a_n$,第三行 $n$ 个整数 $b_1,b_2,\dots,b_n$。 ## 输出格式 对于每一组测试数据,输出一行一个整数,表示所有魔怔的排列的权值之和。 ## 输入输出样例 #1 ### 输入 #1 ``` 3 4 0 0 0 0 0 1 2 3 4 1 0 0 0 0 0 2 2 4 0 0 0 0 0 1 1 1 ``` ### 输出 #1 ``` 10 11 14 ``` ## 说明/提示 |$\text{Subtask}$|$n\le$|$\sum n\le$|分值| |:-:|:-:|:-:|:-:| |$1$|$8$|$800$|$20$| |$2$|$10^3$|$10^4$|$40$| |$3$|$2\times10^5$|$2\times10^6$|$40$| 对于 $100\%$ 的数据,$1 \le n \le 2 \times 10^5,0 \le a_i,b_i < n, \sum n \le 2 \times 10^6$。 **对于每组数据保证存在至少一个魔怔的排列。** c++,不用vector,注释
最新发布
08-09
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值