Practicas de Programacion 1 Ejercicios

Descargar como pdf o txt
Descargar como pdf o txt
Está en la página 1de 100

Programacin I o

Relacin de Ejercicios o y Soluciones


UNIVERSIDAD DE MALAGA Dpto. Lenguajes y CC. Computacin o E.T.S.I. Telecomunicacin o

Sonido e Imagen

Indice
Tema 2: Conceptos Bsicos de Programacin a o Tema 3: Dise o Descendente. Subprogramas n Tema 4: Tipos de Datos Estructurados Tema 5: B squeda y Ordenacin u o 1 25 38 92

Este obra est bajo una licencia Reconocimiento-NoComercial-CompartirIgual 3.0 Unported de a Creative Commons: No se permite un uso comercial de la obra original ni de las posibles obras derivadas, la distribucin de las cuales se debe hacer con una licencia igual a la que regula la obra original. Para o ver una copia de esta licencia, visite http://creativecommons.org/licenses/by-nc-sa/3.0/deed.es ES o envie una carta a Creative Commons, 171 Second Street, Suite 300, San Francisco, California 94105, USA. Nota: en la solucin a los ejercicios, se ha utilizado el tipo array de TR1, que ha sido incorporado a la biblioo teca de C++ en el estndar de 2011. Si su biblioteca estndar no contiene la denicin del tipo array, puede a a o descargarla desde la siguiente direccin: http://www.lcc.uma.es/%7Evicente/docencia/cpplibs/array_tr1.zip o

Tema 2: Conceptos Bsicos de Programacin a o


1. El siguiente programa escrito en C++ calcula la cantidad bruta y neta a pagar por un trabajo realizado en funcin de las horas y d trabajados. Contiene errores, encuntrelos y corr o as e jalos.
# include < iostream > using namespace std ; const tasa : 25.0; const PRECIO_HORA = 60.0; int main () { double horas , dias , total , neto ; cout < < " Introduzca las horas trabajadas : " ; cin < < horas ; cout < < " Introduzca los dias trabajados : " ; cin > > dias ; horas * dias * PRECIO_HORA = total ; neto = total - TASA ; cout > > " El valor total a pagar es : " > > total > > endl ; cout < < " El valor neto a pagar es : " < < NETO < < endl ; }

Solucin o

# include < iostream > using namespace std ; // const double TASA = 25.0; const double PRECIO_HORA = 60.0; // int main () { double horas , dias , total , neto ; cout < < " Introduzca las horas trabajadas : " ; cin > > horas ; cout < < " Introduzca los dias trabajados : " ; cin > > dias ; total = horas * dias * PRECIO_HORA ; neto = total - TASA ; cout < < " El valor total a pagar es : " < < total < < endl ; cout < < " El valor neto a pagar es : " < < neto < < endl ; }

2. Desarrolle un programa que lea dos nmeros de tipo int de teclado y posteriormente los escriba en pantalla. u Ejectelo introduciendo dos nmeros de tipo int vlidos (por ejemplo 1234 y 5678). Posteriormente u u a ejectelo introduciendo por teclado un primer nmero de tipo int (por ejemplo 1234) e introduciendo u u por teclado un segundo dato que no pertenezca al tipo int (por ejemplo hola). Finalmente ejectelo u introduciendo por teclado un primer dato que no pertenezca al tipo int (por ejemplo hola). Evale las u diferencias entre ambas ejecuciones del mismo programa. Solucin o

# include < iostream > using namespace std ; int main () { int dato1 , dato2 ; cout < < " Introduzca un n mero entero : " ; u cin > > dato1 ; cout < < " Introduzca otro n mero entero : " ; u cin > > dato2 ; cout < < " El valor del primer n mero introducido es : " < < dato1 < < endl ; u cout < < " El valor del segundo n mero introducido es : " < < dato2 < < endl ; u } // // Cuando s e i n t r o d u c e por t e c l a d o un v a l o r adecuado a l t i p o de l a // v a r i a b l e que s e e s t a l e y e n d o , l a l e c t u r a s e hace c o r r e c t a m e n t e . // Sin embargo , s i s e i n t r o d u c e por t e c l a d o un v a l o r NO adecuado a l // t i p o de l a v a r i a b l e que s e e s t l e y e n d o , e n t o n c e s l a o p e r a c i n de a o // l e c t u r a f a l l a , l a v a r i a b l e mantiene e l v a l o r que t u v i e s e // a n t e r i o r m e n t e ( en e s t e c a s o como l a v a r i a b l e no ha s i d o // i n i c i a l i z a d a , t e n d r un v a l o r INESPECIFICADO) , y e l f l u j o de a // e n t r a d a (CIN) s e pondr en modo err neo , por l o que c u a l q u i e r o t r a a o // o p e r a c i n de e n t r a d a que s e r e a l i c e p o s t e r i o r m e n t e tambi n f a l l a r a . o e

3. Desarrolle un programa que slo declare variables de tipo int. El programa deber leer dos nmeros o a u enteros desde el teclado, posteriormente los sumar, almacenando el resultado en una variable, y nalmente a escribir por pantalla el resultado de la suma. Ejecute dicho programa introduciendo como datos de entrada a los siguientes nmeros y analice los resultados obtenidos. u
a) -20 y 30. b) 20 y -30. c) 147483647 y 2000000000. d) 200000000 y 2000000000. e) 1 y 2147483647. f ) 1 y 3000000000.

# include < iostream > using namespace std ; int main () {

Solucin o

int dato_1 , dato_2 ; cout < < " Introduzca el primer n mero entero : " ; u cin > > dato_1 ; cout < < " Introduzca el segundo n mero entero : " ; u cin > > dato_2 ; int suma = dato_1 + dato_2 ; cout < < " El valor resultado es : " < < suma < < endl ; } // // El t i p o INT s e r e p r e s e n t a con 32 b i t s en l a s m quinas a c t u a l e s , por a // l o que e l menor n mero que puede r e p r e s e n t a r e s e l 2147483648 u // ( 231) y e l mayor n mero que puede r e p r e s e n t a r e s 2147483647 u // (231 1). Por l o t a n t o en l o s p r i m e r o s c a s o s , t a n t o l o s n meros u // l e d o s como e l r e s u l t a d o de l a o p e r a c i o n pueden s e r r e p r e s e n t a d o s // por e l t i p o INT , s i n embargo en e l c u a r t o y q u i n t o c a s o s e l // r e s u l t a d o de l a o p e r a c i n (2200000000 y 2147483648) no puede s e r o // r e p r e s e n t a d o , y en e l s e x t o caso , i n c l u s o e l segundo n mero l e d o u // (3000000000) tampoco puede s e r r e p r e s e n t a d o por e l t i p o INT .

4. Codique el siguiente programa, ejectelo y analice el resultado obtenido. u


# include < iostream > using namespace std ; int main () { bool ok = (3.0 * (0.1 / 3.0)) = = ((3.0 * 0.1) / 3.0); cout < < " Resultado de (3.0 * (0.1 / 3.0)) == ((3.0 * 0.1) / 3.0): " < < boolalpha < < ok < < endl ; }

Solucin o

# include < iostream > using namespace std ; int main () { bool ok = (3.0 * (0.1 / 3.0)) = = ((3.0 * 0.1) / 3.0); cout < < " Resultado de (3.0 * (0.1 / 3.0)) == ((3.0 * 0.1) / 3.0): " < < boolalpha < < ok < < endl ; } // // El t i p o DOUBLE u t i l i z a una r e p r e s e n t a c i o n INEXACTA y FINITA , por l o // que s e p r o d u c e p r d i d a de p r e c i s i o n en l a s o p e r a c i o n e s de coma e // f l o t a n t e , de t a l forma que p o d ra s u c e d e r que dos v a l o r e s que // matem ticamente son i g u a l e s sean c o m p u t a c i o n a l m e n t e l i g e r a m e n t e a // d i f e r e n t e s ( un v a l o r d e c i m a l muy peque n o ) y por l o t a n t o l a // comparaci n de i g u a l d a d (==) e n t r e n meros r e a l e s no p r o d u z c a o u // l o s r e s u l t a d o s e s p e r a d o s . Por l o t a n t o , l o s n meros r e a l e s nunca s e u // d e b e ra n d e b e ra n comparar por i g u a l d a d , s i n o mediante una // o p e r a c i n s i m i l a r a : ( a b s ( xy ) < 1e 9) , que s i m p l e m e n t e comprueba o // s i dos n meros r e a l e s e s t a n l o s s u f i c i e n t e m e n t e c e r c a n o s . u // // En e s t e e j e m p l o c o n c r e t o , e l r e s u l t a d o matem tico d e b e ra s e r TRUE, a // p e r o a l r e a l i z a r s e en e l ordenador , donde e l n mero de d g i t o s d e c i m a l e s u // e s l i m i t a d o , l a o p e r a c i o n 0 . 1 /3 . 0 p r o d u c e como r e s u l t a d o 0 . 0 3 3 3 3 3 p e r i o d i c o // que a l m u l t i p l i c a r s e por 3 . 0 da como r e s u l t a d o de l a primera s u b e x p r e s i o n // 0 . 0 9 9 9 9 9 p e r i o d i c o , m i e n t r a s que en l a segunda s u b e x p r e s i n , 3 . 0 * 0 . 1 da o // como r e s u l t a d o 0 . 3 , que a l d i v i d i r s e e n t r e 3 . 0 e l r e s u l t a d o f i n a l e s 0 . 1 , // que e s d i f e r e n t e de 0 . 0 9 9 9 9 9 p e r i d i c o , por l o que e l r e s u l t a d o de l a o // comparaci n f i n a l e s FALSE o

5. Desarrolle un programa que declare tres variables de tipo double (a, b y c). El programa deber leer a tres nmeros reales desde el teclado, y posteriormente escribir true si el resultado de multiplicar los dos u a primeros nmeros es igual al tercero (a * b == c), y escribir false en caso contrario. Ejecute dicho u a programa introduciendo como datos de entrada los siguientes nmeros y analice los resultados obtenidos. u Los siguientes resultados han sido realizados en una mquina de 32 bits, en una mquina de 64 bits, el a a resultado de la ultima ejecucin es true (en vez de false como aparece en el enunciado). o

a: 1, b: 1, c: 1 true a: 2, b: 2, c: 4 true a: 3, b: 3, c: 9 true

a: 3, b: 0.1, c: 0.3 false a: 2, b: 0.2, c: 0.4 true a: 3, b: 0.3, c: 0.9 false

a: 0.1, b: 0.1, c: 0.01 false a: 0.2, b: 0.2, c: 0.04 false a: 0.3, b: 0.3, c: 0.09 false

Solucin o

# include < iostream > using namespace std ; int main () { double a , b , c ; cout < < " Introduce 3 n meros reales : " ; u cin > > a > > b > > c ; bool cmp = ( a * b = = c ); cout < < a < < " * " < < b < < " == " < < c < < " " < < boolalpha < < cmp < < endl ; } // // El t i p o DOUBLE u t i l i z a una r e p r e s e n t a c i o n INEXACTA y FINITA , por l o // que s e p r o d u c e p r d i d a de p r e c i s i o n en l a s o p e r a c i o n e s de coma e // f l o t a n t e , de t a l forma que p o d ra s u c e d e r que dos v a l o r e s que // matem ticamente son i g u a l e s sean c o m p u t a c i o n a l m e n t e l i g e r a m e n t e a // d i f e r e n t e s ( un v a l o r d e c i m a l muy peque n o ) y por l o t a n t o l a // comparaci n de i g u a l d a d (==) e n t r e n meros r e a l e s no p r o d u z c a o u // l o s r e s u l t a d o s e s p e r a d o s . Por l o t a n t o , l o s n meros r e a l e s nunca s e u // d e b e ra n d e b e ra n comparar por i g u a l d a d , s i n o mediante una // o p e r a c i n s i m i l a r a : a b s ( xy ) < 1e9 o

6. Desarrolle un programa que lea de teclado dos nmeros enteros y los almacene en dos variables de tipo u int. Posteriormente deber intercambiar los valores almacenados en dichas variables, y nalmente dea ber escribir el valor almacenado en cada una de ellas. a Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca dos n meros naturales : " ; u int dato1 , dato2 ; cin > > dato1 > > dato2 ; int aux = dato1 ; dato1 = dato2 ; dato2 = aux ; cout < < " Valor almacenado en Dato1 : " < < dato1 < < endl ; cout < < " Valor almacenado en Dato2 : " < < dato2 < < endl ; }

7. Desarrolle un programa que lea de teclado una determinada cantidad de euros, calcule y escriba su equivalente en pesetas, considerando que 1 C son 166.386 pts. Solucin o

# include < iostream > using namespace std ; // const double EUR_PTS = 166.386; // int main () { cout < < " Introduzca la cantidad de euros : " ; double euros ; cin > > euros ; double pts = euros * EUR_PTS ; cout < < euros < < " euros son " < < pts < < " pts " < < endl ; }

8. Desarrolle un programa que calcule y escriba la media aritmtica de 3 nmeros enteros le e u dos de teclado. Solucin o

# include < iostream > using namespace std ; // const int N_VALORES = 3; // int main () { cout < < " Introduzca 3 n meros enteros : " ; u int n1 , n2 , n3 ; cin > > n1 > > n2 > > n3 ; double media = double ( n1 + n2 + n3 ) / double ( N_VALORES ); cout < < " Media " < < media < < endl ; }

9. Desarrolle un programa que lea de teclado un nmero entero, y escriba true si el nmero le es par, y u u do false en caso contrario. Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca un n mero natural : " ; u int dato ; cin > > dato ; bool es_par = ( dato % 2 = = 0 ); cout < < " El n mero " < < dato < < " es par ? " < < boolalpha < < es_par < < endl ; u }

10. Desarrolle un programa que lea de teclado una letra minscula (supondremos que la entrada de datos es u correcta), y escriba la letra mayscula correspondiente a la letra minscula le previamente. u u da Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca una letra min scula : " ; u char l et ra _m i nu sc ul a ; cin > > l et ra _m i nu sc ul a ; char l et ra _m a yu sc ul a = char ( ' A ' + ( l et ra _m i nu sc ul a - ' a ' ) ); cout < < l et ra _m i nu sc ul a < < " -> " < < l et ra _m a yu sc ul a < < endl ; }

11. Desarrolle un programa que lea de teclado una cierta cantidad de segundos y muestre su equivalente en semanas, d horas, minutos y segundos, segn el formato de los siguientes ejemplos: as, u 2178585 segundos equivalen a [ 3] semanas, 4 dias, 05:09:45

9127145 segundos equivalen a [ 15] semanas, 0 dias, 15:19:05 Solucin o

# include < iostream > # include < iomanip > using namespace std ; // const int SEG_MIN = 60; const int MIN_HORA = 60; const int HORAS_DIA = 24; const int DIAS_SEMANA = 7; // const int SEG_HORA = SEG_MIN * MIN_HORA ;

const int SEG_DIA = SEG_HORA * HORAS_DIA ; const int SEG_SEMANA = SEG_DIA * DIAS_SEMANA ; // int main () { cout < < " Introduzca los segundos : " ; int s e g u n d o s _ t o t a l e s ; cin > > s e g u n d o s _ t o t a l e s ; // int semanas = s e g u n d o s _ t o t a l e s / SEG_SEMANA ; int resto = s e g u n d o s_ t o t a le s % SEG_SEMANA ; // int dias = resto / SEG_DIA ; resto = resto % SEG_DIA ; // int horas = resto / SEG_HORA ; resto = resto % SEG_HORA ; // int minutos = resto / SEG_MIN ; int segundos = resto % SEG_MIN ; // cout < < s e g u n d o s _ t o t a l e s < < " segundos equivalen a " < < " [ " < < setfill ( ' ' ) < < setw (3) < < semanas < < " ] semanas , " < < dias < < " dias " < < setfill ( ' 0 ' ) < < setw (2) < < horas < < " : " < < setfill ( ' 0 ' ) < < setw (2) < < minutos < < " : " < < setfill ( ' 0 ' ) < < setw (2) < < segundos < < " : " < < endl ; }

12. Una empresa maneja cdigos numricos, donde cada cdigo consta de cuatro d o e o gitos: El primer d gito representa a una provincia. Los dos siguientes d gitos indican el nmero de la operacin. u o El ultimo d gito es un d gito de control. Se desea obtener un programa que lea de teclado un nmero de cuatro d u gitos (el cdigo de provincia es o distinto de cero), lo almacene en una variable de tipo entero (int), y posteriormente imprima en pantalla la informacin desglosada, segn el siguiente ejemplo para el nmero 7362: o u u
Provincia: 7 Numero de operacion: 36 Digito de control: 2 Correcto: true

En caso de que el nmero tenga exactamente cuatro d u gitos, y el d gito de control sea correcto (si su valor coincide con el resto de dividir entre 10 el resultado de multiplicar el nmero de operacin por el cdigo u o o de la provincia), el apartado Correcto: escribir true, y false en otro caso. Por ejemplo, para el nmero a u 6257:
Provincia: 6 Numero de operacion: 25 Digito de control: 7 Correcto: false

Solucin o

# include < iostream > using namespace std ; // const int LIMITE = 10000; const int BASE_PROV = 1000; const int BASE_OP = 10; // int main () { cout < < " Introduzca el c digo num rico de 4 d gitos : " ; o e int codigo ; cin > > codigo ; //

= codigo / BASE_PROV ; = ( codigo % BASE_PROV ) / BASE_OP ; ( codigo % BASE_PROV ) % BASE_OP ; = (( codigo > = BASE_PROV ) & & ( codigo < LIMITE ) & & ( control = = ( operacion * provincia % 10 ))); // cout < < " Provincia : " < < provincia < < endl ; cout < < " N mero de operaci n : " < < operacion < < endl ; u o cout < < " D gito de control : " < < control < < endl ; cout < < " Correcto : " < < boolalpha < < correcto < < endl ; //

int provincia int operacion int control = bool correcto

Estructuras de Control
13. Codique un programa que lea de teclado dos nmeros enteros (x e y) y un carcter (c), y escriba true u a si cumplen las siguientes propiedades, y false en caso contrario: a) x {3, 4, 5, 6, 7} b) x {1, 2, 3, 7, 8, 9} c) x {1, 3, 5, 7, 9} d ) x {2, 5, 6, 7, 8, 9} e) x {3, 4, 6, 8, 9}, y {6, 7, 8, 3} f ) Ni x ni y sean mayores que 10 g) x no sea mltiplo de y u h) c es una letra mayscula u i ) c es una letra j ) c es un alfanumrico (letra o d e gito) Solucin o

# include < iostream > using namespace std ; int main () { int x , y ; char c ; cout < < " Introduzca dos numeros naturales : " ; cin > > x > > y ; cout < < " Introduzca un caracter : " ; cin > > c ; // bool prop_a = ( x > = 3 & & x < = 7); bool prop_b = ( x > = 1 & & x < = 3) | | ( x > = 7 & & x < = 9); bool prop_c = ( x > = 1 & & x < = 9) & & ( x % 2 = = 1); bool prop_d = ( x = = 2) | | ( x > = 5 & & x < = 9); bool prop_e = (( x > = 3 & & x < = 9 & & x ! = 5 & & x ! = 7) & & (( y > = 6 & & y < = 8) | | y = = 3)); bool prop_f = ( x < = 10 & & y < = 10); bool prop_g = ! ( y ! = 0 & & x % y = = 0); bool prop_h = ( c > = ' A ' & & c < = ' Z ' ); bool prop_i = ( c > = ' A ' & & c < = ' Z ' ) | | ( c > = ' a ' & & c < = ' z ' ); bool prop_j = (( c > = ' A ' & & c < = ' Z ' ) ||(c >= 'a ' && c <= 'z ') | | ( c > = ' 0 ' & & c < = ' 9 ' )); // cout < < boolalpha ; cout < < " ( a ) " < < x < < " pertenece a { 3 , 4 , 5 , 6 , 7 }: " < < prop_a < < endl ; cout < < " ( b ) " < < x < < " pertenece a { 1 , 2 , 3 , 7 , 8 , 9 }: " < < prop_b < < endl ; cout < < " ( c ) " < < x < < " pertenece a { 1 , 3 , 5 , 7 , 9 }: " < < prop_c < < endl ; cout < < " ( d ) " < < x < < " pertenece a { 2 , 5 , 6 , 7 , 8 , 9 }: " < < prop_d < < endl ; cout < < " ( e ) " < < x < < " pertenece a { 3 , 4 , 6 , 8 , 9 } , " < < y < < " pertenece a { 6 , 7 , 8 , 3 }: " < < prop_e < < endl ; cout < < " ( f ) Ni " < < x < < " ni " < < y < < " sean mayores que 10: " < < prop_f < < endl ; cout < < " ( g ) " < < x < < " no sea multiplo de " < < y < < " : " < < prop_g < < endl ;

cout < < " ( h ) ' " < < c < < " ' es una letra mayuscula : " < < prop_h < < endl ; cout < < " ( i ) ' " < < c < < " ' es una letra : " < < prop_i < < endl ; cout < < " ( j ) ' " < < c < < " ' es un alfanumerico : " < < prop_j < < endl ;

14. Disee un programa que lea de teclado un nmero real (n) comprendido entre 0 y 10 e imprima la nota n u asociada segn el siguiente esquema: u n = 10 9 n < 10 7n<9 5n<7 0n<5 En otro caso Matr cula de Honor Sobresaliente Notable Aprobado Suspenso Error

Solucin o

# include < iostream > using namespace std ; int main () { double nota ; cout < < " Introduzca la Nota : " ; cin > > nota ; if ( ! (( nota > = 0.0) & & ( nota < = 10.0))) { cout < < " Error : 0 <= n <= 10 " < < endl ; } else if ( nota > = 10.0 - 1 e - 9) { cout < < " Matr cula de Honor " < < endl ; } else if ( nota > = 9.0) { cout < < " Sobresaliente " < < endl ; } else if ( nota > = 7.0) { cout < < " Notable " < < endl ; } else if ( nota > = 5.0) { cout < < " Aprobado " < < endl ; } else { cout < < " Suspenso " < < endl ; } }

15. Disee un programa que lea de teclado cuatro nmeros enteros y escriba en pantalla el mayor de los cuatro. n u Solucin o

# include < iostream > using namespace std ; int main () { int a , b , c , d ; cout < < " Introduzca cuatro n meros : " ; u cin > > a > > b > > c > > d ; int mayor = a ; if ( b > mayor ) { mayor = b ; } if ( c > mayor ) { mayor = c ; } if ( d > mayor ) { mayor = d ; } cout < < " Mayor : " < < mayor < < endl ; }

16. El recibo de la electricidad se elabora de la siguiente forma para una determinada cantidad de Kw consumidos: 1 C de gastos jos. 0.50 C/Kw para los primeros 100 Kw consumidos. 8

0.35 C/Kw para los siguientes 150 Kw consumidos. 0.25 C/Kw para el resto de Kw consumidos. Elabore un programa que lea de teclado dos nmeros, que representan los dos ultimos valores del contador u de electricidad (al restarlos obtendremos el consumo en Kw) y calcule e imprima en pantalla el importe total a pagar en funcin del consumo realizado. o Solucin o

# include < iostream > using namespace std ; // const double GASTOS_FIJOS = 1.0; const double PRECIO_1 = 0.50; const double PRECIO_2 = 0.35; const double PRECIO_3 = 0.25; const int UMBRAL_1 = 100; const int UMBRAL_2 = 150; // int main () { cout < < " Introduzca los dos ltimos valores del contador : " ; u int cont_1 , cont_2 ; cin > > cont_1 > > cont_2 ; if ( cont_1 > cont_2 ) { int aux = cont_1 ; cont_1 = cont_2 ; cont_2 = aux ; } int consumo = cont_2 - cont_1 ; double importe = GASTOS_FIJOS ; if ( consumo < = UMBRAL_1 ) { importe + = consumo * PRECIO_1 ; } else { importe + = UMBRAL_1 * PRECIO_1 ; int resto = consumo - UMBRAL_1 ; if ( resto < = UMBRAL_2 ) { importe + = resto * PRECIO_2 ; } else { importe + = UMBRAL_2 * PRECIO_2 ; resto - = UMBRAL_2 ; importe + = resto * PRECIO_3 ; } } cout < < " Consumo : " < < consumo < < " Kw . " < < " Importe : " < < importe < < " Eur " < < endl ; }

