導航:首頁 > 源碼編譯 > 24點演算法編程

24點演算法編程

發布時間:2024-05-24 15:08:28

Ⅰ C語言24點的演算法

下面是我自己寫的一個程序:

我的解法是把這個問題分解成了兩個子問題,首先求出4個數字的無重復全排列,放到一個數組裡面,再對沒一個排列情況,從頭到尾窮舉所有的四則運算情況。注意到除法是特殊的,我用x/y表示x除以y,用x|y表示x分之y。注意到,如果窮舉的解得到-24的話,只需要把有減法的地方調換一下順序就可以了,代碼如下
/***********************************************************************************/
#include<stdio.h>
#include<stdlib.h>
#include<math.h>
#include<string.h>
int index[4]={0,1,2,3};//used to generate subscription collection
int sub[4]; //used in p() only
float f[4]={8.0f,3.0f,3.0f,8.0f};//the 24 point numbers
float fs[24][4];//all possible permutaions of f
float tmp[4]; //used for buf
int g_number=0; //number of permutations
float RES[4];
char op[3];
void p(int idx){//求全排列的函數
if(idx==4){
for(int i=0;i<4;++i){tmp[i]=f[sub[i]];}
for(int g=0;g<g_number;++g){if(memcmp(fs[g],tmp,sizeof(float)*4)==0)return;}
for(int i=0;i<4;++i){fs[g_number][i]=f[sub[i]];}
g_number++;
return;
}
for(int i=0;i<4;++i){//make subscription collections
bool pflag=false;
for(int j=0;j<idx;++j){if(sub[j]==i)pflag=true;}
if(pflag==true)continue;
sub[idx]=index[i];
p(idx+1);
}
}
void solve(int L){//對某個排列,遞歸求所有四則運算的結果,找到就退出
if(L==3){
if(fabs(fabs(RES[L])-24.0f)<0.01f){
printf("Found solution,RES=%f,((%d%c%d)%c%d)%c%d\n",RES[L],
(int)f[0],op[0],
(int)f[1],op[1],
(int)f[2],op[2],
(int)f[3]);
exit(0);
}
return;
}
for(int j=0;j<5;++j){//j judges for operators
if(j==0){RES[L+1]=RES[L]+tmp[L+1];op[L]='+';solve(L+1);}
if(j==1){RES[L+1]=RES[L]-tmp[L+1];op[L]='-';solve(L+1);}
if(j==2){RES[L+1]=RES[L]*tmp[L+1];op[L]='*';solve(L+1);}
if(j==3&&tmp[L+1]!=0)
{RES[L+1]=RES[L]/tmp[L+1];op[L]='/';solve(L+1);}
if(j==4&&RES[L+1]!=0)
{RES[L+1]=tmp[L+1]/RES[L];op[L]='|';solve(L+1);}
}
}
int main(int argc,char* argv[]){//should avoid 0
f[0]=atoi(argv[1]);
f[1]=atoi(argv[2]);
f[2]=atoi(argv[3]);
f[3]=atoi(argv[4]);
p(0);
for(int i=0;i<g_number;++i){
memcpy(tmp,fs[i],sizeof(float)*4);
RES[0]=tmp[0];
for(int t=0;t<4;++t){ printf("%d,",(int)tmp[t]); }
printf("\n");
solve(0);
}
printf("Found no solution :( \n");
return 0;
}

----------編譯運行,運行時的參數就是4個數字
g++ p.cpp && ./a.out 1 5 5 5
1,5,5,5,
Found solution,RES=-24.000000,((1/5)-5)*5
g++ p.cpp && ./a.out 8 3 3 8
8,3,3,8,
Found solution,RES=-24.000006,((8/3)-3)|8
上面這個解寫出來就是
8
--------- = 24
3-(8/3)
主程序為了簡化,省去了對輸入的檢查,樓主可以自己添加。

python編程:任意輸入4個正整數,編程24點

from __future__ import division

import itertools
n = raw_input('input 4 number sep by comma: 1,2,3,4 -> ')
t = list(itertools.permutations(n.split(','),4))
x = list(itertools.proct(* (['+', '-', '*', '/'],) * 3))
for r in t:
for i in x:
if eval(("(((%s%s%s)%s%s)%s%s)") % (r[0],i[0], r[1], i[1], r[2], i[2], r[3])) == 24 :
print ("(((%s%s%s)%s%s)%s%s)=24") % (r[0],i[0], r[1], i[1], r[2], i[2], r[3])

