洛谷 P2662 牛场围栏

题目背景

小L通过泥萌的帮助,成功解决了二叉树的修改问题,并因此写了一篇论文,

成功报送了叉院(羡慕不?)。勤奋又勤思的他在研究生时期成功转系,考入了北京大学光华管理学院!毕业后,凭着自己积累下的浓厚经济学与计算机学的基础,成功建设了一个现代化奶牛场!

题目描述

奶牛们十分聪明,于是在牛场建围栏时打算和小L斗智斗勇!小L有N种可以建造围栏的木料,长度分别是l1,l2 … lN,每种长度的木料无限。

修建时,他将把所有选中的木料拼接在一起,因此围栏的长度就是他使用的木料长度之和。但是聪明的小L很快发现很多长度都是不能由这些木料长度相加得到的,于是决定在必要的时候把这些木料砍掉一部分以后再使用。

不过由于小L比较节约,他给自己规定:任何一根木料最多只能削短M米。当然,每根木料削去的木料长度不需要都一样。不过由于测量工具太原始,小L只能准确的削去整数米的木料,因此,如果他有两种长度分别是7和11的木料,每根最多只能砍掉1米,那么实际上就有4种可以使用的木料长度,分别是6, 7,10, 11。        

因为小L相信自己的奶牛举世无双,于是让他们自己设计围栏。奶牛们不愿意自己和同伴在游戏时受到围栏的限制,于是想刁难一下小L,希望小L的木料无论经过怎样的加工,长度之和都不可能得到他们设计的围栏总长度。不过小L知道,如果围栏的长度太小,小L很快就能发现它是不能修建好的。因此她希望得到你的帮助,找出无法修建的最大围栏长度。

这一定难不倒聪明的你吧!如果你能帮小L解决这个问题,也许他会把最后的资产分给你1/8哦!

输入输出格式

输入格式:

输入的第一行包含两个整数N,  M,分别表示木料的种类和每根木料削去的最大值。以下各行每行一个整数li(1< li< 3000),表示第i根木料的原始长度。

输出格式:

输出仅一行,包含一个整数,表示不能修建的最大围栏长度。如果任何长度的围栏都可以修建或者这个最大值不存在,输出-1。

输入输出样例

输入样例#1:
2 1
7 11
输出样例#1:
15

说明

40 % :1< N< 10,  0< M< 300

100 % :1< N< 100,  0< M< 3000 



裸的同余类bfs,注意木棍长度不能为负,可以去重减少复杂度。

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
using namespace std;
const int N=105;
const int M=3005;
int n,m,nn,ans,a[N*M],dis[3005];
bool inq[3005];
queue<int>q;
bool pd()
{
	for(int i=2;i<=nn;i++)
		if(a[i]%a[1]!=0)
			return 0;
	return 1;
}
int main()
{
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)
		scanf("%d",&a[i]);
	nn=n;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=m;j++)
			if(a[i]-j>0)
				a[++nn]=a[i]-j;
	sort(a+1,a+nn+1);
	if(pd())
	{
		printf("-1\n");
		return 0;
	}
	nn=unique(a+1,a+nn+1)-a-1;
	memset(dis,0x3f,sizeof(dis));
	dis[0]=0;
	q.push(0);
	inq[0]=1;
	while(!q.empty())
	{
		int u=q.front();
		q.pop();
		for(int i=2;i<=nn;i++)
			if(dis[(u+a[i])%a[1]]>dis[u]+a[i])
			{
				dis[(u+a[i])%a[1]]=dis[u]+a[i];
				if(!inq[(u+a[i])%a[1]])
				{
					inq[(u+a[i])%a[1]]=1;
					q.push((u+a[i])%a[1]);
				}
			}
		inq[u]=0;
	}
	for(int i=1;i<=a[1]-1;i++)
		ans=max(ans,dis[i]);
	printf("%d\n",ans-a[1]);
	return 0;
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值