补退选

Luogu

LOJ

BZOJ

比较裸。

建一棵Trie树,记录一下每个节点的\(sum\)表示经过该点的字符串个数,每次暴力插入、删除。

同时每个节点维护一个vector,记录一下这个点的\(sum\)第一次达到(超过)某个值的时间。

容易证明vector的总的元素个数是\(O(\sum|S|)\)的。总的复杂度为\(O(\sum|S|)\)。

#include<bits/stdc++.h>
using namespace std;
const int N=100007,M=60*N;
int read(){int x=0,c=getchar();while(!isdigit(c))c=getchar();while(isdigit(c))x=x*10+c-48,c=getchar();return x;}
int T=1,cnt,ans,sum[M],ch[M][10];vector<int>t[M];char s[61];
void update(int p)
{
if(sum[p]<=(int)t[p].size()) return ;
t[p].push_back(T);
}
void insert()
{
scanf("%s",s);int len=strlen(s),p=0;++sum[p],update(p);
for(int i=0,c;i<len;++i)
{
if(!ch[p][c=s[i]-'a']) ch[p][c]=++cnt;
++sum[p=ch[p][c]],update(p);
}
}
void remove()
{
scanf("%s",s);int len=strlen(s),p=0;--sum[p];
for(int i=0,c;i<len;++i) --sum[p=ch[p][c=s[i]-'a']];
}
void query()
{
scanf("%s",s);
int a=read(),b=read(),c=read(),len=strlen(s),p=0;
a=(1ll*a*abs(ans)+b)%c;
for(int i=0;i<len;++i) if(!(p=ch[p][s[i]-'a'])) return (void)(printf("%d\n",ans=-1));
if((int)t[p].size()<=a) return (void)(printf("%d\n",ans=-1));
printf("%d\n",ans=t[p][a]);
}
int main()
{
for(int m=read();T<=m;++T)
switch(read())
{
case 1:insert();break;
case 2:remove();break;
case 3:query();break;
}
}

成绩单

Luogu

LOJ

BZOJ

看到数据范围基本上区间dp没得跑了,这东西大概能够容忍\(O(n^5)\)的时间复杂度和\(O(n^4)\)的空间复杂度。(虽然感觉时间复杂度\(O(n^6)\)也完全没问题...)

那么我们就可以把有用的四个东西全部放进我们的状态。

设\(f_{l,r,x,y}\)表示\([l,r]\)区间,剩下的数在\([x,y]\)范围内(记得离散化)的最小代价。然后我们发现这东西不好表示区间全部被消掉的情况,而这正是我们需要求的,所以我们再开一个\(g_{l,r}\)表示消掉\([l,r]\)区间的最小代价。

考虑如何从\([l,r-1]\)扩展到\([l,r]\)。

针对\(w_r\)这一元素,我们有两种转移:(总的取\(\min\))

\(1.\)不管它。

\(f_{l,r,\min(w_r,x),\max(w_r,y)}\leftarrow f_{l,r-1,x,y}\)

\(2.\)在\([l,r-1]\)找一个后缀跟它一起消掉。

\(f_{l,r,x,y}\leftarrow f_{l,k,x,y}+g_{k+1,r}\ k\in[l,r)\)

同时显然有\(g_{l,r}=\min(f_{l,r,x,y}+(t_y-t_x)^2)\)。(\(t_x\)表示去重后\(x\)的原值)

最后的答案就是\(g_{1,n}\)。

注意转移顺序需要满足拓扑序,即先从小到大枚举区间长度再从左往右枚举左端点。

初值\(f_{i,i,w_i,w_i}=0,g_{i,i}=a\),其它的都是\(+\infty\)。

#include<bits/stdc++.h>
using namespace std;
int read(){int x;scanf("%d",&x);return x;}
int max(int a,int b){return a>b? a:b;}
int min(int a,int b){return a<b? a:b;}
int sqr(int x){return x*x;}
const int N=51;
int w[N],t[N],f[N][N][N][N],g[N][N];
int main()
{
int n=read(),m=n,a=read(),b=read();
for(int i=1;i<=n;++i) w[i]=t[i]=read();
sort(t+1,t+n+1),m=unique(t+1,t+n+1)-t-1,memset(f,0x3f,sizeof f),memset(g,0x3f,sizeof g);
for(int i=1;i<=n;++i) w[i]=lower_bound(t+1,t+m+1,w[i])-t,f[i][i][w[i]][w[i]]=0,g[i][i]=a;
for(int len=1,l,r,x,y,k;len<n;++len)
for(l=1,r=l+len;r<=n;++l,++r)
{
f[l][r][w[r]][w[r]]=g[l][r-1];
for(x=1;x<=m;++x) for(y=x;y<=m;++y) f[l][r][min(x,w[r])][max(y,w[r])]=min(f[l][r][min(x,w[r])][max(y,w[r])],f[l][r-1][x][y]);
for(k=l;k<r;++k) for(x=1;x<=m;++x) for(y=x;y<=m;++y) f[l][r][x][y]=min(f[l][r][x][y],f[l][k][x][y]+g[k+1][r]);
for(x=1;x<=m;++x) for(y=x;y<=m;++y) g[l][r]=min(g[l][r],f[l][r][x][y]+a+b*sqr(t[y]-t[x]));
}
printf("%d",g[1][n]);
}

