duda freopen

Iniciado por pony012, 19 Julio 2010, 16:10 PM

0 Miembros y 1 Visitante están viendo este tema.

pony012

bueno, pues según lo que leí el freopen sirve para abrir un archivo y asignarlo para el flujo de datos de entrada o de salida, por ejemplo, yo lo estoy usando para llenar unas matrices en un programa que intenta resolver algun sudoku, y mi problema es que no se como cerrar el archivo, y como lo asigno para que de ahi vengan los datos de entrada, pues ya no puedo aplicar algo para que el usuario salga del programa sin matarlo.
y esa es mi duda, como cerrar un archivo o volver a que los datos de entrada entren por el teclado y no por el archivo.

Por si lo quieren revisar, este es el codigo del programa, aun me falta mucho, pues ahora solo resuelve sudokus faciles, ya despues le intentare meter backtracking, pero por ahora es solo esto  :P
Código (cpp) [Seleccionar]

#include <iostream>

using namespace std;

int sudoku[9][9], matriz[9][9], seguros[9][9], posible=0, unicas=1;
bool usado[10];

void imprimir();
void unicas_soluciones();
void comprobar(int a, int b);

int main()
{
   
   
    freopen("sudoku.txt","r",stdin);
    for(int i=0; i<9; i++){
            for(int j=0; j<9; j++){
                    cin>>sudoku[i][j];
                    }
            }
    freopen("config.txt","r",stdin);
    for(int i=0; i<9; i++){
            for(int j=0; j<9; j++){
                    cin>>matriz[i][j];
                    }
            }
    for(int i=0; i<9; i++){
            for(int j=0; j<9; j++){
                    if(sudoku[i][j]==0)seguros[i][j]=0;
                    else seguros[i][j]=1;
                    }
            }
   
    imprimir();

    unicas_soluciones();
           
    imprimir();
           
           
    while(cin.get() != '\n')
    setbuf(stdin, '\0');
    cin.get();
    return 0;
}
void comprobar(int a, int b){
    int sub_matriz=matriz[a][b];
    int cont=0;
    if(seguros[a][b]==0){
                  for(int i=0; i<10; i++){
                          usado[i]=0;
                          }
                  for(int i=0; i<9; i++){
                               usado[sudoku[a][i]]=1;
                               usado[sudoku[i][b]]=1;
                          }
                  for(int i=0; i<9; i++){
                          for(int j=0; j<9; j++){
                            if(sub_matriz==matriz[i][j]){
                                                         usado[sudoku[i][j]]=1;
                                                         }
                            }
                  }
                  if(unicas==1){
                          for(int i=0; i<10; i++){
                                  if(usado[i]==0)cont++;
                                  }
                          if(cont==1){
                                      posible=1;
                                      cout<<"Posibles para "<<b+1<<","<<a+1<<endl;
                                      for(int i=0; i<10; i++){
                                              if(usado[i]==0){
                                                             
                                                              seguros[a][b]=1;
                                                              sudoku[a][b]=i;
                                                              cout<<i<<endl;
                                                              }
                                              }
                                     
                                      }
                                      }
                          }
    }
void unicas_soluciones(){
     posible=0;
     for(int i=0; i<9; i++){
             for(int j=0; j<9; j++){
                     comprobar(i,j);
                     }
             }
     cout<<"_________________"<<endl;
     if(posible!=0) unicas_soluciones();
     unicas=0;
     }
void imprimir(){
     int counta=0,countb=0;
     for(int i=0; i<9; i++){
            for(int j=0; j<9; j++){
                         cout<<sudoku[i][j];
                         counta++;
                         if(counta==3||counta==6)cout<<" ";
                    }
                    if(counta==9)counta=0;
                    countb++;
                    if(countb==3||countb==6)cout<<endl;
                    cout<<endl;
            }
     }


y los archivos sudoku y config contienen:
sudoku.txt

0 0 0 0 4 3 0 0 6
0 0 0 0 0 0 4 8 0
2 4 8 0 0 0 0 0 7
3 2 0 0 0 6 8 0 9
1 8 0 3 0 0 6 0 2
6 0 0 0 0 0 7 3 5
0 7 0 0 0 0 5 6 0
5 0 2 6 0 0 0 0 3
8 6 0 0 5 0 0 0 0;

y
config.txt

1 1 1 2 2 2 3 3 3
1 1 1 2 2 2 3 3 3
1 1 1 2 2 2 3 3 3
4 4 4 5 5 5 6 6 6
4 4 4 5 5 5 6 6 6
4 4 4 5 5 5 6 6 6
7 7 7 8 8 8 9 9 9
7 7 7 8 8 8 9 9 9
7 7 7 8 8 8 9 9 9;

MIG80

#1
bueno.. no se si freopen() se puede usar dos veces para hacer que la entrada venga desde un archivo..
Código (cpp) [Seleccionar]

freopen("sudoku.txt","r",stdin);
...
freopen("config.txt","r",stdin);

pero creo que el segundo freopen() no es necesario dado que se puede inicializar el arreglo con los valores deseados en lugar hacerlo durante la ejecucion del programa
Código (cpp) [Seleccionar]

#include <iostream>

using namespace std;

