elhacker.net cabecera Bienvenido(a), Visitante. Por favor Ingresar o Registrarse
¿Perdiste tu email de activación?.

 

 


Tema destacado: Usando Git para manipular el directorio de trabajo, el índice y commits (segunda parte)


  Mostrar Mensajes
Páginas: [1] 2
1  Programación / Programación C/C++ / Re: [C++] Resultado extraño en una clase Matriz en: 17 Abril 2011, 21:28 pm
Bueno, tras un poco más de investigación y consultas, lo que fallaba principalmente en el código estaba en el constructor de copia y en la sobrecarga del operador asignación. Como se puede comprobar, ambos crean la estructura de las matrices, pero no les dan el valor de la matriz origen, por tanto, quedan en basura y provocan el mal funcionamiento. Este es el código reescrito:
Código
  1. ......
  2. ......
  3.    //Constructor de copia
  4.    Matriz (const Matriz &matriz){
  5.        NF=matriz.NF;
  6.        NC=matriz.NC;
  7.        //Estructura de matriz
  8.        M=new int *[NF];
  9.        for(int i=0;i<NF;i++){
  10.            M[i]=new int[NC];
  11.        }
  12.        //Contenido de matriz
  13.        for(int k=0;k<NF;k++){
  14.            for(int l=0;l<NC;l++){
  15.                M[k][l]=matriz.M[k][l];
  16.            }
  17.        }
  18.    }
  19. .....
  20. ......
  21.    //Operador asignacion
  22.    Matriz operator = (Matriz m){
  23.        if(NF!=m.NF){
  24.            for(int i=0;i<NF;i++){
  25.                delete []M[i];
  26.            }
  27.            delete []M;
  28.        }
  29.        //Estructura de matriz
  30.        NF=m.NF;
  31.        NC=m.NC;
  32.        M=new int *[NF];
  33.        for(int j=0;j<NF;j++){
  34.            M[j]=new int [NC];
  35.        }
  36.        //Contenido de matriz
  37.        for(int k=0;k<m.NF;k++){
  38.            for(int l=0;l<m.NC;l++){
  39.                M[k][l]=m.M[k][l];
  40.            }
  41.        }
  42.        return(*this);
  43.    }
  44. .....
  45. ......

El tema del return no tiene nada que ver. Igual es por el compilador que utilizo, pero no tengo problemas con los paréntesis ni sin ellos. Gracias por el interés, de todos modos.

