银河英雄传说 (codevs 1540) 题解

【问题描述】

公元五八○一年,地球居民迁移至金牛座α第二行星,在那里发表银河联邦创立宣言,同年改元为宇宙历元年,并开始向银河系深处拓展。

宇宙历七九九年,银河系的两大军事集团在巴米利恩星域爆发战争。泰山压顶集团派宇宙舰队司令莱因哈特率领十万余艘战舰出征,气吞山河集团点名将杨威利组织麾下三万艘战舰迎敌。

杨威利擅长排兵布阵,巧妙运用各种战术屡次以少胜多,难免恣生骄气。在这次决战中,他将巴米利恩星域战场划分成30000列,每列依次编号为1, 2, …, 30000。之后,他把自己的战舰也依次编号为1, 2, …, 30000,让第i号战舰处于第i列(i = 1, 2, …, 30000),形成“一字长蛇阵”,诱敌深入。这是初始阵形。当进犯之敌到达时,杨威利会多次发布合并指令,将大部分战舰集中在某几列上,实施密集攻击。合并指令为M i j,含义为让第i号战舰所在的整个战舰队列,作为一个整体(头在前尾在后)接至第j号战舰所在的战舰队列的尾部。显然战舰队列是由处于同一列的一个或多个战舰组成的。合并指令的执行结果会使队列增大。

然而,老谋深算的莱因哈特早已在战略上取得了主动。在交战中,他可以通过庞大的情报网络随时监听杨威利的舰队调动指令。

在杨威利发布指令调动舰队的同时,莱因哈特为了及时了解当前杨威利的战舰分布情况,也会发出一些询问指令:C i j。该指令意思是,询问电脑,杨威利的第i号战舰与第j号战舰当前是否在同一列中,如果在同一列中,那么它们之间布置有多少战舰。

作为一个资深的高级程序设计员,你被要求编写程序分析杨威利的指令,以及回答莱因哈特的询问。

最终的决战已经展开,银河的历史又翻过了一页……

【样例输入】

4

M 2 3

C 1 2

M 2 4

C 4 2

【样例输出】

-1

1

【解题思路】

本题为NOI2002年的题目,这道题目很明显需要使用并查集,因为题目中说了将一列上的战舰调到另一列上去,然后对于每一个询问,寻找询问的两艘战舰中间所隔的战舰数,一开始的想法是不用路径压缩的并查集。但后面发现这种形式有问题(你需要去找这一列的第一艘战舰的编号,再将其合并到另一列上),这样的时间复杂度会高一些,不过似乎也能过……这里我采用的是多开两个数组,一个记录该战舰所在列的总战舰数,另一个是记录该战舰前面排了多少个战舰,这样一来,如果询问的两艘战舰在同一列上,我们就只需要把两艘战舰前的战舰数相减去绝对值再减一就行了。那么主要问题就是这两个数组在路径压缩和合并的时候应该如何处理,详见代码。

【代码实现】

 1 var father,before,count:array[1..30000] of longint;
 2     i,n,x,y:longint;
 3     ch:char;
 4 function getfather(v:longint):longint;
 5 var f:longint;
 6 begin
 7  if father[v]=v then
 8   exit(v)
 9  else
10   begin
11    f:=getfather(father[v]);
12    before[v]:=before[father[v]]+before[v];//路径压缩,在该战舰前的战舰数为它的祖先前的加上它本身前面的,可自己手推一下。
13    father[v]:=f;
14    exit(father[v]);
15   end;
16 end;
17 procedure union(u,v:longint);
18 var fu,fv:longint;
19 begin
20  fu:=getfather(u);
21  fv:=getfather(v);
22  father[fu]:=fv;
23  before[fu]:=before[fu]+count[fv];//因为两列合并了,所以在fu前面的要加上整个fv所在列的战舰数
24  count[fv]:=count[fv]+count[fu];//fv所在列的战舰数需要加上fu所在的战舰数
25 end;
26 procedure queue(u,v:longint);
27 begin
28  if getfather(u)<>getfather(v) then
29   writeln(-1)
30  else
31   writeln(abs(before[u]-before[v])-1);
32 end;
33 begin
34  readln(n);
35  for i:=1 to 30000 do
36   begin
37    father[i]:=i;
38    before[i]:=0;
39    count[i]:=1;
40   end;//初始化
41  for i:=1 to n do
42   begin
43    readln(ch,x,y);
44    case ch of
45     ‘M‘: union(x,y);
46     ‘C‘: queue(x,y);
47    end;
48   end;
49 end.
时间: 2024-11-10 00:54:59

银河英雄传说 (codevs 1540) 题解的相关文章

code1052 地鼠游戏