17. Codique un programa que se comporte como una calculadora simple. Para ello deber tener las siguientes a caracter sticas: Solo efectuar operaciones con dos operandos (binarias). a Operaciones permitidas: (+,-,*,/). Se trabajar con operandos enteros. a Leer en primer lugar la operacin a realizar, y a continuacin los dos operandos numricos. Si el a o o e operador no se corresponde con alguno de los indicados se emitir un mensaje de error. Ejemplo: a
Operacion : * Operando 1 : 24 Operando 2 : 3 Resultado : 72

# include < iostream > using namespace std ; int main () {

Solucin o

cout < < " Operaci n : " ; o char op ; cin > > op ; if ( ! ( op = = ' + ' | | op = = ' - ' | | op = = ' * ' | | op = = ' / ' )) { cout < < " ERROR : Operaci n no v lida " < < endl ; o a } else { cout < < " Operando 1: " ; int op1 ; cin > > op1 ; cout < < " Operando 2: " ; int op2 ; cin > > op2 ; int res = 0; switch ( op ) { case ' + ' : res = op1 + op2 ; break ; case ' - ' : res = op1 - op2 ; break ; case ' * ' : res = op1 * op2 ; break ; case ' / ' : if ( op2 ! = 0) { res = op1 / op2 ; } else { res = 0; cout < < " Error , divisi n por cero " < < endl ; o } break ; } cout < < " Resultado : " < < res < < endl ; }

18. Calculador repetitivo. Modique el problema anterior para que se repita un nmero indenido de veces. u El calculador dejar de trabajar cuando se introduzca como cdigo de operacin el s a o o mbolo &. Ejemplo:
Operacion : * Operando 1 : 13 Operando 2 : 10 Resultado : 130 Operacion : u ERROR!!!! Operacion : + Operando 1 : 12 Operando 2 : 3 Resultado : 15 Operacion : & FIN DEL PROGRAMA

Solucin o

# include < iostream > using namespace std ; int main () { char op ; cout < < " Operaci n : " ; o cin > > op ; while ( op ! = ' & ' ) { if ( ! ( op = = ' + ' | | op = = ' - ' | | op = = ' * ' | | op = = ' / ' )) { cout < < " ERROR : Operaci n no v lida " < < endl ; o a } else { cout < < " Operando 1: " ; int op1 ; cin > > op1 ; cout < < " Operando 2: " ; int op2 ; cin > > op2 ; int res = 0; switch ( op ) {

10

case ' + ' : res = op1 + op2 ; break ; case ' - ' : res = op1 - op2 ; break ; case ' * ' : res = op1 * op2 ; break ; case ' / ' : if ( op2 ! = 0) { res = op1 / op2 ; } else { res = 0; cout < < " Error , divisi n por cero " < < endl ; o } break ; } cout < < " Resultado : " < < res < < endl ; } cout < < " Operaci n : " ; o cin > > op ; }

19. Disee un programa que lea un nmero entero de teclado y escriba una l n u nea con tantos asteriscos (*) como indique el nmero le u do. Por ejemplo, para un nmero le con valor 5, escribir: u do a
*****

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int i = 0; i < n ; + + i ) { cout < < SIMBOLO ; } cout < < endl ; }

20. Disee un programa que lea un nmero entero de teclado y escriba un cuadrado (relleno) con tantos n u asteriscos (*) de lado como indique el nmero le u do. Por ejemplo, para un nmero le con valor 5, u do escribir: a
***** ***** ***** ***** *****

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n ; + + i ) { cout < < SIMBOLO ; } cout < < endl ;

11

21. Disee un programa que lea un nmero entero de teclado y escriba un cuadrado (hueco) con tantos n u asteriscos (*) de lado como indique el nmero le u do. Por ejemplo, para un nmero le con valor 5, u do escribir: a
***** * * * * * * *****

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n ; + + i ) { if (( j = = 0) | | ( j = = n - 1) | | ( i = = 0) | | ( i = = n - 1)) { cout < < SIMBOLO ; } else { cout < < " " ; } } cout < < endl ; } }

22. Disee un programa que lea un nmero entero de teclado y escriba un tringulo rectngulo (relleno) con n u a a tantos asteriscos (*) de base y altura como indique el nmero le u do. Por ejemplo, para un nmero le u do con valor 5, escribir: a
* ** *** **** *****

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < = j ; + + i ) { cout < < SIMBOLO ; } cout < < endl ; } }

23. Disee un programa que lea un nmero entero de teclado y escriba un tringulo rectngulo (hueco) con n u a a tantos asteriscos (*) de base y altura como indique el nmero le u do. Por ejemplo, para un nmero le u do con valor 5, escribir: a

12

* ** * * * * *****

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < = j ; + + i ) { if (( j = = 0) | | ( j = = n - 1) | | ( i = = 0) | | ( i = = j )) { cout < < SIMBOLO ; } else { cout < < " " ; } } cout < < endl ; } }

24. Disee un programa que lea un nmero entero de teclado y escriba un tringulo (relleno) con tantos n u a asteriscos (*) de altura como indique el nmero le u do. Por ejemplo, para un nmero le con valor 5, u do escribir: a
* *** ***** ******* *********

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n - j - 1; + + i ) { cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) { cout < < SIMBOLO ; } cout < < endl ; } }

25. Disee un programa que lea un nmero entero de teclado y escriba un tringulo (hueco) con tantos n u a asteriscos (*) de altura como indique el nmero le u do. Por ejemplo, para un nmero le con valor 5, u do escribir: a
* * * * * * * *********

13

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n - j - 1; + + i ) { cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) { if (( j = = 0) | | ( j = = n - 1) | | ( i = = 0) | | ( i = = 2 * j )) { cout < < SIMBOLO ; } else { cout < < " " ; } } cout < < endl ; } }

26. Disee un programa que lea un nmero entero de teclado y escriba un rombo (relleno) con asteriscos (*), n u segn el siguiente ejemplo para un nmero le con valor 5: u u do
* *** ***** ******* ********* ******* ***** *** *

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; // for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n - j - 1; + + i ) cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) cout < < SIMBOLO ; } cout < < endl ; } // for ( int j = n - 2; j > = 0; - - j ) { for ( int i = 0; i < n - j - 1; + + i ) cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) cout < < SIMBOLO ; } cout < < endl ; } // }

27. Disee un programa que lea un nmero entero de teclado y escriba un rombo (hueco) con asteriscos (*), n u segn el siguiente ejemplo para un nmero le con valor 5: u u do 14

* * * * * * * * * * * * * * * *

Solucin o

# include < iostream > using namespace std ; const char SIMBOLO = ' * ' ; int main () { int n ; cout < < " Introduzca un n mero : " ; u cin > > n ; // for ( int j = 0; j < n ; + + j ) { for ( int i = 0; i < n - j - 1; + + i ) { cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) { if (( j = = 0) | | ( i = = 0) | | ( i = = 2 * j )) { cout < < SIMBOLO ; } else { cout < < " " ; } } cout < < endl ; } // for ( int j = n - 2; j > = 0; - - j ) { for ( int i = 0; i < n - j - 1; + + i ) { cout < < " " ; } for ( int i = 0; i < 2 * j + 1; + + i ) { if (( j = = 0) | | ( i = = 0) | | ( i = = 2 * j )) { cout < < SIMBOLO ; } else { cout < < " " ; } } cout < < endl ; } // }

28. Disee un programa que lea de teclado dos nmeros enteros y escriba en pantalla el resultado de multiplicar n u ambos nmeros, teniendo en cuenta que no se permite utilizar la operacin de multiplicar (*), por lo que u o se deber realizar mediante sumas sucesivas. Por ejemplo, 2 3 = 2 + 2 + 2 a Solucin o

# include < iostream > using namespace std ; int main () { int m , n ; cout < < " Introduce dos n meros naturales : " ; u cin > > m > > n ; int total = 0; for ( int i = 0; i < n ; + + i ) { total = total + m ; } cout < < " Resultado : " < < total < < endl ; }

15

29. Disee un programa que lea de teclado dos nmeros enteros (x e y) y escriba en pantalla el resultado de n u calcular la potencia de x elevado a y (xy ), mediante multiplicaciones sucesivas. Por ejemplo, 23 = 2 2 2 Solucin o

# include < iostream > using namespace std ; int main () { int b , e ; cout < < " Introduce la base y el exponente : " ; cin > > b > > e ; int total = 1; for ( int i = 0; i < e ; + + i ) { total = total * b ; } cout < < " Resultado : " < < total < < endl ; }

30. Disee un programa que lea de teclado un nmero entero (x) y escriba en pantalla el resultado de calcular n u el factorial de x. Por ejemplo, 4! = 1 2 3 4 Solucin o

# include < iostream > using namespace std ; int main () { int x ; cout < < " Introduce un n mero natural : " ; u cin > > x ; int total = 1; for ( int i = 2; i < = x ; + + i ) { total = total * i ; } cout < < " Resultado : " < < total < < endl ; }

31. Disee un programa que lea de teclado dos nmeros enteros y escriba en pantalla el cociente y resto de n u la divisin de ambos nmeros, teniendo en cuenta que no se permite utilizar las operaciones de divisin y o u o resto (/, %), por lo que se deber realizar mediante restas sucesivas. Por ejemplo, a 7 2 = 7 2 = 5 ; 5 2 = 3 ; 3 2 = 1 = Cociente = 3 ; Resto = 1 Solucin o

# include < iostream > using namespace std ; int main () { int dividendo , divisor ; cout < < " Introduce dividendo y divisor : " ; cin > > dividendo > > divisor ; if ( divisor = = 0) { cout < < " El divisor no puede ser cero " < < endl ; } else { int resto = dividendo ; int cociente = 0; while ( resto > = divisor ) { resto - = divisor ; + + cociente ; } cout < < " Cociente : " < < cociente < < " Resto : " < < resto < < endl ; } }

32. Disee un programa que lea un nmero entero de teclado y escriba true si el nmero es primo, y false n u u en caso contrario.

16

Solucin o

# include < iostream > using namespace std ; int main () { bool es_primo = false ; int num , divisor ; cout < < " Introduzca un n mero natural : " ; u cin > > num ; if ( num > 1) { divisor = 2; while (( num % divisor ) ! = 0) { + + divisor ; } es_primo = ( divisor = = num ); } cout < < " El n mero " < < num < < " es primo : " u < < boolalpha < < es_primo < < endl ; }

33. Desarrolle un programa que lea de teclado una secuencia de caracteres (carcter a carcter) hasta leer un a a punto (.), y que al nal muestre como salida el nmero de comas (,) encontradas, y el nmero de u u caracteres le dos. Nota: considere el concepto de ber de entrada. u Solucin o

# include < iostream > using namespace std ; int main () { int cnt_ct = 0; // c u e n t a de c a r a c t e r e s int cnt_cm = 0; // c u e n t a de comas cout < < " Introduzca una frase hasta ' . ' : " ; char c ; cin . get ( c ); // l e e un c a r c t e r a while ( c ! = ' . ' ) { + + cnt_ct ; if ( c = = ' , ' ) { + + cnt_cm ; } cin . get ( c ); // l e e un c a r c t e r a } cout < < " Cuenta de caracteres : " < < cnt_ct < < " . Cuenta de comas : " < < cnt_cm < < endl ; }

34. Desarrolle un programa que determine si la secuencia "abc" aparece en una sucesin de caracteres le o dos de teclado cuyo nal viene dado por un punto (.). Nota: considere el concepto de ber de entrada. u Solucin o

# include < iostream > using namespace std ; int main () { bool ok = false ; char ant2 = ' ' ; char ant1 = ' ' ; char act ; cout < < " Introduzca una frase hasta ' . ' : " ; cin . get ( act ); while ( act ! = ' . ' ) { if ( ant2 = = ' a ' & & ant1 = = ' b ' & & act = = ' c ' ) { ok = true ; } ant2 = ant1 ; ant1 = act ; cin . get ( act ); } cout < < " La secuencia ' abc ' " ; if ( ok ) {

17

cout < < " SI " ; } else { cout < < " NO " ; } cout < < " aparece en la frase de entrada " < < endl ;

35. Desarrolle un programa que encuentre el mayor, el menor y la media aritmtica de una coleccin de N e o nmeros le u dos por el teclado donde N es el primero de los nmeros. u Solucin o

# include < iostream > using namespace std ; int main () { int n ; cout < < " Introduzca N n meros ( siendo N el primero ): " ; u cin > > n ; if ( n = = 0) { cout < < " Error " < < endl ; } else { int mayor = n ; int menor = n ; int suma = n ; for ( int i = 0; i < n - 1; + + i ) { int num ; cin > > num ; suma + = num ; if ( num > mayor ) { mayor = num ; } else if ( num < menor ) { menor = num ; } } double media = double ( suma ) / double ( n ); cout < < " Mayor : " < < mayor < < endl ; cout < < " Menor : " < < menor < < endl ; cout < < " Media : " < < media < < endl ; } }

36. Desarrolle un programa que lea una secuencia de nmeros enteros terminada en 0, y que encuentre y u escriba en la pantalla la posicin de la primera y de la ultima ocurrencia del nmero 12 dentro de la o u secuencia. Si el nmero 12 no est en la secuencia, el programa deber escribir 0. Por ejemplo, si el u a a octavo nmero de la lista es el unico 12, entonces 8 ser la primera y la ultima posicin de las ocurrencias u a o de 12. Solucin o

# include < iostream > using namespace std ; // const int NUMERO = 12; // int main () { int pos = 0; int pri_oc = 0; int ult_oc = 0; int n ; cout < < " Introduzca una secuencia de n meros terminada en cero (0): " ; u cin > > n ; while ( n ! = 0) { + + pos ; if ( n = = NUMERO ) { if ( pri_oc = = 0) { pri_oc = pos ; } ult_oc = pos ; }

18

cin > > n ; } cout < < " Primera ocurrencia : " < < pri_oc < < endl ; cout < < " Ultima ocurrencia : " < < ult_oc < < endl ;

37. Desarrolle un programa para el siguiente juego: el usuario introduce un nmero como l u mite inferior, un nmero como l u mite superior y piensa un nmero dentro de ese rango. El objetivo del programa es acertar u el nmero pensado por el usuario. Para ello el programa propone un nmero y el usuario responde con >, u u < o =, que signican que el nmero pensado es mayor, menor o igual respectivamente al numero propuesto u por el programa. Si la respuesta es =, entonces el programa terminar satisfactoriamente, en otro caso, si a la respuesta es > o < el programa propondr otro nmero hasta que nalmente acierte el nmero pensado a u u por el usuario. Solucin o

# include < iostream > using namespace std ; int main () { int lim_inf , lim_sup ; cout < < " Introduce el l mite inferior : " ; cin > > lim_inf ; cout < < " Introduce el l mite superior : " ; cin > > lim_sup ; if ( lim_inf > lim_sup ) { cout < < " Error : l mites err neos " < < endl ; u } else { cout < < " Piense un n mero dentro del rango [ " u < < lim_inf < < " -" < < lim_sup < < " ] y pulse ENTER " ; char cod ; cin . get ( cod ); // e l i m i n a ENTER d e l BUFFER ENTRADA de l i m s u p cin . get ( cod ); // e l i m i n a ENTER d e l BUFFER ENTRADA de pausa int num ; do { num = ( lim_inf + lim_sup ) / 2; cout < < " >> Es el n mero " < < num < < " ? " ; u cin > > cod ; switch ( cod ) { case ' < ' : lim_sup = num - 1; break ; case ' > ' : lim_inf = num + 1; break ; case ' = ' : break ; default : cout < < " C digo Erroneo " < < endl ; o } } while (( cod ! = ' = ' ) & & ( lim_inf < = lim_sup )); if ( cod = = ' = ' ) { cout < < " El n mero pensado es el " < < num < < endl ; u } else { cout < < " Has introducido datos err neos " < < endl ; o } } }

38. Desarrolle un programa que lea un nmero N por teclado y calcule el n-simo nmero de la serie de u e u Fibonacci. Los dos primeros nmeros de esta serie son el cero y el uno, y a partir de stos, cada nmero u e u de la secuencia se calcula realizando la suma de los dos anteriores. Solucin o

# include < iostream > using namespace std ; // const int FIB_0 = 0; const int FIB_1 = 1;

19

// int main () { int n ; cout < < " Introduzca N : " ; cin > > n ; int fib_n ; if ( n = = 0) { fib_n = FIB_0 ; } else { int fib_n1 = FIB_0 ; fib_n = FIB_1 ; for ( int i = 2; i < = n ; + + i ) { int fib_n2 = fib_n1 ; fib_n1 = fib_n ; fib_n = fib_n1 + fib_n2 ; } } cout < < " Fibonacci ( " < < n < < " ): " < < fib_n < < endl ; }

El B er de Entrada y el B er de Salida de Datos u u


39. Codique el siguiente programa
# include < iostream > using namespace std ; int main () { cout < < " Introduzca un numero : " ; int dato_1 ; cin > > dato_1 ; cout < < " Introduzca otro numero : " ; int dato_2 ; cin > > dato_2 ; cout < < " Los numeros leidos son : " < < dato_1 < < ' ' < < dato_2 < < endl ; }

Ejectelo introduciendo el nmero 12 y pulse ENTER, posteriormente introduzca el nmero 34 y pulse u u u ENTER, compruebe la salida del programa. A continuacin, vulvalo a ejecutar introduciendo el nmero 12, pulse ESPACIO, nmero 34 y pulse ENTER, o e u u compruebe la salida del programa, comprela con la ejecucin anterior, evale las diferencias y relacinelas a o u o con el concepto de ber de entrada. u Solucin o

// // PRIMERA EJECUCION // // // El programa muestra e l mensaje [ I n t r o d u z c a un numero : ] e n t o n c e s l a // s e n t e n c i a [ c i n >> d a t o 1 ] a c c e d e a l BUFFER DE ENTRADA para l e e r un // d a t o . Como e l b u f f e r e s t v a c i o , d i c h a s e n t e n c i a e s p e r a r a a que a // haya a l g u n d a t o en e l b u f f e r . El u s u a r i o t e c l e a 1 2 [ENTER] que s e // almacenar en e l b u f f e r , por l o que e l programa s e d e s p i e r t a y a // a s i g n a r a e l v a l o r numerico 12 a l a v a r i a b l e [ d a t o 1 ] , e l i m i n a n d o s e // l o s c a r a c t e r e s 1 y 2 d e l b u f f e r ( e l c a r a c t e r [ENTER] permanece en // e l b u f f e r ) // // P o s t e r i o r m e n t e e l programa muestra e l mensaje [ I n t r o d u z c a o t r o // numero : ] e n t o n c e s l a s e n t e n c i a [ c i n >> d a t o 2 ] a c c e d e a l // BUFFER DE ENTRADA para l e e r o t r o d a t o . Como e l b u f f e r NO e s t a // v a c i o , l a l e c t u r a s a l t a l o s espacios/ENTER i n i c i a l e s , por l o que // e l i m i n a r a e l c a r a c t e r [ENTER] d e l b u f f e r ( quedaba a l l de l a // l e c t u r a a n t e r i o r ) , por l o que e l b u f f e r s e queda v a c i o y por l o // t a n t o l a s e n t e n c i a de l e c t u r a e s p e r a r a a que haya a l g u n d a t o en e l // b u f f e r . El u s u a r i o t e c l e a 3 4 [ENTER] que s e almacenar nuevamente en a // e l b u f f e r , por l o que e l programa s e d e s p i e r t a y a s i g n a r e l v a l o r a // numerico 34 a l a v a r i a b l e [ d a t o 2 ] , e l i m i n a n d o s e l o s c a r a c t e r e s 3 y

20

// 4 d e l b u f f e r ( e l c a r a c t e r [ENTER] permanece en e l b u f f e r ) // // Fin al me nt e muestra en p a n t a l l a l o s v a l o r e s de ambas v a r i a b l e s 12 y 34 // // // SEGUNDA EJECUCION // // // El programa muestra e l mensaje I n t r o d u z c a un numero : e n t o n c e s l a // s e n t e n c i a [ c i n >> d a t o 1 ] a c c e d e a l BUFFER DE ENTRADA para l e e r un // d a t o . Como e l b u f f e r e s t v a c i o , d i c h a s e n t e n c i a e s p e r a r a a que a // haya a l g u n d a t o en e l b u f f e r . El u s u a r i o t e c l e a // 1 2 [ESPACIO ] 3 4 [ENTER] que s e almacenar en e l b u f f e r , por l o que e l a // programa s e d e s p i e r t a y a s i g n a r a e l v a l o r numerico 12 a l a v a r i a b l e // [ d a t o 1 ] , e l i m i n a n d o s e l o s c a r a c t e r e s 1 y 2 d e l b u f f e r ( l o s // c a r a c t e r e s [ ESPACIO ] 3 4 [ENTER] permanecen en e l b u f f e r ) // // P o s t e r i o r m e n t e e l programa muestra e l mensaje I n t r o d u z c a o t r o // numero : e n t o n c e s l a s e n t e n c i a [ c i n >> d a t o 2 ] a c c e d e a l // BUFFER DE ENTRADA para l e e r o t r o d a t o . Como e l b u f f e r NO e s t a // v a c i o , l a s e n t e n c i a de l e c t u r a no e s p e r a r a , y por l o t a n t o , l a // l e c t u r a s a l t a l o s espacios/ENTER i n i c i a l e s , por l o que e l i m i n a r a e l // c a r a c t e r [ ESPACIO ] d e l b u f f e r y a s i g n a r a e l v a l o r num rico 34 a l a e // v a r i a b l e [ d a t o 2 ] ( quedaban en e l b u f f e r de l a l e c t u r a a n t e r i o r ) , // e l i m i n a n d o s e l o s c a r a c t e r e s 3 y 4 d e l b u f f e r ( e l c a r a c t e r [ENTER] // permanece en e l b u f f e r ) // // Fin al me nt e muestra en p a n t a l l a l o s v a l o r e s de ambas v a r i a b l e s 12 y 34 // //

40. Codique el siguiente programa


# include < iostream > # include < iomanip > using namespace std ; int main () { cout < < " Introduzca un numero : " ; int num = 123456; cin > > num ; char c = ' # ' ; cin . get ( c ); cout < < " El numero leido es : " < < num < < endl ; if ( c < ' ' ) { cout < < " El separador ( cod ASCII ) es : " < < int ( c ) < < endl ; } else { cout < < " El separador es : ' " < < c < < " ' " < < endl ; } }

Ejectelo para las siguientes entradas de datos, y analice las salidas correspondientes: u Pulse varios espacios, ENTER, varios espacios, el nmero 9876 y ENTER. u Pulse varios espacios, ENTER, varios espacios, el nmero 9876, varios espacios y ENTER. u Pulse varios espacios, el nmero 9876, la letra w y ENTER. u Pulse varios espacios, la letra x, el nmero 9876, la letra w y ENTER. u Solucin o

// // NOTA INFORMATIVA: // // Para s i m p l i f i c a r l a e x p l i c a c i n p o s t e r i o r , s e u t i l i z a r e l sm b o l o o a // [ ] para d e n o t a r e l c a r a c t e r [ ESPACIO ] ( ASCII 3 2 ) , y e l sm b o l o [ $ ] // para d e n o t a r e l c a r c t e r [ENTER] ( ASCII 10) a // //

21

// PRIMERA EJECUCION // // $ . / t 2 e j 0 4 // I n t r o d u z c a un numero : $ // 9876$ // El numero l e i d o e s : 9876 // El c a r a c t e r s e p a r a d o r e s : ' // ' , su c o d i g o ASCII : 10 [ 0 x0a ] // // INSTRUCCION BUFFER NUM C CIN // // [ $ 9876$ ] [123456] [#] [ ok ] // c i n >> num ; // [$] [9876] [#] [ ok ] // c i n . g e t ( c ) ; // [] [9876] [$] [ ok ] // // SEGUNDA EJECUCION // // $ . / t 2 e j 0 4 // I n t r o d u z c a un numero : $ // 9876 $ // El numero l e i d o e s : 9876 // El c a r a c t e r s e p a r a d o r e s : ' ' , su c o d i g o ASCII : 32 [ 0 x20 ] // // INSTRUCCION BUFFER NUM C CIN // // [ $ 9876 $ ] [123456] [#] [ ok ] // c i n >> num ; // [ $] [9876] [#] [ ok ] // c i n . g e t ( c ) ; // [ $] [9876] [ ] [ ok ] // // TERCERA EJECUCION // // $ . / t 2 e j 0 4 // I n t r o d u z c a un numero : 9876w$ // El numero l e i d o e s : 9876 // El c a r a c t e r s e p a r a d o r e s : ' w ' , su c o d i g o ASCII : 119 [ 0 x77 ] // // INSTRUCCION BUFFER NUM C CIN // // [ 9876w$ ] [123456] [#] [ ok ] // c i n >> num ; // [ w$ ] [9876] [#] [ ok ] // c i n . g e t ( c ) ; // [$] [9876] [w] [ ok ] // // CUARTA EJECUCION // // $ . / t 2 e j 0 4 // I n t r o d u z c a un numero : x9876w$ // El numero l e i d o e s : 123456 // El c a r a c t e r s e p a r a d o r e s : ' # ' , su c o d i g o ASCII : 35 [ 0 x23 ] // // INSTRUCCION BUFFER NUM C CIN // // [ x9876w$ ] [123456] [#] [ ok ] // c i n >> num ; // [ x9876w $ ] [123456] [#] [ fail ] // c i n . g e t ( c ) ; // [ x9876w $ ] [123456] [#] [ fail ] //