Sin embargo, creo que nunca descubriré por qué devolvía 3.... :-\
2  Programación / Programación C/C++ / [C++] Resultado extraño en una clase Matriz en: 8 Abril 2011, 19:38 pm
Saludos. A día de hoy me encuentro realizando un programilla que realice operaciones sencillas con matrices. Concretamente, sumas, restas y multiplicaciones. Como todos sabemos, para sumar o restar matrices, deben ser del mismo tamaño, y para multiplicarlas, el número de columnas de la primera debe ser igual al número de filas de la segunda. En este programa en concreto, si no se dan esas características, la operación debe devolver una matriz nula, de 0x0 elementos. Pongo la parte interesante del código y sigo con el problema:
Código
  1. #include <iostream>
  2. #include <stdlib.h>
  3. #include <conio.h>
  4. #include <time.h>
  5.  
  6. using namespace std;
  7.  
  8. //Clase Matriz
  9. class Matriz{
  10. protected:
  11.    int **M;
  12.    int NF,NC;
  13. public:
  14.    //Constructor por defecto
  15.    Matriz (){
  16.        NF=0;
  17.        NC=0;
  18.        M=NULL;
  19.    }
  20.    //Constructor de copia
  21.    Matriz (const Matriz &matriz){
  22.        NF=matriz.NF;
  23.        NC=matriz.NC;
  24.        M=new int *[NF];
  25.        for(int i=0;i<NF;i++){
  26.            M[i]=new int[NC];
  27.        }
  28.    }
  29.    //Constructor con parametros
  30.    Matriz (int nf, int nc){
  31.        M=new int *[nf];
  32.        for(int i=0;i<nf;i++){
  33.            M[i]=new int[nc];
  34.        }
  35.        for(int i=0;i<nf;i++){
  36.            for(int j=0;j<nc;j++){
  37.                M[i][j]=rand()%51;
  38.            }
  39.        }
  40.        return;
  41.    }
  42.    //Destructor
  43.    ~Matriz(){
  44.        for(int i=0;i<NF;i++){
  45.            delete M[i];
  46.        }
  47.        return;
  48.    }
  49.    //Operador asignacion
  50.    Matriz operator = (Matriz m){
  51.        if(NF!=m.NF){
  52.            for(int i=0;i<NF;i++){
  53.                delete M[i];
  54.            }
  55.        }
  56.        NF=m.NF;
  57.        NC=m.NC;
  58.        M=new int *[NF];
  59.        for(int j=0;j<NF;j++){
  60.            M[j]=new int [NC];
  61.        }
  62.        return(*this);
  63.    }
  64.    //Sobrecarga de operador de salida
  65.    friend ostream& operator <<(ostream &medio, Matriz matriz){
  66.        for(int i=0;i<matriz.NF;i++){
  67.            for(int j=0;j<matriz.NC;j++){
  68.                cout<<matriz.M[i][j]<<" ";
  69.            }
  70.            cout<<endl;
  71.        }
  72.        return(medio);
  73.    }
  74.    //Sobrecarga de operador suma
  75.    Matriz operator + (Matriz m){
  76.        Matriz resul;
  77.        if(NF==m.NF&&NC==m.NC){
  78.            for(int i=0;i<NF;i++){
  79.                for(int j=0;j<NC;j++){
  80.                    resul.M[i][j]=M[i][j]+m.M[i][j];
  81.                }
  82.            }
  83.        }
  84.        return(resul);
  85.    }
  86.    //Sobrecarga de operador resta
  87.    Matriz operator - (Matriz m){
  88.        Matriz resul;
  89.        if(NF==m.NF&&NC==m.NC){
  90.            for(int i=0;i<NF;i++){
  91.                for(int j=0;j<NC;j++){
  92.                    resul.M[i][j]=M[i][j]-m.M[i][j];
  93.                }
  94.            }
  95.        }
  96.        return(resul);
  97.    }
  98.    //Sobrecarga de operador producto
  99.    Matriz operator * (Matriz m){
  100.        Matriz resul;
  101.        if(NC==m.NF){
  102.            int N=NC;
  103.            for(int i=0;i<N;i++){
  104.                for(int j=0;j<N;j++){
  105.                    resul.M[i][j]=0;
  106.                    for(int k=0;k<N;k++){
  107.                        resul.M[i][j]=resul.M[i][j]+M[i][k]*m.M[k][j];
  108.                    }
  109.                }
  110.            }
  111.        }
  112.        return(resul);
  113.    }
  114. };
  115.  
  116. int main(void){
  117.    //Reinicio de semilla
  118.    srand(time(0));
  119.    //Prueba de codigo
  120.    Matriz m1(5,5),m2(5,5);
  121.    cout<<"Matriz 1 + Matriz 2 ="<<endl<<m1+m2;
  122.    getche();
  123.    return(0);
  124. }

Como se puede observar, el constructor con parámetros crea una matriz de NFxNC elementos, los cuales son valores aleatorios entre 0 y 50. El caso es que al probar la suma, y a pesar del getche(), el programa se ejecuta y se cierra automáticamente, y lo más extraño de todo, me devuelve 3. Es curioso, porque aunque no se realizara bien la operación, debería devolver 0, ya que compila sin problemas, pero devuelve 3. Por otro lado, si las matrices declaradas en main no llevan parámetros, el programa se ejecuta con normalidad, mostrando el texto del cout sin matriz (porque es nula), espera que pulse una tecla para continuar y devuelve 0. Por tanto, puedo suponer que el problema está, bien en el envío de parámetros, en el constructor de copia, o incluso en la sobrecarga del operador suma, pero no acabo de verlo. Por si sirve de algo, el compilador que utilizo es el QtCreator de Nokia.

Gracias por vuestra paciencia ;)
3  Programación / Programación C/C++ / Re: [C++] Reordenar vector dinámico de objetos en: 17 Marzo 2011, 19:05 pm
Bueno, después de darle mil vueltas logré encontrar los dos fallos que había, como siempre, tonterías, pero que hacen que te vuelvas loco.
Para empezar, el símbolo > funciona, el tema es que el booleano centinela no estaba inicializado, por tanto, no entraba en el bucle de ordenación. Por otro lado, el bucle interno empezaba en j=0, cuando debía hacerlo en j=1. El código resultante del método es el siguiente.
Código
  1. //Metodo reordenacion de contactos
  2.    void Ordenar(void){
  3.        Contacto aux;
  4.        bool centinela=true;
  5.        for(int i=0;i<N&&centinela;i++){
  6.            centinela=false;
  7.            for(int j=1;j<N;j++){
  8.                if(lista[j-1].VerNombre()>lista[j].VerNombre()){
  9.                    aux=lista[j];
  10.                    lista[j]=lista[j-1];
  11.                    lista[j-1]=aux;
  12.                    centinela=true;
  13.                }
  14.            }
  15.        }
  16.        return;
  17.    }
  18.  
