首页 > 其他分享 >CF840C 题解

CF840C 题解

时间:2023-09-19 21:22:19浏览次数:38  
标签:平方 cc 题解 ll tot times CF840C dp

一、题目描述:

  给你一个长度为 $n$ 的序列 $a_1\sim a_n$,$0 \le a_i \le 1\times 10^9$。

  求有多少种 $1\sim n$ 的全排列 $b$,使得对于 $i\in [2,n],a_{b_i}\times a_{b_{i-1}}$ 不是完全平方数。

  本题中完全平方数的定义:若存在某个整数 $k$,使得 $k^2=x$,则 $x$ 是一个完全平方数。

  答案对 $1\times 10^9+7$ 取模。数据范围:$1\le n\le 300$;


 二、解题思路:

  其实题意就是说相邻两个数的乘积不能为完全平方数。

  首先容易发现性质:若正整数 $a,b,c$ 满足 $a\times b,a\times c$ 为完全平方数,则 $b\times c$ 为完全平方数。

  证明:设 $a=x^2\times k,k$ 不含完全平方因子。$x,k$ 均为正整数。

  若正整数 $k$ 可以写作 $p^2\times q$ 且 $p,q$ 均为正整数且 $p>1$,则认为 $k$ 含有完全平方因子。

  那么 $a\times b=x^2\times k\times b,a\times c=x^2\times k\times c$。

  因为 $a\times b,a\times c$ 为完全平方数,则 $b$ 可以写作 $y^2\times k$,$c$ 可以写作 $z^2\times k$。$y,z$ 均为正整数。

  所以 $b\times c=y^2\times z^2\times k^2=(xyk)^2$ 为完全平方数。

  注意 $0$ 是一个特殊存在,它与任何数的乘积都是完全平方数。所以有 $0$ 直接输出 $0$ 就可以了。

  于是我们将这 $n$ 个数分为若干个集合,集合内的数不能相邻,集合外的数可以相邻。设有 $m$ 个集合。

  于是我们发现这个题转化为一个 $dp$ 计数题。

  对于一个集合,因为不能集合内的数不能相邻,可以理解成我们有 $k$ 个空需要填。

  $f_{i,j}$ 表示前 $i$ 个集合还剩下 $j$ 个空需要填的方案数。初值 $f_{0,0}=1$。

  枚举上一步剩余的空,枚举这一步填上的空,枚举这一步新产生的空,转移就可以了。

  不过这个代码里面的 $dfs$ 着实鬼畜。也不知道我是怎么想到的。

  其实这一步是要求有 $x$ 个数要放在 $y$ 个位置的排列数,每个位置可以放多个数。

  乍一看,每个数都有 $y$ 个位置可以选,不就是 $y^x$ 吗?

  其实这样漏掉了多个数在同一位置上的不同排列方案。

  但是我肯定不可能枚举每个位置上的数的个数来求排列,只好一开始就求出排列。

  于是问题变成了顺序在 $y$个位置上放置 $x$ 个数的方案数。

  这个问题可以递归求解,记忆化搜索做到时间复杂度 $O(n^3)$。

  于是这道鬼畜的 $dp$ 组合计数题就做完了,时间复杂度 $O(n^3)$。


 三、完整代码:

 1 #include<bits/stdc++.h>
 2 #define N 1010
 3 #define ll long long
 4 #define M 1000000007
 5 #define rep(i,l,r) for(ll i=l;i<=r;i++)
 6 #define per(i,r,l) for(ll i=r;i>=l;i--)
 7 using namespace std;
 8 ll n,a[N],fac[N],inv[N],f[N][N];
 9 ll cnt,bel[N],sz[N],sum[N],dp[N][N],vis[N][N];
10 ll C(ll x,ll y){
11     return fac[x]*inv[y]%M*inv[x-y]%M;
12 }
13 ll ksm(ll base,ll q){
14     ll res=1;while(q){
15         if(q&1) res*=base,res%=M;
16         base*=base;base%=M;q>>=1;
17     } return res;
18 }
19 ll dfs(ll tot,ll cc){
20     if(cc==0) return 1;
21     if(vis[tot][cc]) return dp[tot][cc]; vis[tot][cc]=1;
22     rep(i,1,tot) (dp[tot][cc]+=dfs(tot-i+1,cc-1))%=M;
23     return dp[tot][cc];
24 }
25 int main(){
26     ios::sync_with_stdio(false);
27     cin.tie(0);cout.tie(0);
28     cin>>n; rep(i,1,n) cin>>a[i];
29     rep(i,1,n) if(!a[i]){
30         cout<<0<<'\n';
31         return 0;
32     }
33     rep(i,1,n){
34         bool flag=0; rep(j,1,cnt){
35             ll val=a[i]*bel[j],sval=sqrt(val);
36             if(sval*sval==val){flag=1;sz[j]++;break;}
37         } if(!flag) bel[++cnt]=a[i],sz[cnt]=1;
38     }
39     fac[0]=1; rep(i,1,n<<1) fac[i]=fac[i-1]*i%M;
40     rep(i,0,n<<1) inv[i]=ksm(fac[i],M-2);
41     f[0][0]=1; rep(i,1,cnt) sum[i]=sum[i-1]+sz[i];
42     rep(i,1,cnt) rep(j,0,max(sum[i-1]-1,0ll))
43         rep(k,0,min(sz[i],j)) rep(l,0,sz[i]-k){
44             ll t1=C(sz[i],k)*fac[k]%M*C(j,k)%M;
45             ll t2=C(sz[i]-k,l)*fac[l]%M*dfs(sz[i]-l,l)%M;
46             ll res=sz[i]-k-l;
47             ll t3=C(sum[i-1]+1-j,res)*fac[res]%M;
48             (f[i][j-k+l]+=f[i-1][j]*t1%M*t2%M*t3%M)%=M;
49         }
50     cout<<f[cnt][0]<<'\n';
51     return 0;
52 }