41. Desarrolle un programa que lea una palabra de cuatro letras por teclado, y posteriormente escriba dicha palabra de manera que cada letra se encuentre codicada sustituyndola por aquel carcter que le sigue e a en la tabla de cdigo ASCII. Nota: considere el concepto de ber de entrada. o u
# include < iostream > using namespace std ;

Solucin o

22

int main () { cout < < " Introduzca una palabra de 4 letras : " ; char l1 , l2 , l3 , l4 ; cin > > ws ; // s a l t a l o s e s p a c i o s i n i c i a l e s cin . get ( l1 ); // l e e un c a r c t e r a cin . get ( l2 ); // l e e un c a r c t e r a cin . get ( l3 ); // l e e un c a r c t e r a cin . get ( l4 ); // l e e un c a r c t e r a char n1 = char ( l1 + 1); char n2 = char ( l2 + 1); char n3 = char ( l3 + 1); char n4 = char ( l4 + 1); cout < < " La palabra [ " < < l1 < < l2 < < l3 < < l4 < < " ] " < < " transformada es [ " < < n1 < < n2 < < n3 < < n4 < < " ] " < < endl ; }

42. Desarrolle un programa que lea una palabra de cuatro letras minsculas y a continuacin la escriba en u o maysculas. Nota: considere el concepto de ber de entrada. u u Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca una palabra de 4 letras min sculas : " ; u char l1 , l2 , l3 , l4 ; cin > > ws ; // s a l t a l o s e s p a c i o s i n i c i a l e s cin . get ( l1 ); // l e e un c a r c t e r a cin . get ( l2 ); // l e e un c a r c t e r a cin . get ( l3 ); // l e e un c a r c t e r a cin . get ( l4 ); // l e e un c a r c t e r a char n1 = char ( ' A ' + ( l1 - ' a ' )); char n2 = char ( ' A ' + ( l2 - ' a ' )); char n3 = char ( ' A ' + ( l3 - ' a ' )); char n4 = char ( ' A ' + ( l4 - ' a ' )); cout < < " La palabra [ " < < l1 < < l2 < < l3 < < l4 < < " ] " < < " transformada es [ " < < n1 < < n2 < < n3 < < n4 < < " ] " < < endl ; }

43. Desarrolle un programa que lea cuatro d gitos (caracteres), calcule y almacene su valor numrico en una e variable entero, y posteriormente muestre el resultado de elevar al cuadrado dicho valor. Nota: considere el concepto de ber de entrada. u Solucin o

# include < iostream > using namespace std ; const int U_MILLAR = 1000; const int CENTENAS = 100; const int DECENAS = 10; const int UNIDADES = 1; int main () { cout < < " Introduzca 4 d gitos : " ; char d1 , d2 , d3 , d4 ; cin > > ws ; // s a l t a l o s e s p a c i o s i n i c i a l e s cin . get ( d1 ); // l e e un c a r c t e r a cin . get ( d2 ); // l e e un c a r c t e r a cin . get ( d3 ); // l e e un c a r c t e r a cin . get ( d4 ); // l e e un c a r c t e r a int n1 = ( d1 - ' 0 ' ); int n2 = ( d2 - ' 0 ' ); int n3 = ( d3 - ' 0 ' ); int n4 = ( d4 - ' 0 ' ); int valor = n1 * U_MILLAR + n2 * CENTENAS + n3 * DECENAS + n4 * UNIDADES ; int cuadrado = valor * valor ; cout < < " El n mero [ " < < d1 < < d2 < < d3 < < d4 < < " ] " u < < " al cuadrado es [ " < < cuadrado < < " ] " < < endl ; }

23

44. Desarrolle un programa que lea una palabra formada por letras minsculas hasta leer ENTER (\n) y a u continuacin la escriba en maysculas. Nota: considere el concepto de ber de entrada. o u u Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca una frase hasta ' ENTER ' : " ; char c ; cin . get ( c ); // l e e un c a r c t e r a while ( c ! = ' \ n ' ) { if ( c > = ' a ' & & c < = ' z ' ) { c = char ( ' A ' + ( c - ' a ' )); } cout < < c ; cin . get ( c ); // l e e un c a r c t e r a } cout < < endl ; }

45. Desarrolle un programa que lea una secuencia de d gitos (caracteres) hasta que lea algo distinto de d gito, almacene su valor numrico en una variable entero, y posteriormente muestre dicho valor. Nota: considere e el concepto de ber de entrada. u Solucin o

# include < iostream > using namespace std ; int main () { cout < < " Introduzca un n mero y pulse ENTER : " ; u char c ; cin > > ws ; cin . get ( c ); // l e e un c a r c t e r a int numero = 0; while ( c > = ' 0 ' & & c < = ' 9 ' ) { int n = ( c - ' 0 ' ); numero = numero * 10 + n ; cin . get ( c ); // l e e un c a r c t e r a } cout < < " Valor num rico : " < < numero < < endl ; e }

24

Tema 3: Dise o Descendente. Subprogramas n


1. Dadas las siguientes declaraciones en un determinado programa:
// -- Prototipos ---bool uno (int x, int y); void dos (int& x, int y); int tres (int x); // -- Principal -------int main () { int a, b, c; bool fin; }

Cules de las siguientes llamadas a subprogramas en el cuerpo del programa principal son vlidas ? a a a) if (uno(a,b)) {/*...*/} b) dos(a, b + 3); c) fin = uno(c, 5); d ) fin = dos(c, 5); e) dos(a,tres(a)); Solucin o h) b = tres(dos(a,5)); i ) dos(4, c); f ) dos(tres(b),c); g) if (tres(a)) {/*...*/}

// C u a l e s de l a s s i g u i e n t e s l l a m a d a s a subprogramas en e l // c u e r p o d e l programa p r i n c i p a l son v a l i d a s ? // // // ( a ) i f ( uno ( a , b ) ) {/ * . . . * /} // Es c o r r e c t a , ya que [ uno ] e s una f u n c i o n que d e v u e l v e [ b o o l ] , // por l o que puede s e r u t i l i z a d a en l a c o n d i c i o n de l a s e n t e n c i a // [ i f ] , y r e c i b e como p a r a m e t r o s por v a l o r dos v a r i a b l e s [ i n t ] // // ( b ) dos ( a , b + 3 ) ; // Es c o r r e c t a , ya que [ dos ] e s un p r o c e d i m i e n t o , por l o que d e b e // s e r u t i l i z a d a como s e n t e n c i a i n d e p e n d i e n t e , r e c i b e como primer // parametro por r e f e r e n c i a una v a r i a b l e [ i n t ] , y como segundo // parametro por v a l o r una e x p r e s i o n de t i p o [ i n t ] // // ( c ) f i n = uno ( c , 5 ) ; // Es c o r r e c t a , ya que [ uno ] e s una f u n c i o n que d e v u e l v e [ b o o l ] , // por l o que puede s e r u t i l i z a d a en l a a s i g n a c i o n a una v a r i a b l e // de t i p o [ b o o l ] , y r e c i b e como p a r a m e t r o s por v a l o r una v a r i a b l e // y una c o n s t a n t e , ambos de t i p o [ i n t ] // // ( d ) f i n = dos ( c , 5 ) ; // Es erronea , ya que [ dos ] e s un p r o c e d i m i e n t o , por l o que no puede // s e r u t i l i z a d a en l a a s i g n a c i o n a ninguna v a r i a b l e . // // ( e ) dos ( a , t r e s ( a ) ) ; // Es c o r r e c t a , ya que [ dos ] e s un p r o c e d i m i e n t o , por l o que d e b e // s e r u t i l i z a d a como s e n t e n c i a i n d e p e n d i e n t e , r e c i b e como primer // parametro por r e f e r e n c i a una v a r i a b l e [ i n t ] , y como segundo // parametro por v a l o r e l v a l o r d e v u e l t o por l a f u n c i o n [ t r e s ] que // e s de t i p o [ i n t ] , y l a l l a m a d a a l a f u n c i o n [ t r e s ] e s c o r r e c t a // p o r q u e s e u t i l i z a su v a l o r [ i n t ] como parametro a l a l l a m a d a a // [ dos ] , y r e c i b e como parametro por v a l o r una v a r i a b l e de t i p o // [ int ] // // ( f ) dos ( t r e s ( b ) , c ) ; // Es erronea , ya que [ dos ] e s un p r o c e d i m i e n t o , por l o que d e b e // s e r u t i l i z a d a como s e n t e n c i a i n d e p e n d i e n t e , p e r o r e c i b e como primer // parametro por r e f e r e n c i a e l v a l o r d e v u e l t o por l a f u n c i o n [ t r e s ] , // que no e s adecuada para e l paso por r e f e r e n c i a ( s o l o e s adecuada // una v a r i a b l e para e l paso por r e f e r e n c i a ) . // // ( g ) i f ( t r e s ( a ) ) {/ * . . . * /} // Es erronea , ya que [ t r e s ] e s una f u n c i o n que d e v u e l v e un v a l o r // [ i n t ] , que no e s adecuado como c o n d i c i o n de l a s e n t e n c i a [ i f ] //

25

// ( h ) b = t r e s ( dos ( a , 5 ) ) ; // Es erronea , ya que [ t r e s ] e s una f u n c i o n que d e v u e l v e un v a l o r // [ i n t ] , que no e s adecuado como a s i g n a c i o n de una v a r i a b l e // [ b o o l ] , ademas r e c i b e como parametro por v a l o r e l v a l o r d e v u e l t o // por e l p r o c e d i m i e n t o [ dos ] que no d e v u e l v e ningun v a l o r . // // ( i ) dos ( 4 , c ) ; // Es erronea , ya que [ dos ] e s un p r o c e d i m i e n t o , por l o que d e b e // s e r u t i l i z a d a como s e n t e n c i a i n d e p e n d i e n t e , p e r o r e c i b e como primer // parametro por r e f e r e n c i a una c o n s t a n t e [ i n t ] , que no e s // adecuada para e l paso por r e f e r e n c i a ( s o l o e s adecuada una v a r i a b l e // para e l paso por r e f e r e n c i a ) .

2. Escribe un programa que lea un nmero entero N por teclado y dibuje un tringulo de asteriscos con u a altura N . Por ejemplo si N = 5 deber dibujarse: a
* *** ***** ******* *********

Solucin o

# include < iostream > # include < cassert > using namespace std ; const char SIMBOLO = ' * ' ; // void esc_caracter ( int n , char simb ) { for ( int i = 0; i < n ; + + i ) { cout < < simb ; } } // void esc_fila ( int f , int nf ) { assert ( f < nf ); esc_caracter ( nf - f - 1 , ' ' ); esc_caracter (2 * f + 1 , SIMBOLO ); cout < < endl ; } // void esc_triangulo ( int nf ) { for ( int f = 0; f < nf ; + + f ) { esc_fila (f , nf ); } } // int main () { cout < < " Introduzca numero de filas : " ; int n_filas ; cin > > n_filas ; esc_triangulo ( n_filas ); }

3. Escribe un programa que imprima una pirmide de d a gitos como la de la gura, tomando como entrada el nmero de las de la misma (se supone menor de 10). u
1 121 12321 1234321 123454321

# include < iostream >

Solucin o

26

# include < cassert > using namespace std ; // const int MAX_FILAS = 10; // void esc_caracter ( int n , char simb ) { for ( int i = 0; i < n ; + + i ) { cout < < simb ; } } // void esc_asce ndente ( int n ) { for ( int i = 1; i < = n ; + + i ) { cout < < i ; } } // void e sc _d es c en de nt e ( int n ) { for ( int i = n ; i > = 1; - - i ) { cout < < i ; } } // void esc_fila ( int f , int nf ) { assert ( f < = nf ); esc_caracter ( nf -f , ' ' ); esc_ ascenden te ( f ); e sc _d es c en de nt e ( f - 1); cout < < endl ; } // void esc_triangulo ( int nf ) { for ( int f = 1; f < = nf ; + + f ) { esc_fila (f , nf ); } } // int main () { cout < < " Introduzca numero de filas : " ; int n_filas ; cin > > n_filas ; if ( n_filas < MAX_FILAS ) { esc_triangulo ( n_filas ); } }

4. Disea un programa que lea de teclado un nmero entero n mayor que cero y muestre las n primeras las n u del siguiente tringulo. a
1 232 34543 4567654 567898765 67890109876 7890123210987 890123454321098 90123456765432109 0123456789876543210 123456789010987654321 .......................

# include < iostream > # include < cassert > using namespace std ;

Solucin o

27

// void esc_caracter ( int n , char simb ) { for ( int i = 0; i < n ; + + i ) { cout < < simb ; } } // void esc_asce ndente ( int a , int b ) { assert ( a < = b ); for ( int i = a ; i < = b ; + + i ) { cout < < ( i %10); } } // void e sc _d es c en de nt e ( int a , int b ) { for ( int i = a ; i > = b ; - - i ) { cout < < ( i %10); } } // void esc_fila ( int f , int nf ) { assert ( f < = nf ); esc_caracter ( nf -f , ' ' ); esc_ ascenden te (f , 2 * f - 1); e sc _d es c en de nt e (2 * f - 2 , f ); cout < < endl ; } // void esc_triangulo ( int nf ) { for ( int f = 1; f < = nf ; + + f ) { esc_fila (f , nf ); } } // int main () { cout < < " Introduzca numero de filas : " ; int n_filas ; cin > > n_filas ; esc_triangulo ( n_filas ); }

5. Escribe un programa que calcule el valor de S para un nmero real X (0 X 1) dado por teclado, u utilizando la serie de Taylor: X2 X3 X4 S =1+X + + + + 2! 3! 4! Nota: No se aadirn ms sumandos cuando se calcule uno con valor menor que 0.0001. n a a Solucin o

# include < iostream > # include < cassert > using namespace std ; // const int L I M I T E _ F A C T O R I A L _ U N S I G N E D = 14; const double LIMITE = 1 e - 4; // double potencia ( double base , int exp ) { assert ( base > = 0 & & base < = 1); double res = 1; for ( int i = 0; i < exp ; + + i ) { res * = base ; } return res ; } //

28

int factorial ( int n ) { assert ( n < L I M I T E _ F A C T O R I A L _ U N S I G N E D ); int res = 1; for ( int i = 2; i < = n ; + + i ) { res * = i ; } return res ; } // double termino ( double x , int i ) { return potencia (x , i ) / double ( factorial ( i )); } // double serie ( double x ) { assert ( x > = 0 & & x < = 1); int i = 0; double res = 1; double term ; do { ++i; term = termino (x , i ); res + = term ; } while ( term > = LIMITE ); return res ; } // int main () { cout < < " Introduzca el valor de X [0..1]: " ; double x ; cin > > x ; if ( ! ( x > = 0 & & x < = 1)) { cout < < " Error . Valor de X fuera de rango " < < endl ; } else { cout < < " Serie : " < < serie ( x ) < < endl ; } }

6. Escribe un programa que calcule el valor de S para un nmero real X (0 X 1) dado por teclado, u utilizando la siguiente serie: S=X+ 1 X3 13 X 5 135 X 7 + + + 2 3 24 5 246 7

Nota: No se aadirn ms de 10 sumandos. n a a Solucin o

# include < iostream > # include < cassert > using namespace std ; // const int L I M I T E _ F A C T O R I A L _ U N S I G N E D = 14; const int MAX_ITER = 10; // double potencia ( double base , int exp ) { assert ( base > = 0 & & base < = 1); double res = 1; for ( int i = 0; i < exp ; + + i ) { res * = base ; } return res ; } // int producto ( int inicio , int limite ) { assert ( limite < = L I M I T E _ F A C T O R I A L _ U N S I G N E D ); int res = 1; for ( int i = inicio ; i < limite ; i + = 2) {

29

res * = i ; } return res ; } // double termino ( double x , int i ) { return ( double ( producto (1 , i )) / double ( producto (2 , i )) * potencia (x , i ) / double ( i )); } // double serie ( double x ) { assert ( x > = 0 & & x < = 1); double res = x ; for ( int i = 1; i < MAX_ITER ; + + i ) { res + = termino (x , 2 * i + 1); } return res ; } // int main () { cout < < " Introduzca el valor de X [0..1]: " ; double x ; cin > > x ; if ( ! ( x > = 0 & & x < = 1)) { cout < < " Error . Valor de X fuera de rango " < < endl ; } else { double s = serie ( x ); cout < < " Serie : " < < s < < endl ; } }

7. Disea un programa que encuentre el primer nmero perfecto mayor que 28. Un nmero es perfecto si n u u coincide con la suma de sus divisores (salvo l mismo). e Por ejemplo, 28 es perfecto ya que 28 = 1 + 2 + 4 + 7 + 14 Solucin o

# include < iostream > # include < cassert > using namespace std ; // int PRIMER_NUMERO = 29; // int suma_div isores ( int n ) { int suma = 0; for ( int i = 1; i < = n / 2; + + i ) { if ( n % i = = 0) { suma + = i ; } } return suma ; } // inline bool es_perfecto ( int n ) { return n = = suma_di visores ( n ); } // int p ri me r_ p er fe ct o () { int i = PRIMER_NUMERO ; while ( ! es_perfecto ( i )) { ++i; } assert ( es_perfecto ( i )); return i ; } //

30

int main () { cout < < " Primer perfecto mayor que " < < PRIMER_NUMERO < < " : " < < p r im er _p e rf ec to () < < endl ; }

8. Dos nmeros a y b se dice que son amigos si la suma de los divisores de a (salvo l mismo) coincide con u e b y viceversa. Disea un programa que tenga como entrada dos nmeros enteros n y m y que muestre en n u la pantalla todas las parejas de nmeros amigos que existan en el intervalo determinado por n y m. u Solucin o

# include < iostream > # include < cassert > using namespace std ; // int suma_div isores ( int n ) { int suma = 0; for ( int i = 1; i < = n / 2; + + i ) { if ( n % i = = 0) { suma + = i ; } } return suma ; } // inline bool son_amigos ( int a , int b ) { return (( a = = suma _divisores ( b )) & & ( b = = sum a_divis ores ( a ))); } // void ordenar ( int & n , int & m ) { if ( n > m ) { int aux = n ; n = m; m = aux ; } assert ( n < = m ); } // void i mp ri mi r _a mi go s ( int n , int m ) { ordenar (n , m ); for ( int i = n ; i < m ; + + i ) { for ( int j = i + 1; j < = m ; + + j ) { if ( son_amigos (i , j )) { cout < < " Amigos : " < < i < < " , " < < j < < endl ; } } } } // int main () { cout < < " Introduzca un intervalo : " ; int n , m ; cin > > n > > m ; i mp ri mi r _a mi go s (n , m ); }

Otra solucin alternativa o

# include < iostream > # include < cassert > using namespace std ; // int suma_div isores ( int n ) { int suma = 0;

31

for ( int i = 1; i < = n / 2; + + i ) { if ( n % i = = 0) { suma + = i ; } } return suma ; } // inline bool son_amigos ( int a , int b ) { return (( a = = suma _divisores ( b )) & & ( b = = sum a_divis ores ( a ))); } // void numero_amigo ( int a , int & b , bool & ok ) { b = suma_div isores ( a ); ok = ( a = = suma_ divisore s ( b )); assert ( ! ok | | son_amigos (a , b )); } // void ordenar ( int & n , int & m ) { if ( n > m ) { int aux = n ; n = m; m = aux ; } assert ( n < = m ); } // void i mp ri mi r _a mi go s ( int n , int m ) { ordenar (n , m ); for ( int i = n ; i < m ; + + i ) { bool amigos ; int j ; numero_amigo (i , j , amigos ); if ( amigos & & ( i < j ) & & ( j < m )) { cout < < " Amigos : " < < i < < " , " < < j < < endl ; } } } // int main () { cout < < " Introduzca un intervalo : " ; int n , m ; cin > > n > > m ; i mp ri mi r _a mi go s (n , m ); }

9. Disea un programa que calcule e imprima en pantalla los N primeros nmeros primos, siendo N un n u nmero que se introduce por teclado. u Solucin o

# include < iostream > using namespace std ; // bool es_primo ( int n ) { int i ; for ( i = 2; ( i < = n / 2) & & ( n % i ! = 0); + + i ) { // v a c i o } return ( i = = n / 2 + 1); } // void i mp ri mi r _p ri mo s ( int n ) { int cnt = 0; int i = 0; while ( cnt < n ) {

32

if ( es_primo ( i )) { + + cnt ; cout < < i < < " , " ; } ++i; } cout < < endl ; } // int main () { int n ; cout < < " Introduzca N : " ; cin > > n ; i mp ri mi r _p ri mo s ( n ); }

10. Disea un programa que calcule e imprima en pantalla los N primeros trminos de la sucesin de Fibonacci, n e o siendo N un nmero que se introduce por teclado. Ej: 0, 1, 1, 2, 3, 5, 8, 13, 21, u Solucin o

# include < iostream > using namespace std ; // const int FIB_0 = 0; const int FIB_1 = 1; // int fibonacci ( int n ) { int fib_n ; if ( n = = 0) { fib_n = FIB_0 ; } else { int fib_n1 = FIB_0 ; fib_n = FIB_1 ; for ( int i = 2; i < = n ; + + i ) { int fib_n2 = fib_n1 ; fib_n1 = fib_n ; fib_n = fib_n1 + fib_n2 ; } } return fib_n ; } // void i m p r i m i r _ f i b o n a c c i ( int n ) { for ( int i = 0; i < n ; + + i ) { cout < < fibonacci ( i ) < < " , " ; } cout < < endl ; } // int main () { int n ; cout < < " Introduzca N : " ; cin > > n ; i m p r i m i r _ f i b o n a c c i ( n ); }

Otra solucin alternativa o

# include < iostream > using namespace std ; // const int FIB_0 = 0; const int FIB_1 = 1; // void i m p r i m i r _ f i b o n a c c i ( int n ) { if ( n > 0) {

33

cout < < FIB_0 < < " , " ; } if ( n > 1) { cout < < FIB_1 < < " , " ; } int fib_n1 = FIB_0 ; int fib_n = FIB_1 ; for ( int i = 2; i < n ; + + i ) { int fib_n2 = fib_n1 ; fib_n1 = fib_n ; fib_n = fib_n1 + fib_n2 ; cout < < fib_n < < " , " ; } cout < < endl ; } // int main () { int n ; cout < < " Introduzca N : " ; cin > > n ; i m p r i m i r _ f i b o n a c c i ( n ); }

11. Escriba un programa que tome como entrada desde teclado dos nmeros enteros (mayores que cero) N e i, u e imprima en pantalla el d gito que ocupa la posicin i-sima del nmero N . Si i es mayor que el nmero o e u u de d gitos de N , se escribir en pantalla 0. Por ejemplo, para N = 25064 e i = 2, el resultado es el d a gito 6, y para i = 6, el resultado es 0. Solucin o

# include < iostream > using namespace std ; // // El n d i c e d e l primer d g i t o e s e l c e r o // int digito ( int n , int i ) { int resto = n ; for ( int j = 0; ( resto > 0) & & ( j < i ); + + j ) { resto / = 10; } return resto % 10; } // int main () { cout < < " Introduzca n mero : " ; u int num ; cin > > num ; cout < < " Introduzca d gito : " ; int dig ; cin > > dig ; if ( dig = = 0) { cout < < " Error , d gito err neo " < < endl ; o } else { cout < < " Valor del d gito : " < < digito ( num , dig - 1) < < endl ; } }

