浙大PTA装箱问题

假设有N项物品,大小分别为s1​​ 、s​2​​ 、…、s​i、…、s​N,其中s​i为满足1≤s​i ≤100的整数。要把这些物品装入到容量为100的一批箱子(序号1-N)中。装箱方法是:对每项物品, 顺序扫描箱子,把该物品放入足以能够容下它的第一个箱子中。请写一个程序模拟这种装箱过程,并输出每个物品所在的箱子序号,以及放置全部物品所需的箱子数目。

输入格式:
输入第一行给出物品个数N(≤1000);第二行给出N个正整数s
​i(1≤s​i​​ ≤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);
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值