四、写题心得:

  今天一上午都用来想这个题了,好在绝杀过大样例。

  希望不要把我的 $rp$ 用完了,保佑我过初赛吧!

  自己想出来这样的题真的很有成就感,所以来写题解了。收获如下:

  $1、只要你敢想,什么都有可能!=> Exp++!$

  $2、只要\ dp\ 的状态设计得好,乱搞都是可以过题的=> Exp++!$

标签:平方,cc,题解,ll,tot,times,CF840C,dp
From: https://www.cnblogs.com/yhy-trh/p/CF840C.html

相关文章

  • 9.18CF1817题解
    9.18CF1817题解A.AlmostIncreasingSubsequence题意给定长度为\(n\)一个序列\(a\)以及\(q\)次询问,每次询问给出\(l\)和\(r\),找出序列\(a\)在\([l,r]\)内最长的几乎递增子序列。对于几乎递增的定义:如果一个序列中不存在连续的三个数\(x\),\(y\),\(z\),使得\(x\g......
  • 题解 AGC058B 【Adjacent Chmax】
    postedon2022-08-1500:08:56|under题解|sourceproblem一个长为\(n\)的排列\(P\),每次可以选择一个\(i\),令\(v=\max(P_i,P_{i+1})\),使\(P_i=P_{i+1}=v\),求若干次操作后有多少种不同的序列。\(1\leqn\leq5000\)。solution显然地,对于一个\(P_i\),它要么被完全覆盖......
  • AT_arc165_d 题解
    AT_arc165_d[ARC165D]SubstringComparison题解Links洛谷AtCoderDescription给定正整数\(n,m\)和\(m\)个形如\((A_{i},B_{i},C_{i},D_{i})\)的限制条件。判断是否存在一个长度为\(n\)的序列\(P\)满足\(\foralli\in[1,m]\),\(P_{A_{i}\dotsB_{i}}\)字典序......
  • AT_arc165_b 题解
    AT_arc165_b[ARC165B]SlidingWindowSort2题解Links洛谷AtCoderDescription给定正整数\(n,k\)和一个长度为\(n\)的整数\(P\),你需要选择一个长度为\(k\)的区间\([l,l+k-1]\),将这个区间从小到大排序。找到操作后最终字典序最大的排列。\(1\leqk\leqn\l......
  • 【Android】折叠效果CoordinatorLayout+AppBarLayout首页效果&& CoordinatorLayout抖
    亲测效果如下:布局结构<?xmlversion="1.0"encoding="utf-8"?><android.support.constraint.ConstraintLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto&qu......
  • Azamon Web Services 题解
    AzamonWebServices看到目前题解都是\(O(n^2)\)的复杂度,来一发\(O(nlogn)\)的贪心题解。思路很简单,先求经过至多一次的交换后,最小的字符串\(S\)。再和\(T\)比较,如果小于就输出,否则无解。问题转化成了两个子问题:求经过至多一次的交换后,最小的字符串\(S\)。和\(T\)......
  • CF762C Two strings 题解
    洛谷传送门CF传送门题意给你两个字符串\(a\)和\(b\),你可以在\(b\)中删去尽量短的子段,使得\(b\)是\(a\)的子序列。求出最后的\(b\)。思路真是奇了怪了,这种题洛谷题解里竟然没有双指针的做法?首先考虑判断一个字符串\(b\)是否是另一个字符串\(a\)的子序列。这......
  • Alice and Hairdresser题解
    AliceandHairdresser第一眼线段树,第二眼好像可以直接用数组模拟。当一根头发长于\(l\),它再长多长其实都一样,所以不用开longlong。如果一根新的头发长到比\(l\)长,那可以分成以下几种情况:如果它左侧和右侧只有一个元素大于\(l\),那答案不变。如果左侧和右侧都大于......
  • c# winform打开外部程序异常问题解决方案
    c#winform中打开外部程序的常规操作是使用Process类,此时,如果外部程序没有对路径的操作或其他路径文件的操作时,通常不会出现报错或异常;反之,会出现找不到路径或者直接抛出异常。此种情况主要是因为外部程序和当前程序不在一个路径下导致的,以下是解决方案:System.IO.Directory.Set......
  • yarn 出现 【 info There appears to be trouble with your network connection. Retr
    第一种解决方案#调整为taobao镜像源yarnconfigsetregistryhttps://registry.npm.taobao.org我用了没用,可以试试第二种解决方案要在项目根目录下创建后缀名为.yarnrc的文件,并设置network-timeout的值为600000,你可以按照以下步骤进行操作:打开文本编辑器,例如Note......