Duda con algoritmos para pasar una fecha a días y calcular la diferencia.

Iniciado por seryioo, 4 Agosto 2015, 13:08 PM

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

seryioo

Buenas, son varios ejercicios, aquí os dejo los enunciados.

12) Teniendo en cuenta que el calendario gregoriano está vigente desde 1583 y que, según
este calendario, son bisiestos los años divisibles por 4, salvo los años seculares (los
divisibles por 100) que además deben ser divisibles por 400 para que sean bisiestos,
defínanse las funciones siguientes: bool esBisiesto(int anno), que determina si un
año del calendario gregoriano es bisiesto o no; int ultimoDeMes(int mes, int anno),
que calcula el último día de cada mes, teniendo en cuenta el año y bool
fechaCorrecta(int dia,int mes,int anno), que determina si los valores de sus
argumentos corresponden a una fecha correcta o no en el calendario gregoriano.
13) Continuando con el ejercicio anterior, defínanse también las funciones void
avanzar(int& d, int& m, int& a), que avanza un día la fecha del argumento y void
retroceder(int& d, int& m, int& a), que retrocede un día la fecha del argumento.
14) Defínanse las funciones: bool esAnterior(int d1,int m1,int a1,int d2,int
m2,int a2), que determina si la fecha d1/m1/a1 es anterior o no a la fecha d2/m2/a2, y
int diferencia(int d1,int m1,int a1,int d2,int m2,int a2), que calcula el
número de días que debe avanzar (resultado positivo) o retroceder (resultado negativo) la
fecha d1/m1/a1 hasta alcanzar la fecha d2/m2/a2.
15) Agrúpense las funciones de los tres ejercicios anteriores en un programa repetitivo que
pida una fecha del teclado, compruebe si es correcta, y ofrezca una serie de opciones para
avanzar o retroceder la fecha, para comparar con otra fecha, que se pedirá del teclado, o
para calcular la diferencia de días con otra fecha que también se deberá leer del teclado.



Me funciona todo excepto calcular la diferencia de días entre 2 fechas y la verdad tengo la cabeza saturada, ya he descubierto unos cuantos fallos que he arreglado, pero aun así sigue sin darme la diferencia correcta. Concretamente el error suceda si introduzco meses o años diferentes.

Código (cpp) [Seleccionar]

#include <iostream>
using namespace std;

bool esbisiesto(int anno);

int ultimodelmes (int mes, int anno);

bool fechacorrecta(int dia, int mes, int anno);

void avanzar(int& d, int& m, int& a);

void retroceder(int& d, int& m, int& a);

bool esAnterior(int d1,int m1,int a1,int d2,int m2,int a2);

int diferencia(int d1,int m1,int a1,int d2,int m2,int a2);

int mesAdias (int mes, int anno);

int annoAdias (int anno);


//-------------------

