旁边的同学小 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
5% 的数据是什么鬼啊啊???(心态大崩*4)
额然而仔细想想发现,这道题不是挺简单的吗?比赛时用时 45 min 就 A 掉了(耶 100 到手,5% 与我无关!!!)~
就是一棵树,画画图,发现贪心可行性显然,于是两次 dfs 搞定!!!
直接附比赛出题人的题解:
题目大意是给你一棵树,要你调整一些边的权值,使根结点 到结点的路径等长。
首先要求该根到各叶子节点的距离相同,实际上也就是要求它 的子树到各个叶子节点相同,而当保证了某棵子树的根到叶子节 点的距离相同的时候,实际上可以把这个看做一条链了,所以这就可以看做一个树形 DP 了。递归处理子树的然后再处理父亲,这种贪心的证明显然,不妨假设某一个边的修改值变小,那么它显 然就需要下传那个差额给其他子树,而子树的个数>=1,所以肯定不会比这个方法优。
说白了,每次对于深度越低的点能放多少陷阱就放多少陷阱,两次深搜解决。
1 var 2 d,delta:array[0..500001] of int64; 3 first,next,en,len,a:array[0..1000001] of longint; 4 i,k,m,n,s,t,flag:longint; 5 v,ans:int64; 6 procedure add(x,y,w:longint); //前向星 7 begin 8 inc(m); 9 next[m]:=first[x]; 10 first[x]:=m; 11 en[m]:=y; 12 len[m]:=w; 13 end; 14 procedure dfs1(x,data:longint); 15 var 16 now,e,bool:longint; 17 begin 18 now:=first[x]; 19 bool:=0; 20 while now<>0 do 21 begin 22 e:=en[now]; 23 bool:=1; 24 dfs1(e,data+len[now]); 25 now:=next[now]; 26 end; 27 if bool=0 then 28 begin 29 inc(k); 30 a[k]:=x; 31 d[k]:=data; 32 if data>v then v:=data; 33 end; 34 end; 35 procedure dfs2(x:longint); 36 var 37 now,e:longint; 38 begin 39 now:=first[x]; 40 if now=0 then exit; 41 while now<>0 do 42 begin 43 e:=en[now]; 44 dfs2(e); 45 if delta[e]<delta[x] then delta[x]:=delta[e]; 46 now:=next[now]; 47 end; 48 now:=first[x]; 49 while now<>0 do 50 begin 51 inc(ans,delta[en[now]]-delta[x]); 52 now:=next[now]; 53 end; 54 end; 55 begin 56 readln(n); 57 for i:=1 to n-1 do 58 begin 59 readln(s,t,v); 60 add(s,t,v); 61 end; 62 v:=-maxlongint; 63 dfs1(1,0); 64 fillchar(delta,sizeof(delta),126); 65 for i:=1 to k do 66 delta[a[i]]:=v-d[i]; 67 dfs2(1); 68 writeln(ans); 69 end.
T2: 蜀传之单刀赴会
3907. 蜀传之单刀赴会 (Standard IO)
本人认为这题是此系列中最难的一道题。
比赛时只用 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 预处理,然后用状压枚举方案,最后和时限做比较求最优方案就可以了。
1 uses math; 2 var 3 n,m,k,t,s,w,i,j,sta,cnt,tt:longint; 4 f:array[0..500001,1..18] of int64; 5 dist:array[1..18,1..18] of longint; 6 py,dui,en,len,c,first,next:array[0..100001] of longint; 7 use:array[0..100001] of boolean; 8 procedure add(x,y,w:longint); //前向星 9 begin 10 inc(sta); 11 next[sta]:=first[x]; 12 first[x]:=sta; 13 en[sta]:=y; 14 len[sta]:=w; 15 end; 16 procedure spfa(s,x:longint); 17 var 18 i,j,t,h,r,head:longint; 19 begin 20 fillchar(c,sizeof(c),63); 21 h:=1; 22 r:=1; 23 use[s]:=true; 24 dui[1]:=s; 25 c[s]:=0; 26 while h<=r do 27 begin 28 head:=dui[h]; 29 t:=first[head]; 30 while t>0 do 31 begin 32 if c[en[t]]>c[head]+len[t] then 33 begin 34 c[en[t]]:=c[head]+len[t]; 35 if use[en[t]]=false then 36 begin 37 use[en[t]]:=true; 38 inc(r); 39 dui[r]:=en[t]; 40 end; 41 end; 42 t:=next[t]; 43 end; 44 use[head]:=false; 45 inc(h); 46 end; 47 for i:=1 to k-1 do 48 dist[x,i]:=c[py[i]]; 49 dist[x,k]:=c[1]; 50 end; 51 begin 52 readln(n,m,k,t); 53 for i:=1 to m do 54 begin 55 readln(s,tt,w); 56 add(s,tt,w); 57 add(tt,s,w); 58 end; 59 py[1]:=1; //把地点 1 加入关键点 60 inc(k); 61 for i:=2 to k do 62 read(py[i]); 63 readln; 64 inc(k); //把地点 n 加入关键点 65 py[k]:=n; 66 inc(k); //把地点 n+1 加入关键点 67 for i:=1 to k-1 do //跑 spfa 记录关键点之间距离 68 spfa(py[i],i); 69 for i:=1 to k-1 do 70 dist[k,i]:=dist[1,i]; 71 for sta:=0 to 1<<k-1 do 72 for i:=1 to k do 73 f[sta,i]:=100000000000000007; 74 f[1,1]:=0; 75 for sta:=1 to 1<<k-1 do //状压 dp 76 for i:=1 to k do 77 for j:=1 to k do 78 if sta and (1<<(j-1))=0 then 79 f[sta or (1<<(j-1)),j]:=min(f[sta or (1<<(j-1)),j],f[sta,i]+dist[i,j]); 80 w:=-1; 81 s:=maxlongint; 82 for sta:=0 to 1<<k-1 do //处理答案 83 if (f[sta,k]<=t) and (sta and (1<<(k-2))<>0) then 84 begin 85 j:=sta; 86 cnt:=0; 87 while j<>0 do 88 begin 89 inc(cnt); 90 j:=j-j and (-j); 91 end; 92 dec(cnt,3); 93 if cnt>w then 94 begin 95 w:=cnt; 96 s:=f[sta,k]; 97 end; 98 if (cnt=w) and (f[sta,k]<s) then s:=f[sta,k]; 99 end; 100 if w=-1 then writeln(-1) else writeln(w,' ',s); 101 end.
然而代码比我想象中要短很多(理想 at least 150+,实际100额)~~~
T3:吴传之火烧连营
3908. 吴传之火烧连营 (Standard IO)
这不是一道常规的二进制 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)
1 var 2 trie:array[0..10000001,0..1] of longint; 3 ans:array[0..10000001] of longint; 4 n,m,i,j,k,t,x,s:longint; 5 begin 6 readln(n,m); 7 k:=1; 8 for i:=1 to n do //建 trie 树 9 begin 10 read(x); 11 s:=0; 12 for j:=31 downto 1 do 13 begin 14 t:=(x>>(j-1)) and 1; 15 if trie[s,t]=0 then 16 begin 17 inc(k); 18 trie[s,t]:=k; 19 s:=k; 20 ans[k]:=i; 21 end else s:=trie[s,t]; 22 end; 23 end; 24 readln; 25 for i:=1 to m do //输出答案 26 begin 27 read(x); 28 s:=0; 29 for j:=31 downto 1 do 30 begin 31 t:=((x>>(j-1)) and 1+1) and 1; 32 //取反,本来原位置上是 1 则 t=0,反之 33 if trie[s,t]=0 then s:=trie[s,(t+1) and 1] 34 else s:=trie[s,t]; 35 //尽可能取反匹配,匹配不了就算了~ 36 end; 37 writeln(ans[s]); 38 end; 39 readln; 40 end.
此系列的题码量总体很(chao)少(40、65、100),三题码量加起来才其他比赛最难那题相当,所以超好改的!!!
但是做完题后魏蜀吴三国我都搞不清了啊啊啊!!!
没什么好总结的,就是时间要控制好。
还有最短路的几种算法忘得差不多了,调 spfa 竟然调了一个小时?!
最后就是敲诈部分分(骗分)的能力还有待提高!!!
(额虽然几乎每次比赛所有的题都是骗分的,很少有比赛中 A 掉的题,但还是不够哎~)