例题:百鸡问题 有一个人有一百块钱,打算买一百只鸡。到市场一看,公鸡五块钱一只,母鸡三块钱一个,小鸡一块钱三只。现在,请你编一程序,帮他计划一下,怎么样买法,才能刚好用一百块钱买一百只鸡?


首先先来认识一下,何谓暴力破解法?是指从可能的解集合(空间)中一一列举各情况,用题目给定的检验条件判定哪些是无用的,哪些是有用的。能使命题成立者,即为问题的解。

基本思路:(1)建立问题的数学模型,确定问题的可能解的集合(可能解的空间)。

                  (2)逐一列出可能解集合中的元素,验证是否是问题的解。

优化方向:通过加强约束条件,缩小可能解的集合的规模。


按照暴力破解法的思想,我们首先会想到的就是直接所有循环破解,管你牛鬼蛇神。

解法一:1.抽象出数学模型,这里指的就是方程组

设公鸡为x,母鸡为y,小鸡为z,

                                        鸡:  x+y+z=100

                                        钱:  5x+3y+1/3z=100

              2.算法

#include<iostream>
using namespace std;

int main(){
	int x,y,z;
	for(x=0;x<=20;x++)//公鸡不会超过20只
		for(y=0;y<=34;y++)//母鸡不会超过34只
			for(z=0;z<=100;z++)//小鸡不会超过100只
				if((x+y+z)==100 && (5*x+3*y+z/3)==100&&z%3==0)//同时要满足z是3的整数
					cout<<"x="<<x<<" "<<"y="<<y<<" "<<"z="<<z<<endl;				
	return 0;
}

               3.结果

java解决百鸡问题_嵌套

算法分析:显然,该算法嵌套三重循环,算法复杂度为O(n^3), 循环次数为20*34*100,数量级为10^4,这种算法根本不能满足我们的需求。所以聪明一点的人可能会想到要减少一个循环的嵌套,以此降低算法复杂度。


解法二:1.减少一个循环的嵌套,模型不变,进一步优化算法

              2.

#include<iostream>
using namespace std;
int main(){
	int x,y,z;
	for(x=0;x<=20;x++)//公鸡不会超过20只
		for(y=0;y<=34;y++)//母鸡不会超过34只
		{
			z=100-x-y;//根据x,y可求出z
			if(z==(300-15*x-9*y)&&z%3==0)
				cout<<"x="<<x<<" "<<"y="<<y<<" "<<"z="<<z<<endl;
		}						
	return 0;
}

算法分析:显然,该算法嵌套二重循环,算法复杂度为O(n^2), 循环次数为20*34,数量级为10^2,这种算法比上一算法足足降低了2个数量级,可见循环嵌套越多,算法就越差。所以还需要继续优化,用我们的初中数学知识就OK.


解法三:1.每使用一个for(x)循环,就相当于将一个未知数(x)变成已知数(x), 两个方程两个未知数,方程就具有可解性了, 这样就可以, 继续减少一个循环的嵌套

                                        鸡:  x+y+z=100

                                        钱:  15x+9y+z=300

                                  =>>7x+4y=100   ->y    ->z

               2.

#include<iostream>
using namespace std;
int main(){
	int x,y,z;
	for(x=0;x<=20;x++)//公鸡不会超过20只
	{
		y=25-1.75*x;//先求y
		z=100-x-y;//再求z
		if(z==(300-15*x-9*y)&&z%3==0&&y>0&&&z>0)
			cout<<"x="<<x<<" "<<"y="<<y<<" "<<"z="<<z<<endl;
	}
	return 0;
}

算法分析:显然,该算法只有一重循环,算法复杂度为O(n), 循环次数为20,数量级为10,这种算法比上一算法降低了1个数量级,说明该算法还是不错的,可以满足我们日常的需求。不过数学思维比较好的人,可以通过结果反过来继续优化。


解法四:

java解决百鸡问题_#include_02

通过上图的结果,我们发现  x以4递增,y以7递减,   z以3递增,所以可以直接打印

#include<iostream>
using namespace std;
int main(){
	int x,y,z;
	for(int k=0;k<=3;k++)
	{
        x = 4 * k;
        y = 25 - 7 * k;
        z = 75 + 3 * k; 
		cout<<"x="<<x<<" "<<"y="<<y<<" "<<"z="<<z<<endl;
	}
	return 0;
}

算法分析:这种算法可以说是接近最优了,不过在AC的时候,强烈建议使用解法三,因为你不可能一开始就知道结果是什么和有多少种解法,以上是我对该算法的理解,如果你觉得还有优化的细节,欢迎留言,一同进步。