纪中OJ 2019.02.15【NOIP提高组】模拟 B 组 梦回三国 比赛题解(第一个)
旁边的同学小 H(胡)对我说: “哟,比赛拿了 140,强!要知道,如果哥第三题 AC 了,哥就 230 了,你个废柴!!!(比赛实际分数 130 额呵)”
顿时,千万草泥马从我心中奔腾而过:你不要每次都把“如果”说得这么理直气壮好吧...... (心态大崩*1)
嗯咳,不和他瞎扯了,骚话一大堆,进入正题。
第一次心情大好 (因为小 H 太搞笑了啊哈),准备写比赛的题解!~
小 H:“明明你是因为以前的比赛题解太长了才懒得写,说得这么好听......”
“额呵,闭嘴!”(心态大崩*2)
嗯,似乎又扯远了。
比赛分数:100+10+30=140 (竟然还有 3 个大佬们高我 160 AK 比赛(心态大崩*3)?太强了,膜拜!!!)
排名:12 (额,比平常平均排名要前)
这次比赛有个特别好听的题目: “梦回三国”
不想排版,就这样了吧哎~
“梦回三国”系列题解如下:
T1:魏传之长坂逆袭
3906. 魏传之长坂逆袭 (standard IO)
Time Limits: 1000 ms Memory Limits: 131072 KB
众所周知,刘备在长坂坡上与他的一众将领各种开挂,硬生生从曹操手中逃了出去,随后与孙权一起火烧赤壁、占有荆益、成就霸业。而曹操则在赤壁一败后再起不能,终生无力南下。
建安二十五年(220年),曹操已到风烛残年,但仍难忘当年长坂的失误,霸业的破灭。他想如果在刘备逃亡的路中事先布下一些陷阱,便能拖延刘备的逃脱时间了。你作为曹操身边的太傅,有幸穿越到了208年的长坂坡,为大魏帝国贡献力量,布置一些陷阱。但时空守卫者告诉你你不能改变历史,不能拖增大刘备的最大逃脱时间,但你身为魏武之仕,忠心报国,希望能添加一些陷阱使得刘备不论怎么逃跑所用的时间都一样。
【问题描述】
已知共有n个据点,n-1条栈道,保证据点联通。1号据点为刘备军逃跑的起点,当刘备军跑到某个据点后不能再前进时视为刘备军逃跑结束。在任意一个栈道上放置1个陷阱会使通过它的时间+1,且你可以在任意一个栈道上放置任意数量的陷阱。
现在问你在不改变刘备军当前最大逃跑时间的前提下,需要添加最少陷阱,使得刘备军的所有逃脱时间都尽量的大。
Input
接下来n-1行每行三个数,ai、bi、ti,表示从据点ai通过第i个栈道到bi耗时ti
Output
Sample Input
3
1 2 1
1 3 3
Sample Output
2
Data Constraint
对于 100% 的数据,1<=n<=500000,0<ti<=1000000
5% 的数据是什么鬼啊啊???(心态大崩*4)
额然而仔细想想发现,这道题不是挺简单的吗?比赛时用时 45 min 就 A 掉了(耶 100 到手,5% 与我无关!!!)~
就是一棵树,画画图,发现贪心可行性显然,于是两次 dfs 搞定!!!
直接附比赛出题人的题解:
题目大意是给你一棵树,要你调整一些边的权值,使根结点 到结点的路径等长。
首先要求该根到各叶子节点的距离相同,实际上也就是要求它 的子树到各个叶子节点相同,而当保证了某棵子树的根到叶子节 点的距离相同的时候,实际上可以把这个看做一条链了,所以这就可以看做一个树形 DP 了。递归处理子树的然后再处理父亲,这种贪心的证明显然,不妨假设某一个边的修改值变小,那么它显 然就需要下传那个差额给其他子树,而子树的个数>=1,所以肯定不会比这个方法优。
说白了,每次对于深度越低的点能放多少陷阱就放多少陷阱,两次深搜解决。
var
d,delta:array[..] of int64;
first,next,en,len,a:array[..] of longint;
i,k,m,n,s,t,flag:longint;
v,ans:int64;
procedure add(x,y,w:longint); //前向星
begin
inc(m);
next[m]:=first[x];
first[x]:=m;
en[m]:=y;
len[m]:=w;
end;
procedure dfs1(x,data:longint);
var
now,e,bool:longint;
begin
now:=first[x];
bool:=;
while now<> do
begin
e:=en[now];
bool:=;
dfs1(e,data+len[now]);
now:=next[now];
end;
if bool= then
begin
inc(k);
a[k]:=x;
d[k]:=data;
if data>v then v:=data;
end;
end;
procedure dfs2(x:longint);
var
now,e:longint;
begin
now:=first[x];
if now= then exit;
while now<> do
begin
e:=en[now];
dfs2(e);
if delta[e]<delta[x] then delta[x]:=delta[e];
now:=next[now];
end;
now:=first[x];
while now<> do
begin
inc(ans,delta[en[now]]-delta[x]);
now:=next[now];
end;
end;
begin
readln(n);
for i:= to n- do
begin
readln(s,t,v);
add(s,t,v);
end;
v:=-maxlongint;
dfs1(,);
fillchar(delta,sizeof(delta),);
for i:= to k do
delta[a[i]]:=v-d[i];
dfs2();
writeln(ans);
end.
T2: 蜀传之单刀赴会
3907. 蜀传之单刀赴会 (Standard IO)
Description
公元215年,刘备取益州,孙权令诸葛瑾找刘备索要荆州。刘备不答应,孙权极为恼恨,便派吕蒙率军取长沙、零陵、桂阳三郡。长沙、桂阳蜀将当即投降。刘备得知后,亲自从成都赶到公安(今湖北公安),派大将关羽争夺三郡。孙权也随即进驻陆口,派鲁肃屯兵益阳,抵挡关羽。双方剑拔弩张,孙刘联盟面临破裂,在这紧要关头,鲁肃为了维护孙刘联盟,不给曹操可乘之机,决定当面和关羽商谈。“肃邀羽相见,各驻兵马百步上,但诸将军单刀俱会”。双方经过会谈,缓和了紧张局势。随后,孙权与刘备商定平分荆州,“割湘水为界,于是罢军”,孙刘联盟因此能继续维持。
【问题描述】
关羽受鲁肃邀请,为了大局,他决定冒险赴会。他带着侍从周仓,义子关平,骑着赤兔马,手持青龙偃月刀,从军营出发了,这就是历史上赫赫有名的“单刀赴会”。关羽平时因为军务繁重,决定在这次出行中拜访几个多日不见的好朋友。然而局势紧张,这次出行要在限定时间内完成,关公希望你能够帮助他安排一下行程,安排一种出行方式,使得从军营出发,到达鲁肃处赴会再回来,同时拜访到尽可能多的朋友,在满足这些条件下行程最短。注意拜访朋友可以在赴会之前,也可以在赴会之后。现在给出地图,请你完成接下来的任务。
Input
接下来m行,每行有x,y,w三个整数,代表x和y之间有长度为w的道路相连。
接下来一行有k个整数,代表朋友所在的都城编号(保证两两不同,且不在1和n)
(我们约定1是关羽的营地,n是鲁肃的营地)
Output
Sample Input
5 7 2 15
1 2 5
1 3 3
2 3 1
2 4 1
3 4 4
2 5 2
4 5 3
2 4
Sample Output
2 14
Data Constraint
有10%数据,k=0;
有10%数据,k=1;
另30%数据,k<=5;
对于100%数据,n<=10000,m<=50000,k<=15,t<=2147483647,w<=10000
本人认为这题是此系列中最难的一道题。
比赛时只用 spfa 跑纯最短路拿了 k=0 的 10 分的部分分(呜呜呜复制粘贴后还调了我快一个小时,才发现是无向边,前向星要建两次边)~ (心态大崩*5)
小 H:“要不是我看那 10 分太少了,还要打最短路。于是哥给你一个面子,让你比我高,不然我 Ctrl C+Ctrl V 然后修改一下就和你一样分数了哈” (心态大崩*6)
额这是比赛啊!!!(虽然我也是偷偷复制粘贴的,只是他没发现,实在不想打啊啊~)
咳咳,思路:先跑一遍 dijkstra(或大约 k 遍 spfa),然后状压 dp 。(太复杂啦!心态大崩*7)
具体思路:
我们观察到 k 很小,不难想到状压 dp 。
dp[sta,i] 表示状态为 sta,最后到达的点为 i 的最少时间。
转移方程不难想到,我们枚举当前状态最后到达的点 i,再枚举未来状态最后到达的点 j,判断一下当前状态里包不包含 j,如果不包含,则:
dp[sta or (1<<(j-1)),j]=min(dp[sta or (1<<(j-1)),j],dp[sta,i]+dist[i,j]);
不过这里有一个小疑问就是 dist[i,j] 之间有一个另外的朋友经过了怎么办,这样不是没有记录么,其实我刚开始也是这样想的,但是其实我们状态都会枚举,不会有错。
那么我们要怎么求 dist[i,j] 呢?
我们先把地点 1 加入到要访问的朋友里,再把地点 n 加入到朋友里。
最后我们加一个 n+1 点,作为最后回到 1 的点(注意我们的 k 已经加了 3 了)。
我们跑 k-2 遍最短路,把两两朋友之间的最小距离算出来。
最后枚举一下状态判断 dp[i,k] 是否小于等于 t 且 i and (1<<(k-2))(判断到达点 n)就可以啦~
思路转自:http://www.itdaan.com/blog/2017/11/08/4bbf264861172e48d24cf044945e7e10.html
再附一段官方题解(最难的题题解最短???心态大崩*8):
题目大意是给定一张图,确定起点和一些关键点,其中至少有一个关键点要到达,要求在时限内安排一种方案使得尽可能经过多的关键点并且回到起点,要求在同等条件下路程最短。
我们对于每一个关键点进行一次 SPFA 预处理,然后用状压枚举方案,最后和时限做比较求最优方案就可以了。
uses math;
var
n,m,k,t,s,w,i,j,sta,cnt,tt:longint;
f:array[..,..] of int64;
dist:array[..,..] of longint;
py,dui,en,len,c,first,next:array[..] of longint;
use:array[..] of boolean;
procedure add(x,y,w:longint); //前向星
begin
inc(sta);
next[sta]:=first[x];
first[x]:=sta;
en[sta]:=y;
len[sta]:=w;
end;
procedure spfa(s,x:longint);
var
i,j,t,h,r,head:longint;
begin
fillchar(c,sizeof(c),);
h:=;
r:=;
use[s]:=true;
dui[]:=s;
c[s]:=;
while h<=r do
begin
head:=dui[h];
t:=first[head];
while t> do
begin
if c[en[t]]>c[head]+len[t] then
begin
c[en[t]]:=c[head]+len[t];
if use[en[t]]=false then
begin
use[en[t]]:=true;
inc(r);
dui[r]:=en[t];
end;
end;
t:=next[t];
end;
use[head]:=false;
inc(h);
end;
for i:= to k- do
dist[x,i]:=c[py[i]];
dist[x,k]:=c[];
end;
begin
readln(n,m,k,t);
for i:= to m do
begin
readln(s,tt,w);
add(s,tt,w);
add(tt,s,w);
end;
py[]:=; //把地点 加入关键点
inc(k);
for i:= to k do
read(py[i]);
readln;
inc(k); //把地点 n 加入关键点
py[k]:=n;
inc(k); //把地点 n+ 加入关键点
for i:= to k- do //跑 spfa 记录关键点之间距离
spfa(py[i],i);
for i:= to k- do
dist[k,i]:=dist[,i];
for sta:= to <<k- do
for i:= to k do
f[sta,i]:=;
f[,]:=;
for sta:= to <<k- do //状压 dp
for i:= to k do
for j:= to k do
if sta and (<<(j-))= then
f[sta or (<<(j-)),j]:=min(f[sta or (<<(j-)),j],f[sta,i]+dist[i,j]);
w:=-;
s:=maxlongint;
for sta:= to <<k- do //处理答案
if (f[sta,k]<=t) and (sta and (<<(k-))<>) then
begin
j:=sta;
cnt:=;
while j<> do
begin
inc(cnt);
j:=j-j and (-j);
end;
dec(cnt,);
if cnt>w then
begin
w:=cnt;
s:=f[sta,k];
end;
if (cnt=w) and (f[sta,k]<s) then s:=f[sta,k];
end;
if w=- then writeln(-) else writeln(w,' ',s);
end.
然而代码比我想象中要短很多(理想 at least 150+,实际100额)~~~
T3:吴传之火烧连营
3908. 吴传之火烧连营 (Standard IO)
蜀汉章武元年(221年),刘备为报吴夺荆州、关羽被杀之仇,率大军攻吴。吴将陆逊为避其锋,坚守不战,双方成对峙之势。蜀军远征,补给困难,又不能速战速决,加上入夏以后天气炎热,以致锐气渐失,士气低落。刘备为舒缓军士酷热之苦,命蜀军在山林中安营扎寨以避暑热。陆逊看准时机,命士兵每人带一把茅草,到达蜀军营垒时边放火边猛攻。蜀军营寨的木栅和周围的林木为易燃之物,火势迅速在各营漫延。蜀军大乱,被吴军连破四十余营。陆逊火烧连营的成功,决定了夷陵之战蜀败吴胜的结果。
【问题描述】
刘备带兵深入吴境,陆逊却避而不出,蜀军只得在山林中安营扎寨。而刘备在扎营时却犯了兵家大忌,将兵营排列成一条直线,远远看去,就像是一条串着珠子的链,美其名曰:链寨。如果吴军将领是一般人,那么这也许不算什么,而陆逊何许人也,他可是江东才子,能力不低于周瑜的一代儒将。他看到刘备这样排阵,心生一计,决定用火攻破阵。然而,火计除了要有风,选定引火点也非常重要,对于刘备的布阵,最佳引火点一定是n个兵营中的一个。而因为风水轮流转,每天的最佳引火点都不一样。我们给每个兵营定下一个固定不变的火攻值Ai,每天定下一个风水值K,对于每天的最佳引火点,显然是所有兵营中火攻值与风水值异或的结果最大的那一个兵营。然而,陆逊是个谨慎的人,他要观察时机,在m天中选定一个最佳的进攻的日期,为此他演算出了这m天每天的风水值,然后他希望你能够告诉他这m天每天最佳引火点的兵营编号。
Input
接下来一行有n个非负整数,代表这n个兵营的火攻值。
接下来一行有m个非负整数,代表这m天的风水值。
Output
如果存在多个最佳引火点使得火攻值与风水值的异或值最大,请任意输出一组解即可。
Sample Input
3 2
1 2 3
4 5
Sample Output
3
2 【样例解释】
对于第1天,由于4 xor 1=5, 4 xor 2=6, 4 xor 3=7,选择第3个引火点是最佳的。
对于第2天,由于5 xor 1=4, 5 xor 2=7, 5 xor 3=6,选择第2个引火点是最佳的。
Data Constraint
对于100%数据,n<=100000,m<=100000, 0<=k,ai<=2147483647
这不是一道常规的二进制 trie 树?
部分分超好拿,n2 的暴力只要不打挂,5 min 就能敲诈走30分!
然后回去打了 1 h 的第二题 10 分暴力分~
比赛后面的 1 h 一直看着这道题,然而什么也没发现!!!(心态大崩*9)
正解:把读入的数的二进制加入 trie 树中,那么询问时,我们要尽量的走与询问数的二进制相反的点,因为这样异或之后就可以让那一位变成 1,很简单不是吗?
(然而我比赛时没想到???心态大崩*10)
附官方正解:
题目大意就是对于一个序列,多次询问求在 xor K 的情况下的最大值。
我们将每一个数拆分成 2 进制,然后映射进一棵 trie 中,然后对于每个询问,根据异或的性质 (1 xor 1=0,1 xor 0=1,0 xor 1=1,0 xor 0=0) 尽可能的匹配,因为是 32 位整数,所以每次询问效率就是 O(32)。
对于水平较高的同学可以思考一下如果改成区间询问该如何做。 (额这个算了吧!心态大崩*11)
var
trie:array[..,..] of longint;
ans:array[..] of longint;
n,m,i,j,k,t,x,s:longint;
begin
readln(n,m);
k:=;
for i:= to n do //建 trie 树
begin
read(x);
s:=;
for j:= downto do
begin
t:=(x>>(j-)) and ;
if trie[s,t]= then
begin
inc(k);
trie[s,t]:=k;
s:=k;
ans[k]:=i;
end else s:=trie[s,t];
end;
end;
readln;
for i:= to m do //输出答案
begin
read(x);
s:=;
for j:= downto do
begin
t:=((x>>(j-)) and +) and ;
//取反,本来原位置上是 则 t=,反之
if trie[s,t]= then s:=trie[s,(t+) and ]
else s:=trie[s,t];
//尽可能取反匹配,匹配不了就算了~
end;
writeln(ans[s]);
end;
readln;
end.
此系列的题码量总体很(chao)少(40、65、100),三题码量加起来才其他比赛最难那题相当,所以超好改的!!!
但是做完题后魏蜀吴三国我都搞不清了啊啊啊!!!
没什么好总结的,就是时间要控制好。
还有最短路的几种算法忘得差不多了,调 spfa 竟然调了一个小时?!
最后就是敲诈部分分(骗分)的能力还有待提高!!!
(额虽然几乎每次比赛所有的题都是骗分的,很少有比赛中 A 掉的题,但还是不够哎~)
纪中OJ 2019.02.15【NOIP提高组】模拟 B 组 梦回三国 比赛题解(第一个)的更多相关文章
- 纪中OJ 2019.01.25【NOIP提高组】模拟 B 组 T2 数字对
声明 数字对 Time Limits: 2000 ms Memory Limits: 262144 KB Description 小 H 是个善于思考的学生,现在她又在思考一个有关序列的问题. ...
- 【纪中集训】2019.08.02【NOIP提高组】模拟 A 组TJ
\(\newcommand{\RNum}[1]{\uppercase\expandafter{\romannumeral #1\relax}}\) T1 一道可以暴力撵标算的题-- Descripti ...
- 纪中集训2020.02.05【NOIP提高组】模拟B 组总结反思——【佛山市选2010】组合数计算,生成字符串 PPMM
目录 JZOJ2290. [佛山市选2010]组合数计算 比赛时 之后 JZOJ2291. [佛山市选2010]生成字符串 比赛时 之后 JZOJ2292. PPMM 比赛时 之后 JZOJ2290. ...
- 纪中集训2020.02.09【NOIP提高组】模拟B 组总结反思
目录 JZOJ.1747[NOIP2014模拟11.5]无穷迷宫 比赛时 之后 总结 JZOJ1478.[NOIP2014模拟11.5]近似乘积 比赛时 之后 总结 JZOJ3926. [NOIP20 ...
- 纪中集训2020.02.03【NOIP提高组】模拟B 组总结反思——登机(board),游戏(game),分组(group)
T1 JZOJ5535. 登机(board) 比赛时 一在题目列表里看到题目标题,就热血沸腾了,不知道为什么,老师居然放了一道之前做过的题目.我清楚地记得这题是DP,于是很快码了出来.讲一讲我的思路, ...
- 【2019.7.15 NOIP模拟赛 T1】夹缝(mirror)(思维题)
思维题 此题应该是比较偏思维的. 假设一次反射后前进的距离是\(2^x(2y+1)\),则显然,它可以看做是前进距离为\(2^x\)的光线经过了\((2y+1)\)次反射,两者是等价的,甚至后者可能还 ...
- 2019.02.15 codechef Favourite Numbers(二分+数位dp+ac自动机)
传送门 题意: 给444个整数L,R,K,nL,R,K,nL,R,K,n,和nnn个数字串,L,R,K,数字串大小≤1e18,n≤65L,R,K,数字串大小\le1e18,n\le65L,R,K,数字 ...
- 【纪中集训2019.3.27】【集训队互测2018】小A的旅行(白)
题目 描述 \(0-n-1\)的图,满足\(n\)是\(2\)的整数次幂, $ i \to j $ 有 $ A_{i,j} $ 条路径: 一条路径的愉悦值定义为起点和终点编号的\(and\)值 ...
- 【纪中集训2019.3.23】Deadline
题意 描述 一个二分图\((A,B)\),每个点额外有一个颜色0或者1: 匹配时,只能相同颜色的点匹配: 给出\(A\)中的颜色,问如何分配\(B\)种的颜色使得\((A,B)\)的最大匹配最小: 范 ...
随机推荐
- PHP根据图片制作缩略图
php中制作缩略图的方法也很简单,是用imagecopyresampled方法根据源图制作一个小一点的图片,来看代码check_image_addthumbs.php <?php //修改图片效 ...
- JVM垃圾收集算法的选择
1. 介绍 JVM提供了多种垃圾收集器,应该根据应用选择一种合适的垃圾收集器. 垃圾回收管理内存通过如下操作: 在年轻代分配对象,把年龄大的对象晋升到老年代. 当年老代超过阈值的时候,并发标记收集. ...
- gluoncv 用已经训练好的模型参数,检测物体
当然这个模型参数,最好用自己的,否则不够精确,我自己的还没训练完. from matplotlib import pyplot as plt import gluoncv from gluoncv i ...
- 【CSS】iconfont的使用
说到浏览器对@font-face的兼容问题,这里涉及到一个字体format的问题,因为不同的浏览器对字体格式支持是不一致的,这样大家有必要了解一下,各种版本的浏览器支持什么样的字体,前面也简单带到了有 ...
- [Python 网络编程] makefile (三)
socket.makefile(mode ='r',buffering = None,*,encoding = None,errors = None,newline = None )返回一个与套接字相 ...
- ServletContextListener在Springboot中的使用
ServletContextListener是servlet容器中的一个API接口, 它用来监听ServletContext的生命周期,也就是相当于用来监听Web应用的生命周期.今天我们就来说说如何在 ...
- jmeter接口测试3-正则表达式提取器的使用
正则表达式的用处很多,最基础的用法 1,断言 2,传参(关联) 例子 1.http请求 2正则表达式提取,想要提取列表列中id,一遍打开列表页 如果是1,每次就会取相同的值!匹配数字的权限高于模板$0 ...
- 【转】上传jar包到nexus私服
原文:https://my.oschina.net/lujianing/blog/297128 1通过网页上传 这种方法只是上传了jar包.通过maven引用当前jar,不能取得jar的依赖 from ...
- python -- 将string转换成dict的方法
装载自:http://smilejay.com/2014/10/convert_string_to_dict_python/ 我将数据库连接相关的一些用户名/密码/host/port等各种东西作为一个 ...
- Order by排序
asc 升序(默认),desc 降序 order by 后面 可以加 列.表达式.别名.序号(从1开始) desc; --表达式 年薪 from emp order by 年薪 desc; --别名 ...