int matriz[9][9]={
1,1,1,2,2,2,3,3,3,
1,1,1,2,2,2,3,3,3,
4,4,4,5,5,5,6,6,6,
4,4,4,5,5,5,6,6,6,
4,4,4,5,5,5,6,6,6,
7,7,7,8,8,8,9,9,9,
7,7,7,8,8,8,9,9,9,
7,7,7,8,8,8,9,9,9
};

int sudoku[9][9], seguros[9][9], posible=0, unicas=1;
bool usado[10];

void imprimir();
void unicas_soluciones();
void comprobar(int a, int b);

int main()
{...}

Og.

#2
Solo vi el codigo por arriba, el config.txt lo usas para saber en que cuadro esta el numero cierto?

yo haria esto:
Código (cpp) [Seleccionar]

bool* posibilidades(int i, int g)
{
   bool *p = new bool[9];
   for(int x=0; x<9; x++)
       p[x] = true;
   for(int x=0; x<9; x++)
   {
       if(sudoku[x][g]!=0)
           p[sudokuo[x][g]-1] = false;
       if(sudoku[i][x]!=0)
           p[sudoku[i][x]-1] = false;
   }
   int cuadro1, cuadro2;
   if(i<=2) cuadro1 = 0;
   else if(i<=5) cuadro1 = 3;
   else cuadro1 = 6;
   if(g<=2) cuadro2 = 0;
   else if(g<=5) cuadro2 = 3;
   else cuadro2 = 6;
   for(int l=0; l<3; l++)
       for(int n=0; n<3; n++)
           if(sudoku[l+cuadro1][n+cuadro2])
               p[sudoku[l+cuadro1][n+cuadro2]-1]=false;
   return p;
}


Esta función te regresara un arreglo de booleanos numerados del 0 al 8 que te dirán que ficha puede entrar.
Por ejemplo si te regresa: 000100011, significa que puedes meter un 4, 8 o 9.
A. no olvides liberar la memoria cuando la desocupes.

Saludos!

Edit: una duda, es la tarea de backtracing de padilla? (no me hagan caso si no son de Jalisco, tal ves estoy confundiéndome xD)
|-

MIG80

#3
lo mismo.. pero mas rapido ::)
Código (cpp) [Seleccionar]

bool *posibilidades(int a,int b)
{
 const int fi=(a/3)*3,ci=(b/3)*3;
 static bool p[10];
 for(int z=0;z<10;z++)
   p[z]=true;
 for(int k=0;k<9;k++)
 {
   p[sudoku[a][k]]=p[sudoku[k][b]]=p[sudoku[fi+k/3][ci+k%3]]=false;
 }
 return &p[1];
}

la verificacion de si la casilla esta ocupada o no se hace fuera de la funcion:
Código (cpp) [Seleccionar]

...
if(sudoku[a][b]==0)
 p=posibilidades(a,b);

y ya no hace falta liberar la memoria ocupada por 'p' despues de cada llamada
Saludos  :)

Littlehorse

Bueno, volviendo al tema principal!  :D

Citarbueno, pues según lo que leí el freopen sirve para abrir un archivo y asignarlo para el flujo de datos de entrada o de salida, por ejemplo, yo lo estoy usando para llenar unas matrices en un programa que intenta resolver algun sudoku, y mi problema es que no se como cerrar el archivo, y como lo asigno para que de ahi vengan los datos de entrada, pues ya no puedo aplicar algo para que el usuario salga del programa sin matarlo.
y esa es mi duda, como cerrar un archivo o volver a que los datos de entrada entren por el teclado y no por el archivo.

Lo primero es saber que no hay una forma estandar de hacer lo que pides, tal vez algo rebuscado se pueda lograr, pero no exactamente lo que pides. Los aspectos subyacentes especificos de cada sistema hacen que sea imposible que el estandar pueda abarcarlos todos.

Dicho eso, la mejor opción dependerá mayormente del sistema en el que estés trabajando. Así que basicamente podes abstraer en una librería los algoritmos que necesites dependiendo de los sistemas en los cuales quieres que el programa funcione, o bien utilizar alguna librería multi-plataforma ya desarrollada y testeada (como Boost, por ejemplo).


Algo como esto:

Código (cpp) [Seleccionar]
freopen("CON","r",stdin);

probablemente te funcione siempre y cuando estés en un sistema Windows, y obviamente, que los streams estén asociados a la consola. Eso si, teniendo en claro que seria la peor forma de hacerlo, por la falta de portabilidad y porque este método puede ser propenso a errores en el caso que las redirecciones tiendan a aumentar de complejidad. 

También cabe aclarar que realmente no veo que sea necesario redirigir los streams en la idea que plantea el algoritmo. Seria bastante mas eficiente y flexible elaborar un modulo que se encargue de la lectura y el procesado de un formato de archivo, y en todo caso en cambios futuros solo necesitarías cambiar ese modulo

Para no extenderme mucho mas, lo primero seria que aclares en que sistema necesitas que funcione para poder guiarte adecuadamente. Si la respuesta es elaborar un algoritmo multi-plataforma, probablemente encuentres algún código hecho con Boost si buscas sobre redirecciones de streams en Google.

Saludos!
An expert is a man who has made all the mistakes which can be made, in a very narrow field.