De todos modos, gracias por el aporte. Lo tendré en cuenta para futuros programas, que se avecinan chungos.  ;)
4  Programación / Programación C/C++ / [C++] Reordenar vector dinámico de objetos en: 16 Marzo 2011, 19:34 pm
Saludos. Liado ando con un gestor de agenda, en el que he de reordenar por nombre los contactos cada vez que añado uno nuevo. Por supuesto, pensé en la burbuja, pero, a pesar de que compila sin problemas, no reordena nada. He aquí mi código:
Código
  1. //Clase Agenda
  2. class Agenda{
  3. private:
  4.    Contacto *lista;
  5.    int N;
  6. public:
  7.    //Constructor por defecto
  8.    Agenda(){
  9.        N=0;
  10.        lista=NULL;
  11.        return;
  12.    }
  13.    //Destructor
  14.    ~Agenda(){
  15.        if(lista){
  16.            delete[]lista;
  17.        }
  18.    }
  19.    //Metodo aniadir contacto
  20.    void NuevoContacto(void){
  21.        //Creacion vector Contacto auxiliar
  22.        Contacto *aux=new Contacto[N+1];
  23.        //Recoleccion de datos anteriores
  24.        for(int i=0;i<N;i++){
  25.            aux[i]=lista[i];
  26.        }
  27.        //Peticion de dato nuevo
  28.        cin>>aux[N];
  29.        cin>>aux[N].fnac;
  30.        cin>>aux[N].dir;
  31.        //Eliminacion de vector Contacto obsoleto
  32.        if(lista!=NULL){
  33.            delete[]lista;
  34.        }
  35.        //Recreacion de vector Contacto util
  36.        lista=aux;
  37.        N++;
  38.        Ordenar(lista,N);
  39.        return;
  40.    }
  41.  
  42. //.........
  43.  
  44.    //Metodo reordenacion de contactos
  45.    void Ordenar(Contacto listado[], int n){
  46.        Contacto aux;
  47.        bool centinela;
  48.        for(int i=0;i<n&&centinela==true;i++){
  49.            centinela=false;
  50.            for(int j=0;j<n;j++){
  51.                if(listado[j-1].VerNombre()>listado[j].VerNombre()){
  52.                    aux=listado[j-1];
  53.                    listado[j-1]=listado[j];
  54.                    listado[j]=aux;
  55.                    centinela=true;
  56.                }
  57.            }
  58.        }
  59.        return;
  60.    }
  61. };
¿Dónde está el fallo?
5  Programación / Programación C/C++ / Re: [C++] Buscar un dato privado en un vector de clase en: 10 Marzo 2011, 18:13 pm
Bueno, ya está resuelto. El error estaba en que inicializaba N en -1, para luego añadirle 1 antes de crear un contacto nuevo. Queda solucionado inicializando N en 0 y añadirle 1 DESPUES de crear el contacto.

Código
  1. class Agenda{
  2. private:
  3.    int N;
  4.    Contacto lista1[NMAX];
  5. public:
  6.    //Constructor por defecto
  7.    Agenda(void){
  8.        N=0;
  9.    }
  10.    //Metodo anyadir contacto
  11.    void NuevoContacto(void){        
  12.        lista1[N].Solicitar();
  13.        N++;
  14.        return;
  15.    }
  16.    //Metodo muestra de contactos
  17.    void ListarContactos(void){
  18.        //Condicion de vector Contacto vacio
  19.        if(N<0){
  20.            cout<<"No hay contactos guardados"<<endl;
  21.            getche();
  22.        }
  23.        else{
  24.            for(int i=0;i<=N;i++){
  25.                cout<<"Contacto n. "<<i+1<<endl;
  26.                lista1[i].Mostrar();
  27.                getche();
  28.            }
  29.        }
  30.        return;
  31.    }
  32.    //Metodo busqueda de contactos
  33.    void BuscarContacto(void){
  34.        string buscar,comparar;
  35.        bool exito=false;
  36.        cout<<"Nombre a buscar: ";
  37.        getline(cin,buscar);
  38.        for(int j=0;j<N;j++){
  39.            //Busqueda de comparador
  40.            comparar=lista1[j].VerNombre();
  41.            //Comprobante
  42.            cout<<comparar<<endl;
  43.            //Comparador de nombres
  44.            if(buscar==comparar){
  45.                lista1[j].Mostrar();
  46.                //Centinela de contacto encontrado
  47.                exito=true;
  48.            }
  49.        }
  50.        if(exito==false){
  51.            cout<<"Contacto no encontrado"<<endl;
  52.        }
  53.        return;
  54.    }
  55. };