贪心算法,从后往前 来自codevs的题解: 我的纠结思考过程:如果每一秒都没有重复的地鼠出现 那么肯定是一个一个挨着打如果有重复的地鼠 那么要考虑打那个更优 当然是选分值最大的 单纯这样想很合理 但是忽略了一种情况 分值小的地鼠早出现了 而后面重复的地鼠都比这个早出现的地鼠分值大我们就不能去打这只小的 而是把时间去用在打后面重复的地鼠身上如 1 2 2    5 6 7 那么按照地鼠的分值排序 然后挨着打怎么样也不对 因为分值大的地鼠也有可能停留时间长 我们早打了他 会导致时间短的地鼠没法打

code1006 等差数列

我绞尽脑汁想一个更好的算法,然而不能如愿,只好写一个n^3的了 很简单,就是暴力搜索(还好n<100) 先排序,然后循环i=1ton,j=i+1ton 把a[i]a[j]确定为等差数列开始的两个数,确定公差,然后用search()搜这个数列的长度 取所有的最大值即可 代码如下: #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<algo

trie tree

trie树也叫字典树,前缀树 字典树(Trie)有如下几条性质 结点不存值,依靠树枝(边)存值 从根节点到某一处标记点为一个单词 每个结点到其子节点的边上的值各不相同 插入和查询复杂度均为O(mn),m为字符串个数,n为字符串平均长度 树深度由最长字符串决定 依次便可做出trie树的结点结构 struct trie{ int cnt=0; bool word=0; trie * son[26]={0}; }; 让我们依次价绍这几个成员的含义 cnt 这条边上有几个单词经过,即有多少指定的前缀相同

codevs 1512 转向游戏 题解

Codevs 1512转向游戏 题解 时间限制: 1 s 空间限制: 1000 KB 题目等级 : 白银 Silver 题解 题目描述 Description 小明自认为方向感很好,请小红来测试.小红先让小明面对北方立正站好,然后发出"向左转""向右转"或"向后转"的命令.每个命令执行后,小明都正确地说出了他面对的方向.小红的命令共N个(1≤n≤10000),请你统计小明说[南]的次数. 命令是以数字方式表达: 0---向左转 1---向右转 2

codevs 1506 传话 题解

Codevs 1506传话 题解 1506 传话--这个题目的解法很多,你能想到几种? 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 白银 Silver 题解 题目描述 Description 一个朋友网络,如果a认识b,那么如果a第一次收到某个消息,那么会把这个消息传给b,以及所有a认识的人. 如果a认识b,b不一定认识a. 所有人从1到n编号,给出所有"认识"关系,问如果i发布一条新消息,那么会不会经过若干次传话后,这个消息传回给了i,1<=i<=n

CODEVS 1048 石子归并 题解

此文为博主原创题解,转载时请通知博主,并把原文链接放在正文醒目位置. 题目链接:http://codevs.cn/problem/1048/ 题目描述 Description 有n堆石子排成一列,每堆石子有一个重量w[i], 每次合并可以合并相邻的两堆石子,一次合并的代价为两堆石子的重量和w[i]+w[i+1].问安排怎样的合并顺序,能够使得总合并代价达到最小. 输入描述 Input Description 第一行一个整数n(n<=100) 第二行n个整数w1,w2...wn  (wi <=

CODEVS 1081 线段树练习 2 题解

此文为博主原创题解,转载时请通知博主,并把原文链接放在正文醒目位置. 题目链接:http://codevs.cn/problem/1081/ 题目描述 Description 给你N个数,有两种操作 1:给区间[a,b]的所有数都增加X 2:询问第i个数是什么? 输入描述 Input Description 第一行一个正整数n,接下来n行n个整数,再接下来一个正整数Q,表示操作的个数. 接下来Q行每行若干个整数.如果第一个数是1,后接3个正整数a,b,X,表示在区间[a,b]内每个数增加X,如果

CODEVS 1080 线段树练习 题解

此文为博主原创题解,转载时请通知博主,并把原文链接放在正文醒目位置. 题目链接:http://codevs.cn/problem/1080/ 题目描述 Description 一行N个方格,开始每个格子里都有一个整数.现在动态地提出一些问题和修改:提问的形式是求某一个特定的子区间[a,b]中所有元素的和:修改的规则是指定某一个格子x,加上或者减去一个特定的值A.现在要求你能对每个提问作出正确的回答.1≤N<100000,,提问和修改的总数m<10000条. 输入描述 Input Descrip

靶形数独 (codevs 1174)题解

[问题描述] 小城和小华都是热爱数学的好学生,最近,他们不约而同地迷上了数独游戏,好胜的他们想用数独来一比高低.但普通的数独对他们来说都过于简单了,于是他们向Z 博士请教,Z 博士拿出了他最近发明的“靶形数独”,作为这两个孩子比试的题目.靶形数独的方格同普通数独一样,在 9 格宽×9 格高的大九宫格中有9 个3 格宽×3 格高的小九宫格(用粗黑色线隔开的).在这个大九宫格中,有一些数字是已知的,根据这些数字,利用逻辑推理,在其他的空格上填入1 到9 的数字.每个数字在每个小九宫格内不能重复出现,