12. Escribe un programa que acepte como entrada desde teclado un nmero entero mayor que cero y d como u e salida el resultado de sumar dos a dos los d gitos que aparecen en posiciones simtricas respecto al d e gito central dentro del nmero dado como entrada. Por ejemplo : u Para el nmero : 2354869 u La salida es: 2 + 9 = 11, 3 + 6 = 9, 5 + 8 = 13, 4 Para el nmero : 6582 u La salida es : 6 + 2 = 8, 5 + 8 = 13

34

Solucin o

# include < iostream > using namespace std ; // int n_digitos ( int n ) { int resto = n ; int i = 1; while ( resto > 9) { ++i; resto / = 10; } return i ; } // int digito ( int n , int i ) { int resto = n ; for ( int j = 0; ( resto > 0) & & ( j < i ); + + j ) { resto / = 10; } return resto % 10; } // void i m p r i m i r _ s u m a _ d i g i t o s ( int n ) { int nd = n_digitos ( n ); for ( int i = 0; i < nd / 2; + + i ) { int di = digito (n , i ); int df = digito (n , nd - i - 1); cout < < di < < " + " < < df < < " = " < < ( di + df ) < < " , " ; } if ( nd %2 ! = 0) { cout < < digito (n , nd / 2); } cout < < endl ; } // int main () { cout < < " Introduzca n mero : " ; u int num ; cin > > num ; i m p r i m i r _ s u m a _ d i g i t o s ( num ); }

13. Dada una sucesin, de longitud indeterminada, de caracteres ceros y unos, construir un programa que o permita calcular el tamao de la mayor subsucesin ordenada de menor a mayor. La sucesin se lee desde n o o el teclado, y el nal viene dado por el carcter punto (.). Ejemplos: a Para la sucesin de entrada: 001001101. imprimir 4. o a Para la sucesin de entrada: 0100101111. imprimir 5. o a Solucin o

# include < iostream > using namespace std ; // inline bool es_fin_sec ( char ant , char act ) { return ( act < ant ); } // void fin_sec_ord ( int & cnt , int & mayor ) { if ( cnt > mayor ) { mayor = cnt ; } cnt = 0; } //

35

void leer ( char & ant , char & act ) { ant = act ; cin > > act ; } // int leer_sucesion () { int mayor = 0; int cnt = 0; char ant = ' 0 ' ; // v a l o r f i c t i c i o char act = ' 0 ' ; // v a l o r f i c t i c i o leer ( ant , act ); while ( act ! = ' . ' ) { if ( es_fin_sec ( ant , act )) { fin_sec_ord ( cnt , mayor ); } + + cnt ; leer ( ant , act ); } fin_sec_ord ( cnt , mayor ); return mayor ; } // int main () { cout < < " Introduzca sucesi n de ceros y unos hasta punto : " ; o int lng = leer_sucesion (); cout < < " Mayor subsucesi n ordenada : " < < lng < < endl ; o }

14. Decimos que una sucesin a1 , a2 , , an de enteros forma una montaa, si existe un h tal que : 1 h n o n y adems a a1 < ...ah1 < ah > ah+1 > an Denimos la anchura de una montaa como el nmero de enteros que la forman. Por ejemplo la sucesin n u o 7, 1, 6, 21, 15 es una montaa de anchura 5. Denimos un valle de la misma forma que una montaa n n pero cambiando el signo de las desigualdades de la denicin anterior: o a1 > ...ah1 > ah < ah+1 < an Por ejemplo 24, 13, 6, 15, 50 ser un valle. a Dada una secuencia de nmeros enteros terminada en cero (0) y separados por espacios en blanco, que u como m nimo contiene una montaa y un valle (suponemos que la secuencia de enteros de entrada es una n secuencia correcta de montaas y valles), disea un programa que calcule la anchura de la montaa y el n n n valle ms largos. Nota: el cero (0) terminador no forma parte de la secuencia. Por ejemplo para la secuencia: a 40 35 30 25 20 22 23 34 50 65 21 13 4 0 producir como mayor montaa 9 y como mayor valle 10. a n

40

35

30

25

20

22

23

34

50

65

21

13

montaa valle
Solucin o

montaa valle

# include < iostream > using namespace std ; // inline bool es_fin_mont ( int ant2 , int ant1 , int act ) {

36

return ( ant1 < ant2 ) & & ( ant1 < act ); } // inline bool es_fin_vall ( int ant2 , int ant1 , int act ) { return ( ant1 > ant2 ) & & ( ant1 > act ); } // void fin_sec ( int & cnt , int & mayor ) { if ( cnt > mayor ) { mayor = cnt ; } cnt = 1; } // void leer ( int & ant2 , int & ant1 , int & act ) { ant2 = ant1 ; ant1 = act ; cin > > act ; } // void leer_sucesion ( int & mont_mayor , int & vall_mayor ) { mont_mayor = 0; vall_mayor = 0; int mont_cnt = 0; int vall_cnt = 0; int ant2 = 0; // v a l o r f i c t i c i o int ant1 = 0; // v a l o r f i c t i c i o int act = 0; // v a l o r f i c t i c i o leer ( ant2 , ant1 , act ); while ( act ! = 0) { if ( es_fin_mont ( ant2 , ant1 , act )) { fin_sec ( mont_cnt , mont_mayor ); } else if ( es_fin_vall ( ant2 , ant1 , act )) { fin_sec ( vall_cnt , vall_mayor ); } + + mont_cnt ; + + vall_cnt ; leer ( ant2 , ant1 , act ); } fin_sec ( mont_cnt , mont_mayor ); fin_sec ( vall_cnt , vall_mayor ); } // int main () { cout < < " Introduzca sucesi n de enteros hasta cero : " ; o int mm , vm ; leer_sucesion ( mm , vm ); cout < < " Mayor Monta~ a : " < < mm < < endl ; n cout < < " Mayor Valle : " < < vm < < endl ; }

37

Tema 4: Tipos de Datos Estructurados


1. Para realizar operaciones con nmeros complejos, podemos denir el siguiente tipo: u
struct Complejo { double real; double img; };

Escribe subprogramas que realicen las operaciones de suma, resta, multiplicacin y divisin de nmeros o o u complejos denidos con el tipo anterior, as como el programa para probar adecuadamente su funcionamien to. Comprese con la solucin del ejercicio 34 de este mismo tema. a o Solucin o

# include < iostream > using namespace std ; // const double E R RO R_ PR E CI SI ON = 1 e - 10; // struct Complejo { double real ; double img ; }; // inline double sq ( double x ) { return x * x ; } // bool iguales ( double x , double y ) { double cmp = x - y ; return ( - ER R OR _P RE C IS IO N < = cmp ) & & ( cmp < = E RR OR _ PR EC IS I ON ); } // Complejo crear ( double real , double img ) { Complejo res ; res . real = real ; res . img = img ; return res ; } // Complejo sumar ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = c1 . real + c2 . real ; res . img = c1 . img + c2 . img ; return res ; } // Complejo restar ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = c1 . real - c2 . real ; res . img = c1 . img - c2 . img ; return res ; } // Complejo multiplicar ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = ( c1 . real * c2 . real ) - ( c1 . img * c2 . img ); res . img = ( c1 . real * c2 . img ) + ( c1 . img * c2 . real ); return res ; } // Complejo dividir ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = (( c1 . real * c2 . real ) + ( c1 . img * c2 . img )) / ( sq ( c2 . real ) + sq ( c2 . img )); res . img = (( c1 . img * c2 . real ) - ( c1 . real * c2 . img )) / ( sq ( c2 . real ) + sq ( c2 . img ));

38

return res ; } // bool iguales ( const Complejo & c1 , const Complejo & c2 ) { return iguales ( c1 . real , c2 . real ) & & iguales ( c1 . img , c2 . img ); } // bool distintos ( const Complejo & c1 , const Complejo & c2 ) { return ! iguales ( c1 , c2 ); } // inline void escribir ( const Complejo & c ) { cout < < " ( " < < c . real < < " , " < < c . img < < " ) " ; } // // // void leer ( Complejo & c ) { cout < < " Introduzca un n mero complejo ( real , img ): " ; u cin > > c . real > > c . img ; } // void prueba_suma ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = sumar ( c1 , c2 ); escribir ( c1 ); cout < < " + " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( distintos ( c1 , restar ( c0 , c2 ))) { cout < < " Error en operaciones de suma/resta " < < endl ; } } // void prueba_resta ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = restar ( c1 , c2 ); escribir ( c1 ); cout < < " - " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( distintos ( c1 , sumar ( c0 , c2 ))) { cout < < " Error en operaciones de suma/resta " < < endl ; } } // void prueba_mult ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = multiplicar ( c1 , c2 ); escribir ( c1 ); cout < < " * " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( distintos ( c1 , dividir ( c0 , c2 ))) { cout < < " Error en operaciones de mult/div " < < endl ; } } // void prueba_div ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = dividir ( c1 , c2 ); escribir ( c1 ); cout < < " / " ; escribir ( c2 );

39

cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( distintos ( c1 , multiplicar ( c0 , c2 ))) { cout < < " Error en operaciones de mult/div " < < endl ; } } // int main () { Complejo c1 , c2 ; leer ( c1 ); leer ( c2 ); // prueba_suma ( c1 , c2 ); prueba_resta ( c1 , c2 ); prueba_mult ( c1 , c2 ); prueba_div ( c1 , c2 ); // }

2. Disee una funcin para buscar la posicin que ocupa un elemento dado (x) en un array de enteros. Si el n o o elemento no se encuentra en el array, entonces devolver un valor fuera del rango vlido para ese array. a a Adems, disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. a n
int buscar (const Vector& v, int x);

Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int NELMS = 5; typedef array < int , NELMS > Vector ; // int buscar ( const Vector & v , int x ) { int i = 0; while (( i < v . size ()) & & ( x ! = v [ i ] )) { ++i; } return i ; } // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // void escribir ( const Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { cout < < v [ i ] < < ' ' ; } cout < < endl ; } // int main () { Vector v ; leer ( v ); cout < < " Introduzca elemento a buscar : " ; int x ; cin > > x ; int i = buscar (v , x ); if ( i > = v . size ()) { cout < < " El elemento " < < x < < " no se encuentra en la lista " < < endl ;

40

} else { cout < < " El elemento " < < x < < " se encuentra en la posici n " < < i o < < " en la lista " < < endl ; } cout < < " Lista : " ; escribir ( v );

3. Disee una funcin para buscar la posicin del menor elemento de un array de enteros. Adems, disee el n o o a n programa y los subprogramas necesarios para probar adecuadamente su funcionamiento.
int buscar_pos_menor (const Vector& v);

Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int NELMS = 5; typedef array < int , NELMS > Vector ; // int b u s c a r _ p o s _ m e n o r ( const Vector & v ) { int pos_menor = 0; for ( int i = pos_menor + 1; i < v . size (); + + i ) { if ( v [ i ] < v [ pos_menor ] ) { pos_menor = i ; } } return pos_menor ; } // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // void escribir ( const Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { cout < < v [ i ] < < ' ' ; } cout < < endl ; } // int main () { Vector v ; leer ( v ); int i = b u s c a r _ p o s _ me n o r ( v ); cout < < " El menor elemento se encuentra en la posici n " < < i < < " en la lista " o < < endl ; cout < < " Lista : " ; escribir ( v ); }

4. Disee una funcin para buscar el menor elemento de un array de enteros. Adems, disee el programa y n o a n los subprogramas necesarios para probar adecuadamente su funcionamiento.
int buscar_menor (const Vector& v);

# include < iostream > # include < tr1/array >

Solucin o

41

using namespace std ; using namespace std :: tr1 ; // const int NELMS = 5; typedef array < int , NELMS > Vector ; // int b u s c a r _ p o s _ m e n o r ( const Vector & v ) { int pos_menor = 0; for ( int i = pos_menor + 1; i < v . size (); + + i ) { if ( v [ i ] < v [ pos_menor ] ) { pos_menor = i ; } } return pos_menor ; } // inline int buscar_menor ( const Vector & v ) { int pos_menor = b u s ca r _ p o s_ m e n o r ( v ); return v [ pos_menor ] ; } // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // void escribir ( const Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { cout < < v [ i ] < < ' ' ; } cout < < endl ; } // int main () { Vector v ; leer ( v ); int menor = buscar_menor ( v ); cout < < " El menor elemento de la lista es " < < menor < < endl ; cout < < " Lista : " ; escribir ( v ); }

5. Disee una funcin para buscar la posicin que ocupa un patrn dado en una cadena de caracteres. Si el n o o o patrn no se encuentra en la cadena, entonces devolver un valor fuera del rango vlido para esa cadena. o a a Adems, disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. a n
int buscar (const string& cadena, const string& patron);

Solucin o

# include < iostream > # include < string > using namespace std ; // int buscar ( const string & cadena , const string & patron ) { int i = 0; while (( i + patron . size () < = cadena . size ()) & & ( patron ! = cadena . substr (i , patron . size ()))) { ++i; } if ( i + patron . size () > cadena . size ()) { i = cadena . size (); } return i ;

42

} // int main () { cout < < " Introduzca una cadena : " ; string cad ; getline ( cin , cad ); cout < < " Introduzca un patr n a buscar : " ; o string pat ; getline ( cin , pat ); int i = buscar ( cad , pat ); if ( i > = cad . size ()) { cout < < " El patr n [ " < < pat < < " ] no se encuentra en la cadena " < < endl ; o } else { cout < < " El patr n [ " < < pat < < " ] se encuentra en la posici n " < < i o o < < " en la cadena " < < endl ; } cout < < " Cadena : " < < cad < < endl ; }

6. Disee un procedimiento para reemplazar en una cadena todas las ocurrencias de un patrn por otro valor. n o Adems, disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. a n
void reemplazar (string& cadena, const string& patron, const string& nuevo);

Solucin o

# include < iostream > # include < string > using namespace std ; // int buscar ( const string & cadena , int ini , const string & patron ) { int i = ini ; while (( i + patron . size () < = cadena . size ()) & & ( patron ! = cadena . substr (i , patron . size ()))) { ++i; } if ( i + patron . size () > cadena . size ()) { i = cadena . size (); } return i ; } // void reemplazar ( string & cadena , const string & patron , const string & nuevo ) { int i = buscar ( cadena , 0 , patron ); while ( i < cadena . size ()) { if ( i + patron . size () < cadena . size ()) { cadena = ( cadena . substr (0 , i ) + nuevo + cadena . substr ( i + patron . size () , cadena . size () - ( i + patron . size ()))); } else { cadena = cadena . substr (0 , i ) + nuevo ; } i = buscar ( cadena , i + nuevo . size () , patron ); } } // int main () { cout < < " Introduzca una cadena : " ; string cad ; getline ( cin , cad ); cout < < " Introduzca un patr n a reemplazar : " ; o string pat ; getline ( cin , pat ); cout < < " Introduzca el nuevo valor : " ; string nv ; getline ( cin , nv ); reemplazar ( cad , pat , nv );

43

cout < < " Salida : " < < cad < < endl ;

7. Disee el programa y los subprogramas necesarios para probar adecuadamente el funcionamiento de los n siguientes apartados. a) Disea una funcin booleana que reciba dos arrays de nmeros enteros, y devuelva true si son iguales n o u y false en otro caso.
bool iguales (const Vector& v1, const Vector& v2);

b) Disea una funcin booleana que reciba dos arrays de nmeros enteros, y devuelva true si ambos n o u contienen los mismos elementos y en el mismo orden relativo, suponiendo que el primer elemento sigue al ultimo, y false en otro caso. Podemos suponer que cada elemento del array aparece a lo sumo una vez.
bool iguales_relativos (const Vector& v1, const Vector& v2);

Por ejemplo, si la entrada fuese la siguiente, la funcin devolver true. o a


v1: [ 1, 3, 4, 9, 6 ] v2: [ 4, 9, 6, 1, 3 ]

Solucin o

# include < iostream > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 20; // typedef array < int , MAX > Vector ; // // IGUALDAD // bool iguales ( const Vector & v1 , const Vector & v2 ) { bool ok = false ; if ( v1 . size () = = v2 . size ()) { int i = 0; while (( i < v1 . size ()) & & ( v1 [ i ] = = v2 [ i ] )) { ++i; } ok = ( i > = v1 . size ()); } return ok ; } // // IGUALDAD RELATIVA // int buscar ( const Vector & v , int x ) { int i = 0; while (( i < v . size ()) & & ( x ! = v [ i ] )) { ++i; } return i ; } // bool i g u a l e s _ r e l a t i v o s ( const Vector & v1 , const Vector & v2 , int i2i ) { assert ( i2i < v2 . size ()); bool ok = false ; if ( v1 . size () = = v2 . size ()) { int i2 = i2i ; int i1 = 0; while (( i1 < v1 . size ()) & & ( v1 [ i1 ] = = v2 [ i2 ] )) { i2 = ( i2 + 1) % v2 . size (); + + i1 ; } ok = ( i1 > = v1 . size ());

44

} return ok ; } // bool i g u a l e s _ r e l a t i v o s ( const Vector & v1 , const Vector & v2 ) { bool ok = false ; if ( v1 . size () = = v2 . size ()) { if ( v1 . size () = = 0) { ok = true ; } else { int i2 = buscar ( v2 , v1 [ 0 ] ); ok = ( i2 < v2 . size ()) & & i g u a l e s _ r e l a t i v o s ( v1 , v2 , i2 ); } } return ok ; } // // LEER ESCRIBIR // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // void escribir ( const Vector & v ) { cout < < " [ " ; for ( int i = 0; i < v . size (); + + i ) { cout < < " ' " < < v [ i ] < < " ' " ; } cout < < " ] " < < endl ; } // // MENU // char menu () { char op ; cout < < endl ; cout < < " - - - - - - - - - - - - - - - - - - - " < < endl ; cout < < " a : Igualdad " < < endl ; cout < < " b : Iguales Orden Relativo " < < endl ; cout < < " x : Fin " < < endl ; cout < < " - - - - - - - - - - - - - - - - - - - " < < endl ; cout < < endl ; do { cout < < " Opcion ? " ; cin > > op ; } while ( ! (( op > = ' a ' & & op < = ' b ' ) | | op = = ' x ' )); cin . ignore (1000 , ' \ n ' ); return op ; } // void check_ig ualdad () { Vector v1 , v2 ; leer ( v1 ); leer ( v2 ); if ( iguales ( v1 , v2 )) { cout < < " Los vectores son iguales " < < endl ; } else { cout < < " Los vectores NO son iguales " < < endl ; } } // void c h e c k _ i g u a l d a d _ r e l a t i v a () { Vector v1 , v2 ; leer ( v1 ); leer ( v2 );

45

if ( i g u a l e s _ r e l a t i v o s ( v1 , v2 )) { cout < < " Los vectores son iguales relativos " < < endl ; } else { cout < < " Los vectores NO son iguales relativos " < < endl ; } } // int main () { char op ; do { op = menu (); switch ( op ) { case ' a ' : chec k_iguald ad (); break ; case ' b ' : c h e c k _ i g u a l d a d _ r e l a t i v a (); break ; } } while ( op ! = ' x ' ); }

8. Se dispone de un array de 10 nmeros enteros en el que al menos hay dos nmeros que son iguales y dos u u que son distintos. Obtenga una funcin que tomando como parmetro dicho array, devuelva un elemento o a del array que sea mayor que el m nimo de ste. Adems, disee el programa y los subprogramas necesarios e a n para probar adecuadamente su funcionamiento. Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int NELMS = 5; typedef array < int , NELMS > Vector ; // int mayor ( int a , int b ) { int res = a ; if ( b > res ) { res = b ; } return b ; } // // Busca un e l e m e n t o que s e a mayor que e l m nimo d e l a r r a y // Al menos hay dos e l e m e n t o s i g u a l e s // Al menos hay dos e l e m e n t o s d i s t i n t o s int b u s c a r _ p r o p i e d a d ( const Vector & v ) { int i = 1; while ( v [ 0 ] = = v [ i ] ) { ++i; } return mayor ( v [ 0 ] , v [ i ] ); } // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // void escribir ( const Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { cout < < v [ i ] < < ' ' ; }

46

cout < < endl ; } // int main () { Vector v ; leer ( v ); int e = b u s c a r _ p r o p ie d a d ( v ); cout < < " El elemento " < < e < < " es mayor que el m nimo de la lista " < < endl ; cout < < " Lista : " ; escribir ( v ); }

9. Disee un programa que lea por teclado las temperaturas medias de los doce meses del ao y calcule la n n temperatura media anual. El programa deber utilizar un tipo enumerado para representar los meses del a ao y considerar que las temperaturas de los meses se pueden introducir en cualquier orden. Como salida n deber imprimir la temperatura de cada mes y la temperatura media anual. Por ejemplo, para la siguiente a entrada:
Mes: Marzo Temperatura: 15.8 Mes: Enero Temperatura: 12.5 Mes: Febrero Temperatura: 13.5 ...

mostrar la siguiente salida: a


Enero: 12.5 Febrero: 13.5 Marzo: 15.8 ... Temperatura media anual: 17.3

Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int N_MESES = 12; typedef array < string , N_MESES > MesStr ; const MesStr MES_STR = {{ " Enero " , " Febrero " , " Marzo " , " Abril " , " Mayo " , " Junio " , " Julio " , " Agosto " , " Septiembre " , " Octubre " , " Noviembre " , " Diciembre " }}; // enum Mes { enero , febrero , marzo , abril , mayo , junio , julio , agosto , septiembre , octubre , noviembre , diciembre }; // typedef array < double , N_MESES > MesTmp ; typedef array < bool , N_MESES > MesOk ; // inline char minuscula ( char c ) { char res = c ; if ( c > = ' A ' & & c < = ' Z ' ) { res = char ( ' a ' + ( c - ' A ' )); } return res ; } // bool iguales ( const string & s1 , const string & s2 ) { bool res = false ; if ( s1 . size () = = s2 . size ()) { int i = 0;

47

while (( i < s1 . size ()) & & ( minuscula ( s1 [ i ] ) = = minuscula ( s2 [ i ] ))) { ++i; } res = ( i > = s1 . size ()); } return res ; } // int buscar ( const MesStr & v , const string & x ) { int i = 0; while (( i < v . size ()) & & ! iguales (x , v [ i ] )) { ++i; } return i ; } // inline void escribir ( Mes m ) { cout < < MES_STR [ int ( m ) ] ; } // inline void leer ( Mes & m ) { string str ; cin > > str ; m = Mes ( buscar ( MES_STR , str )); } // void leer_mes ( Mes & m ) { cout < < " Mes : " ; leer ( m ); while ( m > diciembre ) { cout < < " Error . Introduzca Mes : " ; leer ( m ); } } // void leer_tmp ( double & tmp ) { cout < < " Temperatura : " ; cin > > tmp ; } // void init ( MesOk & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] = false ; } } // void leer_tmp_anyo ( MesTmp & v ) { int nmes = 0; MesOk mok ; init ( mok ); while ( nmes ! = v . size ()) { Mes m ; leer_mes ( m ); if ( mok [ int ( m ) ] ) { cout < < " Error , la temperatura para ese mes ya existe " < < endl ; } else { mok [ int ( m ) ] = true ; + + nmes ; leer_tmp ( v [ int ( m ) ] ); } } } // double media _tmp_any o ( const MesTmp & v ) { double suma = 0; for ( int i = 0; i < v . size (); + + i ) {

48

suma + = v [ i ] ; } return suma / double ( v . size ()); } // void escr_tmp_anyo ( const MesTmp & v ) { for ( int i = 0; i < v . size (); + + i ) { escribir ( Mes ( i )); cout < < " : " < < v [ i ] < < endl ; } } // int main () { MesTmp vt ; leer_tmp_anyo ( vt ); escr_tmp_anyo ( vt ); double media = me dia_tmp_ anyo ( vt ); cout < < " Temperatura media anual : " < < media < < endl ; }

10. Disee un programa que muestra la frecuencia con que aparecen en una lista dada los distintos valores n que la pudieran formar. Por ejemplo, si los valores de una lista pueden estar comprendidos entre 0 y 9, y la lista est formada por: a
6 4 4 1 9 7 5 6 4 2 3 9 5 6 4

el programa mostrar: a
0 1 2 3 4 5 6 7 8 9 * * * **** ** *** * **

