试题一:杀蚂蚁

程序名: antbuster.pas/c/cpp

可执行文件名: antbuster.exe

输入文件名: antbuster.in

输出文件名: antbuster.out

时间限制: 3s

最近,佳佳迷上了一款好玩的小游戏: antbuster

游戏规则非常简单:在一张地图上,左上角是蚂蚁窝,右下角是蛋糕,蚂蚁会源源不断地从窝里爬出来,试图把蛋糕搬回蚂蚁窝。而你的任务,就是用原始资金以及杀蚂蚁获得的奖金造防御塔,杀掉这些试图跟你抢蛋糕的蚂蚁 ~

下附一张游戏截图:

为了拿到尽可能高的分数,佳佳设计了很多种造塔的方案,但在尝试了其中的一小部分后,佳佳发现,这个游戏实在是太费时间了。为了节省时间,佳佳决定写个程序,对于每一种方案,模拟游戏进程,根据效果来判断方案的优劣。

根据自己在游戏中积累的一些经验,以及上网搜到的一些参数,佳佳猜了蚂蚁爬行的算法,并且假设游戏中的蚂蚁也是按这个规则选择路线:

1 、每一秒钟开始的时候,蚂蚁都在平面中的某个整点上。如果蚂蚁没有扛着蛋糕,它会在该点留下 2 单位的信息素,否则它会留下 5 单位的信息素。然后蚂蚁会在正北、正南、正东、正西四个方向中选择一个爬过去。

2 、选择方向的规则是:首先,爬完一个单位长度后到达的那个点上,不能有其他蚂蚁或是防御塔,并且那个点不能是蚂蚁上一秒所在的点(除非上一个时刻蚂蚁就被卡住,且这个时刻它仍无法动),当然,蚂蚁也不会爬出地图的边界(我们定义这些点为不可达点)。如果此时有多个选择,蚂蚁会选择信息素最多的那个点爬过去。

3 、如果此时仍有多种选择,蚂蚁先面向正东,如果正东不是可选择的某个方向,它会顺时针转 90 °,再次判断,如果还不是,再转 90 ° ... 直到找到可以去的方向。

4 、如果将每只蚂蚁在洞口出现的时间作为它的活动时间的第 1 秒,那么每当这只蚂蚁的活动时间秒数为 5 的倍数的时候,它先按规则 1~3 确定一个方向,面对该方向后逆时针转 90 °,若它沿当前方向会走到一个不可达点,它会不停地每次逆时针转 90 °,直到它面对着一个可达的点,这样定下的方向才是蚂蚁最终要爬去的方向。

5 、如果蚂蚁的四周都是不可达点,那么蚂蚁在这一秒内会选择停留在当前点。下一秒判断移动方向时,它上一秒所在点为其当前停留的点。

6 、你可以认为蚂蚁在选定方向后,瞬间移动到它的目标点,这一秒钟剩下的时间里,它就停留在目标点。

7 、蚂蚁按出生的顺序移动,出生得比较早的蚂蚁先移动。

 

然后,是一些有关地图的信息:

1、  每一秒,地图所有点上的信息素会损失 1 单位,如果那个点上有信息素的话。

2、  地图上某些地方是炮台。炮台的坐标在输入中给出。

3、  地图的长、宽在输入中给出,对于 n * m 的地图,它的左上角坐标为( 0 0 ),右下角坐标为( n m )。蚂蚁洞的位置为( 0 0 ),蛋糕的位置为( n m )。

4、  你可以把蚂蚁看做一个直径为 1 单位的圆,圆心位于蚂蚁所在的整点。

5、  游戏开始时,地图上没有蚂蚁,每个点上的信息素含量均为 0

 

一些有关炮塔的信息:

1、  炮塔被放置在地图上的整点处。

2、  为了简单一些,我们认为这些炮塔都是激光塔。激光塔的射速是 1 / 次,它的攻击伤害为 d/ 次,攻击范围为 r 。你可以认为每秒蚂蚁移动完毕后,塔才开始攻击。并且,只有当代表蚂蚁的圆的圆心与塔的直线距离不超过 r 时,塔才算打得到那只蚂蚁。

3、  如果一只蚂蚁扛着蛋糕,那么它会成为 target ,也就是说,任何打得到它的塔的炮口都会对准它。如果蛋糕好好地呆在原位,那么每个塔都会挑离它最近的蚂蚁进行攻击,如果有多只蚂蚁,它会选出生较早的一只。