暴力法行么?

Ⅲ 二十四點演算法是什麼

先取三個數,使它的結果為24,容易想到2×7+10=24,這樣一來,由此構造一個帶分數,使它含有2、7、10這個分數,2或這個帶分數乘以7其結果為24,列式為(2+10÷7)×7=24

Ⅳ C++ 編程,24點游戲

#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<ctime>
usingnamespacestd;

intin[5];

intcalc(intoperName,intx,inty)
{
if(operName==1)//+
{
returnx+y;
}

if(operName==2)//-
{
if(x-y<=0)
{
return-3456731;
}

returnx-y;
}

if(operName==3)//*
{
if(x*y<=0)
{
return-464652;
}

returnx*y;
}

if(operName==4)///
{
if(x*y==0||x<y||x%y!=0)
{
return-5643212;
}

returnx/y;
}
}

charoper[]={'!','+','-','*','/'};

intrandom()
{
returnrand()%12+1;
}

intmain()
{
srand(time(NULL));

cout<<"Initialnumbersare:";
for(inti=1;i<=4;i++)
{
in[i]=random();
cout<<in[i]<<"";
}
cout<<endl;

sort(in+1,in+5);

do
{
for(inti=1;i<=4;i++)//3pos,filloperator
{
for(intj=1;j<=4;j++)
{
for(intk=1;k<=4;k++)
{
ints1=calc(i,in[1],in[2]);
ints2=calc(j,s1,in[3]);
ints3=calc(k,s2,in[4]);
if(s3==24)
{
if(in[1]>=in[2]&&s1>=in[3]&&s2>=in[4])
{
cout<<in[1]<<oper[i]<<in[2]<<'='<<s1<<endl;
cout<<s1<<oper[j]<<in[3]<<'='<<s2<<endl;
cout<<s2<<oper[k]<<in[4]<<'='<<s3<<endl;
return0;
}
}


s1=calc(j,in[2],in[3]);
s2=calc(i,in[1],s1);
s3=calc(k,s2,in[4]);
if(s3==24)
{
if(in[2]>=in[3]&&in[1]>=s1&&s2>=in[4])
{
cout<<in[2]<<oper[j]<<in[3]<<'='<<s1<<endl;
cout<<in[1]<<oper[i]<<s1<<'='<<s2<<endl;
cout<<s2<<oper[k]<<in[4]<<'='<<s3<<endl;
return0;
}
}


s1=calc(j,in[2],in[3]);
s2=calc(k,s1,in[4]);
s3=calc(i,in[1],s2);
if(s3==24)
{
if(in[2]>=in[3]&&s1>=in[4]&&in[1]>=s2)
{
cout<<in[2]<<oper[j]<<in[3]<<'='<<s1<<endl;
cout<<s1<<oper[k]<<in[4]<<'='<<s2<<endl;
cout<<in[1]<<oper[i]<<s2<<'='<<s3<<endl;
return0;
}
}


s1=calc(i,in[1],in[2]);
s2=calc(k,in[3],in[4]);
s3=calc(j,s1,s2);
if(s3==24)
{
if(in[1]>=in[2]&&in[3]>=in[4]&&s1>=s2)
{
cout<<in[1]<<oper[i]<<in[2]<<'='<<s1<<endl;
cout<<in[3]<<oper[k]<<in[4]<<'='<<s2<<endl;
cout<<s1<<oper[j]<<s2<<'='<<s3<<endl;
return0;
}
}


s1=calc(k,in[3],in[4]);
s2=calc(j,in[2],s1);
s3=calc(i,in[1],s2);
if(s3==24)
{
if(in[3]>=in[4]&&in[2]>=s1&&in[1]>=s2)
{
cout<<in[3]<<oper[k]<<in[4]<<'='<<s1<<endl;
cout<<in[2]<<oper[j]<<s1<<'='<<s2<<endl;
cout<<in[1]<<oper[i]<<s2<<'='<<s3<<endl;
return0;
}
}
}
}
}
}while(next_permutation(in+1,in+5));

cout<<"Noanswer!"<<endl;
return0;
}