Esto indica que el 0 y el 8 no aparecen ninguna vez, el 1, 2, 3 y 7 aparecen una vez, el 5 y 9 dos veces, etc. Escriba un programa que lea una lista de nmeros comprendidos entre 0 y 9 (la lista acabar cuando u a se lea un nmero negativo, y a priori no se puede determinar cuantos nmeros contiene) e imprima por u u pantalla una grca como la anterior. a Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 10; typedef array < int , MAX > Vector ; // void ini ( Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] = 0; } } // void leer ( Vector & v ) { ini ( v ); cout < < " Introduzca n meros ( hasta negativo ): " ; u int num ; cin > > num ; while ( num > = 0 & & num < int ( v . size ())) { + + v [ num ] ; cin > > num ;

49

} } // void i m p r i m i r _ a s t e r i s c o s ( int n ) { for ( int i = 0; i < n ; + + i ) { cout < < " * " ; } } // void grafica ( const Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { cout < < i < < " " ; i m p r i m i r _ a s t e r i s c o s ( v [ i ] ); cout < < endl ; } } // int main () { Vector v ; leer ( v ); grafica ( v ); }

11. Un histograma es una grca que muestra la frecuencia con que aparecen en una lista dada los distintos a valores que la pudieran formar. Por ejemplo, si los valores de una lista pueden estar comprendidos entre 0 y 9, y la lista est formada por: a
6 4 4 1 9 7 5 6 4 2 3 9 5 6 4

su histograma vertical ser: a


* * * * * * * * * * * * * * * ------------------0 1 2 3 4 5 6 7 8 9

Esto indica que el 0 y el 8 no aparecen ninguna vez, el 1, 2, 3 y 7 aparecen una vez, el 5 y 9 dos veces, etc. Escriba un programa que lea una lista de nmeros comprendidos entre 0 y 9 (la lista acabar cuando u a se lea un nmero negativo, y a priori no se puede determinar cuantos nmeros contiene) e imprima por u u pantalla un histograma vertical como el anterior. Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 10; typedef array < int , MAX > Vector ; // void ini ( Vector & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] = 0; } } // void leer ( Vector & v ) { ini ( v ); cout < < " Introduzca n meros ( hasta negativo ): " ; u int num ; cin > > num ; while ( num > = 0 & & num < int ( v . size ())) { + + v [ num ] ; cin > > num ;

50

} } // int buscar_mayor ( const Vector & v ) { int mayor = v [ 0 ] ; for ( int i = 1; i < v . size (); + + i ) { if ( v [ i ] > mayor ) { mayor = v [ i ] ; } } return mayor ; } // void imprimir _linea ( int n ) { for ( int i = 0; i < n ; + + i ) { cout < < " -" ; } cout < < endl ; } // void i m p r i m i r _ i n d i c e s ( int n ) { for ( int i = 0; i < n ; + + i ) { cout < < i < < " " ; } cout < < endl ; } // void i m p r i m i r _ a s t e r i s c o s ( const Vector & v , int fila ) { for ( int i = 0; i < v . size (); + + i ) { if ( v [ i ] > = fila ) { cout < < " * " ; } else { cout < < " " ; } } cout < < endl ; } // void histograma ( const Vector & v ) { int mayor = buscar_mayor ( v ); for ( int f = mayor ; f > 0; - - f ) { i m p r i m i r _ a s t e r i s c o s (v , f ); } impr imir_lin ea (2 * v . size () - 1); i m p r i m i r _ i n d i ce s ( v . size ()); } // int main () { Vector v ; leer ( v ); histograma ( v ); }

12. Escribe un programa que lea una sucesin de 10 nmeros enteros, encuentre el valor mximo y lo imprima o u a junto con el nmero de veces que aparece, y las posiciones en que este ocurre. El proceso se repite con u el resto de la sucesin hasta que no quede ningn elemento por tratar. Por ejemplo, para la siguiente o u entrada:
7 10 143 10 52 143 72 10 143 7

producir la siguiente salida: a


143 aparece 3 veces, en posiciones 3 6 9 72 aparece 1 vez, en posicion 7 52 aparece 1 vez, en posicion 5 10 aparece 3 veces, en posiciones 2 4 8 7 aparece 2 veces, en posiciones 1 10

51

Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 10; typedef array < int , MAX > Vector ; typedef array < bool , MAX > Valido ; // void ini ( Valido & x ) { for ( int i = 0; i < x . size (); + + i ) { x [ i ] = true ; } } // void anular ( const Vector & v , Valido & x , int num ) { for ( int i = 0; i < v . size (); + + i ) { if ( v [ i ] = = num ) { x [ i ] = false ; } } } // int b u s c a r _ p r i m e r _ v a l i d o ( const Valido & x ) { int i = 0; while (( i < x . size ()) & & ( ! x [ i ] )) { ++i; } return i ; } // int b u s c a r _ p o s _ m a y o r ( const Vector & v , const Valido & x ) { int pos_mayor = b u s c a r _ p r i m e r _ v a l i d o ( x ); if ( pos_mayor < v . size ()) { for ( int i = pos_mayor + 1; i < v . size (); + + i ) { if (( x [ i ] ) & & ( v [ i ] > v [ pos_mayor ] )) { pos_mayor = i ; } } } return pos_mayor ; } // int frecuencia ( const Vector & v , int num ) { int cnt = 0; for ( int i = 0; i < v . size (); + + i ) { if ( v [ i ] = = num ) { + + cnt ; } } return cnt ; } // void e s c r i b i r _ p o s i c i o n e s ( const Vector & v , int num ) { for ( int i = 0; i < v . size (); + + i ) { if ( v [ i ] = = num ) { cout < < ( i + 1) < < ' ' ; } } } // void procesar _mayor ( const Vector & v , Valido & x , int num ) { int n_veces = frecuencia (v , num ); cout < < num < < " aparece " < < n_veces ; if ( n_veces = = 1) { cout < < " vez , en posici n " ; o

52

} else { cout < < " veces , en posiciones " ; } e s c r i b i r _ p o s i c i o n e s (v , num ); anular (v , x , num ); cout < < endl ; } // void estadisticas ( const Vector & v ) { Valido x ; ini ( x ); int i = b u s c a r _ p o s _ ma y o r (v , x ); while ( i < v . size ()) { proc esar_may or (v , x , v [ i ] ); i = b u s c a r _ p o s _ m a y o r (v , x ); } } // void leer ( Vector & v ) { cout < < " Introduzca " < < v . size () < < " n meros : " ; u for ( int i = 0; i < v . size (); + + i ) { cin > > v [ i ] ; } } // int main () { Vector v ; leer ( v ); estadisticas ( v ); }

13. Disee un programa que lea de teclado una sucesin indenida de nmeros enteros distintos acabada n o u en 0, y calcule la media de los M mayores nmeros de la sucesin, siendo M una constante. El nmero u o u de elementos de la sucesin puede ser menor que M , en cuyo caso calcular la media de los nmeros o a u introducidos. Solucin o

# include < iostream > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 10; // typedef array < int , MAX > Datos ; struct Vector { int nelms ; Datos elm ; }; // void ini ( Vector & v ) { v . nelms = 0; } // void anyadir ( Vector & v , int x ) { if ( v . nelms < v . elm . size ()) { v . elm [ v . nelms ] = x ; + + v . nelms ; } } // int b u s c a r _ p o s _ m e n o r ( const Vector & v ) { int pos_menor = 0; for ( int i = pos_menor + 1; i < v . nelms ; + + i ) {

53

if ( v . elm [ i ] < v . elm [ pos_menor ] ) { pos_menor = i ; } } return pos_menor ; } // void insertar ( Vector & v , int x ) { if ( v . nelms < v . elm . size ()) { anyadir (v , x ); } else { int i = b u s c a r _ p o s _ m en o r ( v ); v . elm [ i ] = x ; } } // double media ( const Vector & v ) { int suma = 0; for ( int i = 0; i < v . nelms ; + + i ) { suma + = v . elm [ i ] ; } return double ( suma ) / double ( v . nelms ); } // void leer ( Vector & v ) { // ini ( v ); // cout < < " Introduzca una secuencia de n meros terminada en cero (0): " ; u int num ; cin > > num ; while ( num ! = 0) { insertar (v , num ); cin > > num ; } } // int main () { Vector v ; leer ( v ); cout < < " Media de los " < < v . nelms < < " mayores : " < < media ( v ) < < endl ; }

14. Escribe un procedimiento que pueda insertar el valor de una variable x en un vector v ordenado de forma creciente, de forma que dicho vector contine estando ordenado. El vector tendr un nmero de elementos u a u vlidos que podr ser menor que el nmero total de elementos del array. Si en el momento de la insercin a a u o el nmero de elementos vlidos coincide con el nmero total de elementos del array, el elemento de mayor u a u valor desaparecer. Adems, disee el programa y los subprogramas necesarios para probar adecuadamente a a n su funcionamiento.
const int MAX = 20; typedef array<char, MAX> Datos; struct Vector { int nelms; Datos elm; }; void insertar (Vector& v, int x);

Solucin o

# include < iostream > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 20;

54

// typedef array < int , MAX > Datos ; struct Vector { int nelms ; Datos elm ; }; // // INICIALIZAR VECTOR // void ini ( Vector & v ) { v . nelms = 0; } // // INSERTAR // int b us ca r_ p os ic io n ( const Vector & v , int x ) { int i = 0; while (( i < v . nelms ) & & ( x > v . elm [ i ] )) { ++i; } return i ; } // void abrir_hueco ( Vector & v , int pos ) { if ( v . nelms < v . elm . size ()) { + + v . nelms ; } for ( int i = v . nelms - 1; i > pos ; - - i ) { v . elm [ i ] = v . elm [ i - 1 ] ; } } // void insertar ( Vector & v , int x ) { int pos = b us ca r_ p os ic io n (v , x ); if ( pos < v . elm . size ()) { abrir_hueco (v , pos ); v . elm [ pos ] = x ; } } // // LEER ESCRIBIR // void leer ( Vector & v ) { // ini ( v ); // cout < < " Introduzca el n mero de elementos ( menor o igual que " u < < v . elm . size () < < " ) del vector : " ; int n_elms ; cin > > n_elms ; if ( n_elms > v . elm . size ()) { n_elms = v . elm . size (); } cout < < " Introduzca " < < n_elms < < " numeros : " ; for ( int i = 0; i < n_elms ; + + i ) { int x ; cin > > x ; insertar (v , x ); } } // void escribir ( const Vector & v ) { cout < < " [ " ; for ( int i = 0; i < v . nelms ; + + i ) { cout < < v . elm [ i ] < < " " ; } cout < < " ] " < < endl ; }

55

// bool esta_ordenado ( const Vector & v ) { bool ok ; if ( v . nelms = = 0) { ok = true ; } else { int i = 0; while (( i < v . nelms - 1) & & ( v . elm [ i ] < = v . elm [ i + 1 ] )) { ++i; } ok = ( i > = v . nelms - 1); } return ok ; } // int main () { Vector v ; leer ( v ); if ( esta_ordenado ( v )) { cout < < " Vector ordenado : " < < endl ; escribir ( v ); } else { cout < < " Error , el vector no se ha ordenado correctamente " < < endl ; escribir ( v ); } }

15. Disea un procedimiento que tome como parmetros de entrada dos vectores ordenados (posiblemente n a incompletos) y devuelva en un tercer parmetro de salida el resultado de realizar la mezcla ordenada de a ambos, de tal forma que este vector resultado (posiblemente incompleto) tambin est ordenado. Adems, e e a disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. n
void mezclar (const Vector& v1, const Vector& v2, Vector& v3);

Solucin o

# include < iostream > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAX = 20; // typedef array < int , MAX > Datos ; struct Vector { int nelms ; Datos elm ; }; // // MEZCLA ORDENADA // // v1 y v2 e s t a n o r d e n a d o s void mezcla ( const Vector & v1 , const Vector & v2 , Vector & v3 ) { v3 . nelms = v1 . nelms + v2 . nelms ; if ( v3 . nelms > v3 . elm . size ()) { v3 . nelms = v3 . elm . size (); } int i1 = 0; int i2 = 0; for ( int i3 = 0; i3 < v3 . nelms ; + + i3 ) { if (( i2 > = v2 . nelms ) | | ( i1 < v1 . nelms & & v1 . elm [ i1 ] < = v2 . elm [ i2 ] )) { v3 . elm [ i3 ] = v1 . elm [ i1 ] ; + + i1 ; } else { v3 . elm [ i3 ] = v2 . elm [ i2 ] ; + + i2 ; } }

56

} // bool esta_ordenado ( const Vector & v ) { bool ok ; if ( v . nelms = = 0) { ok = true ; } else { int i = 0; while (( i < v . nelms - 1) & & ( v . elm [ i ] < = v . elm [ i + 1 ] )) { ++i; } ok = ( i > = v . nelms - 1); } return ok ; } // // LEER ESCRIBIR // void leer ( Vector & v ) { cout < < " Introduzca el n mero de elementos ( menor o igual que " u < < v . elm . size () < < " ) del vector : " ; cin > > v . nelms ; if ( v . nelms > v . elm . size ()) { v . nelms = v . elm . size (); } cout < < " Introduzca " < < v . nelms < < " n meros : " ; u for ( int i = 0; i < v . nelms ; + + i ) { cin > > v . elm [ i ] ; } } // void escribir ( const Vector & v ) { cout < < " [ " ; for ( int i = 0; i < v . nelms ; + + i ) { cout < < " ' " < < v . elm [ i ] < < " ' " ; } cout < < " ] " < < endl ; } // int main () { Vector v1 , v2 , v3 ; cout < < " Introduzca ORDENADOS los elementos de los vectores " < < endl ; leer ( v1 ); leer ( v2 ); if ( esta_ordenado ( v1 ) & & esta_ordenado ( v2 )) { mezcla ( v1 , v2 , v3 ); cout < < " Mezcla ordenada : " ; escribir ( v3 ); } else { cout < < " Error , los vectores no estan ordenados " < < endl ; } }

16. Un determinado juego de cartas consiste en lo siguiente: consideremos un mazo de N cartas, siendo N un nmero triangular, esto es, N = 1 + 2 + 3 + ... + k para algn k N. Se reparte la totalidad de las N u u cartas en un nmero arbitrario de montones, cada uno de ellos con una cantidad arbitraria de cartas. u El lote de montones se puede reorganizar as se toma una carta de cada montn (con lo que desaparecern : o a los montones con una sla carta), y con todas ellas se forma uno nuevo, que se agrega al nal de la lista o de montones. Por ejemplo, la operacin descrita transforma los montones de 1, 8, 1 y 5 cartas, en otros o montones de 7, 4 y 4 respectivamente: [ 1 8 1 5 ][ 0 7 0 4 4 ][ 7 4 4 ] El desarrollo del juego consiste en llevar a cabo la reorganizacin descrita anteriormente cuantas veces sea o necesario hasta que haya un montn de 1 carta, otro de 2 cartas..., otro de k 1 cartas y, nalmente, otro o de k cartas. Por ejemplo, partiendo de la situacin [ 5 7 3 ], las reorganizaciones sucesivas evolucionan o como sigue:

57

[ [ [ [ [ [ [

5 4 3 2 1 2 1

7 6 5 4 3 1 2

3 2 1 1 2 3 3

] 3 2 3 4 4 4

] 4 5 5 5 5

] ] ] ] ]

Realice un programa que lea de teclado el nmero de montones inicial y el nmero inicial de cartas de u u cada montn. El nmero montones mximo k no podr ser mayor de 15. El programa comprobar que los o u a a a valores iniciales de los montones se corresponden con un nmero triangular (un nmero N es triangular u u a si existe un k N tal que N = (1 + k)k/2, es decir, si k = 1+ 21+8N N), y deber mostrar por pantalla la evolucin de los montones hasta nalizar el juego. o Solucin o

# include < iostream > # include < cassert > # include < cmath > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const double E R RO R_ PR E CI SI ON = 1 e - 10; const int MAX = 15; // typedef array < int , MAX > Datos ; struct Vector { int nelms ; Datos elm ; }; // // VECTOR // void ini ( Vector & v ) { v . nelms = 0; } // void anyadir ( Vector & v , int x ) { if ( v . nelms < v . elm . size ()) { v . elm [ v . nelms ] = x ; + + v . nelms ; } } // // REORGANIZAR // void decrementar ( Vector & v ) { for ( int i = 0; i < v . nelms ; + + i ) { v . elm [ i ] = v . elm [ i ] - 1; } } // void compactar ( Vector & v ) { int j = 0; for ( int i = 0; i < v . nelms ; + + i ) { if ( v . elm [ i ] > 0) { v . elm [ j ] = v . elm [ i ] ; ++j; } } v . nelms = j ; } // void reorganizar ( Vector & v ) { int nuevo_monton = v . nelms ; decrementar ( v ); compactar ( v );

58

anyadir (v , nuevo_monton ); } // // OTRA POSIBILIDAD // // v o i d d e c r e m e n t a r c o m p a c t a r ( Vector& v ) // { // i n t j = 0 ; // f o r ( i n t i = 0 ; i < v . nelms ; ++i ) { // i f ( v . elm [ i ] > 1) { // v . elm [ j ] = v . elm [ i ] 1 ; // ++j ; // } // } // v . nelms = j ; // } // // // v o i d r e o r g a n i z a r ( Vector& v ) // { // i n t nuevo monton = v . nelms ; // d e c r e m e n t a r c o m p a c t a r ( v ) ; // a n y a d i r ( v , nuevo monton ) ; // } // int sumar ( const Vector & v ) { int suma = 0; for ( int i = 0; i < v . nelms ; + + i ) { suma + = v . elm [ i ] ; } return suma ; } // bool es_triangular ( const Vector & v ) { int suma = sumar ( v ); double k = ( - 1 + sqrt (1 + 8 * suma )) / 2.0; return ( k - double ( int ( k )) < E RR O R_ PR EC I SI ON ); } // bool es_ascendente ( const Vector & v ) { int i = 0; while (( i < v . nelms ) & & ( v . elm [ i ] = = ( i + 1))) { ++i; } return ( i > = v . nelms ); } // // LEER ESCRIBIR // void leer ( Vector & v ) { // ini ( v ); // cout < < " Introduzca el n mero de montones ( menor o igual que " u < < v . elm . size () < < " ): " ; int n_elms ; cin > > n_elms ; if ( n_elms > v . elm . size ()) { n_elms = v . elm . size (); } for ( int i = 0; i < n_elms ; + + i ) { cout < < " Introduzca el n mero de cartas del monton " < < ( i + 1) < < " : " ; u int x ; cin > > x ; anyadir (v , x ); } } // void escribir ( const Vector & v ) { cout < < " [ " ;

59

for ( int i = 0; i < v . nelms ; + + i ) { cout < < v . elm [ i ] < < " " ; } cout < < " ] " < < endl ; } // // JUEGO // void juego () { Vector v ; leer ( v ); if ( ! es_triangular ( v )) { cout < < " Error , el n mero no es triangular " < < endl ; u } else { while ( ! es_ascendente ( v )) { escribir ( v ); reorganizar ( v ); } escribir ( v ); } } // int main () { juego (); }

17. Disee un programa para contar y mostrar el nmero total de ocurrencias (frecuencia) de cada letra n u minscula en una secuencia de caracteres le por teclado hasta el carcter punto (.). u da a Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int N_LETRAS = ( ' z ' - ' a ' ) + 1; typedef array < int , N_LETRAS > Cnt ; // // c o n v i e r t e de l e t r a a n d i c e int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = ( letra - ' a ' ); } return res ; } // // c o n v i e r t e de n d i c e a l e t r a inline char idx_letra ( int idx ) { return char ( ' a ' + idx ); } // void ini ( Cnt & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] = 0; } } // void e s t a d i s t i c a s _ t e x t o ( Cnt & cnt ) { ini ( cnt ); cout < < " Introduzca un texto hasta punto ( ' . ' ): " ; char c ; cin > > ws ; cin . get ( c ); while ( c ! = ' . ' ) {

60

int idx = letra_idx ( c ); if ( idx < cnt . size ()) { + + cnt [ idx ] ; } cin . get ( c ); } } // void m o s t r a r _ e s t a d i s t i c a s ( const Cnt & cnt ) { for ( int i = 0; i < cnt . size (); + + i ) { if ( i %5 = = 0) { cout < < endl ; } else { cout < < " \ t " ; } cout < < idx_letra ( i ) < < " : " < < cnt [ i ] ; } cout < < endl ; } // int main () { Cnt cnt ; e s t a d i s t i c a s _ t e x t o ( cnt ); m o s t r a r _ e s t a d i s t i c a s ( cnt ); }

18. Disee un programa que tomando como entrada un texto, realice el clculo de la frecuencia con que aparece n a cada palabra de dos letras minsculas en el texto, imprimiendo en pantalla el resultado. u Solucin o

# include < iostream > # include < iomanip > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const string FIN_SEC = " fin " ; const int N_LETRAS = int ( ' z ' ) - int ( ' a ' ) + 1; typedef array < int , N_LETRAS > FrecLetra ; typedef array < FrecLetra , N_LETRAS > Frec2Letras ; // // c o n v i e r t e de l e t r a a n d i c e inline int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = ( letra - ' a ' ); } return res ; } // // c o n v i e r t e de n d i c e a l e t r a inline char idx_letra ( int idx ) { return char ( ' a ' + idx ); } // void init ( Frec2Letras & fr ) { for ( int f = 0; f < fr . size (); + + f ) { for ( int c = 0; c < fr [ f ] . size (); + + c ) { fr [ f ] [ c ] = 0; } } } // void inc_frec uencia ( const string & s , Frec2Letras & fr ) { if ( s . size () = = 2) {

61

int idx_0 = letra_idx ( s [ 0 ] ); int idx_1 = letra_idx ( s [ 1 ] ); if (( idx_0 < fr . size ()) & & ( idx_1 < fr [ idx_0 ] . size ())) { + + fr [ idx_0 ] [ idx_1 ] ; } } } // void f r e c u e n c i a _ p a l a b r a s ( Frec2Letras & fr ) { string palabra ; init ( fr ); cin > > palabra ; while ( palabra ! = FIN_SEC ){ inc_ frecuenc ia ( palabra , fr ); cin > > palabra ; } } // // v o i d e s c r i b i r f r e c u e n c i a s ( c o n s t F r e c 2 L e t r a s& f r ) // { // c o u t << ; // f o r ( i n t c = 0 ; c < f r [ 0 ] . s i z e ( ) ; ++c ) { // c o u t << s e t w ( 2 ) << i d x l e t r a ( c ) << ; // } // c o u t << e n d l ; // f o r ( i n t f = 0 ; f < f r . s i z e ( ) ; ++f ) { // c o u t << << i d x l e t r a ( f ) << : ; // f o r ( i n t c = 0 ; c < f r [ f ] . s i z e ( ) ; ++c ) { // c o u t << s e t w ( 2 ) << f r [ f ] [ c ] << ; // } // c o u t << e n d l ; // } // } // void e s c r i b i r _ f r e c u e n c i a s ( const Frec2Letras & fr ) { cout < < " Frecuencias de las palabras de 2 letras : " < < endl ; for ( int f = 0; f < fr . size (); + + f ) { for ( int c = 0; c < fr [ f ] . size (); + + c ) { if ( fr [ f ] [ c ] > 0) { cout < < idx_letra ( f ) < < idx_letra ( c ) < < " : " < < fr [ f ] [ c ] < < endl ; } } } } // int main () { Frec2Letras fr ; cout < < " Introduzca el texto en min sculas hasta ( fin ) " < < endl ; u f r e c u e n c i a _ p a l a b r a s ( fr ); e s c r i b i r _ f r e c u e n c i a s ( fr ); }

19. Escribe una funcin que encuentre el elemento mayor de un array de dos dimensiones dado. Adems, o a disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. n Solucin o

# include < iostream > # include < iomanip > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAXCOL = 7; const int MAXFIL = 5; typedef array < int , MAXCOL > Fila ; typedef array < Fila , MAXFIL > Matriz ; // int buscar_mayor ( const Matriz & m ) {

62

int mayor = m [ 0 ] [ 0 ] ; for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { if ( m [ f ] [ c ] > mayor ) { mayor = m [ f ] [ c ] ; } } } return mayor ; } // void escribir ( const Matriz & m ) { for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { cout < < setw (3) < < m [ f ] [ c ] < < ' ' ; } cout < < endl ; } } // void leer ( Matriz & m ) { cout < < " Introduzca " < < m . size () < < " filas de " < < m [ 0 ] . size () < < " n meros " < < endl ; u for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { cin > > m [ f ] [ c ] ; } } } // int main () { Matriz m ; leer ( m ); int mayor = buscar_mayor ( m ); cout < < " El n mero " < < mayor < < " es el mayor elemento de la matriz " < < endl ; u escribir ( m ); }

