考场

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的更多相关文章

  1. 基于j2ee的程序代写MVC架构

    人力资源管理系统 完成系统静态页面设计,页面数量不少于10个,页面需用CSS进行美化,并为需要验证的信息利用JavaScript提供客户端验证.要求至少包含部门信息及部门内员工信息的添加.修改.删除和 ...

  2. eShopOnContainers 知多少[4]:Catalog microservice

    引言 Catalog microservice(目录微服务)维护着所有产品信息,包括库存.价格.所以该微服务的核心业务为: 产品信息的维护 库存的更新 价格的维护 架构模式 如上图所示,本微服务采用简 ...

  3. 5.sql2008分组与嵌套

    1.Group by基本介绍;2.Having的使用;3.分组综合应用;4.子查询基本介绍;5.In/Exists/Any/Some/All;6.子查询综合应用; 1.Group by基本介绍:依据B ...

  4. 何时使用SUM()与SUMX()

    概述 SUM()是一个聚合函数.在应用将影响公式的所有过滤器后,它会将您指定的单个列中的所有值相加.SUM()不知道行的存在(它不能逐行求值) - 它所能做的就是在应用过滤器之后将所有内容添加到它所呈 ...

  5. 告诉你 SQL 数据库与 NoSQL 数据库的区别

    简单来说 SQL 数据库和 NoSQL 数据库有着共同的目标:存储数据,但存储的方式不同 一. 表 SQL中的表结构具有严格的数据模式约束: 存储数据很难出错. NoSQL存储数据更加灵活自由:可能导 ...

  6. 数据库系列(五)之 mysql的伸缩性

    这篇文章,主要讲述mysql的伸缩性.在国内mysql一直都是使用得最多的数据库,在国外也排名前三.mysql是一款开源的.性能较高的数据库. 伸缩性是指在软件设计中,软件(数据库.应用程序)通过特定 ...

  7. .NET 微服务 2 架构设计理论(一)

    SOA体系架构 面向服务的体系结构 (SOA) ,通过将应用程序分解为多个服务(通常为 HTTP 服务,WCF服务等),将其分为不同类型(例如子系统或层),从而来划分应用程序的结构. 微服务源自 SO ...

  8. 十四自定义构建购物计算组件&表单组件

    目录: 1.前言 2.组件介绍 3.js业务逻辑层 4.视图层 5.css属性设置 6.效果呈现 1.前言: 在第九篇文章购物车做好后,还忘记了一个至关重要的计算组件.在鸿蒙的组件中并没有提供这样一个 ...

  9. HTML表单特别效果—音量调节,购物数量

    <form oninput="x.value=parseInt(a.value)+parseInt(b.value)">0<input type="ra ...

随机推荐

  1. php-socket通信演示

    client: error_reporting(E_ALL); set_time_limit(0); echo "<h2>TCP/IP Connection</h2> ...

  2. jquery版本更新后无live函数的处理.TypeError: $(...).live is not a function

    jquery live函数语法 jquery版本更新, 发现一个问题: jq自带的live没有了.控制台下会有如下的提示:火狐: TypeError: $(...).live is not a fun ...

  3. 利用共享内存实现比NCCL更快的集合通信

    作者:曹彬 | 旷视 MegEngine 架构师 简介 从 2080Ti 这一代显卡开始,所有的民用游戏卡都取消了 P2P copy,导致训练速度显著的变慢.针对这种情况下的单机多卡训练,MegEng ...

  4. UI_UE在线就业班(2)(Adobe Illustrator软件学习)

    Adobe Illustrator软件的使用     认识AIUI_UE在线就业班(2) .   ▼ AI是Adobe Illustrator的英文缩写,是Adobe公司旗下推出的一款基于矢量图形制作 ...

  5. Python中比较运算符连用的语法规则

    在Python中,比较运用符<.>.<=.>=.== .!=可以连用,但语法规则和其它编程语言不一样 以 == 为例,具体语法规则是: a == b == c == d 等价于 ...

  6. 终极蛇皮上帝视角之铁头娃之鲁迅之暑假闲的慌之bilibili看尚学堂网课的非洲酋长java小复习

    转自https://www.sxt.cn/Java_jQuery_in_action/eight-cache-problem.html 第一个点 自动装箱与拆箱的功能是所谓的"编译器蜜糖(C ...

  7. PHP变量覆盖漏洞整理

    昨天群里HW的大佬们都在传某某服终端检测响应平台edr存在大量RCE的洞 官网上关于EDR的介绍是这么写的 终端检测响应平台EDR,围绕终端资产安全生命周期,通过预防.防御.检测.响应赋予终端更为细致 ...

  8. Jetpack Compose学习(2)——文本(Text)的使用

    原文: Jetpack Compose学习(2)--文本(Text)的使用 | Stars-One的杂货小窝 对于开发来说,文字最为基础的组件,我们先从这两个使用开始吧 本篇涉及到Kotlin和DSL ...

  9. C# 插件编写

    //加载插件 private void LoadPlugins() { string path = Path.Combine(Path.GetDirectoryName(Assembly.GetExe ...

  10. C# ThreadLocal源码追踪

    ThreadLocal 字段成员: private Func<T>? _valueFactory; 一个获取默认值的委托 不同线程共享此成员. [ThreadStatic] private ...