4、  激光塔有个比较奇怪的特性:它在选定了打击目标后,只要目标在其射程内,塔到目标蚂蚁圆心的连线上的所有蚂蚁(这里“被打到”的判定变成了表示激光的线段与表示蚂蚁的圆有公共点)都会被打到并损 d 格血,但激光不会穿透它的打击目标打到后面的蚂蚁。

5、  尽管在真实游戏中,塔是可以升级的,但在这里我们认为塔的布局和等级就此定了下来,不再变动  

再介绍一下蚂蚁窝:

1、  如果地图上的蚂蚁不足 6 只,并且洞口没有蚂蚁,那么窝中每秒会爬出一只蚂蚁,直到地图上的蚂蚁数为 6 只。

2、  刚出生的蚂蚁站在洞口。

3、  每只蚂蚁有一个级别,级别决定了蚂蚁的血量,级别为 k 的蚂蚁的血量为 int(4*1.1^k) int(x) 表示对 x 取下整)。每被塔打一次,蚂蚁的血减少 d 。注意,血量为 0 的蚂蚁仍能精力充沛地四处乱爬,只有一只蚂蚁的血被打成负数时,它才算挂了。

4、  蚂蚁的级别是这样算的:前 6 只出生的蚂蚁是 1 级,第 7~12 只是 2 级,依此类推。

 

最后给出关于蛋糕的介绍:

1、  简单起见,你可以认为此时只剩最后一块蛋糕了。如果有蚂蚁走到蛋糕的位置,并且此时蛋糕没有被扛走,那么这只蚂蚁就扛上了蛋糕。蚂蚁被打死后蛋糕归位。

2、  如果一只扛着蛋糕的蚂蚁走到蚂蚁窝的位置,我们就认为蚂蚁成功抢到了蛋糕,游戏结束。

3、  蚂蚁扛上蛋糕时,血量会增加 int (该蚂蚁出生时血量 / 2 ),但不会超过上限。

 

整理一下 1 秒钟内发生的事件:

1 秒的最初,如果地图上蚂蚁数不足 6 ,一只蚂蚁就会在洞口出生。

接着,蚂蚁们在自己所在点留下一些信息素后,考虑移动。先出生的蚂蚁先移动。

移动完毕后,如果有蚂蚁在蛋糕的位置上并且蛋糕没被拿走,它把蛋糕扛上,血量增加,并在这时被所有塔设成 target

然后所有塔同时开始攻击。如果攻击结束后那只扛着蛋糕的蚂蚁挂了,蛋糕瞬间归位。

攻击结束后,如果发现扛蛋糕的蚂蚁没死并在窝的位置,就认为蚂蚁抢到了蛋糕。游戏也在此时结束。

最后,地图上所有点的信息素损失 1 单位。所有蚂蚁的年龄加 1

漫长的 1 秒到此结束。

 

输入:

输入的第一行是 2 个用空格隔开的整数, n m ,分别表示了地图的长和宽。

第二行是 3 个用空格隔开的整数, s d r ,依次表示炮塔的个数、单次攻击伤害以及攻击范围。

接下来 s 行,每行是 2 个用空格隔开的整数 x y ,描述了一个炮塔的位置。当然,蚂蚁窝的洞口以及蛋糕所在的位置上一定没有炮塔。

最后一行是一个正整数 t ,表示我们模拟游戏的前 t 秒钟。

 

输出:

如果在第 t 秒或之前蚂蚁抢到了蛋糕,输出一行“ Game over after x seconds ”,其中 x 为游戏结束的时间,否则输出“ The game is going on ”。

如果游戏在 t 秒或之前结束,输出游戏结束时所有蚂蚁的信息,否则输出 t 秒后所有蚂蚁的信息。格式如下:

第一行是 1 个整数 s ,表示此时活着的蚂蚁的总数。

接下来 s 行,每行 5 个整数,依次表示一只蚂蚁的年龄(单位为秒)、等级、当前血量,以及在地图上的位置( a b )。输出按蚂蚁的年龄递减排序。

 
 

输入样例:

3 5

1 1 2

2 2

5

 

输出样例:

The game is going on

5

5 1 3 1 4

4 1 3 0 4

3 1 3 0 3

2 1 3 0 2

1 1 4 0 1

 

样例说明:

    3*5 的地图,有 1 个单次伤害为 1 、攻击范围为 2 的激光炮塔,它的位置为( 2 2 ),模拟游戏的前 5 秒。 5 秒内有 5 只蚂蚁出生,都是向东爬行,其中第 1~4 只在路过( 0 2 )点时被激光塔伤了 1 格血。在第 5 秒的时候,最早出生的蚂蚁按移动规则 1~3 本来该向东移动,但由于规则 4 的作用,它在发现向北和向西移动都会到达不可达点后,最终选择了向南移动。

 