Ⅳ 用c語言編寫24點代碼分析

#include<stdio.h>

double fun(double a1,double a2,int b) //用於嘗試著計算的函數,b為運算控制
{
switch(b)
{
case 0:return (a1+a2);
case 1:return (a1-a2);
case 2:return (a1*a2);
case 3:return (a1/a2);
}
}

void main()
{
int i,j,k,l,n,m,r,save[4];
double num[4]={1,1,1,1},tem1,tem2,tem3,abc=1111;
char sign[5]="+-*/"; //列印時候用的符號,需要和fun函數里的順序保持一致

printf("input 4 numbers:");
for(i=0;i<4;i++)
{
scanf("%lf",num+i); //輸入數據
save[i]=num[i]; //保存原始數據
}

//下面程序的思想,就是利用窮舉(其實就是使用的排列組合方法)來計算可能的組合。
//先把輸入的4個數進行排列(前4個for語句就這個用途)
//再依次插入三個運算符(後3個for語句就這個用途)
//事實上,從這里看,這個程序是不怎樣的。七層循環嵌套,這是編程的大忌。一般循環嵌套最好不要超過兩層。
for(i=0;i<4;i++)
for(j=0;j<4;j++)
if(j!=i)
{
for(k=0;k<4;k++)
if(k!=i&&k!=j)
{
for(l=0;l<4;l++)
if(l!=i&&l!=j&&l!=k)
{
for(n=0;n<4;n++)
for(m=0;m<4;m++)
for(r=0;r<4;r++)
{
tem1=fun(num[i],num[j],n);
tem2=fun(tem1,num[k],m);
tem3=fun(tem2,num[l],r);
//以下五種處理方法,涵蓋了有可能的全部運算順序
//這也是本程序最精妙的地方。

if(tem3==24.0)//如果直接算得了24,說明次序不變,直接輸出就是
printf("{(%d%c%d)%c%d}%c%d=24\n",save[i],sign[n],save[j],sign[m],save[k],sign[r],save[l]);
else if(tem3==-24.0)//如果算得的是負的,說明需要顛倒第二次運算(第三次運算不可能是加減)
printf("{%d%c(%d%c%d)}%c%d=24\n",save[k],sign[m],save[i],sign[n],save[j],sign[r],save[l]);
else if(tem3==1.0/24.0)//如果是倒數,說明需要顛倒最後一次運算(第三次運算同樣不可能是加減)
printf("%d%c{(%d%c%d)%c%d}=24\n",save[l],sign[r],save[i],sign[n],save[j],sign[m],save[k]);
else if(tem3==-1.0/24.0)//如果是負倒數,則說明第二次和第三次運算都要顛倒(第三次運算同樣不可能是加或減)
printf("%d%c{%d%c(%d%c%d)}=24\n",save[l],sign[r],save[k],sign[n],save[i],sign[m],save[j]);
else
{ //處理()*/+/-()的情況
tem1=fun(num[i],num[j],n);
tem2=fun(num[k],num[l],r);
tem3=fun(tem1,tem2,m);
if(tem3==24.0)
printf("(%d%c%d)%c(%d%c%d)=24\n",save[i],sign[n],save[j],sign[m],save[k],sign[r],save[l]);
}
}
}
}
}
}

//後面我再研究了下,發現"第三次不可能是加減法"這種思想是錯誤的,而程序作者在設計的時候,確實是這么認為的,所以,這個程序是有問題的.
//但程序里的主體思想沒有問題,如果需要修改這個錯誤,程序需要在運算順序判斷上下功夫.結果只能取==24的情況.

Ⅵ 24點游戲C語言編程。要求 游戲給出4個1-13的整數 用戶輸入一個含這4個整數的四則運算表達式

輸入:A 2 3 4 5 6 7 8 9 10 J Q K
輸出:Yes/No
#include <stdio.h>
#include <string.h>

char t[10][10];
int b[5][1000],bx[3][200],by[3][200],g[5],p[5];
int i,j,k;
bool f;