De todos modos, gracias por las ideas  ;)
6  Programación / Programación C/C++ / Re: [C++] Buscar un dato privado en un vector de clase en: 10 Marzo 2011, 17:27 pm
Pues sí, soy de la UA, del grupo de tarde de teleco. Me has pillado. Pues espero tu código, que a mí se me ha secado el cerebro.
7  Programación / Programación C/C++ / Re: [C++] Buscar un dato privado en un vector de clase en: 6 Marzo 2011, 12:12 pm
Saludos. Gracias por las aportaciones. Por un lado, los atributos deben ser todos privados, así que no puedo hacer ese público. Por otro lado, añadí un método público en "Contacto" que me devolviera el string nombre. Así no tengo errores de compilación ni de ejecución, sin embargo, es posible que no lo haya creado bien, pues cuando voy a buscar un contacto que he añadido, no me encuentra nada. Ni siquiera me salta el centinela, pues me dice "Contacto no encontrado".
Código
  1. class Contacto
  2. {
  3. private:
  4.    string nombre, apellidos;
  5.    FechaNacimiento fnac1;
  6.    Direccion dir1;
  7. public:
  8.    string VerNombre(void)
  9.    {
  10.        return(nombre);
  11.    }
  12.  
  13.    void Mostrar(void)
  14.    {
  15.        cout<<"Nombre: "<<nombre<<endl;
  16.        cout<<"Apellidos: "<<apellidos<<endl;
  17.        fnac1.Mostrar();
  18.        dir1.Mostrar();
  19.        return;
  20.    }
  21.    void Solicitar(void)
  22.    {
  23.        cout<<"Nombre: ";
  24.        getline(cin,nombre);
  25.        cout<<"Apellidos: ";
  26.        getline(cin,apellidos);
  27.        fnac1.Solicitar();
  28.        dir1.Solicitar();
  29.        return;
  30.    }
  31. };
  32.  
  33. class Agenda
  34. {
  35. private:
  36.    int N;
  37.    Contacto lista1[NMAX];
  38.    string buscar,comparar;
  39.    bool exito;
  40. public:
  41.    Agenda(void)
  42.    {
  43.        N=-1;
  44.        exito=false;
  45.    }
  46.    void NuevoContacto(void)
  47.    {
  48.        N++;
  49.        lista1[N].Solicitar();
  50.        return;
  51.    }
  52.    void ListarContactos(void)
  53.    {
  54.        if(N<0)
  55.        {
  56.            cout<<"No hay contactos guardados"<<endl;
  57.            getche();
  58.        }
  59.        else
  60.        {
  61.            for(int i=0;i<=N;i++)
  62.            {
  63.                cout<<"Contacto n. "<<i+1<<endl;
  64.                lista1[i].Mostrar();
  65.                getche();
  66.            }
  67.        }
  68.        return;
  69.    }
  70.    void BuscarContacto(void)
  71.    {
  72.        cout<<"Nombre a buscar: ";
  73.        getline(cin,buscar);
  74.        for(int j=0;j<N;j++)
  75.        {
  76.            comparar=lista1[j].VerNombre();
  77.            if(buscar==comparar)
  78.            {
  79.                lista1[j].Mostrar();
  80.                exito=true;
  81.            }
  82.        }
  83.        if(exito==false)
  84.        {
  85.            cout<<"Contacto no encontrado"<<endl;
  86.        }
  87.        return;
  88.    }
  89. };
8  Programación / Programación C/C++ / [C++] Buscar un dato privado en un vector de clase en: 5 Marzo 2011, 21:19 pm
Saludos. Puede que el título no esté muy claro, así que explicaré cuál es el problema que me ha traído:

En una clase Agenda he de crear un método mediante el cual el usuario introduzca un nombre y el programa muestre todos los datos del vector en los que se encuentre ese nombre. El caso es que el nombre se encuentra en otra clase distinta, y no podemos hacer amigos. Busqué información y descubrí que es posible hacerlo con los métodos get y set, pero no sé cómo aplicarlos.

