ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

[题解] 密码 | 简单计数

2022-05-02 20:05:15  阅读:176  来源: 互联网

标签:排列 计数 一个 题解 sum times 密码 序列 sim


同步发表于 Mina!

题目大意

对于满足以下要求的长度为 \(n\) 的序列进行计数:

  • 序列的值域为 \([1,k]\);

  • 对于序列的任意位置 \(p\in[1,n]\),可以找到至少一个 \(i\) 满足 \(p\in[i,i+k-1]\),且区间 \([i,i+k-1]\) 为一个 \(1\sim k\) 的排列。

\(n\le10^5,k\le100\)

解题思路

其实原本题意不是这样的,试图描述正式之后好像更难懂了。

密码是一个长度为 \(n\) 的序列。

密码由若干个 \(1\sim k\) 的排列拼接而成,且拼接时,不同排列可重叠。

于是不妨设 \(f_i\) 为最后一个完整排列的结尾是 \(i\) 的方案数。于是可以列出转移式:

\[f_i=\sum_{j=1}^{k}f_{i-j}\times g_{j} \]

\(g_j\) 即在一个 \(1\sim k\) 的排列后接上 \(j\) 个数,使得满足以下两个条件的方案数:

  • \([j+1,j+k]\) 是一个 \(1\sim k\) 的排列,

  • 对于任意 \(1<i<=j,[i,i+k-1]\) 不是一个 \(1\sim k\) 的排列。

直接拿 \(1,2,3\cdots k\) 来考虑 \(g_j\) 怎么求,那么即要求一个 \(1\sim j\) 的排列,对于任意 \(i<j\),这个排列 \([1,i]\) 的前缀位置上不能是一个 \(1\sim i\) 的排列,求满足条件的排列个数。

考虑容斥,首先令 \(g_j=j!\),然后考虑减去不合法的,对于一个不合法的排列,它可能存在若干个前缀符合 \([1,i]\) 是一个 \(1\sim i\) 的排列,那么我们枚举每一个不合法排列最后一个违反限制的前缀,在这个位置将其减去。

假设当前枚举到 \(i\),首先 \([1,i]\) 这部分肯定是 \(i!\) 种填法,而 \([i+1,j]\) 这部分,由于我们钦定 \(i\) 是最后一个违反限制的前缀,故 \([1,i]\)与 \([i+1,j]\) 相接不可再违反限制,即对于任意 \(i<p<j,[i+1,p]\) 这一段上不能是将 \(i+1\sim p\) 这些数任意排列的结果,于是就变成子问题了,乘上 \(g_{j-i}\) 就好了。

所以就是两个简单的式子:

\[g_i=i!-\sum_{j=1}^{i-1}j!\times g_{i-j}\\ f_i=\sum_{j=1}^{k}f_{i-j}\times g_{j} \]

\(f_n\) 就是答案了,然后这个题大概还可以搞什么矩阵快速幂或者线性递推,前者感觉没必要,后者我不了解,于是就到此为止了 QwQ。

标签:排列,计数,一个,题解,sum,times,密码,序列,sim
来源: https://www.cnblogs.com/Callis/p/16216609.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有