作业分配问题

测试数据
3
1 2 3
3 2 1
2 1 3

源代码:

#include<stdio.h>

int Works[20][20];
int minConsume=0;

// 给 Works和minConsume赋值
int initWorks(int workerCount)
{
    int t;
    for(int i=1;i<=workerCount;i++)
        for(int j=1;j<=workerCount;j++)
        {
            scanf("%d", &t);
            Works[i][j] = t;
            minConsume+=t;
        }
}

// 统计当前作业分配方式的消费量
int allConsume(int *N, int workerCount)
{
    int sum = 0;
    for(int i = 1;i<=workerCount;i++)
        sum+=Works[i][N[i]];
    return sum;
}

// 判断是否合法(当前工人选择的作业是否已分配给其他工人,如果已分配,返回0,否则返回1)
int isLegal(int *N, int nowWorker)
{
    for(int i=1;i<nowWorker;i++)
    {
        if(N[i]==N[nowWorker])
            return 0;
    }
    return 1;
}

// 子集树实现分配方式(穷举所有的情况)
int backTrackWorks(int *N, int workerCount, int nowWorker)
{
    //找到一种分配方式,求出该方式的总消费量
    if(nowWorker>workerCount)
    {
        int nowConsume = allConsume(N, workerCount);
        if(minConsume>nowConsume)
            minConsume = nowConsume;
        return 0;
    }
    else
    {
        for(int i=1;i<=workerCount;i++)
        {
            N[nowWorker] = i;
            if(isLegal(N, nowWorker))
            {
                backTrackWorks(N, workerCount, nowWorker+1);
            }
        }
    }
}

// workerCount 总工人数
// Works[][] 为第i个人分配工作j所需的费用为Works[i][j]
// N[] 标记已分配好的工人和该工人所做作业 例如N[1]=5,表示第一个工人选择第五个作业
// nowWorker 序号 表示当前为第nowWorker号工人分配作业
// minConsume 最小的费用
int main()
{
    int workerCount;
    scanf("%d", &workerCount);

    initWorks(workerCount);

    int N[workerCount+1];
    int nowWorker = 1;

    backTrackWorks(N, workerCount, nowWorker);

    printf("%d", minConsume);
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值