20. Un array bidimensional a de M las y M columnas es simtrico si sus elementos satisfacen la condicin e o a[i][j] = a[j][i] para todo i, j. Escribe una funcin que determine si un array de ese tipo es simtrico. o e Adems, disee el programa y los subprogramas necesarios para probar adecuadamente su funcionamiento. a n Solucin o

# include < iostream > # include < iomanip > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAXCOL = 5; const int MAXFIL = MAXCOL ; typedef array < int , MAXCOL > Fila ; typedef array < Fila , MAXFIL > Matriz ; // bool es_simetrica ( const Matriz & m ) { bool ok = true ; for ( int f = 0; ok & & f < m . size (); + + f ) { for ( int c = 0; ok & & c < f ; + + c ) { ok = m [ f ] [ c ] = = m [ c ] [ f ] ; } } return ok ; } // void escribir ( const Matriz & m ) { for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) {

63

cout < < setw (3) < < m [ f ] [ c ] < < ' ' ; } cout < < endl ; } } // void leer ( Matriz & m ) { cout < < " Introduzca " < < m . size () < < " filas de " < < m [ 0 ] . size () < < " n meros " < < endl ; u for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { cin > > m [ f ] [ c ] ; } } } // int main () { Matriz m ; leer ( m ); cout < < " La matriz " < < endl ; escribir ( m ); if ( es_simetrica ( m )) { cout < < " SI " ; } else { cout < < " NO " ; } cout < < " es sim trica " < < endl ; e }

21. Un tablero n-goro es un tablero con N (N + 1) casillas de la forma:


0 0 1 N 1 1 N 1 N

Una propiedad interesante es que se pueden visitar todas sus casillas haciendo el siguiente recorrido por diagonales. Empezamos por la casilla (0, 0) y recorremos la diagonal principal hacia la derecha y hacia abajo hasta llegar a la casilla (N 1, N 1). La siguiente casilla a visitar ser la (N, N ) que no existe a porque nos saldr amos del tablero por abajo. En estos casos siempre se pasa a la casilla equivalente en la primera la, es decir, la (0, N ). Ahora seguimos movindonos hacia la derecha y hacia abajo. Pero la e siguiente casilla ser la (1, N + 1) que no existe porque nos hemos salido por la derecha. En estos casos se a contina por la casilla equivalente de la primera columna, es decir, la (1, 0). De nuevo nos movemos hacia u la derecha y hacia abajo, hasta alcanzar la casilla (N 1, N 2). La siguiente casilla ser la (N, N 1), a pero como nos saldr amos por abajo pasamos a la casilla equivalente de la primera la (0, N 1). Si se contina con este proceso se termina visitando todas las casillas del tablero. u Disee un procedimiento que dada una constante N devuelve como parmetro un tablero N-goro con sus n a casillas rellenas con el nmero correspondiente al momento en que se visitan, as como el programa y los u subprogramas necesarios para probar adecuadamente su funcionamiento. Por ejemplo, si N es 4, el tablero a devolver ser a:
0 1 2 3 0 1 6 11 16 1 17 2 7 12 2 13 18 3 8 3 9 14 19 4 4 5 10 15 20

# include < iostream > # include < iomanip > # include < tr1/array > using namespace std ; using namespace std :: tr1 ;

Solucin o

64

// const int MAXFIL = 4; const int MAXCOL = MAXFIL + 1; typedef array < int , MAXCOL > Fila ; typedef array < Fila , MAXFIL > Matriz ; // void escribir ( const Matriz & m ) { for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { cout < < setw (3) < < m [ f ] [ c ] < < ' ' ; } cout < < endl ; } } // void crear_ngoro ( Matriz & m ) { int f = 0; int c = 0; for ( int i = 1; i < = m . size () * m [ 0 ] . size (); + + i ) { m[f][c] = i; f = ( f + 1) %m . size (); c = ( c + 1) %m [ 0 ] . size (); } } // int main () { Matriz m ; crear_ngoro ( m ); escribir ( m ); }

22. Se dispone de una determinada zona cuadrangular dividida en celdas (8 8), por algunas de las cuales se puede circular y por otras no. Un ejemplo se muestra en la gura siguiente, en donde las celdas no visitables se marcan con una X y las visitables se dejan en blanco.
X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X X

X X

Disee un procedimiento recorrido, el programa y los subprogramas necesarios para probar adecuadan mente su funcionamiento. El subprograma recorrido toma como parmetros una zona de ese tipo, un a valor n y una determinada coordenada (la y columna de una celda visitable) y determine la celda en la que nos encontraremos tras realizar n movimientos por la zona partiendo de la coordenada dada, teniendo en cuenta: Un movimiento consiste en un desplazamiento horizontal o vertical desde una celda a una celda vecina visitable, sin considerar la celda de la que se proviene. Cada celda tiene 4 celdas vecinas posibles, teniendo en cuenta slo vecindad horizontal y vertical o (una celda que est en un borde o bordes de la cuadr a cula tiene como vecinas a las celdas del borde o bordes opuestos). Slo existir un camino posible partiendo de la celda dada. o a Si no se puede seguir avanzando sin haber alcanzado los n pasos pedidos, se dar como resultado la a celda ultima en la que nos encontramos. As por ejemplo, considerando la zona de la gura anterior, partiendo de la celda (1, 0) y tras realizar 14 movimientos, nos encontraremos en la celda (5, 2). Si por el contrario pretendemos realizar 19 o ms a movimientos, nos encontraremos en la celda (0, 4).

65

Solucin o

# include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int MAXFIL = 8; const int MAXCOL = MAXFIL ; typedef array < char , MAXCOL > Fila ; typedef array < Fila , MAXFIL > Matriz ; struct Coord { int f ; int c ; }; // const Matriz ZONA = {{ {{ ' X ' , ' X ' , ' ' , ' ' , ' ' , ' X ' , ' X ' , ' X ' }} , {{ ' ' , ' ' , ' X ' , ' X ' , ' X ' , ' X ' , ' X ' , ' X ' }} , {{ ' X ' , ' ' , ' ' , ' ' , ' ' , ' X ' , ' X ' , ' X ' }} , {{ ' X ' , ' X ' , ' X ' , ' X ' , ' ' , ' X ' , ' X ' , ' X ' }} , {{ ' X ' , ' X ' , ' X ' , ' X ' , ' ' , ' X ' , ' X ' , ' X ' }} , {{ ' ' , ' ' , ' ' , ' X ' , ' ' , ' ' , ' ' , ' ' }} , {{ ' X ' , ' X ' , ' ' , ' X ' , ' X ' , ' X ' , ' X ' , ' X ' }} , {{ ' X ' , ' X ' , ' ' , ' X ' , ' X ' , ' X ' , ' X ' , ' X ' }} }}; // inline bool iguales ( const Coord & c1 , const Coord & c2 ) { return ( c1 . f = = c2 . f ) & & ( c1 . c = = c2 . c ); } // inline bool distintos ( const Coord & c1 , const Coord & c2 ) { return ! iguales ( c1 , c2 ); } // inline int abs ( int x ) { int res ; if ( x < 0) { res = int ( - x ); } else { res = int ( x ); } return res ; } // void ini ( Coord & c ) { c . f = MAXFIL ; c . c = MAXCOL ; } // void asg ( Coord & c , int ff , int cc ) { c . f = ff ; c . c = cc ; } // void leer ( Coord & c ) { int ff , cc ; cin > > ff > > cc ; asg (c , ff , cc ); } // void escribir ( const Coord & c ) { cout < < " ( " < < c . f < < " , " < < c . c < < " ) " ; } // void paso ( const Coord & c , int ff , int cc , Coord & n ) { n . f = ( c . f + MAXFIL + ff ) % MAXFIL ;

66

n . c = ( c . c + MAXCOL + cc ) % MAXCOL ; } // inline bool es_visitable ( const Matriz & m , const Coord & c ) { return ( m [ c . f ] [ c . c ] = = ' ' ); } // void mover ( const Matriz & m , Coord & ant , Coord & act , bool & ok ) { ok = false ; for ( int ff = - 1; ! ok & & ff < = + 1; + + ff ) { for ( int cc = - 1; ! ok & & cc < = + 1; + + cc ) { if ( abs ( ff ) + abs ( cc ) = = 1) { Coord n ; paso ( act , ff , cc , n ); if ( distintos (n , ant ) & & es_visitable (m , n )) { ok = true ; ant = act ; act = n ; } } } } } // void recorrido ( const Matriz & m , int n , Coord & c ) { Coord ant ; ini ( ant ); bool ok = true ; for ( int i = 0; ok & & i < n ; + + i ) { mover (m , ant , c , ok ); } } // void escribir ( const Matriz & m ) { for ( int f = 0; f < m . size (); + + f ) { for ( int c = 0; c < m [ f ] . size (); + + c ) { cout < < m [ f ] [ c ] ; } cout < < endl ; } } // void check () { escribir ( ZONA ); cout < < " Introduzca Coordenada de inicio : " ; Coord c ; leer ( c ); cout < < " Introduzca n mero de pasos : " ; u int n ; cin > > n ; recorrido ( ZONA , n , c ); cout < < " Posici n final : " ; o escribir ( c ); cout < < endl ; } // int main () { check (); }

23. Disee un programa que lea dos matrices de nmeros reales de tamao mximo 10 10 (aunque podrn n u n a a tener un tamao menor) y muestre el resultado de multiplicar ambas matrices. Las matrices se almacenarn n a en arrays incompletos de dos dimensiones.
# include < iostream >

Solucin o

67

# include < tr1/array > # include < cassert > using namespace std ; using namespace std :: tr1 ; // C o n s t a n t e s const int MAX = 10; // Tipos typedef array < double , MAX > Fila ; typedef array < Fila , MAX > Tabla ; struct Matriz { int n_fil ; int n_col ; Tabla datos ; }; // void leer_matriz ( Matriz & m ) { cout < < " Dimensiones ?: " ; cin > > m . n_fil > > m . n_col ; cout < < " Escribe valores fila a fila : " < < endl ; for ( int f = 0; f < m . n_fil ; + + f ) { for ( int c = 0; c < m . n_col ; + + c ) { cin > > m . datos [ f ] [ c ] ; } } } // void e sc ri bi r _m at ri z ( const Matriz & m ) { for ( int f = 0; f < m . n_fil ; + + f ) { for ( int c = 0; c < m . n_col ; + + c ) { cout < < m . datos [ f ] [ c ] < < " " ; } cout < < endl ; } } // double s u m a _ f i l a _ p o r _ c o l ( const Matriz & x , const Matriz & y , int f , int c ) { assert ( x . n_col = = y . n_fil ); // PRECOND double suma = 0.0; for ( int k = 0; k < x . n_col ; + + k ) { suma + = x . datos [ f ] [ k ] * y . datos [ k ] [ c ] ; } return suma ; } // void mult_matriz ( Matriz & m , const Matriz & a , const Matriz & b ) { assert ( a . n_col = = b . n_fil ); // PRECOND m . n_fil = a . n_fil ; m . n_col = b . n_col ; for ( int f = 0; f < m . n_fil ; + + f ) { for ( int c = 0; c < m . n_col ; + + c ) { m . datos [ f ] [ c ] = s u m a _ f i l a _ p o r _ c o l (a , b , f , c ); } } } // int main () { Matriz a ,b , c ; leer_matriz ( a ); leer_matriz ( b ); if ( a . n_col ! = b . n_fil ) { cout < < " No se puede multiplicar . " < < endl ; } else { mult_matriz (c , a , b ); e sc ri bi r _m at ri z ( c ); } }

24. Disee un programa que tomando como entrada un texto, realice el listado por pantalla de todas las n

68

palabras del texto que comiencen por ciertas iniciales. Dichas iniciales sern las letras que componen la a primera palabra del texto.
El texto contiene un nmero indenido de palabras en letras minsculas y termina con la palabra fin. u u Cada palabra tiene un nmero indenido pero limitado de caracteres (todos alfabticos minsculas). u e u El carcter separador de palabras es el espacio en blanco. a

Solucin o

# include < iostream > # include < string > using namespace std ; // const string FIN_SEC = " fin " ; // int buscar ( const string & s , char x ) { int i = 0; while (( i < s . size ()) & & ( x ! = s [ i ] )) { ++i; } return i ; } // void i m p r i m i r _ p a l a b r a s ( const string & patron ) { string palabra ; cin > > palabra ; while ( palabra ! = FIN_SEC ){ if ( buscar ( patron , palabra [ 0 ] ) < patron . size ()) { cout < < palabra < < " " ; } cin > > palabra ; } cout < < endl ; } // int main () { cout < < " Introduzca el texto en min sculas hasta ( fin ) con " u < < " el patr n de iniciales al principio . " < < endl ; o string patron ; cin > > patron ; if ( patron ! = FIN_SEC ){ i m p r i m i r _ p a l a b r a s ( patron ); } }

25. Disee un programa que lea de teclado un patrn (una cadena de caracteres) y un texto, y d como n o e resultado las palabras del texto que contengan a dicho patrn. En la salida no habr palabras repetidas. o a Nota: En el texto aparecern un nmero mximo de 20 palabras distintas. Por ejemplo, para la entrada: a u a
Patron : re Texto: creo que iremos a la direccion que nos dieron aunque pienso que dicha direccion no es correcta fin

mostrar como salida: a


creo iremos direccion correcta

El texto contiene un nmero indenido de palabras en letras minsculas y termina con la palabra fin. u u Cada palabra tiene un nmero indenido pero limitado de caracteres (todos alfabticos minsculas). u e u El carcter separador de palabras es el espacio en blanco. a

# include < iostream > # include < string > # include < tr1/array >

Solucin o

69

using namespace std ; using namespace std :: tr1 ; // const string FIN_SEC = " fin " ; const int MAX_PAL = 20; typedef array < string , MAX_PAL > Datos ; struct Vector { int nelms ; Datos elm ; }; // int buscar ( const Vector & v , const string & x ) { int i = 0; while (( i < v . nelms ) & & ( x ! = v . elm [ i ] )) { ++i; } return i ; } // void ini ( Vector & v ) { v . nelms = 0; } // void anyadir ( Vector & v , const string & x ) { if (( buscar (v , x ) > = v . nelms ) & & ( v . nelms < v . elm . size ())) { v . elm [ v . nelms ] = x ; + + v . nelms ; } } // int buscar ( const string & cadena , const string & patron ) { int i = 0; while (( i + patron . size () < = cadena . size ()) & & ( patron ! = cadena . substr (i , patron . size ()))) { ++i; } if ( i + patron . size () > cadena . size ()) { i = cadena . size (); } return i ; } // inline bool contiene ( const string & palabra , const string & patron ) { return buscar ( palabra , patron ) < palabra . size (); } // void leer_palabras ( const string & patron , Vector & v ) { cout < < " Introduzca el texto en min sculas hasta ( fin ) " < < endl ; u ini ( v ); string palabra ; cin > > palabra ; while ( palabra ! = FIN_SEC ){ if ( contiene ( palabra , patron )) { anyadir (v , palabra ); } cin > > palabra ; } } // void e s c r i b i r _ p a l a b r a s ( const Vector & v ) { for ( int i = 0; i < v . nelms ; + + i ) { cout < < v . elm [ i ] < < " " ; } cout < < endl ; } // int main ()

70

{ Vector v ; cout < < " Introduzca el patr n en min sculas : " ; o u string patron ; cin > > patron ; if ( patron ! = FIN_SEC ){ leer_palabras ( patron , v ); e s c r i b i r _ p a l a b r a s ( v ); }

26. Una palabra w es un anagrama de la palabra v, si podemos obtener w cambiando el orden de las letras de v. Por ejemplo, vaca lo es de cava. Disea un programa que lea un texto y determine de cuantas palabras n le das es anagrama la primera dentro de dicho texto.
El texto contiene un nmero indenido de palabras en letras minsculas y termina con la palabra fin. u u Cada palabra tiene un nmero indenido pero limitado de caracteres (todos alfabticos minsculas). u e u El carcter separador de palabras es el espacio en blanco. a

Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const string FIN_SEC = " fin " ; const int N_LETRAS = int ( ' z ' ) - int ( ' a ' ) + 1; typedef array < int , N_LETRAS > Frecuencia ; // // c o n v i e r t e de l e t r a a n d i c e int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = int ( letra ) - int ( ' a ' ); } return res ; } // void init ( Frecuencia & f ) { for ( int i = 0; i < f . size (); + + i ) { f [ i ] = 0; } } // void frecuencia ( const string & s , Frecuencia & f ) { init ( f ); for ( int i = 0; i < s . size (); + + i ) { int idx = letra_idx ( s [ i ] ); if ( idx < f . size ()) { + + f [ idx ] ; } } } // bool es_anagrama ( const string & palabra , const string & patron ) { bool res ; if ( palabra . size () ! = patron . size ()) { res = false ; } else { Frecuencia f1 , f2 ; frecuencia ( palabra , f1 ); frecuencia ( patron , f2 ); res = f1 = = f2 ; } return res ;

71

} // int c o n t a r _ a n a g r a m a s ( const string & patron ) { int contador = 0; string palabra ; cin > > palabra ; while ( palabra ! = FIN_SEC ){ if ( es_anagrama ( palabra , patron )) { + + contador ; } cin > > palabra ; } return contador ; } // int main () { string patron ; int contador = 0; cout < < " Introduzca el texto en m nusculas hasta ( fin ) con " < < " el anagrama a comprobar al principio . " < < endl ; cin > > patron ; if ( patron ! = FIN_SEC ){ contador = c o n t a r _ a n a g r a m a s ( patron ); } cout < < " En este texto hay " < < contador < < " anagramas como <" < < patron < < " >. " < < endl ;

27. Sean u y v dos palabras formadas por letras minsculas. Diremos que u est asociada a v por vocales u a fantasmas, si u se puede obtener a partir de v despus de un cambio de orden de las letras que mantenga e inalterado el orden relativo de las consonantes. Por ejemplo, si u contiene perla las palabras parle, aperl, pearl, paerl, prale, estn asociadas a u por vocales fantasmas. En cambio, lepra no lo est, ya que a a no conserva el orden relativo de las consonantes. Disea un programa que lea desde teclado un texto y n calcule el nmero de palabras le u das asociadas por vocales fantasmas a la primera del mismo.
El texto contiene un nmero indenido de palabras en letras minsculas y termina con la palabra fin. u u Cada palabra tiene un nmero indenido pero limitado de caracteres (todos alfabticos minsculas). u e u El carcter separador de palabras es el espacio en blanco. a

Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const string FIN_SEC = " fin " ; const int N_LETRAS = int ( ' z ' ) - int ( ' a ' ) + 1; typedef array < int , N_LETRAS > Frecuencia ; // // c o n v i e r t e de l e t r a a n d i c e inline int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = int ( letra ) - int ( ' a ' ); } return res ; } // inline bool es_minuscula ( char letra ) {

72

return ( letra > = ' a ' & & letra < = ' z ' ); } // inline bool es_vocal ( char letra ) { return (( letra = = ' a ' ) | | ( letra = = ' e ' ) | | ( letra = = ' i ' ) | | ( letra = = ' o ' ) | | ( letra = = ' u ' )); } // inline bool es_consonante ( char letra ) { return es_minuscula ( letra ) & & ! es_vocal ( letra ); } // void init ( Frecuencia & f ) { for ( int i = 0; i < f . size (); + + i ) { f [ i ] = 0; } } // void frecuencia ( const string & s , Frecuencia & f ) { init ( f ); for ( int i = 0; i < s . size (); + + i ) { int idx = letra_idx ( s [ i ] ); if ( idx < f . size ()) { + + f [ idx ] ; } } } // void c o p i a r _ c o n s o n a n t e s ( const string & palabra , string & consonantes ) { consonantes = " " ; for ( int i = 0; i < palabra . size (); + + i ) { if ( es_consonante ( palabra [ i ] )) { consonantes + = palabra [ i ] ; } } } // bool asoc_voc_fant ( const string & palabra , const string & patron ) { bool res ; if ( palabra . size () ! = patron . size ()) { res = false ; } else { Frecuencia f1 , f2 ; frecuencia ( palabra , f1 ); frecuencia ( patron , f2 ); res = f1 = = f2 ; if ( res ) { string c1 , c2 ; c o p i a r _ c o n s o n a n t e s ( palabra , c1 ); c o p i a r _ c o n s o n a n t e s ( patron , c2 ); res = c1 = = c2 ; } } return res ; } // int c o n t a r _ a s o c _ v o c _ f a n t ( const string & patron ) { int contador = 0; string palabra ; cin > > palabra ; while ( palabra ! = FIN_SEC ){ if ( asoc_voc_fant ( palabra , patron )) { + + contador ; } cin > > palabra ;

73

} return contador ; } // int main () { string patron ; int contador = 0; cout < < " Introduzca el texto en min sculas hasta ( fin ) con " u < < " el patr n a comprobar al principio . " < < endl ; o cin > > patron ; if ( patron ! = FIN_SEC ){ contador = c o n t a r _ a s o c _ v o c _ f a n t ( patron ); } cout < < " En este texto hay " < < contador < < " asociaciones por vocales fantasmas con <" < < patron < < " >. " < < endl ;

28. La distancia entre dos letras en un texto es el nmero de letras que aparecen en el texto entre las dos letras u indicadas. Disee un programa que lea una secuencia de caracteres terminada en punto (.) y muestre n por pantalla la mxima distancia entre cada par de letras minsculas repetidas. Aquellas letras que no se a u repitan no aparecern en la salida. Por ejemplo para la secuencia de entrada: a
abeaddglake.

mostrar la siguiente salida: a


Distancia entre a: 4 Distancia entre d: 0 Distancia entre e: 7

Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int N_LETRAS = ( ' z ' - ' a ' ) + 1; struct Estd { int max_dist ; int ult_pos ; }; typedef array < Estd , N_LETRAS > Cnt ; // // c o n v i e r t e de l e t r a a n d i c e int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = ( letra - ' a ' ); } return res ; } // // c o n v i e r t e de n d i c e a l e t r a inline char idx_letra ( int idx ) { return char ( ' a ' + idx ); } // void ini ( Cnt & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] . max_dist = - 1; v [ i ] . ult_pos = 0; } }

74

// void procesar ( Cnt & cnt , int pos , int idx ) { if ( idx < cnt . size ()) { if ( cnt [ idx ] . ult_pos ! = 0) { int dist = int ( pos - cnt [ idx ] . ult_pos ) - 1; if ( dist > cnt [ idx ] . max_dist ) { cnt [ idx ] . max_dist = dist ; } } cnt [ idx ] . ult_pos = pos ; } } // void e s t a d i s t i c a s _ t e x t o ( Cnt & cnt ) { int pos = 0; ini ( cnt ); cout < < " Introduzca un texto hasta punto ( ' . ' ): " ; char c ; cin > > ws ; cin . get ( c ); while ( c ! = ' . ' ) { + + pos ; procesar ( cnt , pos , letra_idx ( c )); cin . get ( c ); } } // void m o s t r a r _ e s t a d i s t i c a s ( const Cnt & cnt ) { for ( int i = 0; i < cnt . size (); + + i ) { if ( cnt [ i ] . max_dist > = 0) { cout < < " Distancia entre " < < idx_letra ( i ) < < " : " < < cnt [ i ] . max_dist < < endl ; } } } // int main () { Cnt cnt ; e s t a d i s t i c a s _ t e x t o ( cnt ); m o s t r a r _ e s t a d i s t i c a s ( cnt ); }

29. La distancia entre dos palabras en un texto es el nmero de palabras que aparecen en el texto entre las dos u palabras indicadas. Disee un programa que lea un texto de longitud INDEFINIDA, en el que sabemos n que hay un mximo de 20 palabras distintas, y muestre por pantalla la MAXIMA distancia entre cada a par de palabras repetidas. Por ejemplo, para el siguiente texto de entrada:
la casa roja de la esquina es la casa de juan fin

mostrar la salida: a
Distancia entre la: 3 Distancia entre casa: 6 Distancia entre de: 5

