Cita:
Iniciado por rdv316 Me podrías ayudar con esto, no encuentro como reducir el código, para mi todas las líneas son necesarias.
Te intento contestar mañana... que ya es tarde y mañana hay que madrugar :)
-----
Cita:
Iniciado por rdv316 Me podrías ayudar con esto, no encuentro como reducir el código, para mi todas las líneas son necesarias.
Te intento contestar mañana... que ya es tarde y mañana hay que madrugar :)
-----
Variables que se pueden eliminar
Los resultados temporales únicamente interesa almacenarlos en variables si:
- El resultado se reutiliza varias veces dentro de un bucle (sobretodo si la operación es costosa)
- Estamos en fase de desarrollo y facilita las tareas de depuración
Fuera de estos contextos es mejor evitar el uso de variables temporales. Tener una función con multitud de variables es como tener el escritorio lleno de cosas, al final cuesta encontrar la información que necesitamos.
En tu caso puedes llegar a eliminar el 30% de las variables del main:
- regBisAnioF1 y regBisAnioF2 se pueden sustituir directamente por la llamada a la función:
Antes:
Código c:
Ver originalregBisAnioF1 = regularBisiesto(tFecha1);
difDias += regBisAnioF1;
Despues:
- difAnios puede correr la misma suerte con toda tranquilidad:
Antes:
Código c:
Ver originaldifAnios = fecha2.anio - fecha1.anio;
if( difAnios == 0){
Despues:
Código c:
Ver originalif( (fecha2.anio - fecha1.anio) == 0 ){
Refactorización del código
1. Cuando realizas una comparación ( >, <, >=, <=, ==, != ) puedes obtener dos resultados posibles: 1 (TRUE) o 0 (FALSE)
Si miras el código de
regularBisiesto verás que esta función está devolviendo 1 para TRUE y 0 para FALSE. Si eliminas el if y pasas a retornar el resultado de la comparación conseguirás el mismo resultado y todo en una única línea:
Código c:
Ver originalint regularBisiesto( int anio ){
return ((anio %4 == 0 && anio%100 != 0) || (anio%400) == 0);
}
Por cierto, llamar
tFecha al argumento de
regularBisiesto no parece buena idea. Los nombres de las variables tienen que dar una idea acerca de lo que almacenan y de la utilidad que se le pretende dar.
tFecha da a entender que estás almacenando una fecha, cuando realmente está almacenando el valor del año.
2. Cuando estás trabajando con fechas de diferentes años haces uso de un bucle que itera sobre los años intermedios para calcular el número de días de este intervalo. En cada iteración sumas 365 días más la corrección por bisiesto cuando procede. Este bucle te lo puedes ahorrar si tienes en cuenta un par de consideraciones:
- Todos los años tienen, al menos 365 días
- Puedes saber la cantidad de bisiestos que hay entre dos fechas con una sencilla operación matemática
Dicho con código:
Antes:
Código c:
Ver originalfor( i=tFecha1 + 1; i < fecha2.anio ; i++)
{
tFecha1++;
difDias += 365;
difDias += regularBisiesto(tFecha1);
}
Despues:
Código c:
Ver originaldifDias += (fecha2.anio - fecha1.anio - 1) * 365;
difDias += (fecha2.anio-1) / 4 - (fecha2.anio-1) / 100 + (fecha2.anio-1) / 400;
difDias -= (fecha1.anio / 4) - (fecha1.anio / 100) + (fecha1.anio / 400);
Explicación del código:
- La primera línea coge el número de años completos entre las dos fechas y lo multiplica por 365
- En la segunda se calcula el número de años bisiestos hasta la fecha final (año no incluído).
- En la tercera se calcula el número de años bisiestos hasta la fecha inicial.
- Si se restan los dos últimos cálculos se obtiene el número de bisiestos en el rango dado.
Al quitarte un bucle consigues, aunque en este caso no se note, que el tiempo de ejecución sea el mismo, independientemente del número de años que haya entre las dos fechas. El código propuesto es algo más lento que el que tenías si las dos fechas están bastante próximas, pero según se van distanciando las fechas tu código va siendo cada vez más lento, mientras que el mío se mantiene estable.
Por supuesto lo puedes dejar "más bonito" si mueves el cálculo a una función:
Código c:
Ver originaldifDias += (fecha2.anio - fecha1.anio - 1) * 365;
difDias += numeroBisiestos(fecha2.anio-1) - numeroBisiestos(fecha1.anio);
//...
int numeroBisiestos( int anio )
{
return anio / 4 - anio / 100 + anio / 400;
}
3. ¿Y si en vez de un array con los días de cada mes devolvemos el día del año?
A veces, cambiar la perspectiva nos ayuda a encontrar soluciones más sencillas a nuestros problemas. En tu caso estás haciendo distincción entre dos posibles casos:
- Las fechas son del mismo año
- Las fechas son de años diferentes
Conseguir un algoritmo que sea capaz de trabajar indistintamente con ambos casos te evita tener código duplicado y, como has comentado, dificil de refactorizar. Vamos a intentar acabar con la duplicidad.
Lo primero que vamos a cambiar es, como indica el título, calcular el día del año. Esto se puede conseguir reutilizando parte de tu código:
Código c:
Ver originalint diaDelAnio( Fecha fecha )
{
int tot_dias[] = { 00,31,28,31,30,31,30,31,31,30,31,30,31 };
int dias = fecha.dia;
int mes;
for( mes = 1; mes < fecha.mes; ++mes )
dias += tot_dias[mes];
if( fecha.mes > 2 )
dias += regularBisiesto( fecha.anio );
return dias;
}
Con esta sencilla función podemos obrar milagros dentro de tu código. Como verás, ahora ya ha desaparecido el if y todo el código duplicado:
Código c:
Ver original#include <stdio.h>
typedef struct{
int dia;
int mes;
int anio;
}Fecha;
int regularBisiesto( int tFecha );
int numeroBisiestos( int anio );
int diaDelAnio( Fecha fecha );
// Algoritmo para calcular el número de días entre dos fechas
// Los cálculos realizados son:
// Si las fechas son del mismo año se calcula el intervalo:
// - fecha1 -> fecha2
// Si las fechas son de diferentes años se calculan 3 intervalos:
// - fecha1 -> fin año fecha1
// - días por los años intermedios
// - inicio año fecha2 -> fecha2
int main(void){
Fecha fecha1 = { 1,1, 2050 }, fecha2 = { 12,12, 2050 };
int difDias = 0;
// Cálculo del intervalo fecha1 -> fecha2
// no se tienen en cuenta los años en este punto porque la corrección se realiza después
difDias = diaDelAnio( fecha2 ) - diaDelAnio( fecha1 );
if( fecha1.anio != fecha2.anio )
{
// Días por los años completos
difDias += (fecha2.anio - fecha1.anio - 1) * 365;
difDias += numeroBisiestos(fecha2.anio - 1) - numeroBisiestos(fecha1.anio);
// Cálculo correspondiente al intervalo fecha1 -> fin año fecha1
fecha1.dia = 31;
fecha1.mes = 12;
difDias += diaDelAnio( fecha1 );
}
printf("\n Del %d/%d/%d al %d/%d/%d hay %d dias", fecha1.
dia, fecha1.
mes, fecha1.
anio, fecha2.dia, fecha2.mes, fecha2.anio, difDias);
return 0;
}
int regularBisiesto( int anio)
{
return ((anio%4 == 0 && anio%100 != 0) || (anio%400) == 0);
}
int numeroBisiestos( int anio )
{
return anio / 4 - anio / 100 + anio / 400;
}
int diaDelAnio( Fecha fecha )
{
int tot_dias[] = { 00,31,28,31,30,31,30,31,31,30,31,30,31 };
int dias = fecha.dia;
int mes;
for( mes = 1; mes < fecha.mes; ++mes )
dias += tot_dias[mes];
if( fecha.mes > 2 )
dias += regularBisiesto( fecha.anio );
return dias;
}
¿Te gusta la nueva versión? Es bastante corta, es legible y utiliza muy poquitas variables :)
Un saludo