星露谷物语

LOJ

THUSC2016的更多相关文章

  1. thusc2016游记&&滚粗记&&酱油记

    #include <cstdio> using namespace std; int main(){ puts("转载请注明出处:http://www.cnblogs.com/w ...

  2. THUSC2016 游记

    浑浑噩噩地就出发了,只记得可以翘课,不知道自己要干什么去. Day 0    5点起床,到潮汕机场坐飞机.第一次坐飞机非常不爽起飞和降落时的加速度……终于还是转转地铁.动车在下午4点左右抵达目的地,西 ...

  3. BZOJ4896 THUSC2016补退选(trie)

    字符串扔进trie,vector记录每个前缀出现次数的最大值的更新记录即可. #include<iostream> #include<cstdio> #include<c ...

  4. BZOJ4897 THUSC2016成绩单(区间dp)

    拿走一个区间的代价只与最大最小值有关,并且如果最后一次拿走包含区间右端点的子序列一定不会使答案更劣,于是设f[i][j][x][y]为使i~j区间剩余最小值为x最大值为y且若有数剩余一定包含j的最小代 ...

  5. [BZOJ4897][THUSC2016]成绩单(DP)

    4897: [Thu Summer Camp2016]成绩单 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 220  Solved: 132[Subm ...

  6. [BZOJ4896][THUSC2016]补退选(Trie)

    4896: [Thu Summer Camp2016]补退选 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 653  Solved: 228[Subm ...

  7. 【THUSC2016】成绩单(bzoj4897)

    $f(i,j,x,y)$ 表示区间 $[i,j]$中,第 $j$ 个数在最后一次操作中才消去,最后一次操作的最大值为 $x$,最小值为 $y$ 时的最小代价: $g(i,j)$ 表示区间 $[i,j] ...

  8. BZOJ 4896 [Thusc2016]补退选 (Trie树维护vector)

    题目大意:略 这竟然是$thusc$的题... 先把询问里加入的串全拎出来,建出$Trie$树,$Trie$里每个节点都开一个$vector$记录操作标号,再记录操作数量$sum$ 然后瞎**搞搞就行 ...

  9. 洛谷P5335 [THUSC2016]补退选 题解

    传送门 一道字典树的例题吧 先说下思路前1,2两个条件都易满足,字典树插入修改即可,第三个条件可用动态数组来实现,存下它的size表示当前有几个节点经过(即人数),其下标表示第几次出现,里面存入操作次 ...

  10. 【洛谷5335】[THUSC2016] 补退选(指针实现Trie)

    点此看题面 大致题意: 三种操作:加入一个字符串,删除一个字符串,求最早什么时候以某个字符串为前缀的字符串个数超过给定值. \(Trie\) 这道题显然是\(Trie\)的暴力裸题. 考虑我们对于\( ...

随机推荐

  1. Linux下更换为阿里yum源

    更新日期: 2018-08-06 1.yum源的工作原理 yum是为了解决安装包的依赖关系而生的,如果要源码安装一个软件,需要频繁下载各个包,并解决包的依赖关系.这就好比学门课程,要学会这门课程,就要 ...

  2. deep sort

    目录   1. 准备代码与数据 deep_sort开源代码 克隆到本地服务器 git clone https://github.com/nwojke/deep_sort.git 下载MOT16数据集( ...

  3. 使用<bind>元素创建变量

    在使用模糊查询sql时,如果使用${}进行字符拼接,无法防止sql诸如问题,如果使用concat函数则只对mysql有效果,用Oracle则需要用连接符||,这样在数据库变的时候需要修改,不利于移植. ...

  4. web工程中添加自建userLibary与将jar包放到lib文件夹下的区别

    纯 java项目 使用的本地自己的JRE,那么classLoader在加载jar和class时候是分开的,对于我们自己编写的class,会在 APP_HOME/bin下.导入的jar包或者user l ...

  5. vmware 中Linux系统怎么连接外网?

    VMware虚拟机有三种网络模式,分别是Bridged(桥接模式).NAT(网络地址转换模式).Host-only(主机模式). VMware workstation安装好之后会多出两个网络连接,分别 ...

  6. docker容器安装命令

    apt-get update apt-get install vim 转载请注明博客出处:http://www.cnblogs.com/cjh-notes/

  7. vue 动态渲染数据很慢或不渲染

    vue 动态渲染数据很慢或不渲染 原因是因为vue检测速度很慢,因为多层循环了,在VUE 2.x的时候还能渲染出来,1.x的时候压根渲染不出来.解决方式:在动态改变数据的方法,第一行加上 this.$ ...

  8. Graphics 使用一点点注意

    Form_Load 事件下绘制的结果会被 paint 刷新掉.也就等于没有绘制一样. Graphics g = this.CreateGraphics(); g.DrawRectangle(new P ...

  9. Python3+RobotFramewok 循环判断以及Evaluate用法(三)

    本章主要介绍RF的循环,判断以及关键字Evaluate. 1. for循环 在RF中通过 :FOR 编写循环 :FOR ${i} in range 10 log ${i} @{list} create ...

  10. 如何修改Zend Studio代码字体和大小

    Zend Studio的默认字体非常小,看起来很费神,这里教大家怎么修改它的字体和大小.   工具/原料   Zend Studio 方法/步骤     Window-->preferences ...