http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1101

题目大意:

         游戏开始时,他们各自将自己的赌注盖住,同时任何两个赌徒的赌注是不同的,如果其中一个赌徒没有钱了,他可以借一些筹码,但是他的赌注就是负数了。假设,他们的赌注都是整数。

         然后他们揭开所有的赌注。赢家为:他的赌注是其他三个人的赌注的综合。如果赢家不止一家,那么拥有最大赌注的人是赢家。

算法分析:

         对于此题,求三个数n1,n2,n3使其之和为另一个数字n。如果存在多种情况,求n值最大的一个。所以首先将所有的数字从小到大排列,从后向前求解,这样就保证求得的第一个赢家为最大值。

         另外三个数就需要枚举了。三个赌徒,至少有一个在赢家(i)的前面(j),但是由于存在负数,所以另外两个赌徒的编号就有可能在赢家的后面,最初考虑这题的时候,忽略了这点(如果没有负数,三个赌徒肯定都在赢家的前面了)。另外就是如果三个数都要考枚举的话,算法复杂度就高了。但是由于前面已经对数组进行排序,所以可以先枚举其中两个(j,k),则第三个赌徒的赌注为tmp  = jetton[i] – jetton[j] – jetton[k]; 然后进行二分查找tmp,如果存在,则结束枚举就行了,这样可以降低算法复杂度,不过仍为O(n^3lgn)

#include<iostream>
#include<algorithm>
using namespace std;

#define MAXN 1002

int n,jetton[MAXN];

int search(int k,int val)    //二分查找tmp
{
    int left,right,mid;
    left = k+1;    right = n-1;
    while(left<=right)
    {
        mid = (left+right)>>1;
        if(jetton[mid] == val)    return mid;
        if(jetton[mid] < val)    left = mid+1;
        else right = mid-1;
    }
    return 0;
}

int work()
{
    int i,j,k,tmp,pos;
    for(i=n-1;i>0;i--)//从后向前查找
    {
        for(j=0;j<i;j++)    //第一个
        {
            for(k=j+1;k<n;k++)    //第二个
            {
                tmp = jetton[i]-jetton[j]-jetton[k];
                pos = search(k,tmp);    //查找第三个是否存在
                if(pos && pos != i)    //如果符合要求,直接返回
                {
                    return i;
                }
            }
        }
    }
    return -1;
}

int main()
{
    int i;
    while(cin>>n && n)
    {
        for(i=0;i<n;i++)
            cin>>jetton[i];
        sort(jetton,jetton+n);    //从小到大排列
        int ans = work();
        if(ans==-1)    cout<<"no solution"<<endl;
        else cout<<jetton[ans]<<endl;
    }
    return 0;
}

 

posted on 2012-09-02 22:14  pcoda  阅读(1059)  评论(0编辑  收藏  举报