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 ...
随机推荐
- 复杂多变场景下的Groovy脚本引擎实战
一.前言 因为之前在项目中使用了Groovy对业务能力进行一些扩展,效果比较好,所以简单记录分享一下,这里你可以了解: 为什么选用Groovy作为脚本引擎 了解Groovy的基本原理和Java如何集成 ...
- 文件上传靶机DVWA和upload-labs
DVWA靶机 LOW <?php phpinfo() ?> 上传文件 Medium级别 修改Content-Type: application/octet-stream的值为jpg的格式为 ...
- 过渡——transition
① 设置参与过渡的属性 transition-property: ; all / transform / 参与过渡的属性 ② 设置过渡的时长 transition-duration: s ...
- dio框架访问云函数参数传递问题(以腾讯云中的云函数为例子)第一部
dd云函数其实比较普及,这里以腾讯云的云函数为例,传递参数完成简单的账号注册. 一.第一步先注册腾讯云账号,这里不过多阐述,接着点击 控制台 进入开发者界面.(注意提前进行实名认证) 二.开发者界面如 ...
- XSS闯关挑战(1-15)
第一关 关键代码: 这一关两处的输出都没做任何防护,直接将用户的输入拼接到输出里面. payload: 第二关 使用上一关的payload显示如下 闭合一下标签就好了. 第三关 htmlspecial ...
- MySQL-18-MHA+Atlas读写分离架构
Atlas介绍 Atlas是由Qihoo 360 Web平台部基础架构团队开发维护的一个基于MySQL协议的数据中间层项目 它是在mysql-proxy 0.8.2版本的基础上,对其进行了优化,增加了 ...
- brew换源
转自:https://blog.csdn.net/gorwayne/article/details/107359912 第一步,替换brew.git cd "$(brew --repo)&q ...
- Kali 2.0 安装教程
本文适合KALI初学者,将详细介绍Kali Linux 2.0的安装过程. 首先我们到KALI的官网下载镜像,大家可以自己选择下载32或64位的KALI 2.0系统. KALI 官网:https:// ...
- 【现学现卖】th:href标签动态路径设置,thymeleaf获取session中的属性值
update:2020-02-28:按道理来说这个功能在前后端分离的时候应该不怎么用的上,基本到现在我还是没遇到过有这样的需求,不过也是一种方法就是.th:href="@{/{role}/l ...
- NOIP 模拟 $14\; \text{抛硬币}$
题解 \(by\;\;zj\varphi\) 签到题,自己看题解 Code #include<bits/stdc++.h> #define ri register signed #defi ...