int main(){
   int dia, mes, anno;
   int dia2, mes2, anno2;
   do{
       cout<<"Introduce una fecha para saber si es correcta (dia, mes, año): "<<endl;
       cout<<"Dia: ";
       cin>>dia;
       cout<<"Mes: ";
       cin>>mes;
       cout<<"año: ";
       cin>>anno;
       if (fechacorrecta(dia, mes, anno)) cout<<"Ha introducido una fecha correcta."<<endl;
       else cout<<"Fecha incorrecta."<<endl;
   }while(!fechacorrecta(dia, mes, anno));

   int menu=999;
   while(menu!=0){
       cout<<endl
           <<"###########################################################"<<endl
           <<"1.Avanzar un dia la fecha."<<endl
           <<"2.Retroceder un dia la fecha."<<endl
           <<"3.Comparar fecha introducida con otra a introducir."<<endl
           <<"4.Calcular la diferencia en dias de la fechas introducidas."<<endl
           <<"0.Salir"<<endl
           <<"-----------------------------------------------------------"<<endl
           <<endl;

       cout<<"Introduzca una opcion valida: ";
       cin>>menu;

       switch (menu){

           case 0:
           break;

           case 1:
               avanzar(dia, mes, anno);
               cout<<dia<<"/"<<mes<<"/"<<anno<<endl;
           break;

           case 2:
               retroceder(dia, mes, anno);
               cout<<dia<<"/"<<mes<<"/"<<anno<<endl;
           break;

           case 3:
               do{
                   cout<<"Introduce otra fecha para comparar: "<<endl;
                   cout<<"Dia: ";
                   cin>>dia2;
                   cout<<"Mes: ";
                   cin>>mes2;
                   cout<<"año: ";
                   cin>>anno2;
               }while(!fechacorrecta(dia2, mes2, anno2));

               if(esAnterior(dia, mes, anno, dia2, mes2, anno2)) cout<<dia<<"/"<<mes<<"/"<<anno<<" es anterior a "<<dia2<<"/"<<mes2<<"/"<<anno2<<endl;
               else cout<<dia<<"/"<<mes<<"/"<<anno<<" NO ES anterior a "<<dia2<<"/"<<mes2<<"/"<<anno2<<endl;
           break;

           case 4:
               do{
                   cout<<"Introduce otra fecha para comparar: "<<endl;
                   cout<<"Dia: ";
                   cin>>dia2;
                   cout<<"Mes: ";
                   cin>>mes2;
                   cout<<"año: ";
                   cin>>anno2;
               }while(!fechacorrecta(dia2, mes2, anno2));

               cout<<diferencia(dia, mes, anno, dia2, mes2, anno2);
           break;




           }


   }
return 0;
}

//-------------------

bool esbisiesto(int anno){
  if(anno>1583){
       if(anno%4==0){
           if((anno%100==0) and (anno%400==0)){
               return true;
           }else  return true;


       }else return false;

   }else return false;
}

int ultimodelmes (int mes, int anno){
   if (mes==2){
       if(esbisiesto(anno)) return 29;
       else return 28;
   }else if (mes!=2 and mes%2==0) return 30;
   else return 31;
}

bool fechacorrecta(int dia, int mes, int anno){
   if(dia>31 or mes>12 or anno<1583){
       return false;
   }else{
       if(dia>ultimodelmes(mes, anno)) return false;
       else return true;
   }
}

void avanzar(int& d, int& m, int& a){
   ++d;
   if(d>ultimodelmes(m,a)) ++m;
   if(m>12){
       ++a;
       m=1;
       d=1;
   }

}

void retroceder(int& d, int& m, int& a){
   --d;
   if(d<1){
       if(m>1){
           --m;
           d=ultimodelmes(m, a);
       }else{
           m=12;
           --a;
           d=ultimodelmes(m, a);
       }
   }
}

bool esAnterior(int d1,int m1,int a1,int d2,int m2,int a2){
   if(a1>a2) return false;
   else if (a1<a2) return true;
   else{
       if(m1>m2) return false;
       else if(m1<m2) return true;
       else {
           if(d1>d2) return false;
           else if(d1<d2) return true;
           else cout<<"LAS DOS FECHAS SON LA MISMA"<<endl;
       }
   }
}

int mesAdias(int mes, int anno){
   int dias=0;
   for(unsigned contador=1; contador<=mes; contador++){
       if(mes%2==0){
           if(mes==2){
               if(esbisiesto(anno))dias+=29;
               else dias+=28;
           }else dias+=30;
       }else dias+=31;
   }
   return dias;
}

int annoAdias(int anno){
   int dias=0;
   for(unsigned contador=1; contador<= anno; contador++){
       if(esbisiesto(anno)) dias+=366;
       else dias+=365;
   }
   return dias;
}

int diferencia(int d1,int m1,int a1,int d2,int m2,int a2){
   int suma1=0, suma2=0;
   suma1=d1+mesAdias(m1, a1)+annoAdias(a1);
   suma2=d2+mesAdias(m2, a2)+annoAdias(a2);
   if (suma1<=suma2) return suma2-suma1;
   else return suma1-suma2;
}




DarK_FirefoX

Así a simple vista tienes un problema en este método:

