韩信走马分酒问题(回溯)

 泊松是法国数学家、物理学家和力学家。他一生致力科学事业,成果颇多。有许多著名的公式定理以他的名字命名,比如概率论中著名的泊松分布。
 有一次闲暇时,他提出过一个有趣的问题,后称为:“泊松分酒”。在我国古代也提出过类似问题,遗憾的是没有进行彻底探索,其中流传较多是:“韩信走马分油”问题。

 有3个容器,容量分别为12升,8升,5升。其中12升中装满油,另外两个空着。要求你只用3个容器操作,最后使得某个容器中正好有6升油。

 下面的列表是可能的操作状态记录:
 12,0,0
 4,8,0
 4,3,5
 9,3,0
 9,0,3
 1,8,3
 1,6,5

 每行3个数据,分别表示12,8,6升容器中的油量

 第一行表示初始状态,第二行表示把12升倒入8升容器后的状态,第三行是8升倒入5升,...

 当然,同一个题目可能有多种不同的正确操作步骤。

   本题目的要求是,请你编写程序,由用户输入:各个容器的容量,开始的状态,和要求的目标油量,程序则通过计算输出一种实现的步骤(不需要找到所有可能的方法)。     如 果没有可能实现,则输出:“不可能”。

    例如,用户输入:
   12,8,5,12,0,0,6

   用户输入的前三个数是容器容量(由大到小),接下来三个数是三个容器开始时的油量配置,最后一个数是要求得到的油量(放在哪个容器里得到都可以)

    则程序可以输出(答案不唯一,只验证操作可行性):
  12,0,0
  4,8,0
  4,3,5
  9,3,0
  9,0,3
  1,8,3
  1,6,5

   每一行表示一个操作过程中的油量状态。

  注意:

   请仔细调试!您的程序只有能运行出正确结果的时候才有机会得分!

package JiaNan;
import java.util.Scanner;  
import java.util.Stack;  
public class WaterPouring2012_10 
{  
	public static int A,B,C;   //三个杯子的容量  
	public static Stack<Integer[]> road=new Stack<Integer[]>();   //记录倒水步骤  
	public static boolean[][][] vis;   //这个数组不错,记录空间树节点是否被访问
    public static int num=0;    //记录解的数量  
	public static int aim;      //目标水量  
	public static void main(String[] args) 
	{  
		// TODO Auto-generated method stub  
        Scanner sc=new Scanner(System.in);  
		A=sc.nextInt();  
		B=sc.nextInt();  
		C=sc.nextInt();  
		
		vis=new boolean[2*A][2*A][2*A];  //注意这里全部是A,因此注意杯子大小,提前输入数据时就排好序
        for(int i=0;i<A*2;i++)  
            for(int j=0;j<A*2;j++)  
                for(int k=0;k<A*2;k++)  
                    vis[i][j][k]=false;  
				
				int Astart=sc.nextInt();  
				int Bstart=sc.nextInt();  
				int Cstart=sc.nextInt();  
				
				aim=sc.nextInt();   
				road.push(new Integer[]{Astart,Bstart,Cstart});  
				Pouring(Astart,Bstart,Cstart);  
				if(num==0)  
				{  
					System.out.println("不可能");  
				}  
    }  
	
