[ Uva 101 详解 ] the block problem 木块问题

题目

输入 n
编号 0 ~ n-1 的木块,分别摆放在顺序排列编号为 0 ~ n-1 的位置。,要求模拟以下 4 种操作(下面的 a 和 b 都是木块编号)

  • move a onto b:把 a 和 b 上方的木块全部归位,然后把 a 摞在 b 上面
  • move a over b:把 a 上方的木块全部归位,然后把 a 放在 b 所在木块堆的顶部
  • pile a onto b:把 b 上方的木块全部归位,然后把 a 及上面的木块整体摞在 b 上面
  • pile a over b:把 a 及上面的木块整体摞在 b 所在木块堆的顶部

遇到 quit 时终止一组数据。a 和 b 在同一堆的指令是非法指令,应当忽略。

Input

10
move 9 onto 1
move 8 over 1
move 7 over 1
move 6 over 1
pile 8 over 6
pile 8 over 5
move 2 over 1
move 4 over 9
quit

Output

 0: 0
 1: 1 9 2 4
 2:
 3: 3
 4:
 5: 5 8 7 6
 6:
 7:
 8:
 9:

分析

数据存储:
n 表示有多少位置,观察输出结果后,可以用一个定长数组来存储位置,在那个位置上的方块数量会根据操作增加或减少,属于不定长,可以使用 vector 数组来存储

vector<int> pile[100]; // 用这条来创建一个 有100个位置,每个位置一个不定长数组的数组来储存

操作分析:
观察 4 个命令,大体上动作只有 归位 和 移动,可以定义为两个函数,方便复用

接着 4 个命令放一起对比,4 个命令都有 移动 的动作,这意味着我们只需要判断何时进行 归位 的动作即可

前两个命令进行对比后,似乎带有 move 的都会让 a 归位,有 onto 的会把 b 归位

后两个命令也是如此,如果既没有 move 又没有 onto,则只进行 移动 动作

操作实现:
首先是归位,即把 a 以上的木块放回与其编号对应的位置,那么我们需要知道当前 a 所在的位置和处在这个位置中的什么地方,
假定用 p 来表示位置,h 来表示高度,
默认时,编号为 1 的块就在 p = 1, h = 0 的位置,
移动到编号为 2 块上面时,位置变为 p = 2, h = 1 (注意每个位置默认是有一个块的)
那么如何表示位置便清晰了,
接着是如何找到位置,很明显这个操作是会经常复用的,我们来定义一个 find 函数来实现这个功能

void find_block(int a, int& p, int& h)                // a 是查找的木块编号 ,p 和 h 采用引值传导,这样就能修改主函数中位置
{
	for ( p = 0; p < n; p++)                      // 从一切的起点开始遍历,找到对应则直接退出
	{
		for ( h = 0; h < pile[p].size(); h++) // vector容器的 .size() 方法可以返回不定长数组此时的长度,避免越界
		{
			if (a == pile[p][h]) return;
		}
	}
}

那么现在我们已经可以得到任意一个编号木块的位置了,接着实现把上面的木块一个个归位的动作

void goback(int p, int h)
{       int num = 0;
	for (int i = h+1; i < pile[p].size(); i++) // 从当前高度+1处开始归位
	{      num = pile[p][i];                   // 临时储存,你也可以直接用 pile[p][i] 代替 mun,不过那样的话下一行就太过复杂
		pile[num].push_back(num);          // 使用 vector 的 .push_back() 方法,可以将一个值添加到对应不定长数组中的最后
	}
	pile[p].resize(h + 1);                    // .resize() 则是将数组重新限定长度,不够的补齐,多的直接舍弃,因为编号是从0开始,所以要想保存 当前号,需要 +1
}

归位写好之后,移动就很简单了

void move(int p1, int h1, int p2)
{
	for (int i = h1; i < pile[p1].size(); i++) // 这边之所以不对 i +1,因为移动的是 a及a以上
	{
		pile[p2].push_back(pile[p1][i]);  // 同理
	}
	pile[p1].resize(h1);                      // 当前值也删去
}

目前关键操作已经清晰了,接下来就是润色润色,添加一些细节

细节处理:
命令输入是字符串混合数字,使用getchar一个一个读实在麻烦,scanf又不太好实现
所以我们采用流输入,并且用字符串搭配使用,引入 iostream库 和 string库

cin >> s1 >> a >> s2 >> b; //可以直接处理掉

成品代码

#include <iostream>
#include <vector>
#include <cstdio>
#include <string>


using namespace std;

const int MAXN = 100;
vector<int> pile[MAXN];
int n = 0;



void find_block(int a, int& p, int& h);
void goback(int p, int h);
void move(int p1, int h1, int p2);

int command();



int main()
{
	while (scanf_s("%d", &n) != EOF)
	{
		for (int i = 0; i < n; i++)
		{
			pile[i].clear();
			pile[i].push_back(i);
		}

		while (command());											// 开始处理


		for (int i = 0; i < n; i++)									// 输出结果
		{
			printf("%d :", i);
			for (int j = 0; j < pile[i].size(); j++)
			{
				printf("  %d", pile[i][j]);
			}
			printf("\n");
		}


	}
}

int command()
{
	int a, b;
	string s1, s2;
	int n1 = 0;
	for (;;)
	{
		int p1, h1, p2, h2;
		cin >> s1 >> a >> s2 >> b;
		if (s1 == "quit") return 0;
		find_block(a, p1, h1);
		find_block(b, p2, h2);
		if (p1 == p2) continue;
		if (s2 == "onto") goback(p2, h2);
		if (s1 == "move") goback(p1, h1);
		move(p1, h1, p2);
	}
}

void find_block(int a, int& p, int& h)
{
	for ( p = 0; p < n; p++)
	{
		for ( h = 0; h < pile[p].size(); h++)
		{
			if (a == pile[p][h]) return;
		}
	}
}

void goback(int p, int h)
{
	for (int i = h+1; i < pile[p].size(); i++)
	{
		pile[i].push_back(i);
	}
	pile[p].resize(h);
}

void move(int p1, int h1, int p2)
{
	for (int i = h1; i < pile[p1].size(); i++)
	{
		pile[p2].push_back(pile[p1][i]);
	}
	pile[p1].resize(h1);
}
posted @ 2021-09-04 13:13  EdwinAze  阅读(228)  评论(0编辑  收藏  举报