【题解】子序列个数 [51nod1202] [FZU2129]

传送门:子序列个数 \([51nod1202]\) \([FZU2129]\)

【题目描述】

对于给出长度为 \(n\) 的一个序列 \(a\),求出不同的非空子序列个数。答案对 \(10^9+7\) 取模。

【样例】

样例输入:
4
1
2
3
2 样例输出:
13

【数据范围】

\(100\%\) \(1 \leqslant N,a[i] \leqslant 10^5\)


【分析】

先考虑没有相同整数的情况,每个元素有选或不选两种情况,一共 \(n\) 个元素,又不能有空集,答案为 \(2^n-1\)。

如果要写递推方程的话就是 \(dp[i]=dp[i-1]*2+1\),其中 \(dp[i]\) 表示选择原序列中前 \(i\) 个元素所能构成的不同子序列数量,\(dp\) 方程含义为:在长度为 \(i-1\) 的序列中是否加入一个 \(a[i]\) \((\) \(2*dp[i-1]\) 种 \()\) 以及只选 \(a[i]\) 的情况 \((\) \(1\) 种 \()\) 。

假设就按照这个方程推下去,如果前面某一位 \(j\) 上的数与 \(a[i]\) 相同,会对 \(dp[i]\) 造成什么影响呢?

首先,只选 \(a[i]\) 这一个数的情况已经在前面统计过了。

然后,\(j\) 前面的所有方案都不能转移到 \(dp[i]\) 上面来,因为它们已经转移到 \(dp[j]\) 上。

这里的 \(“\) 所有方案 \(”\) 其实就是 \(dp[j-1]\),且 \(j\) 必须为最接近 \(i\) 的那一个位置,否则就不能代表所有被算重复的情况

用 \(w[a[i]]\) 表示前面等于 \(a[i]\) 的且位置最靠近 \(i\) 的数的位置,若不存在则 \(w[a[i]]=0\),\(dp\) 方程为:

\(dp[i]=\begin{cases} dp[i-1]*2+1 & w[a[i]]=0\\dp[i-1]*2+1-dp[w[a[i]]-1]-1 & w[a[i]]!=0 \end{cases}\)

时间复杂度为:\(O(n)\) 。


【Code】

#include<algorithm>
#include<cstring>
#include<cstdio>
#define Re register int
const int N=1e6+3,P=1e9+7;
int n,ans,a[N],W[N],dp[N];
inline void in(Re &x){
int f=0;x=0;char c=getchar();
while(c<'0'||c>'9')f|=c=='-',c=getchar();
while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+(c^48),c=getchar();
x=f?-x:x;
}
int main(){
while(scanf("%d",&n)!=EOF){
memset(dp,0,sizeof(dp));
memset(W,0,sizeof(W));
for(Re i=1;i<=n;++i){
in(a[i]);
dp[i]=((dp[i-1]<<1)%P+1)%P;
if(W[a[i]])((dp[i]-=(dp[W[a[i]]-1]+1)%P)+=P)%=P;
W[a[i]]=i;
}
printf("%d\n",dp[n]%P);
}
}

【题解】子序列个数 [51nod1202] [FZU2129]的更多相关文章

  1. 子序列个数(fzu2129)

    子序列个数 Time Limit:2000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit Status ...

  2. 51nod1202 子序列个数

    看到a[i]<=100000觉得应该从这个方面搞.如果a[x]没出现过,f[x]=f[x-1]*2;否则f[x]=f[x-1]*2-f[pos[a[x]]-1];ans=f[n]-1,然后WA了 ...

  3. 「 LuoguT37042」 求子序列个数

    Description 给定序列 A, 求出 A 中本质不同的子序列 (包含空的子序列) 个数模 10^9+ 7 的结果. 一个序列 B 是 A 的子序列需要满足 A 删掉某些元素后能够得到 B. 两 ...

  4. hdu4632 Palindrome subsequence 回文子序列个数 区间dp

    Palindrome subsequence Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/65535 K (Java/ ...

  5. fzuoj Problem 2129 子序列个数

    http://acm.fzu.edu.cn/problem.php?pid=2129 Problem 2129 子序列个数 Accept: 162    Submit: 491Time Limit: ...

  6. FZU 2129 子序列个数 (递推dp)

    题目链接:http://acm.fzu.edu.cn/problem.php?pid=2129 dp[i]表示前i个数的子序列个数 当a[i]在i以前出现过,dp[i] = dp[i - 1]*2 - ...

  7. 51nod 1202 子序列个数

    1202 子序列个数  题目来源: 福州大学 OJ 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题  收藏  关注 子序列的定义:对于一个序列a=a[1],a[2] ...

  8. 1202 子序列个数(DP)

    1202 子序列个数 题目来源: 福州大学 OJ 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 子序列的定义:对于一个序列a=a[1],a[2],......a[ ...

  9. 51nod 1202 不同子序列个数 [计数DP]

    1202 子序列个数 题目来源: 福州大学 OJ 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 收藏 关注 子序列的定义:对于一个序列a=a[1],a[2],.. ...

随机推荐

  1. C# Net 去除图片白边

    代码根据别人的进行改写,效果更好 直接拷贝使用 名称空间: using System.Drawing; 代码: /// <summary> /// 裁剪图片(去掉百边) /// </ ...

  2. axios用法全解

    [前言] 本文介绍下axios用法,希望对大家有所帮助 这里声明一句:请求数据一般放置到哪里?详见下篇文章 [主体] (1)下载 npm i axios --save (2)引入axios模块 方式1 ...

  3. 洛谷P1192-台阶问题(线性递推 扩展斐波那契)

    占坑 先贴上AC代码 回头来补坑 #include <iostream> using namespace std; int n, k; const int mod = 100003; lo ...

  4. 禁用wordpress模板默认样式

    我们知道wordpress主题比如twentytwenty都会有样式,如果不想使用它们的默认样式怎么处理呢?其实很简单,随ytkah一起来看看吧.进入2020主题的function.php文件,里面有 ...

  5. HTML基础二-DOM操作

    http://www.imdsx.cn/index.php/2017/07/27/html2/ DOM(Document Object Model 文档对象模型) 一个web页面的展示,是由html标 ...

  6. q1095

    一,写题 1,我这个递归的错误我挺想搞出来的 int fa(int x) { ) return cnt; ==) { x=x/; cout<<"测试1:"<< ...

  7. Native Clojure with GraalVM

    转自:https://www.innoq.com/en/blog/native-clojure-and-graalvm/ GraalVM is a fascinating piece of techn ...

  8. ACE在Ubuntu下的安装和编译

    之前写了很多linux下的底层网络API的demo,这些demo可用于了解底层的网络通信过程,但是想做出好的服务器用于实际业务还是非常困难的,需要大量的代码实现,移植性也非常差,想要写出高性能架构的服 ...

  9. js中的自动包装机制

    有一行语句: 1.toString() //error 那么问题来了,为什么报错? 我的第一想法就是1是一个基本类型,所以它没有toString方法 嗯,理直气壮 接下来有大佬show me code ...

  10. android studio 配置相关问题

    1:Unknown AVD name, use -list-avds to see valid list. 下午1:26    Emulator: emulator: ERROR: Unknown A ...