给定序列A,求出A中本质不同的子序列(包含空的子序列)个数模109+7的结果.
一个序列B是A的子序列需要满足A删掉某些元素后能够得到B.
两个子序列中对应位置的数都相同我们就认为这两个子序列本质相同.
标签:本质 while scan git 图片 include 分享 printf pen
5
2 3 1 3 2
27
对于20%的数据,N≤10.
对于40%的数据,N≤20
对于70%的数据,N≤100000,1≤Ai≤100
对于100%的数据,N≤1000000,1≤Ai≤N
中文题意不多解释了。
2 3 1 3 2
以a[i]结尾的序列 1 1+1 3+1 7+1 13+1 (用p[i]记录)
1~i构成的子序列 1 1+2 3+4 8+7-2 13+14-1=26 (用q[i]记录)
对于p[i] :相当于在p[i-1]的所有序列的末尾加上a[i]。
对于q[i] :p[i]加上1 ~ i-1的子序列个数。
/// author:Kissheart /// #include<stdio.h> #include<algorithm> #include<iostream> #include<string.h> #include<vector> #include<stdlib.h> #include<math.h> #include<queue> #include<deque> #include<ctype.h> #include<map> #include<set> #include<stack> #include<string> #define INF 0x3f3f3f3f #define FAST_IO ios::sync_with_stdio(false) const double PI = acos(-1.0); const double eps = 1e-6; const int MAX=1e6+10; const int mod=1e9+7; typedef long long ll; using namespace std; #define gcd(a,b) __gcd(a,b) inline ll lcm(ll a,ll b){return a/gcd(a,b)*b;} inline ll qpow(ll a,ll b){ll r=1,t=a; while(b){if(b&1)r=(r*t)%mod;b>>=1;t=(t*t)%mod;}return r;} inline ll inv1(ll b){return qpow(b,mod-2);} inline ll exgcd(ll a,ll b,ll &x,ll &y){if(!b){x=1;y=0;return a;}ll r=exgcd(b,a%b,y,x);y-=(a/b)*x;return r;} inline ll read(){ll x=0,f=1;char c=getchar();for(;!isdigit(c);c=getchar()) if(c==‘-‘) f=-1;for(;isdigit(c);c=getchar()) x=x*10+c-‘0‘;return x*f;} //freopen( "in.txt" , "r" , stdin ); //freopen( "data.txt" , "w" , stdout ); int n; int vis[MAX],a[MAX]; ll p[MAX],q[MAX]; int main() { ll ans=0; scanf("%d",&n); for(int i=1;i<=n;i++) scanf("%d",&a[i]); p[1]=1; q[1]=1; vis[a[1]]=1; for(int i=2;i<=n;i++) { ya if(!vis[a[i]]) { vis[a[i]]=i; p[i]=(q[i-1]+1)%mod; q[i]=(q[i-1]+p[i])%mod; } else { p[i]=(q[i-1]+1)%mod; q[i]=(q[i-1]+p[i])%mod; q[i]=((q[i]-p[vis[a[i]]])%mod+mod)%mod; //printf("%d %lld\n",vis[a[i]],p[vis[a[i]]]); vis[a[i]]=i; } //printf("%lld %lld\n",p[i],q[i]); } printf("%lld\n",q[n]+1); return 0; }
标签:本质 while scan git 图片 include 分享 printf pen
原文地址:https://www.cnblogs.com/Kissheart/p/9939719.html