数据说明:

100% 的数据满足 1 n,m 8 s 20 t 200,000

 

 

试题二:泡泡堂

 

源程序名: bnb. pas/c/cpp

可执行文件名: bnb.exe

输入文件名: bnb.in

输出文件名: bnb.out

时限: 1s

 

XXXX NOI 期间,为了加强各省选手之间的交流,组委会决定组织一场省际电子竞技大赛,每一个省的代表队由 n 名选手组成,比赛的项目是老少咸宜的网络游戏泡泡堂。每一场比赛前,对阵双方的教练向组委会提交一份参赛选手的名单,决定了选手上场的顺序,一经确定,不得修改。比赛中,双方的一号选手,二号选手……, n 号选手捉对厮杀,共进行 n 场比赛。每胜一场比赛得 2 分,平一场得 1 分,输一场不得分。最终将双方的单场得分相加得出总分,总分高的队伍晋级 ( 总分相同抽签决定 )

作为浙江队的领队,你已经在事先将各省所有选手的泡泡堂水平了解的一清二楚,并将其用一个实力值来衡量。为简化问题,我们假定选手在游戏中完全不受任何外界因素干扰,即实力强的选手一定可以战胜实力弱的选手,而两个实力相同的选手一定会战平。由于完全不知道对手会使用何种策略来确定出场顺序,所以所有的队伍都采取了这样一种策略,就是完全随机决定出场顺序。

当然你不想这样不明不白的进行比赛。你想事先了解一下在最好与最坏的情况下,浙江队最终分别能得到多少分。

 

输入:

       输入文件的第一行为一个整数 n ,表示每支代表队的人数。

       接下来 n 行,每行一个整数,描述了 n 位浙江队的选手的实力值。

       接下来 n 行,每行一个整数,描述了你的对手的 n 位选手的实力值。

       20 %的数据中, 1<=n<=10

40 %的数据中, 1<=n<=100

60 %的数据中, 1<=n<=1000

100 %的数据中, 1<=n<=100000 ,且所有选手的实力值在 0 10000000 之间。

 

输出:

				
						样例1
				
		

输入文件中包括两个用空格隔开的整数,分别表示浙江队在最好与最坏的情况下分别能得多少分。不要在行末输出多余的空白字符。

bnb.in

2

1

3

2

4

bnb.out

2 0

样例说明

我们分别称 4 位选手为 A,B,C,D 。则可能出现以下 4 种对战方式,最好情况下可得 2 分,最坏情况下得 0 分。

 

 

浙江

???

结果

浙江

???

结果

浙江

???

结果

浙江

???

结果

一号选手

A

C

A

D

B

C

B

D

二号选手

B

D

B

C

A

D

A

C

总得分

0

2

2

0

 

样例 2

bnb.in

6

10000000

10000000

10000000

10000000

10000000

10000000

0

0

0

0

0

0

bnb.out

12 12

样例说明

对手都是认真学习的好孩子,不会打游戏。无论如何排列出场顺序都无法改变被蹂躏的结果。浙江队总能取得全胜的结果。

 

 

 

 

 

试题三 Risk

源程序名 :risk. pas/c/cpp

可执行文件名 :risk.exe

输入文件名 :risk.in

输出文件名 :risk.out

时限 :1s

 

     经过连续若干年的推广, Risk 这个游戏已经风靡全国,成为大众喜闻乐见的重要娱乐方式。 Risk 这个游戏可以理解为一种简易的策略游戏,游戏者的目的是占领所有的土地。由于游戏规则的规定,只要两个国家相邻,就认为两个国家有交战的可能性。我们现在希望知道在当前的局面下,哪些国家之间有交战的可能性。注意,我们认为只有当两个国家的国界线有公共边的时候才认为相邻,若两个国家的领土只有公共点,则认为两个国家不相邻。

       每一个国家的边界由一系列线段组成,保证这个边界是一个简单多边形,即严格不自交。为了定位每个国家的位置,我们还给出每个国家最庞大的一支军队的位置,保证这个位置一定出现在某一个形内,而不是出现在某条边界上。

 

输入:

