回 帖 发 新 帖 刷新版面

主题:[原创]看灌水(倒酒)问题有感

昨日在“三思小百科” http://www.oursci.org/ency/math/002.htm看到一篇好文章。该文较详细的介绍了灌水(倒酒)问题的一般解法。根据文章提示的算法,我写了一个程序。
问题和算法简介:
倒水问题的经典形式是这样的:
“假设有一个池塘,里面有无穷多的水。现有2个空水壶,容积分别为5升和6升。问题是如何只用这2个水壶从池塘里取得3升的水。”
  当然题外是有一些合理的限制的,比如从池塘里灌水的时候,不管壶里是不是已经有水了,壶一定要灌满,不能和另一个壶里的水位比照一下“毛估估”(我们可以假设壶是不透明的,而且形状也不同);同样的,如果要把水从壶里倒进池塘里,一定要都倒光;如果要把水从一个壶里倒进另一个壶里,也要都倒光,除非在倒的过程中另一个壶已经满了;倒水的时候水没有损失(蒸发溢出什么的)等等等等。
 
算法介绍:“灌水定理”:
“如果有n个壶容积分别为A1,A2,……,An(Ai均为大于0的整数)设w为另一大于0的整数。则用此n个壶可倒出w升水的充要条件为:
 1) w小于等于A1+A2+......+An;
 2) w可被(A1,A2,......,An)(这n个数的最大公约数)整除。”
 这两个条件都显然是必要条件,如果1)不被满足的话,你连放这么多水的地方都没有。2)的道理和上面两个壶的情况完全一样,因为在任何步骤中,任何壶中永远只有(A1,A2,......,An)的倍数的水。

如果A1,A2,……,An是n个整数,(A1,A2,......,An)=s,那么存在整数U1,U2,……,Un,使得
       U1A1 + U2A2 + ...... + UnAn = s.    (*)
在代数学上称此结果为“整数环是主理想环”。
回头看“灌水定理”。w是(A1,A2,......,An)的倍数,根据上节的公式(*),两边乘以这个倍数,我们就有整数V1,V2,……,Vn使得 V1A1 + V2A2 + ...... + VnAn = w.注意到Vi是有正有负的。这就说明,只要分别把A1,A2,……,An壶,灌上V1,V2,……,Vn次(如果Vi是负的话,“灌上Vi次”要理解成“倒空-Vi次”),就可以得到w升水了。

具体操作: 先求出各Vi。然后遍历所有的壶,往Vi是正数的壶里灌水,灌1次就把Vi减1。再把这些水到进Vi是负数的壶里,等某个壶灌满了,就把它倒空,然后给这个负的Vi加1,壶之间倒来倒去不变更各Vi的值。要注意的是要从池塘里灌水,一定要用空壶灌,要倒进池塘里的水,一定要是整壶的。这样一直到所有Vi都是0为止。
注意:为了得到较优解,应先把Vi是负数的壶全部处理好,等到所有的壶都不能往池中倒水了,再把Vi是正数的壶里的水倒入Vi等于0的壶中。 

我的程序:
#include <stdio.h> 
#include <stdlib.h> 
#include <math.h>
#define MAX 10 //所需壶的最大数量 

int Result(int cubage[], int times[], int n);
void GetElements(int cubage[], int times[], int n, int w);
void GetBest(int cubage[], int times[], int temp[], int n, int w, int top, int *min);
int IsStepEnd(int times[], int n);
void Process(int cubage[], int times[], int n);
int Sum(int cup[], int n);
int CommonDivisor(int m, int n);
int ComDiv(int cubage[], int n);

int main(void) 

    int cubage[MAX] = {0};//用来存储各壶的容积
    int times[MAX] = {0};//用来存储从池中加水(或倒水)次数,正数表示加水,负数表示倒水 
    int n; //壶的实际数量 
    int d;//存储所有壶的容积的最大公约数 
    int w;//存储最终需要的水的数量 
    int i;
    
    while (1)//可以处理多次 
    {
        puts("请输入壶的数量(输入0结束程序)");
        scanf("%d", &n);
        
        if (n == 0)//如果输入0,结束程序 
            break;
        
        puts("请输入各个壶的容积");
        for (i=0; i<n; i++)
            scanf("%d", &cubage[i]);
            
        puts("请输入最终需要的水的数量");
        scanf("%d", &w);
        
        d = ComDiv(cubage, n);//获取所有壶的容积的最大公约数 
        
        if (Sum(cubage, n) < w || w % d != 0)//如果不满足条件,则无解 
        {
            puts("NO");
            continue;
        }
        
        //获取各个壶从池中加水(或倒水)次数,times[i]>0表示从池中加水;times[i]<0表示往池中倒水 
        GetElements(cubage, times, n, w);
        for (i=0; i<n; i++)
            printf("%d: %d\n", cubage[i], times[i]);
            
        Process(cubage, times, n);//输出详细操作过程 
    }
        
      system("pause");
      return 0; 
}