Código (cpp) [Seleccionar]
int ultimodelmes (int mes, int anno){
    if (mes==2){
        if(esbisiesto(anno)) return 29;
        else return 28;
    }else if (mes!=2 and mes%2==0) return 30; //Agosto entra por aquí (FALSE)
    else return 31;
}


Recuerda que Agosto tiene 31 días al igual que julio. En este método agosto es el mes 8, es distinto de 2 y deja resto 0 con 2, por lo tanto te va devolver 30. Lo cual es falso.

Salu2s

seryioo

cierto, no me había dado cuenta, ahora cuando llegue a casa lo pruebo. Puede que fuese por esa tontería.

Gracias.

seryioo

Buenas, he corregido un par de cosas basadas en el error de que agosto tiene 31 días y no 30.

Código (cpp) [Seleccionar]

int ultimodelmes (int mes, int anno){
    if (mes==2){
        if(esbisiesto(anno)) return 29;
        else return 28;
    }else if (mes!=2 and mes%2==0){
        if(mes==8) return 31;
        else return 30;
    }
    else return 31;
}


y

Código (cpp) [Seleccionar]

int mesAdias(int mes, int anno){
    int dias=0;
    for(unsigned contador=1; contador<=mes; contador++){
        if(mes%2==0){
            if(mes==2){
                if(esbisiesto(anno))dias+=29;
                else dias+=28;
            }else if (mes==8) dias+=31;
             else dias+=30;
        }else dias+=31;
    }
    return dias;
}




He comprobado que funcione con las siguientes  2 fechas: 2/2/2000 y 2/1/2000 y me dice que la diferencia son 27 días... sigo sin ver el problema...

DarK_FirefoX

#4
Cita de: seryioo en  4 Agosto 2015, 21:19 PM
Código (cpp) [Seleccionar]

int mesAdias(int mes, int anno){
   int dias=0;
   for(unsigned contador=1; contador<=mes; contador++){
       if(mes%2==0){
           if(mes==2){
               if(esbisiesto(anno))dias+=29;
               else dias+=28;
           }else if (mes==8) dias+=31;
            else dias+=30;
       }else dias+=31;
   }
   return dias;
}


Con respecto a este método, explícame "semánticamente" que significa? Pues cogí papel y lápiz y la verdad no le encuentro sentido. Date cuenta que la variable mes la mantienes fija, o sea, estas sumando con el ejemplo que dices:

Código (cpp) [Seleccionar]
suma1=d1+mesAdias(m1, a1)+annoAdias(a1);
suma2=d2+mesAdias(m2, a2)+annoAdias(a2);


Esto te está dando así:

Citarsuma1 = 2 + 31 + 366 = 399
suma2 = 2 + 58 + 366 = 426

Luego la resta te da 27. Lo cuál por supuesto que está mal. Porque entre el 2/2/2000 y 2/1/2000 hay 31 días.

Lo que quiero que expliques es como funciona tu algoritmo para ver si trabajamos desde ahí, porque en mi opinión no está bien diseñado

EDITO: No hagas doble post

Salu2s

seryioo

Vale, quizás debería haber puesto lo siguiente, ya que como dices mes está estático y contador es la variable que va sumando los días de los respectivos meses hasta la variable mes.

Código (cpp) [Seleccionar]

int mesAdias(int mes, int anno){
   int dias=0;
   for(unsigned contador=1; contador<=mes; contador++){
       if([b]contador[/b]%2==0){
           if([b]contador[/b]==2){
               if(esbisiesto(anno))dias+=29;
               else dias+=28;
           }else if (mes==8) dias+=31;
            else dias+=30;
       }else dias+=31;
   }
   return dias;
}



PD: Acabo de probarlo y me devuelve 29 días :(

Por cierto, la función annoAdias lo que hace es pasar cuantos días sería en este caso 2000 años con su bucle for en el que contador va de 1 a 2000 sumando cada pasada 365 dias (o 366 según sea bisiesto).

mesAdias hace algo similar, con mes=2 el bucle for recorre con contador del 1 al 2 y suma a dias 31+28 (o 29 días si es bisiesto)