输入文件的第一行中包括两个整数 n,m 。分别表示地图上的国家数和描述国家的边界的线段的数量。 1<=n<=600 1<=m<=4000 。接下来 n 行,每行用一对数描述了某个国家的主力军队的坐标。接下来 m 行,每行有 4 个数 x1,y1,x2,y2 ,( x1,y1 -(x2,y2) 描述了一条国界线。所有点的坐标都是 0-10000 之间的整数。

保证输入的所有线段至多只会在线段交点处相交。整张地图上有且仅有一块面积无限的空白区域不属于任何国家。每一条国界线两侧的区域或者隶属于两个不同的国家,或者分隔了一个国家与那块无穷大的空白区域。即保证一条国界线两侧的区域不同时属于同一个国家或是同时都是空白区域。所有封闭区域内部包含且仅包含一支主力军队,表示了该区域的归属。

 

输入说明:

o_ZJOI 08 risk.bmp

       例如上图中第一行的数据是合法的。而第二行中的数据都是不合法的。左边的那幅图包含线段两侧都是空白区域;中间的图包含线段两侧区域同时属于同一个国家;右边的图中军队被布置在了国界线上,因此非法;此外若最右侧的图中若没有军队也是非法的。保证输入文件提供的数据都是合法的,你的程序不需要进行数据合法性的判定。

 

输出:

       输出文件包括 n 行,每行第一个数字 x 表示有 x 个国家可能与这个国家交战,接着在同一行中升序输出 x 个整数,表示可能与这个国家交战的国家的编号。国家按输入中给出的顺序从 1 n 编号。注意数字间严格以一个空格隔开,并且不要在行末输出多余的空白字符。


				
						样例1
				
		

risk.in

4 12

3 2

11 8

12 17

1 19

0 0 10 0

10 0 20 0

20 0 20 10

20 10 20 20

20 20 10 20

10 20 0 20

0 20 0 10

0 10 0 0

10 0 10 10

0 10 10 10

20 10 10 10

10 20 10 10

 

risk.out

2 2 4

2 1 3

2 2 4

2 1 3


				
						样例2
				
		

risk.in

4 16

170 13

24 88

152 49

110 130

60 60 140 60

140 60 140 140

140 140 60 140

60 140 60 60

0 0 200 0

200 0 200 200

200 200 0 200

0 200 0 0

40 40 160 40

160 40 160 160

160 160 40 160

40 160 40 40

20 20 180 20

180 20 180 180

180 180 20 180

20 180 20 20

 

risk.out

1 2

2 1 3

2 2 4

1 3

 

试题四:树的统计

 

源程序名 :count. pas/c/cpp

可执行文件名 : count.exe

输入文件名 :count.in

输出文件名 : count.out

时限 : 2s

 

一棵树上有 n 个节点,编号分别为 1 n ,每个节点都有一个权值 w

我们将以下面的形式来要求你对这棵树完成一些操作:

I.                   CHANGE u t : 把结点 u 的权值改为 t

II.                QMAX u v: 询问从点 u 到点 v 的路径上的节点的最大权值

III.             QSUM u v: 询问从点 u 到点 v 的路径上的节点的权值和

 

注意:从点 u 到点 v 的路径上的节点包括 u v 本身

 

输入:

输入文件的第一行为一个整数 n ,表示节点的个数。

       接下来 n – 1 行,每行 2 个整数 a b ,表示节点 a 和节点 b 之间有一条边相连。

       接下来 n 行,每行一个整数,第 i 行的整数 wi 表示节点 i 的权值。

       接下来 1 行,为一个整数 q ,表示操作的总数。

接下来 q 行,每行一个操作,以“ CHANGE u t ”或者“ QMAX u v ”或者“ QSUM u v ”的形式给出。

 

对于 100 %的数据,保证 1<=n<=30000 0<=q<=200000 ;中途操作中保证每个节点的权值 w -30000 30000 之间。

输出:

       对于每个“ QMAX ”或者“ QSUM ”的操作,每行输出一个整数表示要求输出的结果。


				
						样例
				
		

count.in

4

1 2

2 3

4 1

4 2 1 3

12

QMAX 3 4

QMAX 3 3

QMAX 3 2

QMAX 2 3

QSUM 3 4

QSUM 2 1

CHANGE 1 5

QMAX 3 4

CHANGE 3 6

QMAX 3 4

QMAX 2 4

QSUM 3 4

count.out

4

1

2

2

10

6

5

6

5

16

 

 

 

 

 

 

 

 

posted on 2009-03-14 22:49 250 阅读(914) 评论(1)  编辑 收藏 引用

FeedBack:
# re: ZJOI 08
2010-06-05 19:07 | ccsdu2009
有exe可以下载吗 ?   回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理


<2009年6月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

留言簿(6)

随笔分类

随笔档案

文章档案

相册

搜索

  •  

最新评论