//**********************************************
//
// 假币算法求解(三分算法和二分算法)
// xwlee 2006/12/20
//**********************************************
#include<iostream>
#include<cmath>
#include <ctime>
#include <iomanip>
using namespace std;
int find2(int coin[], int low, int high);//二分算法
int find3(int coin[], int low, int high);//三分算法
int main(void)
{
int i, *coin,n,temp;
cout<<"请输入硬币的个数(n最大为机器整数值):";
cin>>n;
srand( (unsigned)time( NULL ) );//srand()函数产生一个以当前时间开始的随机种子
cout<<"正在随机产生数据(用0和1分别代表假币和真币).../n/n";
coin=new int[n+1];//多产生一个位置,用coin[0]来记录本次调用称重的次数
temp=1+rand()%n;//随机产生1~n之间的一个数
coin[0]=0;//计数器
for(i=1; i<n+1; ++i)
i!=temp ? coin[i]=1 : coin[i]=0;
for(i=1; i<n+1; ++i)
if(i!=temp)
cout<<"第"<<setw(3)<<i<<" 枚硬币是真币(1)"<<endl;
else
cout<<"第"<<setw(3)<<i<<" 枚硬币是假币(0)"<<endl;
cout<<endl<<endl<<"假币二分算法结果:"<<endl;
cout<<"假币在第"<<setw(3)<<find2(coin,1,n)<<"个位置!"<<endl;
cout<<"总共称的次数为"<<setw(3)<<coin[0]<<"次!"<<endl;
cout<<endl<<endl<<"假币三分算法结果:"<<endl;
cout<<"假币在第"<<setw(3)<<find3(coin,1,n)<<"个位置!"<<endl;
cout<<"总共称的次数为"<<setw(3)<<coin[0]<<"次!"<<endl<<endl<<endl;
return 0;
}
int find2(int coin[], int low, int high)
{
int i,mid,sum1,sum2,sign;
coin[0]=0;
do //下界和上界所指示的初始位置
{
mid=(low+high)/2;
sign= (low+high)%2;//标记奇数和偶数
//cout<<sign<<endl;
sum1=sum2=0;
if(sign==0)//处理奇数的情况
{
for(i=low; i<mid; i++)
sum1 += coin[i];
for(i=mid+1; i<=high; i++)
sum2 += coin[i];
if( sum1==0 && sum2==0 )
;
else
coin[0] += 1;
if(sum1==sum2)
return mid;
else if(sum1<sum2)
{
high=mid-1;
}
else
{
low=mid+1;
}
}
else//处理偶数的情况
{
for(i=low; i<=mid; i++)
sum1 += coin[i];
for(i=mid+1; i<=high; i++)
sum2 += coin[i];
if( sum1==0 && sum2==0 )
;
else
coin[0] += 1;
if(sum1<sum2)
high=mid;
else
low=mid+1;
}
}while(1);//end while
return -1;
}
int find3(int coin[], int low, int high)
{
int i,mid1,mid2,sum1,sum2,sum3,flag,my_space;
coin[0]=0;
if(high<3)
{cout<<"您必须键入的个数要大于3"<<endl<<endl<<endl; exit(1);}
while(low<=high) //下界和上界所指示的初始位置
{
sum1=sum2=sum3=0;
my_space=(high-low+1)/3;
mid1=low+my_space-1;
mid2=mid1+my_space;
flag= (high-low+1) % 3;//标记被3除的余数,分别是余数为0,1,2
for(i=low; i<=mid1; i++)
sum1 += coin[i];
for(i=mid1+1; i<=mid2; i++)
sum2 += coin[i];
for(i=mid2+1; i<=mid2+my_space; i++) //注意这里
sum3 += coin[i];
if( sum1==0 && sum2==0 && sum3==0 )
;
else
coin[0] += 1;
if(flag==0)//处理余数为0的情况
{
if(sum1==sum2)
low=mid2+1;
if(sum1==sum3)
{ low=mid1+1; high=mid2; }
if(sum2==sum3)
high=mid1;
}
else if(flag==1)处理余数为1的情况
{
if( sum1==sum2)
{
if(sum1==sum3)
{ return high;}
else
{low=mid2+1; high=high-1;}
}
else
{
if(sum1==sum3)
{ low=mid1+1; high=mid2;}
if(sum2==sum3)
{ high=mid1; }
}
}
else处理余数为2的情况
{
if( sum1==sum2)
{
if(sum1==sum3)
{
coin[0] += 1;
if(coin[high-1]==0)
return high-1;
else
return high;
}
else
{low=mid2+1; high=high-2;}
}
else
{
if(sum1==sum3)
{ low=mid1+1; high=mid2;}
if(sum2==sum3)
{ high=mid1; }
}
}
//coin[0] += 1;
}//end while
return -1;
}
//
// 假币算法求解(三分算法和二分算法)
// xwlee 2006/12/20
//**********************************************
#include<iostream>
#include<cmath>
#include <ctime>
#include <iomanip>
using namespace std;
int find2(int coin[], int low, int high);//二分算法
int find3(int coin[], int low, int high);//三分算法
int main(void)
{
int i, *coin,n,temp;
cout<<"请输入硬币的个数(n最大为机器整数值):";
cin>>n;
srand( (unsigned)time( NULL ) );//srand()函数产生一个以当前时间开始的随机种子
cout<<"正在随机产生数据(用0和1分别代表假币和真币).../n/n";
coin=new int[n+1];//多产生一个位置,用coin[0]来记录本次调用称重的次数
temp=1+rand()%n;//随机产生1~n之间的一个数
coin[0]=0;//计数器
for(i=1; i<n+1; ++i)
i!=temp ? coin[i]=1 : coin[i]=0;
for(i=1; i<n+1; ++i)
if(i!=temp)
cout<<"第"<<setw(3)<<i<<" 枚硬币是真币(1)"<<endl;
else
cout<<"第"<<setw(3)<<i<<" 枚硬币是假币(0)"<<endl;
cout<<endl<<endl<<"假币二分算法结果:"<<endl;
cout<<"假币在第"<<setw(3)<<find2(coin,1,n)<<"个位置!"<<endl;
cout<<"总共称的次数为"<<setw(3)<<coin[0]<<"次!"<<endl;
cout<<endl<<endl<<"假币三分算法结果:"<<endl;
cout<<"假币在第"<<setw(3)<<find3(coin,1,n)<<"个位置!"<<endl;
cout<<"总共称的次数为"<<setw(3)<<coin[0]<<"次!"<<endl<<endl<<endl;
return 0;
}
int find2(int coin[], int low, int high)
{
int i,mid,sum1,sum2,sign;
coin[0]=0;
do //下界和上界所指示的初始位置
{
mid=(low+high)/2;
sign= (low+high)%2;//标记奇数和偶数
//cout<<sign<<endl;
sum1=sum2=0;
if(sign==0)//处理奇数的情况
{
for(i=low; i<mid; i++)
sum1 += coin[i];
for(i=mid+1; i<=high; i++)
sum2 += coin[i];
if( sum1==0 && sum2==0 )
;
else
coin[0] += 1;
if(sum1==sum2)
return mid;
else if(sum1<sum2)
{
high=mid-1;
}
else
{
low=mid+1;
}
}
else//处理偶数的情况
{
for(i=low; i<=mid; i++)
sum1 += coin[i];
for(i=mid+1; i<=high; i++)
sum2 += coin[i];
if( sum1==0 && sum2==0 )
;
else
coin[0] += 1;
if(sum1<sum2)
high=mid;
else
low=mid+1;
}
}while(1);//end while
return -1;
}
int find3(int coin[], int low, int high)
{
int i,mid1,mid2,sum1,sum2,sum3,flag,my_space;
coin[0]=0;
if(high<3)
{cout<<"您必须键入的个数要大于3"<<endl<<endl<<endl; exit(1);}
while(low<=high) //下界和上界所指示的初始位置
{
sum1=sum2=sum3=0;
my_space=(high-low+1)/3;
mid1=low+my_space-1;
mid2=mid1+my_space;
flag= (high-low+1) % 3;//标记被3除的余数,分别是余数为0,1,2
for(i=low; i<=mid1; i++)
sum1 += coin[i];
for(i=mid1+1; i<=mid2; i++)
sum2 += coin[i];
for(i=mid2+1; i<=mid2+my_space; i++) //注意这里
sum3 += coin[i];
if( sum1==0 && sum2==0 && sum3==0 )
;
else
coin[0] += 1;
if(flag==0)//处理余数为0的情况
{
if(sum1==sum2)
low=mid2+1;
if(sum1==sum3)
{ low=mid1+1; high=mid2; }
if(sum2==sum3)
high=mid1;
}
else if(flag==1)处理余数为1的情况
{
if( sum1==sum2)
{
if(sum1==sum3)
{ return high;}
else
{low=mid2+1; high=high-1;}
}
else
{
if(sum1==sum3)
{ low=mid1+1; high=mid2;}
if(sum2==sum3)
{ high=mid1; }
}
}
else处理余数为2的情况
{
if( sum1==sum2)
{
if(sum1==sum3)
{
coin[0] += 1;
if(coin[high-1]==0)
return high-1;
else
return high;
}
else
{low=mid2+1; high=high-2;}
}
else
{
if(sum1==sum3)
{ low=mid1+1; high=mid2;}
if(sum2==sum3)
{ high=mid1; }
}
}
//coin[0] += 1;
}//end while
return -1;
}