El texto contiene un nmero indenido de palabras en letras minsculas y termina con la palabra fin. u u Cada palabra tiene un nmero indenido pero limitado de caracteres (todos alfabticos minsculas). u e u El carcter separador de palabras es el espacio en blanco. a

Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; //

75

const string FIN_SEC = " fin " ; struct Distancia { string palabra ; int ult_pos ; int max_dist ; }; const int MAX_PAL = 20; typedef array < Distancia , MAX_PAL > Datos ; struct Vector { int nelms ; Datos elm ; }; // int buscar ( const Vector & v , const string & x ) { int i = 0; while (( i < v . nelms ) & & ( x ! = v . elm [ i ] . palabra )) { ++i; } return i ; } // void ini ( Vector & v ) { v . nelms = 0; } // void anyadir ( Vector & v , const string & x , int pos ) { if ( v . nelms < v . elm . size ()) { v . elm [ v . nelms ] . palabra = x ; v . elm [ v . nelms ] . ult_pos = pos ; v . elm [ v . nelms ] . max_dist = - 1; + + v . nelms ; } } // void procesar ( Vector & v , int pos , const string & palabra ) { int i = buscar (v , palabra ); if ( i < v . nelms ) { int dist = int ( pos - v . elm [ i ] . ult_pos ) - 1; if ( dist > v . elm [ i ] . max_dist ) { v . elm [ i ] . max_dist = dist ; } v . elm [ i ] . ult_pos = pos ; } else { anyadir (v , palabra , pos ); } } // void e s t a d i s t i c a s _ t e x t o ( Vector & v ) { int pos = 0; ini ( v ); cout < < " Introduzca el texto en min sculas hasta ( fin ) " < < endl ; u string palabra ; cin > > palabra ; while ( palabra ! = FIN_SEC ){ + + pos ; procesar (v , pos , palabra ); cin > > palabra ; } } // void m o s t r a r _ e s t a d i s t i c a s ( const Vector & v ) { for ( int i = 0; i < v . nelms ; + + i ) { if ( v . elm [ i ] . max_dist > = 0) { cout < < " Distancia entre " < < v . elm [ i ] . palabra < < " : " < < v . elm [ i ] . max_dist < < endl ; } } }

76

// int main () { Vector v ; e s t a d i s t i c a s _ t e x t o ( v ); m o s t r a r _ e s t a d i s t i c a s ( v ); }

30. Un grupo de M soldados (M es una constante) est rodeado por el enemigo y no hay posibilidad de a victoria sin refuerzos, pero hay slo un caballo para escapar y pedir dichos refuerzos. Los soldados llegan a o un acuerdo para determinar quien va a escapar y pedir ayuda. Forman un c rculo y sacan de un sombrero un nmero n y un nombre de uno de los soldados que forman el c u rculo. Empezando por el soldado cuyo nombre se ha sacado cuentan, en el sentido de las agujas del reloj, n soldados y sacan fuera del c rculo al soldado encontrado en el lugar n-simo. La cuenta empieza de nuevo con el siguiente soldado (el que sigue e al eliminado segn el sentido de las agujas del reloj). El proceso contina de forma que cada vez que la u u cuenta llega a n se saca un soldado del c rculo. Una vez que un soldado se saca del c rculo ya no se vuelve a contar. El soldado que queda al nal es el que coge el caballo y escapa. Disea un subprograma (y programa para prueba) con la siguiente cabecera que realiza el proceso anterior: n
typedef array<string, M> Soldados; int escapa (const Soldados& nom_soldados, int n, const string& pr_sol);

El signicado de cada parmetro es el siguiente: nom_soldados es un array de cadenas de caracteres con a los nombres de los M soldados que forman el c rculo y en el orden en que se encuentran en el mismo. Al soldado que ocupa la ultima posicin del array le sigue en el c o rculo (segn las agujas del reloj) el que se u encuentra en la primera posicin del array. El nmero n para realizar la cuenta segn el proceso indicado. o u u pr_sol es nombre del soldado que se escoge inicialmente del sombrero. La funcin devuelve el o ndice en el array donde se encuentra el soldado elegido para escapar. Solucin o

# include < cassert > # include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int N_SOLDADOS = 5; typedef array < string , N_SOLDADOS > Soldados ; typedef array < bool , N_SOLDADOS > Validos ; // void ini_validos ( Validos & v ) { for ( int i = 0; i < v . size (); + + i ) { v [ i ] = true ; } } // int cnt_validos ( const Validos & v ) { int cnt = 0; for ( int i = 0; i < v . size (); + + i ) { if ( v [ i ] ) { + + cnt ; } } return cnt ; } // void siguiente ( const Validos & v , int & i ) { assert ( cnt_validos ( v ) > 0); do { i = ( i + 1) %v . size (); } while ( ! v [ i ] ); } // void siguiente_n ( const Validos & v , int n , int & i )

77

{ for ( int k = 0; k < n ; + + k ) { siguiente (v , i ); } } // void liberar ( Validos & v , int i ) { assert ( v [ i ] ); v [ i ] = false ; } // void l ib er ar _ so ld ad o ( Validos & v , int n , int & i ) { siguiente_n (v , n , i ); liberar (v , i ); } // int buscar ( const Soldados & s , const string & x ) { int i = 0; while (( i < s . size ()) & & ( x ! = s [ i ] )) { ++i; } return i ; } // int escapa ( const Soldados & nombres , int n , const string & pr_sol ) { Validos v ; ini_validos ( v ); int i = buscar ( nombres , pr_sol ); if ( i < nombres . size ()) { for ( int k = 0; k < nombres . size () - 1; + + k ) { l ib er ar _ so ld ad o (v , n , i ); cout < < " Sale : " < < nombres [ i ] < < endl ; } siguiente (v , i ); } return i ; } // void leer_nombres ( Soldados & nombres ) { cout < < " Introduzca nombres de los soldados " < < endl ; for ( int i = 0; i < nombres . size (); + + i ) { cout < < " Nombre " < < i < < " : " ; cin > > nombres [ i ] ; } } // int leer_numero () { cout < < " Introduzca un n mero : " ; u int num ; cin > > num ; return num ; } // void leer_nombre ( string & nombre ) { cout < < " Introduzca el nombre del primer soldado : " ; cin > > nombre ; } // void circulo () { Soldados nombres ; leer_nombres ( nombres ); int n = leer_numero (); string pr_sol ; leer_nombre ( pr_sol ); int i_esc = escapa ( nombres , n , pr_sol );

78

if ( i_esc < nombres . size ()) { cout < < " El soldado que escapa es : " < < nombres [ i_esc ] < < endl ; } else { cout < < " Error : " < < pr_sol < < endl ; } } // int main () { circulo (); }

31. Disee un programa que permita leer una clave y una secuencia de caracteres terminada en punto (.) que n representa un mensaje codicado, y lo decodique segn la clave le u da. Los signos de puntuacin y d o gitos que aparezcan en el mensaje deben escribirse como tales. La clave consiste en una sucesin de las 26 letras o minsculas del alfabeto, las cuales se hacen corresponder la primera letra de la clave con la letra a, la u segunda con la letra b, etc....Por ejemplo, una entrada de la forma:
Introduzca la clave: ixmrklstnuzbowfaqejdcpvhyg [las 26 letras minsculas] u Introduzca el texto: milk.

de tal forma que la letra i se corresponde con la letra a, la letra x se corresponde con la letra b, la letra m se corresponde con la letra c, y as sucesivamente, por lo que el ejemplo anterior deber a dar como salida: cafe. Solucin o

# include < iostream > # include < string > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // const int N_LETRAS = ( ' z ' - ' a ' ) + 1; typedef array < char , N_LETRAS > Clave ; // // c o n v i e r t e de l e t r a a n d i c e int letra_idx ( char letra ) { int res = N_LETRAS ; if ( letra > = ' a ' & & letra < = ' z ' ) { res = ( letra - ' a ' ); } return res ; } // // c o n v i e r t e de n d i c e a l e t r a inline char idx_letra ( int idx ) { return char ( ' a ' + idx ); } // void init ( Clave & c ) { for ( int i = 0; i < c . size (); + + i ) { c[i] = ' '; } } // // Ejemplo : i x m r k l s t n u z b o w f a q e j d c p v h y g // void leer_clave ( bool & ok , Clave & clave_cifrado , Clave & c l a ve _ d e s c i f r a d o ) { ok = true ; init ( clave_cifrado ); init ( c l a v e _ d e s c i f r a d o ); cout < < " Introduzca la clave : " ; int i_letra = 0; while ( ok & & i_letra < N_LETRAS ) { char letra ; cin > > letra ; int idx = letra_idx ( letra );

79

if (( idx < N_LETRAS ) & & ( c l a v e _ d e s c i f r a d o [ idx ] = = ' ' )) { c l a v e _ d e s c i f r a do [ idx ] = idx_letra ( i_letra ); clave_cifrado [ i_letra ] = letra ; + + i_letra ; } else { ok = false ; } } cin . ignore (1000 , ' \ n ' ); } // void transformar ( const Clave & clave ) { cout < < " Introduzca el texto : " ; char c ; cin > > ws ; cin . get ( c ); while ( c ! = ' . ' ) { if ( c > = ' a ' & & c < = ' z ' ) { c = clave [ letra_idx ( c ) ] ; } cout < < c ; cin . get ( c ); } cout < < c < < endl ; cin . ignore (1000 , ' \ n ' ); } // char menu () { char op ; cout < < endl ; cout < < " - - - - - - - - - - - - - - - - - - - " < < endl ; cout < < " c : Cifrar " < < endl ; cout < < " d : Descifrar " < < endl ; cout < < " x : Fin " < < endl ; cout < < " - - - - - - - - - - - - - - - - - - - " < < endl ; cout < < endl ; do { cout < < " Opcion ? " ; cin > > op ; } while ( op ! = ' c ' & & op ! = ' d ' & & op ! = ' x ' ); cin . ignore (1000 , ' \ n ' ); return op ; } // int main () { bool ok ; Clave cc , cd ; leer_clave ( ok , cc , cd ); if ( ! ok ) { cout < < " clave err nea " < < endl ; o } else { char op ; do { op = menu (); switch ( op ) { case ' c ' : transformar ( cc ); break ; case ' d ' : transformar ( cd ); break ; } } while ( op ! = ' x ' ); } }

32. Se deber realizar un programa que lea una cadena de caracteres que ser utilizada como clave de cifrado, a a y posteriormente, se leer un texto de longitud indenida hasta un carcter punto (.) el cual deber ser a a a cifrado y mostrado por pantalla. El cifrado se realizar de la siguiente manera: a

80

Slo se cifrarn las letras minsculas. o a u Cada carcter de la clave (se supone que slo esta formada por letras minsculas) se corresponde a o u con un valor numrico que representa un incremento igual a la distancia alfabtica de dicha letra e e respecto a la letra a. Por ejemplo al carcter a de la clave le corresponde un incremento de 0, al a carcter b un incremento de 1, a c de 2, y as sucesivamente. a A cada carcter del texto a cifrar se le asocia para realizar el cifrado la letra de la clave correspondiente a a su posicin (el primer carcter ocupa la posicin 0) mdulo el nmero de caracteres de la clave. o a o o u Nota: mdulo signica el resto de la divisin. o o Cada carcter del texto a cifrar ser cifrado mediante un incremento circular (el siguiente a la z a a es la a) correspondiente a la letra de la clave asociada. Por ejemplo, para la clave abx y el texto de entrada hola y adios. mostrar como resultado hpia z xdjls. a como se indica en el siguiente esquema:
abxa b xabxa hola y adios. ------------hpia z xdjls. ------------|||| | ||||| |||| | ||||+--> |||| | |||+---> |||| | ||+----> |||| | |+-----> |||| | +------> |||| +--------> |||+----------> ||+-----------> |+------------> +------------->

s o i d a y a l o h

+ + + + + + + + + +

0 = s 23 = l 1 = j 0 = d 23 = x 1 = z 0 = a 23 = i 1 = p 0 = h

Solucin o

# include < cassert > # include < iostream > # include < string > using namespace std ; // bool es_minuscula ( char c ) { return (( c > = ' a ' ) & & ( c < = ' z ' )); } // void sig_circular ( int & i , int limite ) { assert ( i < limite ); ++i; // i = ( i + 1) % l i m i t e ; if ( i = = limite ) { // i = 0; // } // } // void inc_circular ( int & x , int inc , int min , int max ) { assert (( x > = min ) & & ( x < = max ) & & ( inc < max - min + 1)); x = x + inc ; // x = ( ( ( ( x+i n c )min) %(maxmin+1))+min ) ; if ( x > max ) { // x = min + x - max - 1; // } // } // void cifrar ( char & c , char cl ) { assert ( es_minuscula ( c ) & & es_minuscula ( cl )); const int inc = ( cl - ' a ' ); int x = c ; inc_circular (x , inc , int ( ' a ' ) , int ( ' z ' )); c = char ( x ); } //

81

void cifrar_texto ( const string & clave ) { assert ( clave . size () > 0); int i = 0; cout < < " Texto : " ; cin > > ws ; char c ; cin . get ( c ); cout < < " Resultado : " ; while ( c ! = ' . ' ) { if ( es_minuscula ( c )) { cifrar (c , clave [ i ] ); sig_circular (i , clave . size ()); } cout < < c ; cin . get ( c ); } cout < < c ; } // int main () { string clave ; cout < < " Clave : " ; cin > > clave ; cifrar_texto ( clave ); cout < < endl ;

33. Un importante vulcanlogo, con objeto de estudiar las consecuencias destructoras de un volcn en una o a determinada rea, pretende desarrollar un sistema informtico que le ayude en su tarea. Para ello, nos a a pide que realicemos el siguiente subprograma junto con el programa necesario para realizar la prueba.
const int N_FILAS = 10; const int N_COLUMNAS = 12; typedef array<int, N_COLUMNAS> FilaSup; typedef array<FilaSup, N_FILAS> Superficie; typedef array<char, N_COLUMNAS> FilaLava; typedef array<FilaLava, N_FILAS> LavaSup; void flujo_lava (const Superficie& sup, int fila, int columna, LavaSup& lava);

El cual recibe un array bidimensional (sup) que representa el plano de la zona a estudiar, donde cada elemento contiene un nmero entero que representa la altura de ese punto respecto al nivel del mar. u As mismo, recibe un punto (fila y columna) de dicho plano donde surge el crter de un volcn. El a a subprograma predecir el recorrido que realiza la lava, y la mostrar en el array bidimensional de salida a a (lava), donde el asterisco (*) representa que la lava ha pasado por ese punto, y el espacio en blanco ( ) representa que la lava no ha pasado por dicho punto. El ujo de lava se desplaza segn el siguiente esquema a partir del crter: u a Desde un determinado punto, siempre se mueve hacia los puntos circundantes que se encuentran a menor altura. Los puntos circundantes de uno dado sern el superior, inferior, izquierdo y derecho a (no se considerarn los diagonales). a As sucesivamente se repite el proceso para todos los puntos donde haya alcanzado el ujo de lava. Por ejemplo, para un crter del volcn en el punto (2, 2) a a sup 0 1 2 3 4 0 4 5 7 5 4 1 7 6 7 6 6 2 8 9 8 8 6 3 7 9 8 6 9 4 6 7 7 8 5 0 lava 1 2 3 4 0 1 2 3 4

82

Solucin o

# include < cassert > # include < iostream > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // C o n s t a n t e s const int N_FILAS = 10; const int N_COLUMNAS = 12; // Tipos typedef array < int , N_COLUMNAS > FilaSup ; typedef array < FilaSup , N_FILAS > Superficie ; typedef array < char , N_COLUMNAS > FilaLava ; typedef array < FilaLava , N_FILAS > LavaSup ; // struct Coord { int fil ; int col ; }; const int MAX_COORD = N_FILAS * N_COLUMNAS * 2; typedef array < Coord , MAX_COORD > Datos ; struct Vector { Datos elm ; int nelms ; }; // inline void ini ( Vector & v ) { v . nelms = 0; } // void anyadir ( Vector & v , const Coord & c ) { if ( v . nelms < v . elm . size ()) { v . elm [ v . nelms ] = c ; + + v . nelms ; } } // void nueva_coord ( Vector & v , Coord & c ) { assert ( v . nelms > 0); - - v . nelms ; c = v . elm [ v . nelms ] ; } // void ini ( LavaSup & lava ) { for ( int f = 0; f < lava . size (); + + f ) { for ( int c = 0; c < lava [ f ] . size (); + + c ) { lava [ f ] [ c ] = ' ' ; } } } // inline bool es_valida ( const Superficie & sup , const LavaSup & lava , const Coord & c ) { return (( c . fil < sup . size ()) & & ( c . col < sup [ c . fil ] . size ()) & & ( lava [ c . fil ] [ c . col ] = = ' ' )); } // inline bool es_menor ( const Superficie & sup , const LavaSup & lava , const Coord & act , const Coord & nueva ) { return ( es_valida ( sup , lava , nueva ) & & ( sup [ nueva . fil ] [ nueva . col ] < sup [ act . fil ] [ act . col ] )); } // inline int abs ( int x ) { int res ; if ( x < 0) { res = int ( - x );

83

} else { res = int ( x ); } return res ; } // inline void inc ( int & val , int i ) { int x = int ( val ) + i ; if ( x < 0) { x = MAX_COORD ; } val = int ( x ); } // void paso ( const Coord & c , int ff , int cc , Coord & n ) { n = c; inc ( n . fil , ff ); inc ( n . col , cc ); } // void p r o c e s a r _ v e c i n o s ( const Superficie & sup , const Coord & act , const LavaSup & lava , Vector & v ) { for ( int ff = - 1; ff < = + 1; + + ff ) { for ( int cc = - 1; cc < = + 1; + + cc ) { if ( abs ( ff ) + abs ( cc ) = = 1) { Coord n ; paso ( act , ff , cc , n ); if ( es_menor ( sup , lava , act , n )) { anyadir (v , n ); } } } } } // void asg ( Coord & c , int ff , int cc ) { c . fil = ff ; c . col = cc ; } // void flujo_lava ( const Superficie & sup , int fil , int col , LavaSup & lava ) { Vector v ; ini ( v ); ini ( lava ); Coord pos ; asg ( pos , fil , col ); if ( es_valida ( sup , lava , pos )) { anyadir (v , pos ); while ( v . nelms > 0) { nueva_coord (v , pos ); if ( lava [ pos . fil ] [ pos . col ] = = ' ' ) { lava [ pos . fil ] [ pos . col ] = ' * ' ; p r o c e s a r _ v e c i n o s ( sup , pos , lava , v ); } } } } // void linea ( int n ) { for ( int i = 0; i < n ; + + i ) { cout < < ' - ' ; } cout < < endl ; } // void imprimir ( const LavaSup & lava ) {

84

linea ( lava [ 0 ] . size () + 2); for ( int f = 0; f < lava . size (); + + f ) { cout < < " | " ; for ( int c = 0; c < lava [ f ] . size (); + + c ) { cout < < lava [ f ] [ c ] ; } cout < < " | " < < endl ; } linea ( lava [ 0 ] . size () + 2); } // const Superficie ALTURAS = {{ {{ 4 , 7 , 8 , 7 , 6 , 6 , 8 , 6 , 8 , 9 , 5 , {{ 5 , 6 , 9 , 9 , 7 , 3 , 4 , 7 , 8 , 7 , 6 , {{ 7 , 7 , 8 , 8 , 7 , 4 , 7 , 8 , 7 , 6 , 4 , {{ 5 , 6 , 9 , 9 , 7 , 3 , 4 , 7 , 8 , 7 , 6 , {{ 4 , 6 , 6 , 9 , 5 , 5 , 6 , 9 , 9 , 7 , 6 , {{ 5 , 6 , 8 , 6 , 8 , 9 , 5 , 4 , 3 , 2 , 1 , {{ 4 , 7 , 8 , 7 , 6 , 6 , 8 , 6 , 8 , 9 , 5 , {{ 7 , 7 , 8 , 8 , 7 , 4 , 7 , 8 , 7 , 6 , 4 , {{ 4 , 7 , 8 , 7 , 6 , 6 , 8 , 6 , 8 , 9 , 5 , {{ 4 , 6 , 6 , 9 , 5 , 5 , 6 , 9 , 9 , 7 , 6 , }}; // int main () { cout < < " Introduzca las coordenadas int f , c ; cin > > f > > c ; LavaSup lava ; flujo_lava ( ALTURAS , f , c , lava ); imprimir ( lava ); }

4 9 6 9 8 0 4 6 4 8

}} , }} , }} , }} , }} , }} , }} , }} , }} , }}

del cr ter : " ; a

34. Para realizar operaciones con nmeros complejos, podemos denir el siguiente tipo: u
struct Complejo { double real; double img; };

Escribe subprogramas (sobrecargando los operadores necesarios) que realicen las operaciones de suma, resta, multiplicacin y divisin de nmeros complejos denidos con el tipo anterior, as como el programa o o u para probar adecuadamente su funcionamiento. Comprese con la solucin del ejercicio 1 del tema 4. a o Solucin o

# include < iostream > using namespace std ; // const double E R RO R_ PR E CI SI ON = 1 e - 10; // struct Complejo { double real ; double img ; }; // inline double sq ( double x ) { return x * x ; } // bool iguales ( double x , double y ) { double cmp = x - y ; return ( - ER R OR _P RE C IS IO N < = cmp ) & & ( cmp < = E RR OR _ PR EC IS I ON ); } // Complejo crear ( double real , double img ) { Complejo res ; res . real = real ;

85

res . img = img ; return res ; } // Complejo operator + ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = c1 . real + c2 . real ; res . img = c1 . img + c2 . img ; return res ; } // Complejo operator - ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = c1 . real - c2 . real ; res . img = c1 . img - c2 . img ; return res ; } // Complejo operator * ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = ( c1 . real * c2 . real ) - ( c1 . img * c2 . img ); res . img = ( c1 . real * c2 . img ) + ( c1 . img * c2 . real ); return res ; } // Complejo operator/ ( const Complejo & c1 , const Complejo & c2 ) { Complejo res ; res . real = (( c1 . real * c2 . real ) + ( c1 . img * c2 . img )) / ( sq ( c2 . real ) + sq ( c2 . img )); res . img = (( c1 . img * c2 . real ) - ( c1 . real * c2 . img )) / ( sq ( c2 . real ) + sq ( c2 . img )); return res ; } // bool operator = = ( const Complejo & c1 , const Complejo & c2 ) { return iguales ( c1 . real , c2 . real ) & & iguales ( c1 . img , c2 . img ); } // bool operator ! = ( const Complejo & c1 , const Complejo & c2 ) { return ! ( c1 = = c2 ); } // inline void escribir ( const Complejo & c ) { cout < < " ( " < < c . real < < " , " < < c . img < < " ) " ; } // // // void leer ( Complejo & c ) { cout < < " Introduzca un n mero complejo ( real , img ): " ; u cin > > c . real > > c . img ; } // void prueba_suma ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = c1 + c2 ; escribir ( c1 ); cout < < " + " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( c1 ! = c0 - c2 ) { cout < < " Error en operaciones de suma/resta " < < endl ; } } // void prueba_resta ( const Complejo & c1 , const Complejo & c2 )

86

{ Complejo c0 = c1 - c2 ; escribir ( c1 ); cout < < " - " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( c1 ! = c0 + c2 ) { cout < < " Error en operaciones de suma/resta " < < endl ; } } // void prueba_mult ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = c1 * c2 ; escribir ( c1 ); cout < < " * " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( c1 ! = c0 / c2 ) { cout < < " Error en operaciones de mult/div " < < endl ; } } // void prueba_div ( const Complejo & c1 , const Complejo & c2 ) { Complejo c0 = c1 / c2 ; escribir ( c1 ); cout < < " / " ; escribir ( c2 ); cout < < " = " ; escribir ( c0 ); cout < < endl ; if ( c1 ! = c0 * c2 ) { cout < < " Error en operaciones de mult/div " < < endl ; } } // int main () { Complejo c1 , c2 ; leer ( c1 ); leer ( c2 ); // prueba_suma ( c1 , c2 ); prueba_resta ( c1 , c2 ); prueba_mult ( c1 , c2 ); prueba_div ( c1 , c2 ); // }

35. Disee un programa para gestionar una agenda personal, donde la informacin que se almacena de cada n o persona es la siguiente: Nombre, Telfono, Direccin, Calle, Nmero, Piso, Cdigo Postal y Ciudad. La e o u o agenda permitir realizar las siguientes operaciones: a a) Aadir los datos de una persona n b) Acceder a los datos de una persona a partir de su nombre. c) Borrar una persona a partir de su nombre. d ) Modicar los datos de una persona a partir de su nombre. e) Listar el contenido completo de la agenda.
# include # include # include # include < iostream > < string > < cassert > < tr1/array >

