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 ...
随机推荐
- DC-9 靶机渗透测试
DC-9 渗透测试 冲冲冲,好好学习 DC系列的9个靶机做完了,对渗透流程基本掌握,但是实战中出现的情况千千万万,需要用到的知识面太广了,学不可以已. 靶机IP: 172.66.66.139 kali ...
- 为什么说使用 Linux 系统学习 OpenGL 更方便
前言 上一篇随笔介绍了我的电脑,同时也介绍了 Ubuntu 20.10 系统的安装和美化.这一篇,我将正式开始 OpenGL 之旅.使用 Ubuntu 来进行开发,不仅仅只是因为我对 Linux 桌面 ...
- Leetcode_丑数问题
1. 263. 丑数 给你一个整数 n ,请你判断 n 是否为 丑数 .如果是,返回 true :否则,返回 false . 丑数 就是只包含质因数 2.3 或 5 的正整数. 示例 1: 输入:n ...
- Vue 脚手架学习
首先就是安装脚手架 npm install @vue/cil -g 全局安装 在这里我遇到一个问题:安装不了脚手架,报错显示: 通过苦逼的查找原因就是 以前使用的taobao镜像 导致的,删除镜像换成 ...
- AMD CMD commonJS es6
看到很多人傻傻的分不清楚 AMD.CMD . commonJS . es6的区别,实际上这跟这几年前段技术的发展有很大的关系. 实际上这是JavaScript的模块化思想演进的一个过程. 最开始的时候 ...
- 跟我一起写 Makefile(九)
使用函数 ---- 在Makefile中可以使用函数来处理变量,从而让我们的命令或是规则更为的灵活和具有智能.make所支持的函数也不算很多,不过已经足够我们的操作了.函数调用后,函数的返回值可以当做 ...
- Linux 基础学习篇笔记 Linux基础知识
哎呀,翻到第一篇,映出眼帘的标题:从Unix到Linux(我就知道学习不能急,不能像我,看个简介,就赶忙去查了,原来作者在这里给出详细的介绍了) 1.1根据书上写的,原来linux的内核是被Linus ...
- JVM内存调整
JVM内存调整 先试着调整一下idea的 找到软件安装位置/bin/idea64.exe.vmoptions 给他直接整个起飞的,改成 -Xms512m -Xmx1500m 找到Java安装的位置/j ...
- 【原创】Dubbo 2.7.8多个远程代码执行漏洞
马上年底了,发现年初定的几个漏洞的KPI还没来得及完成,趁着最近有空赶紧突击一波,之前业务部门被爆过Dubbo的漏洞,干脆就把Dubbo拖过来挖一把.之前没用过Dubbo,既然要挖它就先大体了解了一下 ...
- 008 PCI设备BAR空间的初始化
一.PCI设备BAR空间的初始化 在PCI Agent设备进行数据传送之前,系统软件需要初始化PCI Agent设备的BAR0~5寄存器和PCI桥的Base.Limit寄存器.系统软件使用DFS算法对 ...