luoguP2267 琪琪的项链

时间:2022-12-18 06:17:38

题目:http://www.luogu.org/problem/show?pid=2267

题解:这题略吊。

看了之后发现不能用组合数学直接得出公式,然后如果直接暴力也不知道如何去排除两个颜色序列相同的情况。

然后去膜拜题解

---------------------------------------------------------------------------------------------------

令f[i]表示结尾 i 必选的DIY方法的种数。所有珠子颜色不一样的话:f[i] = f[0] + f[1] + ... + f[i-1]

对于颜色不同的话,假设珠子i颜色为c,颜色c在i之前出现的位置为j,那么在式子:f[i] = f[0] + f[1] + ... + f[i-1] 中,我们多算了f[0] + f[1] + ... + f[j-1]。

所以对于此时的f[i] = f[j] + ... +f[i-1],最后 ans = sum(f[i] , 1 <= i <= n)。

----------------------------------------------------------------------------------------------------

说到这里应该就很明显了,搞个前缀和就行了。

这种与颜色序列有关的题貌似记录一下该颜色上一次出现的位置是个不错的想法?@HH的项链 @采花

代码:

 #include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<iostream>
#include<string>
#include<set>
#include<map>
#include<vector>
#include<algorithm>
#include<queue>
#define for0(i,n) for(int i=0;i<=n;i++)
#define for1(i,n) for(int i=1;i<=n;i++)
#define for2(i,x,y) for(int i=(x);i<=(y);i++)
#define for3(i,y,x) for(int i=(y);i>=(x);i--)
#define maxn 500000+100
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();}
return x*f;
}
int n,m,tot,f[maxn],g[maxn],a[maxn],b[maxn],c[maxn],pre[maxn],head[maxn];
inline bool cmp(int x,int y){return a[x]<a[y];}
int main()
{
freopen("input.txt","r",stdin);
freopen("output.txt","w",stdout);
n=read();m=read();
for1(i,n)a[i]=read(),c[i]=i;
sort(c+,c+n+,cmp);
for1(i,n)
{
if(i==||a[c[i]]!=a[c[i-]])tot++;
b[c[i]]=tot;
}
for1(i,n)
{
pre[i]=head[b[i]];head[b[i]]=i;
}
g[]=f[]=;
for1(i,n)
{
f[i]=g[i-];if(pre[i])f[i]=(f[i]-g[pre[i]-]+m)%m;
g[i]=(g[i-]+f[i])%m;
}
printf("%d\n",g[n]-);
return ;
}