Solucin o

87

using namespace std ; using namespace std :: tr1 ; // C o n s t a n t e s const int MAX_PERSONAS = 50; // Tipos struct Direccion { int num ; string calle ; string piso ; string cp ; string ciudad ; }; struct Persona { string nombre ; string tel ; Direccion direccion ; }; // Tipos typedef array < Persona , MAX_PERSONAS > Personas ; struct Agenda { int n_pers ; Personas pers ; }; enum Cod_Error { OK , AG_LLENA , NO_ENCONTRADO , YA_EXISTE }; // S u b a l g o r i t m o s void Inicializar ( Agenda & ag ) { ag . n_pers = 0; } // S u b a l g o r i t m o s void Leer_Dir eccion ( Direccion & dir ) { cin > > dir . calle ; cin > > dir . num ; cin > > dir . piso ; cin > > dir . cp ; cin > > dir . ciudad ; } // void Leer_Persona ( Persona & per ) { cin > > per . nombre ; cin > > per . tel ; Leer _Direcci on ( per . direccion ); } // S u b a l g o r i t m o s void E s c r i b i r _ D i r e c c i o n ( const Direccion & dir ) { cout < < dir . calle < < " " ; cout < < dir . num < < " " ; cout < < dir . piso < < " " ; cout < < dir . cp < < " " ; cout < < dir . ciudad < < " " ; } // void E s c r i b i r _ P e r s o n a ( const Persona & per ) { cout < < per . nombre < < " " ; cout < < per . tel < < " " ; E s c r i b i r _ D i r e c c i o n ( per . direccion ); cout < < endl ; } // S u b a l g o r i t m o s // Busca una Persona en l a Agenda // D e v u e l v e su p o s i c i o n s i s e e n c u e n t r a , o b i e n >= ag . n p e r s en o t r o c a s o int Buscar_P ersona ( const string & nombre , const Agenda & ag ) { int i = 0; while (( i < ag . n_pers ) & & ( nombre ! = ag . pers [ i ] . nombre )) { ++i; } return i ;

88

} // void Anyadir ( Agenda & ag , const Persona & per ) { ag . pers [ ag . n_pers ] = per ; + + ag . n_pers ; } // void Eliminar ( Agenda & ag , int pos ) { ag . pers [ pos ] = ag . pers [ ag . n_pers - 1 ] ; - - ag . n_pers ; } // S u b a l g o r i t m o s void A ny ad ir _ Pe rs on a ( const Persona & per , Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( per . nombre , ag ); if ( i < ag . n_pers ) { ok = YA_EXISTE ; } else if ( ag . n_pers = = ag . pers . size ()) { ok = AG_LLENA ; } else { ok = OK ; Anyadir ( ag , per ); } } // void Borrar_P ersona ( const string & nombre , Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; Eliminar ( ag , i ); } } // S u b a l g o r i t m o s void M o d i f i c a r _ P e r s o n a ( const string & nombre , const Persona & nuevo , Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { Eliminar ( ag , i ); A ny ad ir _ Pe rs on a ( nuevo , ag , ok ); } } // void I m p r i m i r _ P e r s o n a ( const string & nombre , const Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } } // S u b a l g o r i t m o s void I mp ri mi r _A ge nd a ( const Agenda & ag , Cod_Error & ok ) { for ( int i = 0; i < ag . n_pers ; + + i ) { E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } ok = OK ; } // char Menu () { char opcion ; cout < < endl ; cout < < " a . - Anadir Persona " < < endl ; cout < < " b . - Buscar Persona " < < endl ;

89

cout cout cout cout do {

<< << << <<

"c. "d. "e. "x.

Borrar Persona " < < endl ; Modificar Persona " < < endl ; Imprimir Agenda " < < endl ; Salir " < < endl ;

cout < < " Introduzca Opcion : " ; cin > > opcion ; } while ( ! ((( opcion > = ' a ' ) & & ( opcion < = ' e ' )) | | ( opcion = = ' x ' ))); return opcion ; } // S u b a l g o r i t m o s void E s c r i b i r _ C o d _ E r r o r ( Cod_Error cod ) { switch ( cod ) { case OK : cout < < " Operacion correcta " < < endl ; break ; case AG_LLENA : cout < < " Agenda llena " < < endl ; break ; case NO_ENCONTRADO : cout < < " La persona no se encuentra en la agenda " < < endl ; break ; case YA_EXISTE : cout < < " La persona ya se encuentra en la agenda " < < endl ; break ; } } // P r i n c i p a l int main () { Agenda ag ; char opcion ; Persona per ; string nombre ; Cod_Error ok ; Inicializar ( ag ); do { opcion = Menu (); switch ( opcion ) { case ' a ' : cout < < " Introduzca los datos de la Persona " < < endl ; cout < < " ( nombre , tel , calle , num , piso , cod_postal , ciudad ) " < < endl ; Leer_Persona ( per ); A ny ad ir _ Pe rs on a ( per , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' b ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; I m p r i m i r _ P e r s o na ( nombre , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' c ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; Borr ar_Perso na ( nombre , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' d ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; cout < < " Nuevos datos de la Persona " < < endl ; cout < < " ( nombre , tel , calle , num , piso , cod_postal , ciudad ) " < < endl ; Leer_Persona ( per ); M o d i f i c a r _ P e r s o n a ( nombre , per , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' e ' : I mp ri mi r _A ge nd a ( ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; } } while ( opcion ! = ' x ' );

90

91

Tema 5: B squeda y Ordenacin u o


1. Disee un programa para gestionar una agenda personal, donde la informacin que se almacena de can o da persona es la siguiente: Nombre, Telfono, Direccin, Calle, Nmero, Piso, Cdigo Postal y Ciudad. e o u o La agenda se encontrar ordenada segn el nombre de la persona, y permitir realizar las siguientes a u a operaciones: a) Aadir los datos de una persona n b) Acceder a los datos de una persona a partir de su nombre. c) Borrar una persona a partir de su nombre. d ) Modicar los datos de una persona a partir de su nombre. e) Listar el contenido completo de la agenda. Solucin o

# include < iostream > # include < string > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // C o n s t a n t e s const int MAX_PERSONAS = 50; // Tipos struct Direccion { int num ; string calle ; string piso ; string cp ; string ciudad ; }; struct Persona { string nombre ; string tel ; Direccion direccion ; }; // Tipos typedef array < Persona , MAX_PERSONAS > Personas ; struct Agenda { int n_pers ; Personas pers ; }; enum Cod_Error { OK , AG_LLENA , NO_ENCONTRADO , YA_EXISTE }; // S u b a l g o r i t m o s void Inicializar ( Agenda & ag ) { ag . n_pers = 0; } // S u b a l g o r i t m o s void Leer_Dir eccion ( Direccion & dir ) { cin > > dir . calle ; cin > > dir . num ; cin > > dir . piso ; cin > > dir . cp ; cin > > dir . ciudad ; } // void Leer_Persona ( Persona & per ) { cin > > per . nombre ; cin > > per . tel ; Leer _Direcci on ( per . direccion ); } // S u b a l g o r i t m o s void E s c r i b i r _ D i r e c c i o n ( const Direccion & dir ) { cout < < dir . calle < < " " ;

92

cout cout cout cout

<< << << <<

dir . num < < " " ; dir . piso < < " " ; dir . cp < < " " ; dir . ciudad < < " " ;

} // void E s c r i b i r _ P e r s o n a ( const Persona & per ) { cout < < per . nombre < < " " ; cout < < per . tel < < " " ; E s c r i b i r _ D i r e c c i o n ( per . direccion ); cout < < endl ; } // S u b a l g o r i t m o s // Busca una Persona en l a Agenda Ordenada // D e v u e l v e su p o s i c i o n s i s e e n c u e n t r a , o b i e n >= ag . n p e r s en o t r o c a s o int Buscar_P ersona ( const string & nombre , const Agenda & ag ) { int i = 0; int f = ag . n_pers ; int m = ( i + f ) / 2; while (( i < f ) & & ( nombre ! = ag . pers [ m ] . nombre )) { if ( nombre < ag . pers [ m ] . nombre ) { f = m; } else { i = m + 1; } m = ( i + f ) / 2; } if ( i > = f ) { m = ag . n_pers ; } return m ; } // S u b a l g o r i t m o s int B us ca r_ P os ic io n ( const string & nombre , const Agenda & ag ) { int i = 0; while (( i < ag . n_pers ) & & ( nombre > = ag . pers [ i ] . nombre )) { ++i; } return i ; } // void Abrir_Hueco ( Agenda & ag , int pos , const Persona & per ) { for ( int i = ag . n_pers ; i > pos ; - - i ) { ag . pers [ i ] = ag . pers [ i - 1 ] ; } ag . pers [ pos ] = per ; + + ag . n_pers ; } // void Cerrar_Hueco ( Agenda & ag , int pos ) { - - ag . n_pers ; for ( int i = pos ; i < ag . n_pers ; + + i ) { ag . pers [ i ] = ag . pers [ i + 1 ] ; } } // S u b a l g o r i t m o s void A ny ad ir _ Pe rs on a ( const Persona & per , Agenda & ag , Cod_Error & ok ) { int pos = B us ca r_ P os ic io n ( per . nombre , ag ); if (( pos < ag . n_pers ) & & ( per . nombre = = ag . pers [ pos ] . nombre )) { ok = YA_EXISTE ; } else if ( ag . n_pers = = ag . pers . size ()) { ok = AG_LLENA ; } else { ok = OK ; Abrir_Hueco ( ag , pos , per ); } } //

93

void Borrar_P ersona ( const string & nombre , Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; Cerrar_Hueco ( ag , i ); } } // S u b a l g o r i t m o s void M o d i f i c a r _ P e r s o n a ( const string & nombre , const Persona & nuevo , Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; Cerrar_Hueco ( ag , i ); A ny ad ir _ Pe rs on a ( nuevo , ag , ok ); } } // void I m p r i m i r _ P e r s o n a ( const string & nombre , const Agenda & ag , Cod_Error & ok ) { int i = Busc ar_Perso na ( nombre , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } } // void I mp ri mi r _A ge nd a ( const Agenda & ag , Cod_Error & ok ) { for ( int i = 0; i < ag . n_pers ; + + i ) { E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } ok = OK ; } // S u b a l g o r i t m o s char Menu () { char opcion ; cout < < endl ; cout < < " a . - Anadir Persona " < < endl ; cout < < " b . - Buscar Persona " < < endl ; cout < < " c . - Borrar Persona " < < endl ; cout < < " d . - Modificar Persona " < < endl ; cout < < " e . - Imprimir Agenda " < < endl ; cout < < " x . - Salir " < < endl ; do { cout < < " Introduzca Opcion : " ; cin > > opcion ; } while ( ! ((( opcion > = ' a ' ) & & ( opcion < = ' e ' )) | | ( opcion = = ' x ' ))); return opcion ; } // S u b a l g o r i t m o s void E s c r i b i r _ C o d _ E r r o r ( Cod_Error cod ) { switch ( cod ) { case OK : cout < < " Operacion correcta " < < endl ; break ; case AG_LLENA : cout < < " Agenda llena " < < endl ; break ; case NO_ENCONTRADO : cout < < " La persona no se encuentra en la agenda " < < endl ; break ; case YA_EXISTE : cout < < " La persona ya se encuentra en la agenda " < < endl ; break ;

94

} } // P r i n c i p a l int main () { Agenda ag ; char opcion ; Persona per ; string nombre ; Cod_Error ok ; Inicializar ( ag ); do { opcion = Menu (); switch ( opcion ) { case ' a ' : cout < < " Introduzca los datos de la Persona " < < endl ; cout < < " ( nombre , tel , calle , num , piso , cod_postal , ciudad ) " < < endl ; Leer_Persona ( per ); A ny ad ir _ Pe rs on a ( per , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' b ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; I m p r i m i r _ P e r s o na ( nombre , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' c ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; Borr ar_Perso na ( nombre , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' d ' : cout < < " Introduzca Nombre " < < endl ; cin > > nombre ; cout < < " Nuevos datos de la Persona " < < endl ; cout < < " ( nombre , tel , calle , num , piso , cod_postal , ciudad ) " < < endl ; Leer_Persona ( per ); M o d i f i c a r _ P e r s o n a ( nombre , per , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; case ' e ' : I mp ri mi r _A ge nd a ( ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; } } while ( opcion ! = ' x ' ); }

2. Dado una estructura de datos que puede contener hasta un nmero mximo de 50 personas, donde cada u a persona contiene su nombre, la fecha de nacimiento y su telfono: e a) Disee un subprograma para ordenar dicha estructura de datos. El criterio de ordenacin es en orden n o creciente por nombre de la persona, y en caso de nombres iguales, entonces se considera el orden creciente por fecha de nacimiento. b) Dada la estructura de datos ordenada segn el criterio de ordenacin del apartado anterior, disee u o n un subprograma para realizar la bsqueda binaria de una persona en la estructura de datos, para u ello, el subprograma recibir tanto el nombre de la persona, como su fecha de nacimiento. a c) Disee el programa principal y los subprogramas necesarios para comprobar el funcionamiento aden cuado de los subprogramas anteriores.
Antonio 15/05/1997 952.234.567 Lucas 23/11/1989 952.135.246 Lucas 17/03/1992 952.235.711 Lucas 14/08/1992 952.123.456 Lucas 25/08/1992 952.987.654 Mar a 13/01/1994 952.567.234

# include < iostream >

Solucin o

95

# include < iomanip > # include < string > # include < cassert > # include < tr1/array > using namespace std ; using namespace std :: tr1 ; // C o n s t a n t e s const int MAX_PERSONAS = 50; // Tipos struct Fecha { int dia ; int mes ; int anyo ; }; struct Persona { string nombre ; Fecha fnac ; string tel ; }; struct Clave { string nombre ; Fecha fnac ; }; // Tipos typedef array < Persona , MAX_PERSONAS > Personas ; struct Agenda { int n_pers ; Personas pers ; }; enum Cod_Error { OK , AG_LLENA , NO_ENCONTRADO , YA_EXISTE }; // S u b a l g o r i t m o s void Inicializar ( Agenda & ag ) { ag . n_pers = 0; } // S u b a l g o r i t m o s void Leer_Fecha ( Fecha & f ) { cin > > f . dia ; cin > > f . mes ; cin > > f . anyo ; } // void Leer_Clave ( Clave & clave ) { cin > > clave . nombre ; Leer_Fecha ( clave . fnac ); } // void Leer_Persona ( Persona & per ) { cin > > per . nombre ; Leer_Fecha ( per . fnac ); cin > > per . tel ; } // void Escribir _Fecha ( const Fecha & f ) { cout < < setfill ( ' 0 ' ) < < setw (2) < < f . dia < < " / " ; cout < < setfill ( ' 0 ' ) < < setw (2) < < f . mes < < " / " ; cout < < setfill ( ' 0 ' ) < < setw (4) < < f . anyo < < " " ; } // void E s c r i b i r _ P e r s o n a ( const Persona & per ) { cout < < per . nombre < < " " ; Escr ibir_Fec ha ( per . fnac ); cout < < per . tel < < " " ; cout < < endl ; } // void Crear_Clave ( Clave & clave , const Persona & per )

96

{ clave . nombre = per . nombre ; clave . fnac = per . fnac ; } // // COMPARACION // inline bool es_igual ( const Fecha & f1 , const Fecha & f2 ) { return ( f1 . anyo = = f2 . anyo ) & & ( f1 . mes = = f2 . mes ) & & ( f1 . dia = = f2 . dia ); } inline bool es_distinto ( const Fecha & f1 , const Fecha & f2 ) { return ! es_igual ( f1 , f2 ); } inline bool es_menor ( const Fecha & f1 , const Fecha & f2 ) { return (( f1 . anyo < f2 . anyo ) | | (( f1 . anyo = = f2 . anyo ) & & (( f1 . mes < f2 . mes ) | | (( f1 . mes = = f2 . mes ) & & ( f1 . dia < f2 . dia ))))); } inline bool es_mayor ( const Fecha & f1 , const Fecha & f2 ) { return es_menor ( f2 , f1 ); } inline bool es _menor_i gual ( const Fecha & f1 , const Fecha & f2 ) { return ! es_menor ( f2 , f1 ); } inline bool es _mayor_i gual ( const Fecha & f1 , const Fecha & f2 ) { return ! es_menor ( f1 , f2 ); } // inline bool es_igual ( const Persona & p1 , const Persona & p2 ) { return ( p1 . nombre = = p2 . nombre ) & & es_igual ( p1 . fnac , p2 . fnac ) & & ( p1 . tel = = p2 . tel ); } inline bool es_distinto ( const Persona & p1 , const Persona & p2 ) { return ! es_igual ( p1 , p2 ); } inline bool es_menor ( const Persona & p1 , const Persona & p2 ) { return (( p1 . nombre < p2 . nombre ) | | (( p1 . nombre = = p2 . nombre ) & & es_menor ( p1 . fnac , p2 . fnac ))); } inline bool es_mayor ( const Persona & p1 , const Persona & p2 ) { return es_menor ( p2 , p1 ); } inline bool es _menor_i gual ( const Persona & p1 , const Persona & p2 ) { return ! es_menor ( p2 , p1 ); } inline bool es _mayor_i gual ( const Persona & p1 , const Persona & p2 ) { return ! es_menor ( p1 , p2 ); } // inline bool es_igual ( const Clave & p1 , const Persona & p2 ) { return ( p1 . nombre = = p2 . nombre ) & & es_igual ( p1 . fnac , p2 . fnac ); } inline bool es_distinto ( const Clave & p1 , const Persona & p2 ) { return ! es_igual ( p1 , p2 ); } inline bool es_menor ( const Clave & p1 , const Persona & p2 ) {

97

return (( p1 . nombre < p2 . nombre ) | | (( p1 . nombre = = p2 . nombre ) & & es_menor ( p1 . fnac , p2 . fnac ))); } inline bool es_mayor ( const Clave & p1 , const Persona & p2 ) { return (( p1 . nombre > p2 . nombre ) | | (( p1 . nombre = = p2 . nombre ) & & es_mayor ( p1 . fnac , p2 . fnac ))); } inline bool es _menor_i gual ( const Clave & p1 , const Persona & p2 ) { return ! es_mayor ( p1 , p2 ); } inline bool es _mayor_i gual ( const Clave & p1 , const Persona & p2 ) { return ! es_menor ( p1 , p2 ); } // // ANYADIR PERSONA SIN ORDENACION // // Busca una Persona en l a Agenda // D e v u e l v e su p o s i c i o n s i s e e n c u e n t r a , o b i e n >= ag . n p e r s en o t r o c a s o int Buscar_P ersona ( const Clave & clave , const Agenda & ag ) { int i = 0; while (( i < ag . n_pers ) & & es_distinto ( clave , ag . pers [ i ] )) { ++i; } return i ; } // void Anyadir ( Agenda & ag , const Persona & per ) { ag . pers [ ag . n_pers ] = per ; + + ag . n_pers ; } // void A ny ad ir _ Pe rs on a ( const Persona & per , Agenda & ag , Cod_Error & ok ) { Clave clave ; Crear_Clave ( clave , per ); int i = Busc ar_Perso na ( clave , ag ); if ( i < ag . n_pers ) { ok = YA_EXISTE ; } else if ( ag . n_pers = = int ( ag . pers . size ())) { ok = AG_LLENA ; } else { ok = OK ; Anyadir ( ag , per ); } } // // BUSCAR PERSONA CON ORDENACION // // Busca una Persona en l a Agenda Ordenada // D e v u e l v e su p o s i c i o n s i s e e n c u e n t r a , o b i e n >= ag . n p e r s en o t r o c a s o int B u s c a r _ P e r s o n a _ B i n a r i a ( const Clave & clave , const Agenda & ag ) { int i = 0; int f = ag . n_pers ; int m = ( i + f ) / 2; while (( i < f ) & & es_distinto ( clave , ag . pers [ m ] )) { if ( es_menor ( clave , ag . pers [ m ] )) { f = m; } else { i = m + 1; } m = ( i + f ) / 2; } if ( i > = f ) { m = ag . n_pers ; } return m ; }

98

// void I m p r i m i r _ P e r s o n a _ O r d ( const Clave & clave , const Agenda & ag , Cod_Error & ok ) { int i = B u s c a r _ P e r s o n a _ B i n a r i a ( clave , ag ); if ( i > = ag . n_pers ) { ok = NO_ENCONTRADO ; } else { ok = OK ; E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } } // // ORDENACION // int B us ca r_ P os ic io n ( const Clave & clave , const Agenda & ag ) { int i = 0; while (( i < ag . n_pers ) & & es_may or_igual ( clave , ag . pers [ i ] )) { ++i; } return i ; } // void Anyadir_Ord ( Agenda & ag , int pos , const Persona & per ) { for ( int i = ag . n_pers ; i > pos ; - - i ) { ag . pers [ i ] = ag . pers [ i - 1 ] ; } ag . pers [ pos ] = per ; + + ag . n_pers ; } // void A n y a d i r _ P e r s o n a _ O r d ( const Persona & per , Agenda & ag ) { Clave clave ; Crear_Clave ( clave , per ); int pos = B us ca r_ P os ic io n ( clave , ag ); Anyadir_Ord ( ag , pos , per ); } // void Ordenar_ Agenda ( Agenda & ag , Cod_Error & ok ) { ok = OK ; int npers = ag . n_pers ; ag . n_pers = 1; while ( ag . n_pers < npers ) { Persona per = ag . pers [ ag . n_pers ] ; A n y a d i r _ P e r s o n a _ O r d ( per , ag ); } } // void I mp ri mi r _A ge nd a ( const Agenda & ag , Cod_Error & ok ) { for ( int i = 0; i < ag . n_pers ; + + i ) { E s c r i b i r _ P e r s o na ( ag . pers [ i ] ); } ok = OK ; } // // // char Menu ( bool ord ) { char opcion ; cout < < endl ; if ( ord ) { cout < < " Estado : Ordenado " < < endl ; } else { cout < < " Estado : Desordenado " < < endl ; } cout < < " a . - Anadir Persona " < < endl ; cout < < " b . - Buscar Persona " < < endl ; cout < < " c . - Ordenar " < < endl ; cout < < " d . - Imprimir Agenda " < < endl ;

99

cout < < " x . - Salir " < < endl ; do { cout < < " Introduzca Opci n : " ; o cin > > opcion ; } while ( ! ((( opcion > = ' a ' ) & & ( opcion < = ' d ' )) | | ( opcion = = ' x ' ))); return opcion ; } // S u b a l g o r i t m o s void E s c r i b i r _ C o d _ E r r o r ( Cod_Error cod ) { switch ( cod ) { case OK : cout < < " Operaci n correcta " < < endl ; o break ; case AG_LLENA : cout < < " Agenda llena " < < endl ; break ; case NO_ENCONTRADO : cout < < " La persona no se encuentra en la agenda " < < endl ; break ; case YA_EXISTE : cout < < " La persona ya se encuentra en la agenda " < < endl ; break ; } } // P r i n c i p a l int main () { bool ordenada = true ; Agenda ag ; char opcion ; Clave clave ; Persona nuevo ; Cod_Error ok ; Inicializar ( ag ); do { opcion = Menu ( ordenada ); switch ( opcion ) { case ' a ' : cout < < " Introduzca los datos de la Persona " < < endl ; cout < < " ( nombre , dia , mes , anyo , tel ) " < < endl ; Leer_Persona ( nuevo ); A ny ad ir _ Pe rs on a ( nuevo , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); ordenada = false ; break ; case ' b ' : if ( ordenada ) { cout < < " Introduzca Nombre , dia , mes , anyo " < < endl ; Leer_Clave ( clave ); I m p r i m i r _ P e r s o n a _ O r d ( clave , ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); } else { cout < < " Error , la agenda no esta ordenada " < < endl ; } break ; case ' c ' : Orde nar_Agen da ( ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); ordenada = true ; break ; case ' d ' : I mp ri mi r _A ge nd a ( ag , ok ); E s c r i b i r _ C o d _ E r r o r ( ok ); break ; } } while ( opcion ! = ' x ' ); }

100

También podría gustarte