void GetElements(int cubage[], int times[], int n, int w)
{
    int temp[MAX] = {0};//用来存储中间结果
    int min = 200; //存储最少的操作次数,初始化为200(假设操作次数不超过200) 
    
    GetBest(cubage, times, temp, n, w, 0, &min); 
}

//计算U1A1 + U2A2 + ...... + UnAn 
int Result(int cubage[], int times[], int n)
{
    int sum = 0;
    int i;
    
    for (i=0; i<n; i++)
        sum += times[i] * cubage[i];
        
    return sum;
}

//求满足 ∑times[i] * cubage[i] = w的最佳times[]
void GetBest(int cubage[], int times[], int temp[], int n, int w, int top, int *min)
{    
    int i, sum;
    
    if (top == n)
    {
        if (Result(cubage, temp, n) == w)//如果满足条件 
        {
            sum = Sum(temp, n);//计算加(倒)水次数之和 
            if (sum < *min)//取加(倒)水次数之和最小的结果作为最佳结果(意味着操作次数最少) 
            {
                for (i=0; i<n; i++)//存储最佳结果 
                    times[i] = temp[i];
                *min = sum;
            }
        }
    }
    else//回溯寻找最佳结果 
    {
        for (i=-10; i<11; i++)
        {
            temp[top] = i;//存储中间过程 
            GetBest(cubage, times, temp, n, w, top+1, min);
        }
    }
}

int IsEnd(int times[], int n)//判断是否已经处理完毕,处理完毕返回1,否则返回0 
{
    int i;
    
    for (i=0; i<n; i++)//当所有的times[i] == 0时,表示处理完毕 
    {
        if (times[i] != 0)
            return 0;
    }
    
    return 1;
}

int IsStepEnd(int times[], int n)//判断是否已经把该倒水的壶处理完毕,处理完毕返回1,否则返回0 
{
    int i;
    
    for (i=0; i<n; i++)//当所有的times[i] >= 0时,表示所有的壶都不再往池中倒水 
    {
        if (times[i] < 0)
            return 0;
    }
    
    return 1;
}

void Process(int cubage[], int times[], int n)//输出详细操作过程 
{
    int water[MAX] = {0};//存储各个壶当前的储水量 
    int count = 0;//累计操作次数 
    int need, fact;//need表示当前壶需要加入的水量,fact表示当前壶实际加入的水量 
    int i, j, k;
    
    //输出表头 
    printf("step: ");
    for (j=0; j<n; j++)
        printf("c[%d] ", j);
    for (j=0; j<n; j++)
        printf("t[%d] ", j);
    printf("\n");
    printf("step: ");
    for (j=0; j<n; j++)
        printf("%3d  ", cubage[j]);
    for (j=0; j<n; j++)
        printf("%3d  ", times[j]);
    printf("\n\n");
                    
    while (!IsStepEnd(times, n))//一直处理,直到已经把该倒水的壶处理完毕 
    {
        for (i=0; i<n; i++)//遍历数组,进行从池中加水和倒水操作 
        {
            if (water[i] == 0 && times[i] > 0)//若当前壶空,且仍可以进行加水操作,则从池中加水 
            {
                water[i] = cubage[i];
                --times[i];
                //输出操作步骤 
                printf("%4d: ", ++count);
                for (j=0; j<n; j++)
                    printf("%3d  ", water[j]);
                for (j=0; j<n; j++)
                    printf("%3d  ", times[j]);
                printf("\n");
            }
            if (water[i] == cubage[i] && times[i] < 0)//若当前壶满,且仍可以进行倒水操作,则往池中倒水
            {
                water[i] = 0;
                ++times[i];
                //输出操作步骤     
                printf("%4d: ", ++count);
                for (j=0; j<n; j++)
                    printf("%3d  ", water[j]);
                for (j=0; j<n; j++)
                    printf("%3d  ", times[j]);
                printf("\n");
            }
        }
        //遍历数组,壶之间倒来倒去,以调整各壶的储水量 
        for (i=0; !IsStepEnd(times, n) && i<n; i++)
        {  
            if (times[i] < 0)//如果当前壶仍可以进行倒水操作,则从别的壶倒水给它,等它满后就可往池中倒水了 
            {
                for (j=0; j<n; j++)//遍历所有的壶,寻找可以倒水给cup[i]的壶,注意:可倒多次 
                {
                    need = cubage[i] - water[i];//计算cup[i]需要加入的水量 
                    if (need == 0)//如果不需要加水,退出循环 
                        break;
                    
                    if (times[j] >= 0)//cup[j]不能往池中倒水了,可以把它的水倒给cup[i] 
                    {
                        fact = water[j] < need ? water[j] : need;//计算cup[i]实际加入的水量
                        water[i] += fact;
                        water[j] -= fact;
                        
                        if (fact > 0)//如果真的进行了倒水操作,则输出该过程 
                        {
                            printf("%4d: ", ++count);
                            for (k=0; k<n; k++)
                                printf("%3d  ", water[k]);
                            for (k=0; k<n; k++)
                                printf("%3d  ", times[k]);
                            printf("\n");
                        }
                    }
                }
            } 
        }
    }
    
    while (!IsEnd(times, n))//一直操作直至结束 
    {
        for (i=0; i<n; i++)//遍历数组,进行从池中加水操作 
        {
            if (water[i] == 0 && times[i] > 0)//若当前壶空,且仍可以进行加水操作,则从池中加水 
            {
                water[i] = cubage[i];
                --times[i];
                //输出操作步骤 
                printf("%4d: ", ++count);
                for (j=0; j<n; j++)
                    printf("%3d  ", water[j]);
                for (j=0; j<n; j++)
                    printf("%3d  ", times[j]);
                printf("\n");
            }
        }
        //遍历数组,壶之间倒来倒去,以调整各壶的储水量 
        for (i=0; !IsEnd(times, n) && i<n; i++)
        {  
            if (times[i] == 0)//cup[i]不能从池中取水和倒水了,可以把其他壶中的水倒给它
            {
                for (j=0; j<n; j++)//遍历所有的壶,寻找可以倒水给cup[i]的壶,注意:可倒多次 
                {
                    need = cubage[i] - water[i];//计算cup[i]需要加入的水量 
                    if (need == 0)//如果不需要加水,退出循环 
                        break;
                    
                    if (times[j] > 0)//若cup[j]可以从池中取水,则把它的水倒给cup[i] 
                    {
                        fact = water[j] < need ? water[j] : need;//计算cup[i]实际加入的水量
                        water[i] += fact;
                        water[j] -= fact;
                        
                        if (fact > 0)//如果真的进行了倒水操作,则输出该过程 
                        {
                            printf("%4d: ", ++count);
                            for (k=0; k<n; k++)
                                printf("%3d  ", water[k]);
                            for (k=0; k<n; k++)
                                printf("%3d  ", times[k]);
                            printf("\n");
                        }
                    }
                }
            } 
        }
    }
}

