算法设计 邮票问题

算法设计 邮票问题

1. 问题描述
设有n种不同面值a1, a2,…, an的邮票,规定每封信最多贴m张邮票。对于给定的m,n,求出最大的邮资连续区间。例如,给定n=3,m=3,邮票面值分别为2, 3, 5,则最大的邮资连续区间为[2,13]。
2. 具体要求
Input
输入的第一行是一个正整数n,表示测试例个数。接下来几行是n个测试例的数据,每个测试例的数据由两行组成,其中第一行含两个正整数n和m (1<=n, m<=10),表示有n种邮票,每封信最多贴m张邮票;第二行含n个正整数,表示n种邮票的面值。同一行整数之间用一个空格隔开。
Output
对于每个测试例输出一行,含两个整数,依次为最大的邮资连续区间的下界和上界,当存在多个最大的邮资连续区间时,输出下界值最小的一个。同一行两个整数之间用一个空格隔开。
3. 测试数据
Sample Input
2
3 3
2 3 5
4 5
1 4 12 21

Sample Output
2 13
1 71

代码:

/// <summary>
/// 是否可以选出目标价值
/// </summary>
/// <param name="stamp">邮票面值</param>
/// <param name="n">邮票数组大小</param>
/// <param name="tar">目标价值</param>
/// <param name="m">最多可以选择的邮票张数</param>
/// <param name="find">是否找到目标价值的解</param>
/// <param name="count">当前已经选择的邮票张数</param>
/// <param name="cur">当前价值</param>
/// <param name="next">接下来要选择的邮票的价值</param>
void select(int* stamp, int n, int tar, int m, bool& find, int count, int cur, int next)
{
	if (count > m || find == true)	// 张数超出或已经找到
	{
		return;
	}
	else
	{
		cur += next;
		if (cur == tar)	// 找到目标价值
		{
			find = true;
		}
		for (int i = 0; i < n && !find; i++)	// 按照顺序每次加入一张邮票
		{
			select(stamp, n, tar, m, find, count + 1, cur, stamp[i]);
		}
	}
}

void test2()
{
	int n;
	cin >> n;
	while (n--)
	{
		int n, m;
		cin >> n >> m;
		int* stamp = new int[n];
		for (int i = 0; i < n; i++)
		{
			cin >> stamp[i];
		}

		int minstamp = *min(stamp, stamp + n - 1);
		int maxstamp = *max(stamp, stamp + n - 1);
		int start = minstamp;
		int maxstart = start;
		int end = minstamp;
		int maxend = end;
		int len = 0;
		int maxlen = len;
		bool find = false;
		for (int i = minstamp; i <= maxstamp * m; i++)
		{
			select(stamp, n, i, m, find, 0, 0, 0);
			if (find)	// 找到目标价值的邮票组合
			{
				end = i;
				len++;
				find = false;
			}
			else
			{
				if (len > maxlen)	// 如果当前找到的区间更长则替换
				{
					maxlen = len;
					maxstart = start;
					maxend = end;
				}
				start = i + 1;
				end = i + 1;
				len = 0;
			}
		}
		cout << maxstart << " " << maxend << endl;

		delete[] stamp;
	}
}

截图:
在这里插入图片描述

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值