だから私はこのフォーマットのテキストファイルからロードされたCの不等式パズルを解くはずのこのプログラムを持っています:
5 0 | 0 | 0 | 0 | 0 --- --- v --- --- 0> 0 | 0 | 0 | 3 ---- ---- ---- ---- ---- 0 | 0 <2 | 0 | 0 --- --- v --- --- 0 | 0 | 0 | 0 | 4 ^ --v ---- ---- ---- 0 | 0 | 0 | 0 | 0
ここで、5は行列のサイズであり、演算子に隣接する数値、、、は、それら<
によって>
課せられる条件を満たす必要があります。ファイルから、行のすべての文字がスペースで分割されます。ファイル、それが数学演算子の条件を満たしているかどうかを確認するために、しかし私は再帰関数で立ち往生しています^
v
0 |
私が知りたいこと:
行列を格納する正しい方法を選択しましたか、それとも論理演算子から数値を除算する必要がありましたか?
マトリックスに対して再帰的な展開を実行するにはどうすればよいですか?また、特定のステップで使用された数を追跡するにはどうすればよいですか(バックトラックする必要がある場合)?
例えば。index[j][j]
(行列のサイズ)の場所に到達したとしましょう。j<n
そこから始めて、数値のみをデクリメントj
(「タッチ」)して、部分行列が条件を満たすかどうかを確認する必要があります。
これが私がこれまでにコーディングできたものです。
どこ :
char **readmat(int *n);
//ファイルから行列を読み取り、文字間のスペースを削除します
void print(char **mat,int n);
//保存された行列を出力します
int check(char **mat,int n);
//サイズnの行列の項目が数学演算子を満たすかどうかをチェックします
int expand (char **mat,int n,int i);
//これは、一度に要素を取得し、満たす必要のある条件があるかどうかをチェックする再帰関数である必要があります。満たす必要がある場合は、それをインクリメントします
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
char **readmat(int *n);
void print(char **mat,int n);
int check(char **mat,int n);
int expand (char **mat,int n,int i);
int main(int argc, char *argv[])
{
char **mat;
int n, j;
mat=readmat(&n);
if(mat == NULL)
return 1;
if(check(mat,n)){
print(mat,n);
}
else if(expand(mat,n,0)==1){
print(mat,n);
}
else {
printf("Nessuna soluzione trovata.\n");
}
for(j=0; j<=n;j++)
free(mat[j]);
free(mat);
system("PAUSE");
return 0;
}
char **readmat(int *n){
FILE *fp;
char *line,nome[100];
int i,j,k;
char **mat;
printf("Inserire il nome del file: ");
scanf("%s",nome);
fp=fopen(nome,"r");
if(fp==NULL){
printf("Errore apertura file");
return NULL;
}
if(fgets(nome,100,fp)==NULL){
printf("Formato file non valido\n");
fclose(fp);
return NULL;
}
if(sscanf(nome,"%d",n)!=1){
printf("Errore nei parametri del file\n");
fclose(fp);
return NULL;
}
(*n)=(((*n)*2)-1);
mat=(char**)malloc((*n)*sizeof(char*));
for(i=0;i<=(*n);i++)
mat[i]=(char*)malloc((*n)*sizeof(char));
line=(char*)malloc(2*(*n)*sizeof(char));
i=0;
while(i<=2*(*n) && fgets(line,2*(*n)+2,fp)!=NULL){
j=0;
k=0;
while(j<=2*(*n)){
if(line[j]!=' '){
mat[i][k]=line[j];
k++;
}
j++;
}
i++;
}
return mat;
//print(mat, (*n));
}
void print(char **mat,int n){
int i=0,j=0;
for (i=0; i<n; i++) {
for (j=0; j<n; j++) {
printf("%c", mat[i][j]);
}
printf("\n");
}
}
int check(char **mat,int n) {
int i,j;
int k=1;
for(i=0;i<n;i++){
for(j=0;j<n;j++){
if(mat[i][j]=='<'){
if(mat[i][j-1] >= mat[i][j+1])
k=0;
}
else if(mat[i][j]=='>'){
if(mat[i][j-1] <= mat[i][j+1])
k=0;
}
else if(mat[i][j]=='^'){
if(mat[i-1][j] >= mat[i+1][j])
k=0;
}
else if(mat[i][j]=='v'){
if(mat[i-1][j] <= mat[i+1][j])
k=0;
}
}
}
return k;
}
int expand (char **mat,int n,int i){
int j=i/n;
int k=i%n;
int p;
if(i>=n*n){
return 1;
}
else{
if((mat[j][k]>47)&&(mat[j][k]<58)){
if(mat[j][k]=='0'){
expand(mat,n,i+2);
}
for (p=(mat[j][k]-48); p<(10-(mat[j][k]-48)); p++) {
mat[j][k]=48+p;
if (check(mat,i)) {
if (expand(mat, n, i+2)) {
return 1;
}
}
}
i-=2;
mat[j][k]='0';
}
}
return 0;
}
例の解決策:論理条件領域が明確に満たされていることがわかります。
0 | 0 | 1 | 0 | 0
- - - - v - - - -
1 > 0 | 0 | 0 | 3
- - - - - - - - -
0 | 0 < 2 | 0 | 0
- - - - v - - - -
0 | 1 | 0 | 0 | 4
^ - v - - - - - -
1 | 0 | 0 | 0 | 0