CodeForces1051E EXKMP + 线段树dp
http://codeforces.com/problemset/problem/1051/E
题意:给你一个很大的数字,然后你可以把这个数字拆分成为任意多个部分,要求每一个部分的数字大小要在一个区间内,问有多少种拆分方式。
很容易看出这是一个dp,用dp[i]表示到i之前位置总共的数量,再用l[i]和r[i]表示i位置到l和r区间内的字符串全都满足上下限的条件,将dp[i - 1]加到l到r上更新即可,是一个很显然的n²dp,当然n²是不可能的,随随便便搞个数据结构进行一下区间修改就行了,这里用的是线段树。
问题就给到了预处理l[i]和r[i]这个问题上,暴力预处理又是一个n²的操作,考虑到大数比较事实上是去掉最大公共前缀之后比较下一位即可,可以容易的想到用EXKMP去处理下就好了。
(线段树一开始就开了0 ~ N,RE了一个小时)
#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
inline int read(){int now=;register char c=getchar();for(;!isdigit(c);c=getchar());
for(;isdigit(c);now=now*+c-'',c=getchar());return now;}
#define For(i, x, y) for(int i=x;i<=y;i++)
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Sca3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define Scl(x) scanf("%lld",&x);
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second
typedef vector<int> VI;
const double eps = 1e-;
const int maxn = 1e6 + ;
const int INF = 0x3f3f3f3f;
const LL mod = ;
char a[maxn],down[maxn],up[maxn];
int l[maxn],r[maxn];
int nxt[maxn],edown[maxn],eup[maxn];
void pre_EKMP(char x[],int m,int next[]){
next[] = m;
int j = ;
while(j + < m && x[j] == x[j + ]) j++;
next[] = j;
int k = ;
for(int i = ; i < m ; i ++){
int p = next[k] + k - ;
int L = next[i - k];
if(i + L < p + ) next[i] = L;
else{
j = max(,p - i + );
while(i + j < m && x[i + j] == x[j]) j ++;
next[i] = j;
k = i;
}
}
}
void EKMP(char x[],int m,char y[],int n,int next[],int extend[]){
pre_EKMP(x,m,next);
int j = ;
while(j < n && j < m && x[j] == y[j]) j ++;
extend[] = j;
int k = ;
for(int i = ; i < n ; i ++){
int p = extend[k] + k - ;
int L = next[i - k];
if(i + L < p + ) extend[i] = L;
else{
j = max(,p - i + );
while(i + j < n && j < m && y[i + j] == x[j]) j ++;
extend[i] = j;
k = i;
}
}
}
struct Tree{
int l,r;
LL lazy;
}tree[maxn << ];
void Build(int t,int l,int r){
if(l > r) exit();
tree[t].l = l; tree[t].r = r;
tree[t].lazy = ;
if(l == r) return;
int m = (l + r) >> ;
Build(t << ,l,m); Build(t << | ,m + ,r);
}
void Pushdown(int t){
if(tree[t].lazy){
tree[t << ].lazy = (tree[t << ].lazy + tree[t].lazy) % mod;
tree[t << | ].lazy = (tree[t << | ].lazy + tree[t].lazy) % mod;
tree[t].lazy = ;
}
}
void update(int t,int l,int r,LL v){
if(l <= tree[t].l && tree[t].r <= r){
tree[t].lazy = (v + tree[t].lazy) % mod;
return;
}
Pushdown(t);
int m = (tree[t].l + tree[t].r) >> ;
if(r <= m) update(t << ,l,r,v);
else if(l > m) update(t << | ,l,r,v);
else{
update(t << ,l,m,v);
update(t << | ,m + ,r,v);
}
}
LL query(int t,int p){
if(tree[t].l >= tree[t].r) return tree[t].lazy % mod;
Pushdown(t);
int m = (tree[t].l + tree[t].r) >> ;
if(p <= m) return query(t << ,p);
else return query(t << | ,p);
}
int main()
{
scanf("%s%s%s",a,down,up);
int N = strlen(a);
int l1 = strlen(down),l2 = strlen(up);
EKMP(down,l1,a,N,nxt,edown);
EKMP(up,l2,a,N,nxt,eup);
for(int i = ; i <= N ; i ++){
l[i] = i + l1 - ,r[i] = i + l2 - ;
if(a[i - ] == ''){
if(down[] == ''){
l[i] = r[i] = i;
}else{
l[i] = ,r[i] = ;
}
continue;
}
int len = edown[i - ];
if((len < l1) && down[len] > a[i + len - ]) l[i]++;
len = eup[i - ];
if((len < l2) && up[len] < a[i + len - ]) r[i]--;
}
Build(,,N * ); update(,,,);
for(int i = ; i <= N ; i ++){
if(l[i] > r[i]) continue;
LL x = query(,i - );
update(,l[i],r[i],x);
}
Prl(query(,N));
return ;
}
CodeForces1051E EXKMP + 线段树dp的更多相关文章
- Tsinsen A1219. 采矿(陈许旻) (树链剖分,线段树 + DP)
[题目链接] http://www.tsinsen.com/A1219 [题意] 给定一棵树,a[u][i]代表u结点分配i人的收益,可以随时改变a[u],查询(u,v)代表在u子树的所有节点,在u- ...
- HDU 3016 Man Down (线段树+dp)
HDU 3016 Man Down (线段树+dp) Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Ja ...
- lightoj1085 线段树+dp
//Accepted 7552 KB 844 ms //dp[i]=sum(dp[j])+1 j<i && a[j]<a[i] //可以用线段树求所用小于a[i]的dp[j ...
- [CF 474E] Pillars (线段树+dp)
题目链接:http://codeforces.com/contest/474/problem/F 意思是给你两个数n和d,下面给你n座山的高度. 一个人任意选择一座山作为起始点,向右跳,但是只能跳到高 ...
- HDU-3872 Dragon Ball 线段树+DP
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3872 题意:有n个龙珠按顺序放在一列,每个龙珠有一个type和一个权值,要求你把这n个龙珠分成k个段, ...
- HDU4521+线段树+dp
题意:在一个序列中找出最长的某个序列.找出的序列满足题中的条件. 关键:对于 第 i 个位置上的数,要知道与之相隔至少d的位置上的数的大小.可以利用线段树进行统计,查询.更新的时候利用dp的思想. / ...
- Codeforces Round #343 (Div. 2) D - Babaei and Birthday Cake 线段树+DP
题意:做蛋糕,给出N个半径,和高的圆柱,要求后面的体积比前面大的可以堆在前一个的上面,求最大的体积和. 思路:首先离散化蛋糕体积,以蛋糕数量建树建树,每个节点维护最大值,也就是假如节点i放在最上层情况 ...
- Special Subsequence(离散化线段树+dp)
Special Subsequence Time Limit: 5 Seconds Memory Limit: 32768 KB There a sequence S with n inte ...
- hdu 4117 GRE Words (ac自动机 线段树 dp)
参考:http://blog.csdn.net/no__stop/article/details/12287843 此题利用了ac自动机fail树的性质,fail指针建立为树,表示父节点是孩子节点的后 ...
随机推荐
- CSS3 flexbox 布局 ---- flex 容器属性介绍
flexbox布局是CSS3中新增的属性,它可以很轻松地帮我们解决掉一些常见的布局问题,比如导航栏. 我们用普通的方法写导航栏,通常会在ul, li 结构写好后,让li 元素左浮动,然后再给ul 清浮 ...
- vuejs 单文件组件.vue 文件
vuejs 自定义了一种.vue文件,可以把html, css, js 写到一个文件中,从而实现了对一个组件的封装, 一个.vue 文件就是一个单独的组件.由于.vue文件是自定义的,浏览器不认识,所 ...
- bzoj2152-[国家集训队]聪聪可可
Description 聪聪和可可是兄弟俩,他们俩经常为了一些琐事打起来,例如家中只剩下最后一根冰棍而两人都想吃.两个人都想玩儿电脑(可是他们家只有一台电脑)--遇到这种问题,一般情况下石头剪刀布就好 ...
- Spring01-Ioc基本使用
一. Spring简介 1. Spring介绍 Spring框架主页: Spring官网 Spring资源地址:下载地址 Spring框架,由Rod Johnson开发 Spring是一个非常活跃的开 ...
- POJ 2449 Remmarguts' Date (算竞进阶习题)
A* + dijkstra/spfa 第K短路的模板题,就是直接把最短路当成估价函数,保证估价函数的性质(从当前状态转移的估计值一定不大于实际值) 我们建反图从终点跑最短路,就能求出从各个点到终点的最 ...
- Codeforces379 F. New Year Tree
Codeforces题号:#379F 出处: Codeforces 主要算法:LCA+树的直径 难度:4.4 思路分析: 给出q个操作,每次在一个节点上接上两个叶子.每一次询问树的直径. 暴力做法:每 ...
- 【BZOJ3771】Triple 生成函数 FFT 容斥原理
题目大意 有\(n\)把斧头,不同斧头的价值都不同且都是\([0,m]\)的整数.你可以选\(1\)~\(3\)把斧头,总价值为这三把斧头的价值之和.请你对于每种可能的总价值,求出有多少种选择方案. ...
- 【BZOJ3522】【BZOJ4543】【POI2014】Hotel 树形DP 长链剖分 启发式合并
题目大意 给你一棵树,求有多少个组点满足\(x\neq y,x\neq z,y\neq z,dist_{x,y}=dist_{x,z}=dist_{y,z}\) \(1\leq n\leq 1 ...
- github 快速部署
在github上 新建一个项目后,并且未提交任何代码,会有一个页面提示我们如何快速部署.在此备份一下那个页面 Quick setup — if you’ve done this kind of thi ...
- 使用ROME解析rss,如何获取icon图标
问题 网站使用SpringMVC框架,实现了RSS订阅,但是在阅读器里面,只有网站的标题,没有网站的图标. 分析 ROME会从<channel>标签下指定的<link>地址中, ...