20210821 打表,蛇,购物,ants
考场
T1 没看懂
T4 一眼回滚莫队,但忘记怎么写了,小慌
模拟 T1 题意的时候教练让 zsy 澄清了一下,确定了我不会做。。。
T2 一看就是毒瘤题,T3 感觉比较可做
T4 确定了回滚的细节,除了板子不会写其他都会了
7.40 开写,1h 后写完前三题暴力
先回想了一下普通莫队和回滚的思想,结合题目具体确定了实现方法,9.00 开写,比想象中顺利多了,9.40 过样例,10.00 过拍
剩下的时间没事干,先把 T3 的两个部分分写了,然后给 T1 加了个记忆化,对着数据卡常(一开始要 7s),随机的极限数据能在 0.8~1.1s 内跑过,记忆化后有点像状压 DP,但没啥时间了,就没细想。(卡常期间想起 T4 就是区间 mex,貌似可以主席树,感觉要被全场切了)
交题的时候发现 T1 还有个地方可以卡,但想着不差这点,且最后 10min 不想再改,就放弃了
res
rk1 95+65+70+100
T1 真被卡了 5ms
rk2 gjy 50+100+50+50
rk3 付文暄 50+65+100+20
总结
学东西的时候一定要注重思想,并想清楚代码为什么这么实现,这样即使忘了也能现推
发现自己心态还是不太好,如果有会做的非一眼题就比较正常,该拼部分分拼部分分,该卡常卡常;但如果一个题都不会,就很烦,暴力懒得想优化,部分分稍微想想就丢了,然后肝题到快结束,最后打 tetris。
最近倒是不打 tetris 了,但还是会心态爆炸,考场不是自闭的地方,即使啥都不会也要打满暴力,想不出题就优化暴力,卡常,不要把考场的时间用来发呆。
改题时发现对于拼接 hash 这种细节很多的东西很容易调不出来,以后遇到这类问题先在草稿纸上列好,尤其是边界,不要写完之后再改,效率太低。
打表
考场做法:考虑先枚举出每次由哪个 CPU 决策(称为 CPU 的决策顺序),在 dfs 出它当前这位填 \(0\) 还是 \(1\),记忆化未决策的 CPU 和当前已填位状态即可。
正确性:显然不论哪个 CPU 做决策,选择的是同一位,但填的数相反(可以用反证法证明),因此被选择的位的顺序是固定的(称为最优顺序)。先强制令每次决策时决策剩下的最高位,那么对于一个 CPU 的决策序列,可以将它的顺序掉换成最优顺序,由于所有 CPU 的决策顺序都会被枚举到,那么这么做是等价的。
(其实到这里就很接近正解了, 但考场上没多想)
code
const int mod = 1e9+7;
int n,m,a[1<<18];
int all,ans;
unordered_map<int,int> f[1<<19];
int Pow(int x,int y=mod-2)
{ int res=1; for(;y;y>>=1,x=(LL)x*x%mod)if(y&1)res=(LL)res*x%mod; return res; }
int dfs(int u,int cpu,int p) {
if( u > n ) return abs(a[p]-a[m]);
if( f[cpu].count(p) ) return f[cpu][p];
int res0 = dfs(u+1,cpu>>1,p<<1), res1 = dfs(u+1,cpu>>1,p<<1|1);
return f[cpu][p] = cpu&1 ? min(res0,res1) : max(res0,res1);
}
signed main() {
read(n,m); all = (1<<n--)-1;
For(i,0,all) read(a[i]);
for(int i = 1<<n+1, ii = i|all; i <= ii; ++i) ans = (ans + dfs(0,i,0)) %mod;
write((LL)ans*Pow(1<<n+1)%mod);
return iocl();
}
蛇
只考虑蛇向右的情况,它的路径一定是先向左走一段(可能为 \(0\)),再调头向右,然后上下扭动地走,最后再调头向左。
考虑 hash 求出路径的左右的部分,然后中间的路径数可以 DP。(需要特判 \(|S|\le2\) 的情况)
细节很多。
code
const int N = 2e3+5, mod = 1e9+7, dx[]={0,-1,0,1}, dy[]={-1,0,1,0};
char s[2][N],t[N];
const LL P = 999999001;
int n,m;
LL ans,f[2][N][N];
LL base[N],hs[4][N],ht[2][N];
void ckadd(LL &x,LL y) { x+=y; if(x>=mod) x-=mod; }
namespace sub {
bool vis[2][N];
void dfs(int x,int y,int u) {
if( u == m ) return ++ans, void();
For(i,0,3) {
int xx = x+dx[i], yy = y+dy[i];
if( xx<0||xx>1||yy<1||yy>n || vis[xx][yy] || s[xx][yy] != t[u+1] )
continue;
vis[xx][yy] = 1, dfs(xx,yy,u+1), vis[xx][yy] = 0;
}
}
void main() {
if( m > 2 ) return; cerr<<"sub"<<endl;
For(i,0,1) For(j,1,n) if( s[i][j] == t[1] )
vis[i][j] = 1, dfs(i,j,1), vis[i][j] = 0;
write(ans%mod);
iocl(), exit(0);
}
}
LL hsh(LL *h,int l,int r) { return (h[r] - h[l-1]*base[r-l+1] %P+P)%P; }
LL rhsh(LL *h,int l,int r) { return (h[l] - h[r+1]*base[r-l+1] %P+P)%P; }
void work() {
memset(f,0,sizeof f);
For(i,1,n) hs[0][i] = (hs[0][i-1] * 997 + s[0][i]) %P, // 1为高位
hs[1][i] = (hs[1][i-1] * 997 + s[1][i]) %P;
rFor(i,n,1) hs[2][i] = (hs[2][i+1] * 997 + s[0][i]) %P, // n为高位
hs[3][i] = (hs[3][i+1] * 997 + s[1][i]) %P;
For(i,1,m) ht[0][i] = (ht[0][i-1] + t[i] * base[i-1]) %P;
rFor(i,m,1) ht[1][i] = (ht[1][i+1] * 997 + t[i]) %P;
For(i,1,n) {
f[0][i][1] = s[0][i]==t[1], f[1][i][1] = s[1][i]==t[1];
For(j,1,i) {
int len = i-j+1;
f[1][i][len*2] =
(hsh(hs[0],j,i)+rhsh(hs[3],j,i)*base[len])%P == ht[0][len*2];
f[0][i][len*2] =
(hsh(hs[1],j,i)+rhsh(hs[2],j,i)*base[len])%P == ht[0][len*2];
}
}
For(i,1,n) For(j,2,m) {
if( s[0][i] == t[j] ) {
ckadd(f[0][i][j],f[0][i-1][j-1]);
if( s[1][i] == t[j-1] ) ckadd(f[0][i][j],f[1][i-1][j-2]);
}
if( s[1][i] == t[j] ) {
ckadd(f[1][i][j],f[1][i-1][j-1]);
if( s[0][i] == t[j-1] ) ckadd(f[1][i][j],f[0][i-1][j-2]);
}
}
For(i,1,n) ckadd(ans,f[0][i][m]), ckadd(ans,f[1][i][m]);
For(i,1,n) rFor(j,i-1,1) {
int len = i-j+1; if( len*2 > m ) break;
if( (hsh(hs[0],j,i)*base[len]+rhsh(hs[3],j,i))%P == ht[1][m-len*2+1] )
ckadd(ans,f[1][j-1][m-len*2]);
if( (hsh(hs[1],j,i)*base[len]+rhsh(hs[2],j,i))%P == ht[1][m-len*2+1] )
ckadd(ans,f[0][j-1][m-len*2]);
}
}
signed main() {
base[0] = 1; For(i,1,2e3) base[i] = base[i-1] * 997 %P;
scanf("%s%s%s",s[0]+1,s[1]+1,t+1); n = strlen(s[0]+1), m = strlen(t+1);
sub::main();
work(), reverse(s[0]+1,s[0]+n+1), reverse(s[1]+1,s[1]+n+1), work();
write(ans);
return iocl();
}
购物
显然如果 \(a_i\) 能凑出 \(x\),那么区间 \([\lceil\frac x2\rceil,x]\) 中的 \(k\) 都合法。
结论:将 \(a\) 排序,记 \(s_i\) 为 \(a_i\) 前缀和,那么 \((1,s_n)\) 中不合法的 \(k\) 就是 \([s_{i-1},\frac{a_i}2]\)(这个区间如果为空则不考虑)的并集。
证明:
已处理完了 \([1,s_{i-1}]\),当前处理的区间 \((s_{i-1},\frac{a_i}2)\)。如果这个区间为空显然正确;如果不为空,则前 \(i-1\) 个数最多只能凑到 \(s_{i-1}\),而第 \(i\) 个数之后的数都 \(\ge a_i\),最少也是 \(\frac{a_i}2\),因此这个区间是不可能被补上的。
再考虑证明 \([\frac{a_i}2,s_i]\) 一定合法。如果前面没有空隙显然合法;如果有,那么空隙大小一定 \(\le\frac{a_i}2\)(前面的总长只有 \(\frac{a_i}2\)),设空隙为 \((l,r)\),则将 \(l\) 左边、\(r\) 右边的区间加 \(a_i\) (总和加 \(a_i\),左端点加 \(\frac{a_i}2\),右端点加 \(a_i\))后这个区间变为 \((l+a_i,r+\frac{a_i}2)\),一定为空。
code
const int N = 1e5+5;
int n,a[N];
LL ans,s[N];
signed main() {
read(n);
For(i,1,n) read(a[i]); sort(a+1,a+n+1);
For(i,1,n) {
s[i] = s[i-1] + a[i];
ans -= max((a[i]+1)/2-s[i-1]-1,0ll);
}
write(ans+s[n]);
return iocl();
}
ants
裸的回滚莫队
值域上每个点维护它向左、向右能连续到哪,插入 \(x\) 时与 \(x-1,x+1\) 合并,每次有三个点需要回滚。
考场代码
const int N = 1e5+5;
int n,m,a[N];
struct Q { int l,r,id; } q[N];
int size,mm,now,be[N],le[N],ri[N],ans[N];
bool vis[N];
int tp; struct Node { int p,v,l,r; } stk[N*3];
bool operator < (const Q &x,const Q &y)
{ return be[x.l]!=be[y.l] ? x.l<y.l : x.r<y.r; }
void push(int x) { stk[++tp] = Node{x,vis[x],le[x],ri[x]}; }
void add(int x,bool op) {
int l = le[x-1], r = ri[x+1];
if( vis[x-1] ) { if( op ) push(l); } else l = x;
if( vis[x+1] ) { if( op ) push(r); } else r = x;
if( op ) push(x);
vis[x] = 1, le[x] = l, ri[x] = r, ri[l] = r, le[r] = l;
ckmax(now,ri[x]-le[x]+1);
}
void clear() {
while( tp ) {
auto u = stk[tp--];
vis[u.p] = u.v, le[u.p] = u.l, ri[u.p] = u.r;
}
}
int bf(int l,int r) {
For(i,l,r) add(a[i],1);
int res = now;
now = 0, clear();
return res;
}
signed main() {
// freopen("d.in","r",stdin);
// freopen("d.out","w",stdout);
read(n,m); size = sqrt(n);
For(i,1,n) read(a[i]), be[i] = (i-1)/size+1, le[i] = ri[i] = i;
For(i,1,m) {
int l,r; read(l,r);
if( be[l] == be[r] ) ans[i] = bf(l,r);
else q[++mm] = Q{l,r,i};
}
sort(q+1,q+mm+1);
for(int i = 1, r; i <= mm; ++i) {
if( be[q[i].l] != be[q[i-1].l] ) {
now = 0, r = size * be[q[i].l];
For(i,1,n) vis[i] = 0, le[i] = ri[i] = i;
}
while( r < q[i].r ) add(a[++r],0);
int l = size * be[q[i].l] + 1, old = now;
while( l > q[i].l ) add(a[--l],1);
ans[q[i].id] = now;
now = old, clear();
}
For(i,1,m) write(ans[i]);
return iocl();
}
20210821 打表,蛇,购物,ants的更多相关文章
- 基于j2ee的程序代写MVC架构
人力资源管理系统 完成系统静态页面设计,页面数量不少于10个,页面需用CSS进行美化,并为需要验证的信息利用JavaScript提供客户端验证.要求至少包含部门信息及部门内员工信息的添加.修改.删除和 ...
- eShopOnContainers 知多少[4]:Catalog microservice
引言 Catalog microservice(目录微服务)维护着所有产品信息,包括库存.价格.所以该微服务的核心业务为: 产品信息的维护 库存的更新 价格的维护 架构模式 如上图所示,本微服务采用简 ...
- 5.sql2008分组与嵌套
1.Group by基本介绍;2.Having的使用;3.分组综合应用;4.子查询基本介绍;5.In/Exists/Any/Some/All;6.子查询综合应用; 1.Group by基本介绍:依据B ...
- 何时使用SUM()与SUMX()
概述 SUM()是一个聚合函数.在应用将影响公式的所有过滤器后,它会将您指定的单个列中的所有值相加.SUM()不知道行的存在(它不能逐行求值) - 它所能做的就是在应用过滤器之后将所有内容添加到它所呈 ...
- 告诉你 SQL 数据库与 NoSQL 数据库的区别
简单来说 SQL 数据库和 NoSQL 数据库有着共同的目标:存储数据,但存储的方式不同 一. 表 SQL中的表结构具有严格的数据模式约束: 存储数据很难出错. NoSQL存储数据更加灵活自由:可能导 ...
- 数据库系列(五)之 mysql的伸缩性
这篇文章,主要讲述mysql的伸缩性.在国内mysql一直都是使用得最多的数据库,在国外也排名前三.mysql是一款开源的.性能较高的数据库. 伸缩性是指在软件设计中,软件(数据库.应用程序)通过特定 ...
- .NET 微服务 2 架构设计理论(一)
SOA体系架构 面向服务的体系结构 (SOA) ,通过将应用程序分解为多个服务(通常为 HTTP 服务,WCF服务等),将其分为不同类型(例如子系统或层),从而来划分应用程序的结构. 微服务源自 SO ...
- 十四自定义构建购物计算组件&表单组件
目录: 1.前言 2.组件介绍 3.js业务逻辑层 4.视图层 5.css属性设置 6.效果呈现 1.前言: 在第九篇文章购物车做好后,还忘记了一个至关重要的计算组件.在鸿蒙的组件中并没有提供这样一个 ...
- HTML表单特别效果—音量调节,购物数量
<form oninput="x.value=parseInt(a.value)+parseInt(b.value)">0<input type="ra ...
随机推荐
- 题解 P6892 [ICPC2014 WF]Baggage
解题思路 非常好的一道构造题. 在手动模拟几个样例(也许不止几个)之后呢. 就可以发现其实这些操作的开始以及最后几步是有相通之处的. 关于手动模拟的样例放在了文章末尾,需要的自取. 先考虑操作次数. ...
- D. 旅游景点 Tourist Attractions 状压DP
题目描述 FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺序不是完全随意的,比如说FGD 不希望在刚吃过一顿大餐之后立刻去下一 ...
- [解决方案]docker: Error response from daemon: OCI runtime create failed
错误原因 在新服务器上安装好docker后,发现无法运行,经常一顿搜索后,发现是docker安装的版本过高,最新版本docker-18.06 的核心好像没有经过充分的测试就发布了. 导致一运行,就提示 ...
- 卷向字节码-Java异常到底是怎么被处理的?
你好呀,我是why,你也可以叫我歪歪. 比如下面这位读者: 他是看了我<神了!异常信息突然就没了?>这篇文章后产生的疑问. 既然是看了我的文章带来的进一步思考,恰巧呢,我又刚好知道. 虽然 ...
- no-strings-attached writeup
no-strings-attach writeup 1.程序分析 主函数如图所示,关键函数在authenticate中,进入函数. 分析可得,decrypt代码段为关键代码段,进入关键函数decryp ...
- RHCSA_DAY02
Linux:一切皆文件 分区:/boot:做引导盘 /swap:虚拟内存----最大20gb /data:自己放文件用 /:根分区 - 图形界面: - Ctrl+Shift +号 //调整命令 ...
- 痞子衡嵌入式:在IAR开发环境下将关键函数重定向到RAM中执行的三种方法
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是在IAR开发环境下将关键函数重定向到RAM中执行的三种方法. 嵌入式项目里应用程序代码正常是放在 Flash 中执行的,但有时候也需要将 ...
- Golang语言系列-06-map数据类型和指针
Map数据类型和指针 Map数据类型 Map基本概念 package main import "fmt" // map // make()函数和new()函数的区别 // make ...
- luogu P6239 奇怪的道路
奇怪的道路 我看不出来是状压的状压 好吧,其实看到k的范围应该去往状压方面想的. 然后,题目中说"任何一个城市都与恰好偶数条道路相连(0也被认为是偶数)". 所以,奇偶,两种状态可 ...
- 题解 block
传送门 如果不想让next_permutation()自动忽略重复元素,可以在比较函数里加个rk之类的东西使它们不同(next_permutation()不用等于号) 关于第一问:貌似也是一个挺常见的 ...