Aquí pongo parte del código:
Código
  1. class Contacto
  2. {
  3. private:
  4.    string nombre, apellidos;
  5.    FechaNacimiento fnac1;
  6.    Direccion dir1;
  7. public:
  8.    void Mostrar(void)
  9.    {
  10.        cout<<"Nombre: "<<nombre<<endl;
  11.        cout<<"Apellidos: "<<apellidos<<endl;
  12.        fnac1.Mostrar();
  13.        dir1.Mostrar();
  14.    }
  15.    void Solicitar(void)
  16.    {
  17.        cout<<"Nombre: ";
  18.        getline(cin,nombre);
  19.        cout<<"Apellidos: ";
  20.        getline(cin,apellidos);
  21.        fnac1.Solicitar();
  22.        dir1.Solicitar();
  23.    }
  24. };
  25.  
  26. class Agenda
  27. {
  28. private:
  29.    int N;
  30.    Contacto lista1[NMAX];
  31.    string buscar;
  32.    bool exito;
  33. public:
  34.    Agenda(void)
  35.    {
  36.        N=-1;
  37.        exito=false;
  38.    }
  39.    void NuevoContacto(void)
  40.    {
  41.        N++;
  42.        lista1[N].Solicitar();
  43.    }
  44. //...
  45. //Codigo entre medias
  46. //...
  47.    void BuscarContacto(void)
  48.    {
  49.        cout<<"Nombre a buscar: ";
  50.        getline(cin,buscar);
  51.        for(int j=0;j<N;j++)
  52.        {
  53.            if(lista1[j].nombre==buscar)//Esta condicion no es correcta
  54.            {
  55.                lista1[j].Mostrar();
  56.                exito=true;//Centinela de contacto encontrado
  57.            }
  58.        }
  59.        if(exito==false)
  60.        {
  61.            cout<<"Contacto no encontrado"<<endl;
  62.        }
  63.    }
  64. };
9  Programación / Programación C/C++ / [C] Mostrando datos de un vector de enteros en: 5 Diciembre 2010, 20:44 pm
Saludos. Sigo con el black jack avanzando a mi ritmo, pero con otra traba. Y si resuelvo esto, el resto del programa lo doy por solucionado.

Bien, resulta que el formato de salida debe ser algo como:

"BLACK JACK

Tus cartas: 1 3 J 2 8

Valor total: 25

Deseas otra carta?"

Según este ejemplo, la primera jugada debe mostrar las dos primeras cartas, el valor total y la pregunta de coger otra carta, y al decir que sí, lo mismo, pero con la siguiente. En definitiva, que mientras aparece el texto de alrededor, primero se muestran dos cartas y luego de una en una.

Mi idea al hacer el código fue poner el primer texto de modo forzado, directamente con printf y punto, y las siguientes cartas ya en un do while. Esto es lo que tengo hecho de momento:

Código
  1.        system("cls");
  2.        printf("**************");
  3.        printf("\n* BLACK JACK *");
  4.        printf("\n**************");
  5.        printf("\nTus cartas: ");
  6.        if(baraja[0]==11){
  7.            printf("J ");
  8.            val=10;
  9.        }else if(baraja[0]==12){
  10.            printf("Q ");
  11.            val=10;
  12.        }else if(baraja[0]==13){
  13.            printf("K ");
  14.            val=10;
  15.        }else{
  16.            printf("%d ",baraja[0]);
  17.            val=baraja[0];
  18.        }
  19.        cartas=cartas+val;
  20.  
  21.        for(i=1;cartas<21;i++){
  22.            if(baraja[i]==11){
  23.                printf("J ");
  24.                val=10;
  25.            }else if(baraja[i]==12){
  26.                printf("Q ");
  27.                val=10;
  28.            }else if(baraja[i]==13){
  29.                printf("K ");
  30.                val=10;
  31.            }else{
  32.                printf("%d ",baraja[i]);
  33.                val=baraja[i];
  34.            }
  35.            cartas=cartas+val;
  36.            printf("\n\nEl valor de tus cartas es: %d",cartas);
  37.        }

La teoría es buena, obviando el hecho de que de este modo no se muestre el resto del texto, sin embargo, lo que me sale es algo como:

"BLACK JACK

Tus cartas: 1 3

J lor total: 4

2 lor total: 14

Valor total: 16"

Es decir, que sobreescribe la línea de valor total de la anterior tirada.

Una cosa que se me ocurrió fue meter todo el texto en un for o un do while, que borrase todo y lo volviera a escribir, aunque se me presentaban dos problemas: el primero, no saldrían las dos primeras cartas de golpe, y el segundo, que probablemente sólo saliera la última carta, pese a que cambie el valor total.
10  Programación / Programación C/C++ / Re: [C] Entremezclar valores de un vector en: 4 Diciembre 2010, 19:19 pm
Cierto, qué error más tonto. Había creado un bucle infinito por una letrita de nada.
¡Muchas gracias!
Páginas: [1] 2
WAP2 - Aviso Legal - Powered by SMF 1.1.21 | SMF © 2006-2008, Simple Machines