	public static void Pouring(int x,int y,int z)  
    {  
        if(x==aim||y==aim||z==aim)  //如果得到了目标解,就将计数器加1,打印出倒水过程
        {  
			num++;  
			System.out.println("方案"+num+":");  
			PrintRoad(); 
			System.out.println();
			return;
        }  
		else if(x<=A&&y<=B&&z<=C&&!vis[x][y][z])   //如果空间树节点没被访问过  
        {  
			vis[x][y][z]=true;     //该节点的所有子结点全部被访问  
			//A->B A倒空  
			if(x+y<B && x>0)  
			{  
				//System.out.print("A->B A倒空");  
				road.push(new Integer[]{0, x+y, z});  
                Pouring(0, x+y, z);  
                road.pop();                            //这里巧妙运用堆栈,用push代表执行某过程,下面的pop代表恢复
            }  
			//A->B B倒满  
			if(y<B && x>0 && (x+y)>=B)  
            {  
				//System.out.print("A->B B倒满");  
				road.push(new Integer[]{x+y-B, B, z});  
				Pouring(x+y-B, B, z);  
                road.pop();  
			}  
            //A->C A倒空  
            if(x+z<C && x>0)  
			{  
				//System.out.print("A->C A倒空");  
				road.push(new Integer[]{0,y,x+z});  
				Pouring(0,y,x+z);  
				road.pop();  
			}  
            //A->C C倒满  
			if(z<C && x>0 && x+z>=C)  
			{  
				//System.out.print("A->C C倒满");  
                road.push(new Integer[]{x+z-C,y,C});  
				Pouring(x+z-C,y,C);  
                road.pop();  
            }  
			
			//B->A B倒空  
			if(x+y<=A && y>0)  
			{  
				//System.out.print("B->A B倒空");  
                road.push(new Integer[]{x+y,0,z});  
				Pouring(x+y,0,z);  
                road.pop();  
			}  
            //B->A A倒满  
            if(x<A && y>0 && y+x>=A)  
            {  
				//System.out.print("B->A A倒满");  
				road.push(new Integer[]{A,x+y-A,z});  
                Pouring(A,x+y-A,z);  
				road.pop();  
			}  
			//B->C B倒空  
			if(y+z<C && y>0)  
            {  
				//System.out.print("B->C B倒空");  
				road.push(new Integer[]{x,0,y+z});  
                Pouring(x,0,y+z);  
				road.pop();  
			}  
            //B->C C倒满  
            if(z<C && y>0 && y+z>=C)  
			{  
                //System.out.print("B->C C倒满");  
				road.push(new Integer[]{x,y+z-C,C});  
                Pouring(x,y+z-C,C);  
                road.pop();  
			}  
			
            //C->A C倒空  
            if(x+z<=A && z>0)  
			{  
				//System.out.print("C->A C倒空");  
				road.push(new Integer[]{x+z,y,0});  
				Pouring(x+z,y,0);  
				road.pop();  
			}
			//C->A A倒满  
			if(x<A && z>0 && x+z>=A)  
			{  
				//System.out.print("C->A A倒满");  
				road.push(new Integer[]{A,y,x+z-A});  
				Pouring(A,y,x+z-A);  
				road.pop();  
			}  
			//C->B C倒空  
			if((B-y)>z&&z>0)  
			{  
				//System.out.print("C->B C倒空");  
				road.push(new Integer[]{x,y+z,0});  
				Pouring(x,y+z,0);  
				road.pop();  
			}  
			//C->B B倒满  
			if(y<B && z>0 && y+z>=B)  
			{  
				//System.out.print("C->B B倒满");  
				road.push(new Integer[]{x,B,y+z-B});  
				Pouring(x,B,y+z-B);  
				road.pop();  
			}    
        }    
    }  
	
	public static void PrintRoad()  
    {  
		for(int i=0;i<road.size();i++)  
		{  
            Integer[] tmp=road.get(i);  
            System.out.println("("+tmp[0]+","+tmp[1]+","+tmp[2]+")");  
		}  
        //road.push(new Integer[]{A,0,0});  
	}  
	
}  
/*
12 8 5 12 0 0 6
方案1:
(12,0,0)
(4,8,0)
(0,8,4)
(8,0,4)
(8,4,0)
(3,4,5)
(3,8,1)
(11,0,1)
(11,1,0)
(6,1,5)

方案2:
(12,0,0)
(4,8,0)
(4,3,5)
(9,3,0)
(9,0,3)
(1,8,3)
(1,6,5)

*/


 

你可能感兴趣的:(韩信走马分酒问题(回溯))