int main()
{
while (~scanf("%s",&t[0]))
{
for (i=1;i<4;i++)
scanf("%s",&t[i]);
for (i=0;i<4;i++)
if (t[i][0]=='A')
g[i+1]=1;
else if (t[i][0]=='J')
g[i+1]=11;
else if (t[i][0]=='Q')
g[i+1]=12;
else if (t[i][0]=='K')
g[i+1]=13;
else if (t[i][0]=='1')
g[i+1]=10;
else
g[i+1]=t[i][0]-48;
memset(b,255,sizeof(b));
for (p[1]=1;p[1]<=4;p[1]++) if (b[4][24]!=4)
for (p[2]=1;p[2]<=4;p[2]++) if ((p[1]!=p[2]) && (b[4][24]!=4))
for (p[3]=1;p[3]<=4;p[3]++) if ((p[1]!=p[3]) && (p[2]!=p[3]) && (b[4][24]!=4))
{
p[4]=10-p[1]-p[2]-p[3];
memset(b,255,sizeof(b));
b[1][g[p[1]]]=1;
for (i=2;i<=4;i++)
for (j=500;j>0;j--)
if (b[i-1][j]==i-1)
{
b[i][j+g[p[i]]]=i;
if (j>g[p[i]])
b[i][j-g[p[i]]]=i;
if (j<g[p[i]])
b[i][g[p[i]]-j]=i;
if (j*g[p[i]]<1000)
b[i][j*g[p[i]]]=i;
if (j%g[p[i]]==0)
b[i][j/g[p[i]]]=i;
if ((j!=0) && (g[p[i]]%j==0))
b[i][g[p[i]]/j]=i;
}
if (b[4][24]!=4)
{
memset(bx,255,sizeof(bx));
memset(by,255,sizeof(by));
bx[1][g[p[1]]]=1;
bx[2][g[p[1]]+g[p[2]]]=2;
if (g[p[1]]>g[p[2]])
bx[2][g[p[1]]-g[p[2]]]=2;
if (g[p[1]]<g[p[2]])
bx[2][g[p[2]]-g[p[1]]]=2;
bx[2][g[p[1]]*g[p[2]]]=2;
if (g[p[1]]%g[p[2]]==0)
bx[2][g[p[1]]/g[p[2]]]=2;
if (g[p[2]]%g[p[1]]==0)
bx[2][g[p[2]]/g[p[1]]]=2;
by[1][g[p[3]]]=1;
by[2][g[p[3]]+g[p[4]]]=2;
if (g[p[3]]>g[p[4]])
by[2][g[p[3]]-g[p[4]]]=2;
if (g[p[3]]<g[p[4]])
by[2][g[p[4]]-g[p[3]]]=2;
by[2][g[p[3]]*g[p[4]]]=2;
if (g[p[3]]%g[p[4]]==0)
by[2][g[p[3]]/g[p[4]]]=2;
if (g[p[4]]%g[p[3]]==0)
by[2][g[p[4]]/g[p[3]]]=2;
for (i=200;i>0;i--)
if (bx[2][i]==2)
for (j=200;j>0;j--)
if (by[2][j]==2)
if ((i+j==24) || (i-j==24) || (j-i==24) || (i*j==24) || ((i%j==0) && (i/j==24) || ((j%i==0) && (j/i==24))))
{
b[4][24]=4;
break;
}
}
}
if (b[4][24]==4)
puts("Yes");
else
puts("No");
}
return 0;
}

閱讀全文

與24點演算法編程相關的資料

熱點內容
阿里雲物理伺服器 瀏覽:951
靈狐視頻app哪個好 瀏覽:255
大廠退役程序員自述 瀏覽:252
linux命令watch 瀏覽:889
加密幣哪些平台不撤出中國 瀏覽:553
max加線命令 瀏覽:424
app胖瘦模式哪個好用 瀏覽:724
可以下載源碼的軟體 瀏覽:487
程序員寫一天代碼累嗎 瀏覽:628
ie文件夾禁止訪問 瀏覽:543
百川互聯網程序員 瀏覽:783
linuxpython解釋器 瀏覽:667
興安得力軟體加密狗 瀏覽:494
智能網路攝像頭加密 瀏覽:574
軟體畢業程序員培訓 瀏覽:652
安卓陀螺儀低怎麼辦 瀏覽:247
一級建造師復習題集pdf 瀏覽:904
法理學pdf海默 瀏覽:393
伺服器內存儲器是用什麼的 瀏覽:819
微幫同城分類信息源碼 瀏覽:808