假设有N项物品,大小分别为s1 、s2 、…、si、…、sN,其中si为满足1≤si ≤100的整数。要把这些物品装入到容量为100的一批箱子(序号1-N)中。装箱方法是:对每项物品, 顺序扫描箱子,把该物品放入足以能够容下它的第一个箱子中。请写一个程序模拟这种装箱过程,并输出每个物品所在的箱子序号,以及放置全部物品所需的箱子数目。
输入格式:
输入第一行给出物品个数N(≤1000);第二行给出N个正整数s
i(1≤si ≤100,表示第i项物品的大小)。
输出格式:
按照输入顺序输出每个物品的大小及其所在的箱子序号,每个物品占1行,最后一行输出所需的箱子数目。
输入样例:
8
60 70 80 90 30 40 10 20
输出样例:
60 1
70 2
80 3
90 4
30 1
40 5
10 1
20 2
5
思路:由于物品的大小是不变的,而且只需遍历(存放)一次,而箱子是动态的(不断地放入物体,直到放不下),链表的顺序遍历很方便,数组的重复调用很方便,因此将物品的信息放入链表,箱子的容量放进数组。(当然都用链表或者都用数组实现也可以)。对于每一个物品, 从前往后遍历箱子,若放的下,则放进去,此时箱子的容量需要更改成原来的容量减去当前放入物品的容量,反复操作,直至所有的物品全部放入
ac代码:
#include <iostream>
using namespace std;
typedef struct LNode * List;
struct LNode{
int Data;
int id;
struct LNode * Next;
};
List ReadList();
List Processed(List L);
void PrintList(List L);
int cnt = 0;//物品个数
int main()
{
int remain[1005];
List L;
L = ReadList();//输入
Processed(L);//装箱
return 0;
}
List ReadList()
{
List L, head, s;
L = (List)malloc(sizeof(struct LNode));
L->Next = NULL;
head = L;//带有头节点的链表
int n;
scanf("%d", &n);
for(int i = 1; i <= n; i++)
{
int e;
scanf("%d", &e);
s = (List)malloc(sizeof(struct LNode));
s->Data = e;
s->Next = NULL;
L->Next = s;
L = L->Next;
}
L = head->Next;
free(head);//释放头节点
cnt = n;
return L;
}
void PrintList(List L)
{
List head = L;
while(L){
printf("%d %d\n", L->Data, L->id);
L = L->Next;
}
return;
}
List Processed(List L)
{
int remain[1005], index = 0;//index 记录最大的箱子编号
for(int i = 1; i <= cnt; i++)
remain[i] = 100;
List head = L;
int cnt = 1;
while(L){//从第一个物体往后遍历
for(int j = 1; ; j++){//从第一个箱子依次往后遍历,若箱子的容量比物体的大小来的大,则放进去,箱子的容量=之前的容量-物品大小
if(remain[j] >= L->Data){
L->id = j;
remain[j] -= L->Data;
index = max(index, j);
break;
}
}
L = L->Next;
}
L = head;
PrintList(L);
printf("%d\n", index);
}