//对所有壶的容积或加(倒)水次数求和
int Sum(int cup[], int n)
{
    int sum = 0;
    int i;
    
    for (i=0; i<n; i++)
        sum += abs(cup[i]);//注意因为times[]的元素值有正有负,故累计次数时应取绝对值 
    
    return sum;
}

//返回m,n的最大公约数 
int CommonDivisor(int m, int n)
{
    int temp;
    
    while (n > 0)
    {
        temp = m % n;
        m = n;
        n = temp;
    }
    return m;
}

//返回所有壶的容积的最大公约数
int ComDiv(int cubage[], int n)
{
    int d = cubage[0];
    int i;
    
    for (i=1; i<n; i++)
        d = CommonDivisor(d, cubage[i]);
    
    return d;
}

回复列表 (共6个回复)

沙发

上面的程序在求times[i]时采用的是回溯的方法,因为要寻求最优解,非常耗时, 当n值超过5时不能忍受.我注意到所谓的最优是从池中加水和倒水的次数最少,所以我们穷举寻找times[i]时,可以从0开始,然后依次判断1,-1,2,-2..直到10,-10(只考虑加(倒)水次数在10次以内的,多于10次的太麻烦,无实际意义--当然也可以增大次数,但耗时过多),只要找到满足条件的解,立即跳出,这样可以节省大把时间,但得到的解不是最优解.

修改了两个函数,其他部分不变如下:
void GetElements(int cubage[], int times[], int n, int w)
{
    int flag = 0;
    GetBest(cubage, times, n, w, 0, &flag); 
}
void GetBest(int cubage[], int times[], int n, int w, int top, int *flag)
{    
    int i, sum;
    
    if (top == n)
    {
        if (Result(cubage, times, n) == w)//如果满足条件 
            *flag = 1;
    }
    else//回溯寻找最快,但非最佳结果 
    {
        for (i=0; i<11; ++i)
        {
            times[top] = i;//存储中间过程 
            GetBest(cubage, times, n, w, top+1, flag);
            if (*flag == 1)
                return;
                
            times[top] = -i;//存储中间过程 
            GetBest(cubage, times, n, w, top+1, flag);
            if (*flag == 1)
                return;
        }
    }
}

板凳

留个记号
慢慢看

3 楼


[quote]如果A1,A2,……,An是n个整数,(A1,A2,......,An)=s,那么存在整数U1,U2,……,Un,使得
       U1A1 + U2A2 + ...... + UnAn = s.    (*)
[/quote]

这个结论的证明好象没看到

4 楼

[quote]要注意的是要从池塘里灌水,一定要用空壶灌,要倒进池塘里的水,一定要是整壶的。
[/quote]

为什么?
尤其是为什么“要倒进池塘里的水,一定要是整壶的”

5 楼

[quote]#include <math.h>[/quote]

这个用的有些夸张了吧?
我在代码中找了半天
没有发现使用这个编译预处理命令的理由

6 楼

蒽,看不懂,留着慢慢琢磨~

我来回复

您尚未登录,请登录后再回复。点此登录或注册