Programacion Fundamentos

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

Fundamentos 

de la programación

RB
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

 Programming. Principles and Practice Using C++


B. Stroustrup. Pearson Education, 2009
 C++: An Introduction to Computing (2ª edición)
J. Adams, S. Leestma, L. Nyhoff. Prentice Hall, 1998
 El lenguaje de programación C++ (Edición especial)
B. Stroustrup. Addison‐Wesley, 2002
 Programación y resolución de problemas con C++
N. Dale, C. Weems. McGraw‐Hill Interamericana, 2007
 Problem Solving, Abstraction, Design Using C++ (3ª edición)
F.L. Friedman, E.B. Koffman. Addison‐Wesley, 2000.
Luis Hernández Yáñez

 Programación en C++ para ingenieros


F. Xhafa et al. Thomson, 2006

Fundamentos de la programación
Programming. Principles and Practice Using C++
Del autor del lenguaje C++, un amplio tutorial que enseña a programar
en C++; hace un uso temprano de conceptos de orientación a objetos y
de la STL, que quedan fuera del temario de este curso

C++: An Introduction to Computing (2ª edición)


Buena introducción a la programación en C++; buena organización de
los contenidos, bien desarrollado y con secciones prácticas

El lenguaje de programación C++ (Edición especial)


Del autor del lenguaje C++, la referencia absoluta sobre el lenguaje C++
en la que consultar dudas y detalles técnicos sobre los elementos del
Luis Hernández Yáñez

lenguaje

Fundamentos de la programación

Programación y resolución de problemas con C++


Un enfoque práctico al desarrollo de programas con C++ con
numerosos ejemplos

Problem Solving, Abstraction, Design Using C++ (3ª edición)


Introducción a la programación en C++ con un enfoque de desarrollo
de software y numerosos casos de estudio

Programación en C++ para ingenieros


Introducción a la programación en C++ con explicaciones sencillas y
una organización clara
Luis Hernández Yáñez

Fundamentos de la programación
Fundamentos de la programación

1
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Informática, computadoras y programación 3
Lenguaje máquina y ensamblador 12
Lenguajes de programación de alto nivel 15
Un poco de historia 19
Programación e Ingeniería del Software 24
El lenguaje de programación C++ 27
Sintaxis de los lenguajes de programación  30
Un primer programa en C++ 35
Herramientas de desarrollo 39
C++: Un mejor C 45
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 3

Informática (Ciencia de la computación)


Conjunto de conocimientos científicos y técnicas
que hacen posible el tratamiento automático
de la información por medio de ordenadores

Computadora
Máquina electrónica, analógica o digital,
dotada de una memoria de gran capacidad
y de métodos de tratamiento de la información,
capaz de resolver problemas matemáticos y lógicos
Luis Hernández Yáñez

mediante la ejecución de programas informáticos

Fundamentos de la programación: Computadoras y programación Página 4
En todas partes y con muchas formas
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 5

Hardware
Componentes que integran
la parte material
de una computadora

Software
Programas, instrucciones
y reglas informáticas
para ejecutar tareas
en una computadora
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 6
Programar
Indicar a la computadora qué es lo que tiene que hacer

Programa
 Secuencia de instrucciones
 Instrucciones que entiende la computadora
 Y que persiguen un objetivo: ¡resolver un problema!
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 7

Trabajo en equipo
Múltiples roles...
 Gestores
 Analistas
Parque Jurásico  Diseñadores
 Programadores
 Probadores
 Administradores de
sistemas
...
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 8
Esquema general

Memoria
temporal

Unidad Central de Proceso
Central Processor Unit

Dispositivos Dispositivos
C.P.U.
de entrada de salida

Teclado Monitor
Ratón Impresora
Escáner Altavoz
Luis Hernández Yáñez

Táctil Almacenamiento …
… permanente

Fundamentos de la programación: Computadoras y programación Página 9

La arquitectura de Von Neumann

Dispositivos de E/S

Una ALU de 2 bits (Wikipedia)
C.P.U. (Procesador)

A.L.U.
Unidad Aritmético‐Lógica
Memoria
Luis Hernández Yáñez

Unidad de Control

Fundamentos de la programación: Computadoras y programación Página 10
La memoria
Memoria Cada celda en una dirección
01 Celdas de 8 / 16 / 32 / 64 bits
02
03
Información volátil
04

Bus  05 1 Bit = 0 / 1
de  06 1 Byte = 8 bits = 1 carácter
datos 07
1 Kilobyte (KB) = 1024 Bytes
08
1 Megabyte (MB) = 1024 KB
. . .
1 Gigabyte (GB) = 1024 MB
1 Terabyte (TB) = 1024 GB
1 Petabyte (PB) = 1024 TB
Luis Hernández Yáñez

Dirección
210 = 1024  1000

Fundamentos de la programación: Computadoras y programación Página 11
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 12
Los procesadores trabajan con ceros y unos (bits)
Unidad de memoria básica: Byte (8 bits)
(2 dígitos hexadecimales: 01011011  0101 1011  5B)

Lenguaje máquina
Códigos hexadecimales que representan instrucciones,
registros de la CPU, direcciones de memoria o datos
Instrucción Significado Lenguaje de bajo nivel
A0 2F Acceder a la celda de memoria 2F Dependiente de la máquina
3E 01 Copiarlo el registro 1 de la ALU Programación difícil
A0 30 Acceder a la celda de memoria 30
Luis Hernández Yáñez

3E 02 Copiarlo en el registro 2 de la ALU


1D Sumar
B3 31 Guardar el resultado en la celda de memoria 31

Fundamentos de la programación: Computadoras y programación Página 13

Nemotécnicos para los códigos hexadecimales:


A0  READ   3E  REG   1D  ADD   …

Mayor legibilidad:
READ 2F Código fuente
REG 01  (lenguaje ensamblador)
READ 30
REG 02 
ADD    Programa
WRITE 31 ensamblador

Lenguaje de nivel medio


Código objeto
Luis Hernández Yáñez

(lenguaje máquina)

Fundamentos de la programación: Computadoras y programación Página 14
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 15

 Más cercanos a los lenguajes natural y matemático


resultado = dato1 + dato2;
 Mayor legibilidad, mayor facilidad de codificación
 Estructuración de datos / abstracción procedimental
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 16
Traducción #include <iostream>
using namespace std;
Código fuente int main()
{
cout << "Hola Mundo!" << endl;
Compiladores: }
return 0;

Compilan y enlazan Compilador


programas completos

Intérpretes: Código objeto 0100010100111010011100…


Compilan, enlazan
y ejecutan instrucción Código
Enlazador objeto de
a instrucción
biblioteca
Luis Hernández Yáñez

Programa Para una arquitectura concreta
ejecutable y un sistema operativo

Fundamentos de la programación: Computadoras y programación Página 17

Genealogía de lenguajes Versiones / Estándares


Prolog
1970 Java
COBOL PL/I C++
1959 1995
1964 1983
FORTRAN C#
1954 CPL C
2000
1963 1971
Python
ALGOL Pascal Modula 1991
1958 1970 1975
BASIC
1964
Ada Eiffel
1979 1986
Simula
Fuente: 1964 Smalltalk
http://www.levenez.com/lang/ Ruby
1971 1993
Luis Hernández Yáñez

Haskell
Lisp Scheme 1987
1958 1975
Logo
1968

Fundamentos de la programación: Computadoras y programación Página 18
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 19

La prehistoria
El ábaco

Siglo XIX (Wikipedia)


Máquina analítica de Charles Babbage

Lady Ada Lovelace


es considerada
la primera
programadora
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 20
Siglo XX
1936 Máquina de Turing
1946 ENIAC: Primera computadora digital
de propósito general
1947 El transistor
ENIAC (Wikipedia)
1953 IBM 650: Primera
computadora a gran escala
1966 ARPANET: Origen de Internet
1967 El disquete
1970 Sistema operativo UNIX
1972 Primer virus informático (Creeper)
Luis Hernández Yáñez

Lenguaje de programación C
1974 Protocolo TCP. Primera red local

Fundamentos de la programación: Computadoras y programación Página 21

1975 Se funda Microsoft


1976 Se funda Apple
1979 Juego Pacman
1981 IBM PC
Sistema operativo MS‐DOS Apple II (Wikipedia)

1983 Lenguaje de programación C++


1984 CD‐ROM
1985 Windows 1.0
1990 Lenguaje HTML
World Wide Web Linux
Luis Hernández Yáñez

IBM PC (Wikipedia)
1991 Sistema operativo Linux

Fundamentos de la programación: Computadoras y programación Página 22
1992 Windows 3.1
1995 Lenguaje de programación Java
DVD
1998 Se funda Google
1999 MSN Messenger

Siglo XXI
2001 Windows XP
Mac OS X
2002 Mozilla Firefox
2007 iPhone
Luis Hernández Yáñez

2008 Android ...

Fundamentos de la programación: Computadoras y programación Página 23
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 24
¿Qué es programar?
Decirle a un tonto muy rápido exactamente lo que tiene que hacer
Especificar la estructura y el comportamiento de un programa,
así como probar que el programa realiza su tarea
adecuadamente y con un rendimiento aceptable

Programa: Transforma entrada en salida

Entrada Programa Salida

Algoritmo: Secuencia de pasos y operaciones que debe realizar


Luis Hernández Yáñez

el programa para resolver el problema


El programa implementa el algoritmo en un lenguaje concreto

Fundamentos de la programación: Computadoras y programación Página 25

La programación es sólo una etapa del proceso de desarrollo

Modelo de desarrollo “en cascada”:

Planificación Recursos necesarios, presupuesto, plan, …

Análisis ¿Qué?

Diseño ¿Cómo?

Programación Implementación

Prueba y depuración
Luis Hernández Yáñez

Mantenimiento

Fundamentos de la programación: Computadoras y programación Página 26
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 27

Bjarne Stroustrup (1983)


Hola Mundo!

#include <iostream>
using namespace std;

int main()
{
cout << "Hola Mundo!" << endl;
// Muestra Hola Mundo!
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Computadoras y programación Página 28
Instrucciones
Datos: literales, variables, tipos
Subprogramas (funciones)
Comentarios
Directivas Directiva
#include <iostream>
...
using namespace std;

Subprograma int main()


{ Dato
Instrucción cout << "Hola Mundo!" << endl;
// Muestra Hola Mundo!
Luis Hernández Yáñez

Comentario
Dato
Instrucción return 0;
}

Fundamentos de la programación: Computadoras y programación Página 29
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 30
Sintaxis y semántica de los lenguajes
Sintaxis
— Reglas que determinan cómo se pueden construir
y secuenciar los elementos del lenguaje

Semántica
— Significado de cada elemento del lenguaje
¿Para qué sirve?
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 31

Especificación
 Lenguajes (BNF)
 Diagramas
Ejemplo: Números enteros (sin decimales)
BNF
<numero entero> ::= <signo opcional><secuencia de dígitos>
<signo opcional> ::= +|‐|<nada>
<secuencia de dígitos> ::= <dígito>|<dígito><secuencia de dígitos>
<dígito> ::= 0|1|2|3|4|5|6|7|8|9 +23   
<nada> ::=  | significa ó
‐159  
1374  
1‐34  
+
3.4   
Luis Hernández Yáñez

0 .. 9 002   

Fundamentos de la programación: Computadoras y programación Página 32
<numero entero> ::= <signo opcional><secuencia de dígitos>
<signo opcional> ::= +|‐|<nada>
<secuencia de dígitos> ::= <dígito>|<dígito><secuencia de dígitos>
<dígito> ::= 0|1|2|3|4|5|6|7|8|9
<nada> ::= 
+23
<numero entero> ::= <signo opcional><secuencia de dígitos>
::= +<secuencia de dígitos> ::= +<dígito><secuencia de dígitos> 
::= +2<secuencia de dígitos> ::= +2<dígito> ::= +23
1374

<numero entero> ::= <signo opcional><secuencia de dígitos> 
::= <secuencia de dígitos> ::= <dígito><secuencia de dígitos> 
::= 1<secuencia de dígitos> ::= 1<dígito><secuencia de dígitos> 
::= 13<secuencia de dígitos> ::= 13<dígito><secuencia de dígitos>  
::= 137<secuencia de dígitos> ::= 137<dígito> ::= 1374
Luis Hernández Yáñez

1‐34
<numero entero> ::= <signo opcional><secuencia de dígitos>
::= <secuencia de dígitos> ::= <dígito><secuencia de dígitos> 
::= 1<secuencia de dígitos> ::= ERROR (‐ no es <dígito>)

Fundamentos de la programación: Computadoras y programación Página 33

+23 +


0 .. 9 +23 

1374 +


0 .. 9 1374 

1‐34 + ?
0 .. 9 1‐ 
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 34
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 35

Hola Mundo!
Un programa que muestra un saludo en la pantalla:
#include <iostream>
using namespace std;

int main()
// main() es donde empieza la ejecución
{
cout << "Hola Mundo!" << endl; // Muestra Hola Mundo!
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Computadoras y programación Página 36
Análisis del programa
Biblioteca

Directiva #include <iostream> Espacio de nombres

Instrucción using namespace std;
Coloreado sintáctico
Tipo Palabras reservadas

Declaración int main() Cabecera de la función

{
Bloque de código

Variable Cadena de caracteres Constante


Instrucción cout << "Hola Mundo!" << endl;
Operador Operador
Datos literales
return 0;
Luis Hernández Yáñez

Instrucción
Número
} Cuerpo de la función

Las instrucciones terminan en ;

Fundamentos de la programación: Computadoras y programación Página 37

Hola Mundo!
Casi todo es infraestructura
Sólo
cout << "Hola Mundo!" << endl
hace algo palpable

La infraestructura (notación, bibliotecas y otro soporte)


hace nuestro código simple, completo, confiable y eficiente

¡El estilo importa!


Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 38
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 39

Editor
 Bloc de notas, Wordpad, Word, Writer, Gedit, Kwrite, …
(texto simple, sin formatos)
 Editores específicos: coloreado sintáctico
 Recomendación: Notepad++

Instalación y uso:
Sección
Herramientas de desarrollo
en el Campus Virtual
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 40
hola.cpp Compilador hola.obj
(código fuente) (código objeto)

Código objeto de
Enlazador
la biblioteca iostream

Hola Mundo!

Cargador hola.exe
Luis Hernández Yáñez

(ejecutable)

Fundamentos de la programación: Computadoras y programación Página 41

Compilador
 Importante: C++ estándar
 Recomendación: GNU G++ (MinGW en Windows)

Instalación y uso:
Sección
Herramientas de desarrollo
en el Campus Virtual
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 42
Entornos de desarrollo
 Para editar, compilar y probar el código del programa
 Recomendaciones:
— Windows: MS Visual Studio / C++ Express o Eclipse
— Linux: Netbeans o Eclipse

Instalación y uso:
Sección
Herramientas de desarrollo
en el Campus Virtual
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 43

¿Qué hace el programa?


 La ejecución del programa siempre empieza en main()
 Se ejecutan las instrucciones en secuencia de principio a fin

Inicio
Pantalla (cout)
_ Hola Mundo!
cout << "Hola Mundo!" << endl; _
Muestra Hola Mundo!
en la pantalla y salta de línea
return 0;
Devuelve 0 como código
Luis Hernández Yáñez

de terminación del programa


Fin

Fundamentos de la programación: Computadoras y programación Página 44
Luis Hernández Yáñez

Fundamentos de la programación: Computadoras y programación Página 45

El lenguaje C
 Lenguaje creado por Dennis M. Ritchie en 1972
 Lenguaje de nivel medio:
— Estructuras típicas de los lenguajes de alto nivel

— Construcciones para control a nivel de máquina

 Lenguaje sencillo (pocas palabras reservadas)


 Lenguaje estructurado (no estrictamente estructurado en bloques)
 Compartimentalización de código (funciones) y datos (ámbitos)
 Componente estructural básico: la función (subprograma)
 Programación modular
Luis Hernández Yáñez

 Distingue entre mayúsculas y minúsculas


 Palabras reservadas (o clave): en minúsculas

Fundamentos de la programación: Computadoras y programación Página 46
Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Computadoras y programación Página 47
Fundamentos de la programación

2
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Un ejemplo de programación 50 Operadores relacionales 177


El primer programa en C++ 64 Toma de decisiones (if) 180
Las líneas de código del programa 80 Bloques de código 183
Cálculos en los programas 86 Bucles (while) 186
Variables 92 Entrada/salida por consola 190
Expresiones 98 Funciones definidas
Lectura de datos desde el teclado 108 por el programador 199
Resolución de problemas 119
Los datos de los programas 127
Identificadores 129
Tipos de datos 133
Declaración y uso de variables 142
Instrucciones de asignación 147
Operadores 152
Más sobre expresiones 160
Luis Hernández Yáñez

Constantes 167
La biblioteca cmath 171
Operaciones con caracteres 174

Fundamentos de la programación: Tipos e instrucciones I
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 50

Una computadora de un coche


Instrucciones que entiende:
<instrucción> ::= <inst> ;
<inst> ::= Start | Stop | <avanzar>
<avanzar> ::= Go <dirección> <num> Blocks
<dirección> ::= North | East | South | West
<num> ::= 1 | 2 | 3 | 4 | 5
Ejemplos:
Start;
Luis Hernández Yáñez

Go North 3 Blocks;
Stop;

Fundamentos de la programación: Tipos e instrucciones I Página 51
Sintaxis del lenguaje de programación
= Literales
Start

instrucción Stop ;

avanzar

avanzar Go dirección num Blocks

North 1

East 2
Luis Hernández Yáñez

dirección num 3
South
4
West
5

Fundamentos de la programación: Tipos e instrucciones I Página 52

N
El problema a resolver
Estando el coche en la posición A,
conseguir llegar al Cine Tívoli (B) B

¿Qué pasos hay que seguir?


Arrancar
Ir un bloque al Norte
Ir dos bloques al Este
Ir cinco bloques al Norte A
Ir dos bloques al Este
Parar Bloque:
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 53
N
El algoritmo
Secuencia de pasos que hay que

seguir para resolver el problema

1.‐ Arrancar
2.‐ Ir un bloque al Norte
3.‐ Ir dos bloques al Este
4.‐ Ir cinco bloques al Norte
5.‐ Ir dos bloques al Este
6.‐ Parar
Luis Hernández Yáñez

Esos pasos sirven tanto para


una persona como para una computadora.

Fundamentos de la programación: Tipos e instrucciones I Página 54

N
El programa
Instrucciones escritas en
el lenguaje de programación B

Start;
Go North 1 Blocks;
Go East 2 Blocks;
Go North 5 Blocks;
A
Go East 2 Blocks;
Stop;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 55
El programa
Escribimos el código del programa en un editor
y lo guardamos en un archivo:

Stat;
Go North 1 Blocks Copiamos el archivo
Go East Blocks; en una llave USB
Go Noth 5 Blocks; y lo llevamos al coche
Go West 2 Blocks;
Luis Hernández Yáñez

Stop;

Fundamentos de la programación: Tipos e instrucciones I Página 56

La compilación
Introducimos la llave USB en el coche
y pulsamos el botón de ejecutar el programa:

Stat;
‐‐‐‐^ Unknown word.
Go North 1 Blocks
‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐^ ; missing.
Go East Blocks; Errores
‐‐‐‐‐‐‐‐^ Number missing.
de sintaxis
Go Noth 5 Blocks;
‐‐‐‐‐‐‐^ Unknown word.
Go West 2 Blocks;
Luis Hernández Yáñez

Stop;
There are errors. Impossible to run the program.

Fundamentos de la programación: Tipos e instrucciones I Página 57
Depuración
Editamos el código para corregir los errores sintácticos:

Stat; Start;
Go North 1 Blocks Go North 1 Blocks;
Go East Blocks; Go East 3 Blocks;
Go Noth 5 Blocks; Go North 5 Blocks;
Go West 2 Blocks; Go West 2 Blocks;
Stop; Stop;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 58

N
La ejecución
Se realiza lo que pide
cada instrucción: B
Start;
Go North 1 Blocks;
Go East 3 Blocks;
!

Error de ejecución
Luis Hernández Yáñez

¡Una instrucción no se puede ejecutar!

Fundamentos de la programación: Tipos e instrucciones I Página 59
Depuración
Editamos el código para arreglar el error de ejecución:

Start; Start;
Go North 1 Blocks; Go North 1 Blocks;
Go East 3 Blocks; Go East 2 Blocks;
Go North 5 Blocks; Go North 5 Blocks;
Go West 2 Blocks; Go West 2 Blocks;
Stop; Stop;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 60

N
La ejecución
Se realiza lo que pide ?
cada instrucción: B
Start;
Go North 1 Blocks;
Go East 2 Blocks;
Go North 5 Blocks;
Go West 2 Blocks;
Stop;
Luis Hernández Yáñez

Error lógico
¡El programa no llega al resultado deseado!

Fundamentos de la programación: Tipos e instrucciones I Página 61
Depuración
Editamos el código para arreglar el error lógico:

Start; Start;
Go North 1 Blocks; Go North 1 Blocks;
Go East 2 Blocks; Go East 2 Blocks;
Go North 5 Blocks; Go North 5 Blocks;
Go West 2 Blocks; Go East 2 Blocks;
Stop; Stop;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 62

N
La ejecución

Se realiza lo que pide
cada instrucción:

Start;
Go North 1 Blocks;
Go East 2 Blocks;
Go North 5 Blocks;
Go East 2 Blocks;
Stop;
Luis Hernández Yáñez

¡Conseguido!

Fundamentos de la programación: Tipos e instrucciones I Página 63
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 64

Hola Mundo!
De vuelta en el programa que muestra un saludo en la pantalla:
#include <iostream>
using namespace std;

int main() // main() es donde empieza la ejecución


{
cout << "Hola Mundo!" << endl;

return 0;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 65
Hola Mundo!
La única instrucción que produce algo tangible:
#include <iostream>
using namespace std;

int main() // main() es donde empieza la ejecución
{
cout << "Hola Mundo!" << endl;

return 0;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 66

cout (iostream) character output stream


Visualización en la pantalla: operador << (insertor)

cout << "Hola Mundo!" << endl; 

cout << "Hola Mundo!" << endl;

Hola Mundo!
_

endl  end line


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 67
Pantalla en modo texto
 Líneas de 80 caracteres (textos)

Aplicación en modo texto

80 caracteres
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 68

Ventanas de consola o terminal


Las aplicaciones en modo texto se ejecutan dentro de ventanas:
 Windows: ventanas de consola (Símbolo del sistema)
 Linux: ventanas de terminal

H o l a M u n d o !

...
Luis Hernández Yáñez

...

Cursor parpadeante: Donde se colocará el siguiente carácter.

Fundamentos de la programación: Tipos e instrucciones I Página 69
El insertor << cout << ...; 
Inserta textos en la pantalla de modo texto
Representación textual de los datos
A partir de la posición del cursor
Line wrap (continúa en la siguiente línea si no cabe)
Se pueden encadenar:
cout << ... << ... << ...;

Recuerda: las instrucciones terminan en ;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 70

Con el insertor << podemos mostrar...


 Cadenas de caracteres literales
Textos encerrados entre comillas dobles: "..."
cout << "Hola Mundo!";
¡Las comillas no se muestran!
 Números literales
Con o sin decimales, con signo o no: 123, ‐37, 3.1416, ...
cout << "Pi = " << 3.1416;
Se muestran los caracteres que representan el número
Luis Hernández Yáñez

 endl ¡Punto decimal, NO coma!

Fundamentos de la programación: Tipos e instrucciones I Página 71
El programa principal
La función main(): donde comienza la ejecución...
#include <iostream>
using namespace std;

int main() // main() es donde empieza la ejecución


{
cout << "Hola Mundo!" << endl;
return 0;
}
Luis Hernández Yáñez

Contiene las instrucciones que hay que ejecutar

Fundamentos de la programación: Tipos e instrucciones I Página 72

El programa principal
La función main():

Tipo de la función (int = entero): Tipo de valor que devuelve


Nombre de la función
int main() ¡Es una función!
{
...
return 0; Cuerpo de la función (bloque de código)
}
Luis Hernández Yáñez

return 0; Devuelve el resultado (0) de la función

Fundamentos de la programación: Tipos e instrucciones I Página 73
Documentando el código...
Comentarios (se ignoran):
#include <iostream>
using namespace std;

int main()   // main() es donde empieza la ejecución
{
cout << "Hola Mundo!" << endl;
...
Hasta el final de la línea: // Comentario de una línea
Luis Hernández Yáñez

De varias líneas: /* Comentario de varias


líneas seguidas */

Fundamentos de la programación: Tipos e instrucciones I Página 74

La infraestructura
Código para reutilizar:
#include <iostream> Una directiva: empieza por #
using namespace std;

int main()   // main() es donde empieza la ejecución
{
cout << "Hola Mundo!" << endl; 
return 0;
}
Luis Hernández Yáñez

Bibliotecas de funciones a nuestra disposición

Fundamentos de la programación: Tipos e instrucciones I Página 75
Bibliotecas
Se incluyen con la directiva #include:
#include <iostream>
(Utilidades de entrada/salida por consola)
Para mostrar o leer datos hay que incluir la biblioteca iostream

Espacios de nombres
En iostream hay espacios de nombres; ¿cuál queremos?
#include <iostream>
using namespace std; Es una instrucción: termina en ;
Luis Hernández Yáñez

Siempre usaremos el espacio de nombres estándar (std)


Muchas bibliotecas no tienen espacios de nombres

Fundamentos de la programación: Tipos e instrucciones I Página 76

Compilación y enlace
A menudo en un paso

hola.cpp Compilador hola.obj


(código fuente) (código objeto)

Código objeto de
Enlazador
la biblioteca iostream

Hola Mundo!
Luis Hernández Yáñez

Cargador hola.exe
(ejecutable)

Fundamentos de la programación: Tipos e instrucciones I Página 77
Elementos del programa
Biblioteca

Directiva #include <iostream> Espacio de nombres

Instrucción using namespace std; Coloreado sintáctico:


Directivas              Tipos
Palabras reservadas generales
Tipo Palabras reservadas Datos literales   Comentarios

Declaración int main() Cabecera de la función

{
Bloque de código

Variable Cadena de caracteres Constante


Instrucción cout << "Hola Mundo!" << endl;
Operador Operador
Datos literales
Instrucción return 0;
Luis Hernández Yáñez

Número
} Cuerpo de la función

Las instrucciones terminan en ;

Fundamentos de la programación: Tipos e instrucciones I Página 78

Uso de espacio en blanco


Separación de elementos por uno o más espacios en blanco
(espacios, tabuladores y saltos de línea)
El compilador los ignora
#include <iostream> using namespace std;
int main(){cout<<"Hola Mundo!"<<endl; 
return 0;}
#include <iostream>
using namespace std;

int main()
{   ¿Cuál se lee mejor?
cout << "Hola Mundo!" << endl;
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 79
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 80

Programa con E/S por consola


Una plantilla para empezar:
#include <iostream>
using namespace std;

int main()
{

¡Tu código aquí!


Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 81
... recitado en la consola
Mostrar los textos con cout <<:
#include <iostream>
using namespace std;

int main()
{
cout << "En un lugar de la Mancha," << endl;
cout << "de cuyo nombre no quiero acordarme," << endl;
cout << "no ha mucho tiempo que vivía un hidalgo de los de 
lanza en astillero, ..." << endl;
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 82

Introducción del código del programa


Terminamos cada línea de código con un salto de línea (↲):
#include <iostream> ↲
using namespace std; ↲

int main() ↲
{↲
cout << "En un lugar de la Mancha," << endl; ↲
cout << "de cuyo nombre no quiero acordarme," << endl; ↲
cout << "no ha mucho tiempo que vivía un hidalgo de los de 
lanza en astillero, ..." << endl; ↲
Luis Hernández Yáñez

return 0; ↲
}↲

Fundamentos de la programación: Tipos e instrucciones I Página 83
Introducción del código del programa
No hay que partir una cadena literal entre dos líneas:
cout << "no ha mucho tiempo que vivía un hidalgo de ↲
los de lanza en astillero, ..." << endl; ↲

¡La cadena no termina (1ª línea)!


¡No se entiende los (2ª línea)!
Luis Hernández Yáñez

Veamos cómo nos muestra los errores el compilador...

Fundamentos de la programación: Tipos e instrucciones I Página 84

Mantenimiento y reusabilidad
 Usa espacio en blanco para separar los elementos:
cout << "En un lugar de la Mancha," << endl;

mejor que
cout<<"En un lugar de la Mancha,"<<endl;

 Usa sangría (indentación) para el código de un bloque:


{
Tab cout << "En un lugar de la Mancha," << endl;
ó ...
3 esp. return 0;
Luis Hernández Yáñez

}
¡El estilo importa!
Fundamentos de la programación: Tipos e instrucciones I Página 85
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 86

Operadores aritméticos
+ Suma
‐ Resta
* Multiplicación
/ División
Operadores binarios
operando_izquierdo operador   operando_derecho

Operación Resultado
3 + 4 7
2.56 ‐ 3 ‐0.44
Luis Hernández Yáñez

143 * 2 286


45.45 / 3 15.15

Fundamentos de la programación: Tipos e instrucciones I Página 87
Números literales (concretos)
 Enteros: sin parte decimal
Signo negativo (opcional) + secuencia de dígitos
3   143   ‐12   67321   ‐1234

No se usan puntos de millares


 Reales: con parte decimal
Signo negativo (opcional) + secuencia de dígitos
+ punto decimal + secuencia de dígitos
3.1416  357.0  ‐1.333  2345.6789  ‐404.1
Luis Hernández Yáñez

Punto decimal (3.1416), NO coma (3,1416)

Fundamentos de la programación: Tipos e instrucciones I Página 88

cálculos.cpp

Ejemplo
#include <iostream>
using namespace std;

int main()
Un texto Un número
{
cout << "133 + 1234 = " << 133 + 1234 << endl;
cout << "1234 ‐ 111.5 = " << 1234 ‐ 111.5 << endl;
cout << "34 * 59 = " << 34 * 59 << endl;
cout << "3.4 * 5.93 = " << 3.4 * 5.93 << endl;
cout << "500 / 3 = " << 500 / 3 << endl; // Div. entera
cout << "500.0 / 3 = " << 500.0 / 3 << endl; // Div. real
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 89
División entera
Luis Hernández Yáñez

División real

Fundamentos de la programación: Tipos e instrucciones I Página 90

¿División entera o división real?


Ambos operandos enteros  División entera
Algún operando real  División real

División Resultado
500 / 3 166
500.0 / 3 166.667
500 / 3.0 166.667
500.0 / 3.0 166.667

Comprueba siempre que el tipo de división sea el que quieres


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 91
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 92

Datos que se mantienen en memoria


Variable: dato que se accede por medio de un nombre
Dato literal: un valor concreto
Variable: puede cambiar de valor (variar)
edad = 19; // variable edad y literal 19

Las variables deben ser declaradas


¿Qué tipo de dato queremos mantener?
 Valor numérico sin decimales (entero): tipo int
 Valor numérico con decimales (real): tipo double
Luis Hernández Yáñez

Declaración: tipo nombre;

Fundamentos de la programación: Tipos e instrucciones I Página 93
Declaración de variables tipo nombre;
int cantidad;
Memoria
double precio; cantidad ?
Se reserva espacio suficiente precio ?

...
LAS VARIABLES NO SE INICIALIZAN
No se deben usar hasta que se les haya dado algún valor

¿Dónde colocamos las declaraciones?


Siempre, antes del primer uso
Luis Hernández Yáñez

Habitualmente al principio de la función

Fundamentos de la programación: Tipos e instrucciones I Página 94

Declaración de variables Memoria

#include <iostream> cantidad ?
using namespace std;
precio ?

int main()
total ?
{
int cantidad; ...
double precio, total;

Podemos declarar varias de un mismo tipo


return 0; separando los nombres con comas
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 95
Capacidad de las variables
int
‐2.147.483.648 ... 2.147.483.647
‐2147483648 .. 2147483647

double
2,23 x 10‐308 ... 1,79 x 10+308 y sus negativos

[+|‐] 2.23e‐308 .. 1.79e+308 Notación científica


Luis Hernández Yáñez

Problemas de precisión

Fundamentos de la programación: Tipos e instrucciones I Página 96

Asignación de valores a las variables (operador =)


variable = expresión; Instrucción: termina en ;

cantidad = 12; // int cantidad  12


precio = 39.95; // double
total = cantidad * precio; // Asigna 479.4

Concordancia de tipos: cantidad = 12.5;


Luis Hernández Yáñez

¡¡¡A la izquierda del = debe ir siempre una variable!!!

Fundamentos de la programación: Tipos e instrucciones I Página 97
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 98

Expresiones
Secuencias de operandos y operadores
operando operador operando operador operando ...
total = cantidad * precio * 1.18;

Expresión
A igual prioridad se evalúan de izquierda a derecha

Paréntesis para forzar ciertas operaciones


total = cantidad1 + cantidad2 * precio;
total = (cantidad1 + cantidad2) * precio; 
Luis Hernández Yáñez

Unos operadores se evalúan antes que otros

Fundamentos de la programación: Tipos e instrucciones I Página 99
Precedencia de los operadores
cantidad1 = 10;
cantidad2 = 2;
precio = 40.0;

* y / se evalúan antes que + y ‐

total = cantidad1 + cantidad2 * precio;
* antes que +  10 + 2 * 40,0  10 + 80,0  90,0

total = (cantidad1 + cantidad2) * precio;
+ antes que *  (10 + 2) * 40,0  12 * 40,0  480,0
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 100

variables.cpp

Ejemplo de uso de variables y expresiones


#include <iostream>
using namespace std;

int main()
{
int cantidad;
double precio, total;
cantidad = 12;
precio = 39.95;
total = cantidad * precio;
cout << cantidad << " x " << precio << " = "
<< total << endl;
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 101


Ejemplo de uso de variables Memoria

#include <iostream> cantidad ?
using namespace std;
precio ?
int main()
{
total ?
int cantidad;
double precio, total;
...
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 102

Ejemplo de uso de variables Memoria

#include <iostream> cantidad 12
using namespace std;
precio ?
int main()
{
total ?
int cantidad;
double precio, total;
cantidad = 12; ...
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 103


Ejemplo de uso de variables Memoria

#include <iostream> cantidad 12
using namespace std;
precio 39.95
int main()
{
total ?
int cantidad;
double precio, total;
cantidad = 12; ...
precio = 39.95;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 104

Ejemplo de uso de variables Memoria

#include <iostream> cantidad 12
using namespace std;
precio 39.95
int main()
{
total 479.4
int cantidad;
double precio, total;
cantidad = 12; ...
precio = 39.95;
total = cantidad * precio;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 105


Ejemplo de uso de variables Memoria

#include <iostream> cantidad 12
using namespace std;
precio 39.95
int main()
{
total 479.4
int cantidad;
double precio, total;
cantidad = 12; ...
precio = 39.95;
total = cantidad * precio;
cout << cantidad << " x " << precio << " = "
<< total << endl;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 106

Ejemplo de uso de variables


#include <iostream>
using namespace std;

int main()
{
int cantidad;
double precio, total;
cantidad = 12;
precio = 39.95;
total = cantidad * precio;
cout << cantidad << " x " << precio << " = "
<< total << endl;
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 107


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 108

cin (iostream) character input stream


Lectura de valores de variables: operador >> (extractor)

cin >> cantidad; 
Memoria
cin >> cantidad; cantidad 12
?
...

12
1 2 ↲ _
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 109


El extractor >> cin >> variable; 
Transforma los caracteres introducidos en datos
Cursor parpadeante: lugar de lectura del siguiente carácter
La entrada termina con Intro (cursor a la siguiente línea)

¡El destino del extractor debe ser SIEMPRE una variable!

Se ignoran los espacios en blanco iniciales


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 110

Lectura de valores enteros (int)


Se leen dígitos hasta encontrar un carácter que no lo sea
12abc↲ 12 abc↲ 12   abc↲ 12↲
Se asigna el valor 12 a la variable
El resto queda pendiente para la siguiente lectura
Recomendación: Lee cada variable en una línea 12↲

Lectura de valores reales (double)


Se leen dígitos, el punto decimal y otros dígitos
39.95.5abc↲ 39.95 abc↲ 39.95↲
Luis Hernández Yáñez

Se asigna el valor 39,95 a la variable; el resto queda pendiente


Recomendación: Lee cada variable en una línea 39.95↲

Fundamentos de la programación: Tipos e instrucciones I Página 111


¿Qué pasa si el usuario se equivoca?
El dato no será correcto
Aplicación profesional: código de comprobación y ayuda
Aquí supondremos que los usuarios no se equivocan
En ocasiones añadiremos comprobaciones sencillas

Para evitar errores, lee cada dato en una instrucción aparte


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 112

¿Qué pasa si el usuario se equivoca?


int cantidad; ¡Amigable con el usuario!
double precio, total; ¿Qué tiene que introducir?
cout << "Introduce la cantidad: ";
cin >> cantidad;
cout << "Introduce el precio: ";
cin >> precio;
cout << "Cantidad: " << cantidad << endl;
cout << "Precio: " << precio << endl;
Luis Hernández Yáñez

No se puede leer un entero  0 para cantidad y Error


La lectura del precio falla: precio no toma valor (basura)

Fundamentos de la programación: Tipos e instrucciones I Página 113


¿Qué pasa si el usuario se equivoca?

12 para cantidad
No se puede leer un real
 0 para precio y Error

12 para cantidad
.5  0,5 para precio
Lo demás queda pendiente

¡¡¡Lectura correcta!!!
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 114

División de dos números


Pedir al usuario dos números y mostrarle el resultado
de dividir el primero entre el segundo
Algoritmo.‐
Datos / cálculos
1. Pedir el numerador
Variable numerador (double)
2. Pedir el denominador
Variable denominador (double)
3. Realizar la división, guardando el resultado
Variable resultado (double)
Luis Hernández Yáñez

resultado = numerador / denominador
4. Mostrar el resultado

Fundamentos de la programación: Tipos e instrucciones I Página 115


Entrada‐Proceso‐Salida
Muchos programas se ajustan a un sencillo esquema:

Declaraciones Entrada Procesamiento Salida

1. Leer numerador
2. Leer denominador

3. Calcular división en resultado


Luis Hernández Yáñez

4. Mostrar resultado

Fundamentos de la programación: Tipos e instrucciones I Página 116

División de dos números


Pedir al usuario dos números y mostrarle el resultado de dividir el
primero entre el segundo.
1. Leer numerador

cin >> numerador;
2. Leer denominador

cin >> denominador;
3. Calcular división en resultado

resultado = numerador / denominador;
Luis Hernández Yáñez

4. Mostrar resultado

cout << resultado;

Fundamentos de la programación: Tipos e instrucciones I Página 117


División de dos números división.cpp

#include <iostream> Numerador: _129
using namespace std; _
Denominador: _
2
_
Resultado: _
64.5
64.5
int main() _
{
Declaraciones double numerador, denominador, resultado;
cout << "Numerador: ";
cin >> numerador;
Entrada
cout << "Denominador: ";
cin >> denominador;
Procesamiento resultado = numerador / denominador;
Luis Hernández Yáñez

Salida cout << "Resultado: " << resultado << endl;


return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 118


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 119


Problema
Dadas la base y la altura de un triángulo, mostrar su área

Refinamiento

Mostrar en la pantalla un texto que pida la base del triángulo. El usuario


introducirá el valor con el teclado. Mostrar en la pantalla un texto que
pida la altura del triángulo. El usuario introducirá el valor con el teclado.
Se calculará el área del triángulo y se mostrará en la pantalla.
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 120

Objetos: Datos que maneja el programa


variable cin
cout cadena literal

Mostrar en la pantalla un texto que pida la base del triángulo. El usuario


introducirá la base con el teclado. Mostrar en la pantalla un texto que
pida la altura del triángulo. El usuario introducirá la altura con el
teclado. Se calculará el área del triángulo y se mostrará en la pantalla.

cadena literal variable


variable
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 121


Datos que maneja el programa: tipos
Objeto Tipo ¿Varía? Nombre
Pantalla Variable cout
"Introduzca la base del triángulo: " Constante ninguno
Base del triángulo double Variable base
Teclado Variable cin
"Introduzca la altura del triángulo: " Constante ninguno
Altura del triángulo double Variable altura
Área del triángulo double Variable area
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 122

Operaciones (acciones)

cout << ... cin >> ...

Mostrar en la pantalla un texto que pida la base del triángulo. El usuario


introducirá la base con el teclado. Mostrar en la pantalla un texto que
pida la altura del triángulo. El usuario introducirá la altura con el
teclado. Se calculará el área del triángulo y se mostrará en la pantalla.

area = base * altura / 2
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 123


Secuencia de acciones que ha de realizar el programa
para conseguir resolver el problema

1. Mostrar en la pantalla el texto que pida la base del triángulo


2. Leer del teclado el valor para la base del triángulo
3. Mostrar en la pantalla el texto que pida la altura
4. Leer del teclado el valor para la altura del triángulo
5. Calcular el área del triángulo
6. Mostrar el área del triángulo
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 124

#include <iostream>
using namespace std;
int main()
{
Declaraciones
1. Mostrar en la pantalla el texto que pida la base del triángulo

Algoritmo 2. Leer del teclado el valor para la base del triángulo

traducido 3. Mostrar en la pantalla el texto que pida la altura del triángulo


a código 4. Leer del teclado el valor para la altura del triángulo
en C++ 5. Calcular el área del triángulo
6. Mostrar el área del triángulo
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 125


triángulo.cpp

El programa: implementación
#include <iostream>
using namespace std; ¿triβngulo?

int main()
{
double base, altura, area;                // Declaraciones
cout << "Introduzca la base del triángulo: ";         // 1
cin >> base;                                          // 2
cout << "Introduzca la altura del triángulo: ";       // 3
cin >> altura;                                        // 4
area = base * altura / 2;                             // 5
cout << "El área de un triángulo de base " << base // 6
<< " y altura " << altura << " es: " << area << endl;
Luis Hernández Yáñez

return 0;
} Recuerda: las instrucciones terminan en ;

Fundamentos de la programación: Tipos e instrucciones I Página 126


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 127


Variabilidad de los datos
"Introduzca la base del triángulo: "
3.141592653589

Literales

Constantes

Con nombre
Datos

Pi = 3.141592653589
Variables
Luis Hernández Yáñez

base, altura, area Identificadores

Fundamentos de la programación: Tipos e instrucciones I Página 128


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 129


 palabras reservadas

Para variables y constantes con nombre


— Nombre de un dato (para accederlo/modificarlo)

— Deben ser descriptivos

Sintaxis:

a..z, A..Z, _

0..9, a..z, A..Z, _

cantidad  prrecio total  base  altura  area numerador


Luis Hernández Yáñez

Al menos 32 caracteres significativos

¡Ni eñes ni vocales acentuadas!

Fundamentos de la programación: Tipos e instrucciones I Página 130

Palabras reservadas del lenguaje C++


asm auto  bool  break  case  catch  char  class  const  
const_cast continue  default  delete  do  double  
dynamic_cast else  enum  explicit  extern  false  
float  for  friend  goto if  inline  int  long  
mutable  namespace  new  operator  private  protected  
public  register  reinterpret_cast return  short  
signed  sizeof  static  static_cast struct  switch  
template  this  throw  true  try  typedef  typeid
Luis Hernández Yáñez

typename union  unsigned  using  virtual  void  
volatile  while

Fundamentos de la programación: Tipos e instrucciones I Página 131


a..z, A..Z, _

¿Qué identificadores son válidos? 0..9, a..z, A..Z, _

balance  interesAnual 
_base_imponible  años 
EDAD12  salario_1_mes 
cálculoNómina 
__edad

valor%100  AlgunValor 

100caracteres  valor? 
Luis Hernández Yáñez

_12_meses
 ____valor

Fundamentos de la programación: Tipos e instrucciones I Página 132
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 133


Tipos 125
true
Cada dato, de un tipo concreto 'a'
3.14159
Cada tipo establece: "Hola"
— El conjunto (intervalo) de valores válidos
— El conjunto de operaciones que se pueden realizar

Expresiones con datos de distintos tipos (compatibles):


Transformación automática de tipos (promoción de tipo)
Luis Hernández Yáñez

Anexo del Tema 2: detalles técnicos

Fundamentos de la programación: Tipos e instrucciones I Página 134

int
Números enteros (sin decimales)
float
1363, ‐12, 49 
Números reales 12.45, ‐3.1932, 1.16E+02
double
Números reales (mayores intervalo y precisión)
char

Caracteres 'a' , '{', '\t'
bool
Valores lógicos (verdadero/falso) true, false
string
Cadenas de caracteres (biblioteca string) "Hola Mundo!"
Luis Hernández Yáñez

void
Nada, ausencia de tipo, ausencia de dato (funciones)

Fundamentos de la programación: Tipos e instrucciones I Página 135


Caracteres
Intervalo de valores: Juego de caracteres (ASCII) 1 byte
Literales:
'a'   '%'   '\t'
Constantes de barra invertida (o secuencias de escape):
Caracteres de control
'\t' = tabulador '\n' = salto de línea …

ISO‐8859‐1
Luis Hernández Yáñez

(ASCII extendido: códigos 128..255)


ASCII (códigos 32..127)

Fundamentos de la programación: Tipos e instrucciones I Página 136

Valores lógicos
Sólo dos valores posibles:
— Verdadero (true)

— Falso (false)

Literales:
true   false

Cualquier número distinto de 0 es equivalente a true


El 0 es equivalente a false
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 137


C++ distingue entre mayúsculas y minúsculas

int: palabra reservada de C++ para declarar datos enteros

Int, INT o inT no son palabras reservadas de C++

true: palabra reservada de C++ para el valor verdadero

True o TRUE no son palabras reservadas de C++


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 138

Cadenas de caracteres
"Hola"  "Introduce el numerador: "  "X142FG5TX?%A"

" "
char
Secuencias de caracteres
Programas con variables de tipo string:
#include <string>
using namespace std;

Las comillas tipográficas (apertura/cierre) “…” NO sirven


Luis Hernández Yáñez

Asegúrate de utilizar comillas rectas: "…"

Fundamentos de la programación: Tipos e instrucciones I Página 139


tipos.cpp

#include <iostream>
#include <string>
using namespace std; // Un solo using... para ambas bibliotecas

int main()
{
int entero = 3; // Podemos asignar (inicializar) al declarar
double real = 2.153;
char caracter = 'a';
bool cierto = true;
string cadena = "Hola";
cout << "Entero: " << entero << endl;
cout << "Real: " << real << endl;
cout << "Carácter: " << caracter << endl;
cout << "Booleano: " << cierto << endl;
Luis Hernández Yáñez

cout << "Cadena: " << cadena << endl;

return 0; ¿Cuántos números hay en total en el programa?


} ¿Y caracteres? ¿Y cadenas? ¿Y booleanos?
Fundamentos de la programación: Tipos e instrucciones I Página 140

— signed / unsigned : con signo (por defecto) / sin signo


— short / long : menor / mayor intervalo de valores

Tipo Intervalo
int ‐2147483648 .. 2147483647
unsigned int 0 .. 4294967295
short int ‐32768 .. 32768
unsigned short int 0 .. 65535
long int ‐2147483648 .. 2147483647
unsigned long int 0 .. 4294967295
+|‐ 2.23e‐308 .. 1.79e+308
Luis Hernández Yáñez

double
long double +|‐ 3.37E‐4932 .. 1.18E+4932

Fundamentos de la programación: Tipos e instrucciones I Página 141


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 142

[modificadores] tipo lista_de_variables;
Opcional
lista_de_variables Identificador

int i, j, l; ,
short int unidades;
unsigned short int monedas;
double balance, beneficio, perdida;

Programación con buen estilo:


Identificadores descriptivos
Espacio tras cada coma
Luis Hernández Yáñez

Nombres de las variables en minúsculas


(Varias palabras: capitaliza cada inicial: interesPorMes)

Fundamentos de la programación: Tipos e instrucciones I Página 143


Se reserva memoria suficiente para cada tipo de dato

int inicio; Memoria


01

short int unidades; inicio
02

03
double balance; 04

05
unidades
06

07
balance
08

09

10

11

12

13
Luis Hernández Yáñez

14

15

. . .

Fundamentos de la programación: Tipos e instrucciones I Página 144

¡En C++ las variables no se inicializan automáticamente!


¡Una variable debe ser haber sido inicializada antes de ser accedida!
¿Cómo se inicializa una variable?
— Al leer su valor (cin >>)

— Al asignarle un valor (instrucción de asignación)

— Al declararla

Inicialización en la propia declaración:

… Identificador = Expresión Expresión: valor compatible

int i = 0, j, l = 26;
Luis Hernández Yáñez

En particular, una expresión 
short int unidades = 100; puede ser un literal

Fundamentos de la programación: Tipos e instrucciones I Página 145


Obtención del valor de una variable
 Nombre de la variable en una expresión
cout << balance;
cout << interesPorMes * meses / 100;

Modificación del valor de una variable


 Nombre de la variable a la izquierda del =
balance = 1214;
porcentaje = valor / 30;

Las variables han de haber sido previamente declaradas


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 146


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 147


El operador =
Variable = Expresión ;

A la izquierda, SIEMPRE una variable

int i, j = 2;
i = 23 + j * 5; // i toma el valor 33
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 148

Errores
int a, b, c;

5 = a;      
// ERROR: un literal no puede recibir un valor
a + 23 = 5; 
// ERROR: no puede haber una expresión a la izda.
b = "abc";  
// ERROR: un entero no puede guardar una cadena
c = 23 5;  
// ERROR: expresión no válida (falta operador)
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 149


int i, j = 2;
i = 23 + j * 5;

Memoria Memoria
01 01

i i
02

03

04
? 23 + 2 * 5
02

03

04
33
05 05
j j
2 2
06 06

07 07

08 08

09 09

10 10

. . . . . .
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 150

Necesitamos una variable auxiliar


double a = 3.45, b = 127.5, aux;

a 3.45 a 3.45
b 127.5 aux = a; b 127.5
aux ? aux 3.45

a 127.5
a = b; b 127.5
aux 3.45

a 127.5
Luis Hernández Yáñez

b = aux; b 3.45
aux 3.45

Fundamentos de la programación: Tipos e instrucciones I Página 151


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 152

Operaciones sobre valores de los tipos


Cada tipo determina las operaciones posibles
Tipos de datos numéricos (int, float y double):
— Asignación (=)

— Operadores aritméticos

— Operadores relacionales (menor, mayor, igual, ...)

Tipo de datos bool:


— Asignación (=)

— Operadores lógicos (Y, O, NO)

Tipos de datos char y string:


Luis Hernández Yáñez

— Asignación (=)

— Operadores relacionales (menor, mayor, igual, ...)

Fundamentos de la programación: Tipos e instrucciones I Página 153


Operadores para tipos de datos numéricos

Operador Operandos Posición int float / double


‐ 1 (monario) Prefijo Cambio de signo
+ 2 (binario) Infijo Suma
‐ 2 (binario) Infijo Resta
* 2 (binario) Infijo Producto
/ 2 (binario) Infijo Div. entera División real
% 2 (binario) Infijo Módulo No aplicable
++ 1 (monario) Prefijo / postfijo Incremento
‐‐ 1 (monario) Prefijo / postfijo Decremento
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 154

Operadores monarios y operadores binarios


Operadores monarios (unarios)
— Cambio de signo (‐):

Delante de variable, constante o expresión entre paréntesis


‐saldo     ‐RATIO     ‐(3 * a ‐ b)
— Incremento/decremento (sólo variables) (prefijo/postfijo):
++interes ‐‐meses     j++ // 1 más ó 1 menos

Operadores binarios
— Operando izquierdo operador operando derecho
Operandos: literales, constantes, variables o expresiones
Luis Hernández Yáñez

2 + 3 a * RATIO     ‐a + b
(a % b) * (c / d)

Fundamentos de la programación: Tipos e instrucciones I Página 155


¿División entera o división real? /
Ambos operandos enteros: división entera
int i = 23, j = 2;
cout << i / j; // Muestra 11

Algún operando real: división real


int i = 23;
double j = 2;
cout << i / j; // Muestra 11.5
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 156

Módulo (resto de la división entera) %


Ambos operandos han de ser enteros
int i = 123, j = 5;
cout << i % j; // Muestra 3

División entera:
No se obtienen decimales  Queda un resto

123 5
3 24
Luis Hernández Yáñez

123 % 5

Fundamentos de la programación: Tipos e instrucciones I Página 157


Operadores de incremento y decremento ++/‐‐
Incremento/decremento de la variable numérica en una unidad
Prefijo: Antes de acceder
int i = 10, j;
i=i+1;
j = ++i; // Incrementa antes de copiar
j=i;
cout << i << " ‐ " << j; // Muestra 11 ‐ 11
Postfijo: Después de acceder
int i = 10, j;
j=i; j = i++; // Copia y después incrementa
i=i+1;
cout << i << " ‐ " << j; // Muestra 11 ‐ 10
Luis Hernández Yáñez

No mezcles ++ y ‐‐ con otros operadores

Fundamentos de la programación: Tipos e instrucciones I Página 158

#include <iostream> operadores.cpp
using namespace std;

int main() {
int entero1 = 15, entero2 = 4;
double real1 = 15.0, real2 = 4.0;
cout << "Operaciones entre los números 15 y 4:" << endl;
cout << "División entera (/): " << entero1 / entero2 << endl;
cout << "Resto de la división (%): " << entero1 % entero2 << endl;
cout << "División real (/): " << real1 / real2 << endl;
cout << "Num = " << real1 << endl;
real1 = ‐real1;
cout << "Cambia de signo (‐): " << real1 << endl;
real1 = ‐real1;
cout << "Vuelve a cambiar (‐): " << real1 << endl;
cout << "Se incrementa antes (++ prefijo): " << ++real1 << endl;
cout << "Se muestra antes de incrementar (posfijo ++): " 
Luis Hernández Yáñez

<< real1++ << endl;
cout << "Ya incrementado: " << real1 << endl;
return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 159


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 160

¿En qué orden se evalúan los operadores?


3 + 5 * 2 / 2 ‐ 1
¿De izquierda a derecha?
¿De derecha a izquierda?
¿Unos antes que otros?

Precedencia de los operadores (prioridad):


Se evalúan antes los de mayor precedencia
¿Y si tienen igual prioridad?
Normalmente, de izquierda a derecha
Luis Hernández Yáñez

Paréntesis: fuerzan a evaluar su subexpresión

Fundamentos de la programación: Tipos e instrucciones I Página 161


Precedencia Operadores
Mayor prioridad ++ ‐‐ (postfijos)
++ ‐‐ (prefijos)
‐ (cambio de signo)
* / %
Menor prioridad + ‐

3 + 5 * 2 / 2 ‐ 1  3 + 10 / 2 ‐ 1  3 + 5 ‐ 1  8 ‐ 1  7

Misma precedencia: Mayor Misma precedencia:


Luis Hernández Yáñez

Izquierda antes precedencia Izquierda antes

Fundamentos de la programación: Tipos e instrucciones I Página 162

((3 + 5) * 4 + 12) / 4 ‐ (3 * 2 ‐ 1) Primero, los paréntesis...


* antes que ‐
(8 * 4 + 12) / 4 ‐ (6 ‐ 1)
* antes que +
(32 + 12) / 4 ‐ 5

44 / 4 ‐ 5
/ antes que ‐
11 ‐ 5 Pon espacio antes y después
de cada operador binario
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 163


fórmula.cpp

#include <iostream>
using namespace std;

int main()
{
double x, f;
cout << "Introduce el valor de X: ";
cin >> x;
f = 3 * x * x / 5 + 6 * x / 7 ‐ 3;
cout << "f(x) = " << f << endl;
return 0;
}
Luis Hernández Yáñez

Usa paréntesis para mejorar la legibilidad:


f = (3 * x * x / 5) + (6 * x / 7) ‐ 3;

Fundamentos de la programación: Tipos e instrucciones I Página 164

variable = variable operador op_derecho;


La misma 
variable operador= op_derecho;

Asignación Abreviatura
a = a + 12; a += 12;
Igual precedencia
a = a * 3; a *= 3; que la asignación
a = a ‐ 5; a ‐= 5;
a = a / 37; a /= 37; De momento,
mejor evitarlas
Luis Hernández Yáñez

a = a % b; a %= b;

Fundamentos de la programación: Tipos e instrucciones I Página 165


¿Valor siguiente al máximo?
Valor mayor del máximo (o menor del mínimo) del tipo
short int i = 32767; // Valor máximo para short int
i++; // 32768 no cabe en un short int
cout << i; // Muestra ‐32768

Bit de signo 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 32767


0 = positivo
+ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 + 1
1 = negativo
1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 ‐32768
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 166


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 167


Declaración de constantes Modificador de acceso const
Variables inicializadas a las que no dejamos variar

const Declaración de variable con inicializador

const short int Meses = 12;
La constante no podrá volver a
const double Pi = 3.141592,
aparecer a la izquierda de un =
RATIO = 2.179 * Pi;

Programación con buen estilo:


Luis Hernández Yáñez

Pon en mayúscula la primera letra


de una constante o todo su nombre

Fundamentos de la programación: Tipos e instrucciones I Página 168

 Aumentan la legibilidad del código


cambioPoblacion = (0.1758 ‐ 0.1257) * poblacion; vs.
cambioPoblacion = (RatioNacimientos ‐ RatioMuertes) * poblacion;

 Facilitan la modificación del código


double compra1 = bruto1 * 18 / 100;
double compra2 = bruto2 * 18 / 100; 3 cambios
double total = compra1 + compra2;
cout << total << " (IVA: " << 18 << "%)" << endl;

const int IVA = 18; ¿Cambio del IVA al 21%?


double compra1 = bruto1 * IVA / 100;
double compra2 = bruto2 * IVA / 100; 1 cambio
Luis Hernández Yáñez

double total = compra1 + compra2;
cout << total << " (IVA: " << IVA << "%)" << endl;

Fundamentos de la programación: Tipos e instrucciones I Página 169


constantes.cpp

#include <iostream>
using namespace std;

int main() {
const double Pi = 3.141592;
double radio = 12.2, circunferencia;
circunferencia = 2 * Pi * radio;
cout << "Circunferencia de un círculo de radio " 
<< radio << ": " << circunferencia << endl;
const double Euler = 2.718281828459; // Número e
cout << "Número e al cuadrado: " << Euler * Euler << endl;
const int IVA = 21;
int cantidad = 12;
double precio = 39.95, neto, porIVA, total;
neto = cantidad * precio;
porIVA = neto * IVA / 100;
Luis Hernández Yáñez

total = neto + porIVA;
cout << "Total compra: " << total << endl;
return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 170


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 171


#include <cmath>

Algunas ... abs(x) Valor absoluto de x


pow(x, y) x elevado a y
sqrt(x) Raíz cuadrada de x
ceil(x) Menor entero que es mayor o igual que x
floor(x) Mayor entero que es menor o igual que x
exp(x) ex
log(x) Ln x (logaritmo natural de x)
log10(x) Logaritmo en base 10 de x
sin(x) Seno de x
cos(x) Coseno de x
Tangente de x
Luis Hernández Yáñez

tan(x)
round(x) Redondeo al entero más próximo
trunc(x) Pérdida de la parte decimal (entero)

Fundamentos de la programación: Tipos e instrucciones I Página 172

mates.cpp

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

int main() {
double x, y, f; pow() con argumento entero:
cout << "Valor de X: "; Usa el molde double():
cin >> x; pow(double(i), 5)
cout << "Valor de Y: ";
cin >> y;
f = 2 * pow(x, 5) + sqrt(pow(x, 3) / pow(y, 2))
/ abs(x * y) ‐ cos(y);
cout << "f(x, y) = " << f << endl;
return 0;
Luis Hernández Yáñez

}
Pon un espacio detrás de cada coma en las listas de argumentos

Fundamentos de la programación: Tipos e instrucciones I Página 173


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 174

char
Asignación, ++/‐‐ y operadores relacionales

Funciones para caracteres (biblioteca cctype)


isalnum(c) true si c es una letra o un dígito
isalpha(c) true si c es una letra
isdigit(c) true si c es un dígito
islower(c) true si c es una letra minúscula
isupper(c) true si c es una letra mayúscula
false en caso contrario
toupper(c) devuelve la mayúscula de c
Luis Hernández Yáñez

tolower(c) devuelve la minúscula de c


Fundamentos de la programación: Tipos e instrucciones I Página 175


caracteres.cpp

...
#include <cctype>
int main() {
char caracter1 = 'A', caracter2 = '1', caracter3 = '&';
cout << "Carácter 1 (" << caracter1 << ").‐" << endl;
cout << "Alfanumérico? " << isalnum(caracter1) << endl;
cout << "Alfabético? " << isalpha(caracter1) << endl;
cout << "Dígito? " << isdigit(caracter1) << endl;
cout << "Mayúscula? " << isupper(caracter1) << endl;
caracter1 = tolower(caracter1);
cout << "En minúscula: " << caracter1 << endl;
cout << "Carácter 2 (" << caracter2 << ").‐" << endl;
cout << "Alfabético? " << isalpha(caracter2) << endl;
cout << "Dígito? " << isdigit(caracter2) << endl;
cout << "Carácter 3 (" << caracter3 << ").‐" << endl;
cout << "Alfanumérico? " << isalnum(caracter3) << endl;
Luis Hernández Yáñez

cout << "Alfabético? " << isalpha(caracter3) << endl;
cout << "Dígito? " << isdigit(caracter3) << endl;
return 0;
} 1  true / 0  false
Fundamentos de la programación: Tipos e instrucciones I Página 176
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 177


Operadores relacionales
Comparaciones (condiciones)
Condición simple ::= Expresión Operador_relacional Expresión
Concordancia de tipo entre las expresiones
Resultado: bool (true o false)

Operadores (prioridad)
< menor que
...
<= menor o igual que
* / %
> mayor que
+ ‐
>= mayor o igual que
Luis Hernández Yáñez

< <= > >=
== igual que
== !=
!= distinto de
= += ‐= *= /= %=

Fundamentos de la programación: Tipos e instrucciones I Página 178

Menor prioridad que los operadores aditivos y multiplicativos


bool resultado;
int a = 2, b = 3, c = 4;
resultado = a < 5;              // 2 < 5  true
resultado = a * b + c >= 12;    // 10 >= 12  false
resultado = a * (b + c) >= 12;  // 14 >= 12  true
resultado = a != b;             // 2 != 3  true
resultado = a * b > c + 5;      // 6 > 9  false
resultado = a + b == c + 1;     // 5 == 5  true

No confundas el operador de igualdad (==)


Luis Hernández Yáñez

con el operador de asignación (=)

Fundamentos de la programación: Tipos e instrucciones I Página 179


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 180

Selección: bifurcación condicional


if (condición) {
true false códigoT
Condición
}
códigoT códigoF
else {
códigoF
int num; }
cout << "Número: ";
Opcional: puede no haber else
cin >> num;
if (num % 2 == 0) {
cout << num << " es par";
Luis Hernández Yáñez

}
else {
cout << num << " es impar";
}
Fundamentos de la programación: Tipos e instrucciones I Página 181
selección.cpp

#include <iostream>
using namespace std;

int main() {
int op1 = 13, op2 = 4;
int opcion;
cout << "1 ‐ Sumar" << endl;
cout << "2 ‐ Restar" << endl;
cout << "Opción: ";
cin >> opcion;
if (opcion == 1) {
cout << op1 + op2 << endl;
}
else {
cout << op1 ‐ op2 << endl;
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 182


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 183


Agrupación de instrucciones
Grupo de instrucciones a ejecutar en una rama del if

{ instrucción } {
intrucción1
Tab ó intrucción2
3 esp. ...
int num, total = 0; intrucciónN
cin >> num; }
if (num > 0) 
{
cout << "Positivo"; Ámbito local
total = total + num; (declaraciones locales)
Luis Hernández Yáñez

}
cout << endl;

Fundamentos de la programación: Tipos e instrucciones I Página 184

Posición de las llaves: cuestión de estilo


if (num > 0) if (num > 0) {
{ cout << "Positivo";
cout << "Positivo"; total = total + num;
total = total + num; }
} cout << endl;
cout << endl;

No necesitamos las llaves si sólo hay una instrucción


if (num > 0) { if (num > 0)
cout << "Positivo";  cout << "Positivo";
}
Luis Hernández Yáñez

Usaremos siempre llaves por simplicidad...


Evita poner el if y la instrucción objetivo en la misma línea:
if (num > 0) cout << "Positivo";

Fundamentos de la programación: Tipos e instrucciones I Página 185


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 186

Repetición o iteración condicional

while ( condición ) cuerpo


Bloque
de código

true false
Condición
while (condición) {
Cuerpo cuerpo
}
Luis Hernández Yáñez

Si la condición es false al empezar,


no se ejecuta el cuerpo ninguna vez

Fundamentos de la programación: Tipos e instrucciones I Página 187


serie.cpp

#include <iostream>
using namespace std;

int main() {
int i = 1, n = 0, suma = 0;
while (n <= 0) { // Sólo n positivo
cout << "¿Cuántos números quieres sumar? ";
cin >> n;
}
while (i <= n) {
suma = suma + i;
i++;
}
cout << "Sumatorio de i (1 a " << n << ") = " 
Luis Hernández Yáñez

<< suma << endl;
return 0;
}

Fundamentos de la programación: Tipos e instrucciones I Página 188

Iteración condicional
while (i <= n) {
suma = suma + i;
i++;
}
n 5
i 6
2
3
4
5
1
true false suma 15
10
1
3
60
i <= n

suma += i;
i++;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 189


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 190

#include <iostream>
Flujos de texto (streams) using namespace std;
Conectan la ejecución del programa con los dispositivos de E/S
Son secuencias de caracteres
Entrada por teclado: flujo de entrada cin (tipo istream)
Salida por pantalla: flujo de salida cout (tipo ostream)

cin cout
7 3 5 . 3 5 1 Programa 1 6 = l a t o T

Biblioteca iostream con espacio de nombres std


Luis Hernández Yáñez

Extractor Flujo de entrada >> Variable

Insertor Flujo de salida << Expresión

Fundamentos de la programación: Tipos e instrucciones I Página 191


cin >> Variable

Salta los espacios en blanco (espacios, tabuladores o saltos de línea)


— char
Se lee un carácter en la variable
— int
Se leen dígitos y se transforman en el valor a asignar
— float/double:
Se leen dígitos (quizá el punto y más dígitos) y se asigna el valor
— bool:
Si se lee 1, se asigna true; con cualquier otro valor se asigna false

Se amigable con el usuario


Lee cada dato en una línea
Luis Hernández Yáñez

cout << "Introduce tu edad: ";
cin >> edad;

Fundamentos de la programación: Tipos e instrucciones I Página 192

#include <string>
using namespace std;

cin >> cadena termina con el primer espacio en blanco


cin.sync() descarta la entrada pendiente
string nombre, apellidos; string nombre, apellidos;
cout << "Nombre: "; cout << "Nombre: ";
cin >> nombre; cin >> nombre;
cout << "Apellidos: "; cin.sync();
cin >> apellidos; cout << "Apellidos: ";
cout << "Nombre completo: " cin >> apellidos;
<< nombre << " " cout << ...
<< apellidos << endl;

apellidos recibe "Antonio"


Luis Hernández Yáñez

¿Cómo leer varias palabras?


Siguiente página...

Fundamentos de la programación: Tipos e instrucciones I Página 193


Lectura sin saltar los espacios en blanco iniciales
Llamada a funciones con el operador punto (.) :
El operador punto permite llamar a una función sobre una variable
variable.función(argumentos)

Lectura de un carácter sin saltar espacios en blanco:


cin.get(c); // Lee el siguiente carácter

Lectura de cadenas sin saltar los espacios en blanco:


getline(cin, cad);
Lee todo lo que haya hasta el final de la línea (Intro)
Luis Hernández Yáñez

Recuerda:
Espacios en blanco son espacios, tabuladores, saltos de línea, ...

Fundamentos de la programación: Tipos e instrucciones I Página 194

cout << Expresión

Representación textual de los datos


int meses = 7;
cout << "Total: " << 123.45 << endl << " Meses: " << meses;
El valor double 123.45 se guarda en memoria en binario
Su representación textual es:'1' '2' '3' '.' '4' '5'
double d = 123.45;

d 123.45 ¡Un número real!

cout << d;
5 4 . 3 2 1
La biblioteca iostream
Luis Hernández Yáñez

define la constante endl ¡Un texto!


como un salto de línea (secuencia de caracteres)

Fundamentos de la programación: Tipos e instrucciones I Página 195


cout << Expresión

T o t a l : 1 2 3 . 4 5  M e s e s : 7 Programa
cout

int meses = 7;
cout << "Total: " << 123.45 << endl << " Meses: " << meses;

cout       << 123.45 << endl << " Meses: " << meses;

cout        << endl << " Meses: " << meses;

cout        << " Meses: " << meses;
Total: 123.45
Luis Hernández Yáñez

Meses: 7 cout           << meses;

Fundamentos de la programación: Tipos e instrucciones I Página 196

#include <iomanip>

Constantes y funciones a enviar a cout para ajustar el formato de salida

Biblioteca Constante/función Propósito


showpoint / Mostrar o no el punto decimal para reales sin
iostream
noshowpoint decimales (34.0)
fixed Notación de punto fijo (reales) (123.5)
scientific Notación científica (reales) (1.235E+2)
boolalpha Valores bool como true / false
left / right Ajustar a la izquierda/derecha (por defecto)
iomanip setw(anchura)* Nº de caracteres (anchura) para el dato
Precisión: Nº de dígitos (en total)
setprecision(p)
Con fixed o scientific, nº de decimales
Luis Hernández Yáñez

*setw() sólo afecta al siguiente dato que se escriba,


mientras que los otros afectan a todos

Fundamentos de la programación: Tipos e instrucciones I Página 197


bool fin = false;
cout << fin << "‐>" << boolalpha << fin << endl; 0‐>false
double d = 123.45;
char c = 'x';
int i = 62;
cout << d << c << i << endl; 123.45x62
cout << "|" << setw(8) << d << "|" << endl; |  123.45|
cout << "|" << left << setw(8) << d << "|" << endl; |123.45  |
cout << "|" << setw(4) << c << "|" << endl; |x   |
cout << "|" << right << setw(5) << i << "|" << endl; |   62|
double e = 96;
cout << e << " ‐ " << showpoint << e << endl; 96 ‐ 96.0000
Luis Hernández Yáñez

cout << scientific << d << endl; 1.234500e+002
cout << fixed << setprecision(8) << d << endl; 123.45000000

Fundamentos de la programación: Tipos e instrucciones I Página 198


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I Página 199


Los programas pueden incluir otras funciones además de main()
Forma general de una función en C++:
tipo nombre(parámetros) // Cabecera
{
// Cuerpo
}
 Tipo de dato que devuelve la función como resultado
 Parámetros para proporcionar datos a la función
Declaraciones de variables separadas por comas
 Cuerpo: secuencia de declaraciones e instrucciones
¡Un bloque de código!
Luis Hernández Yáñez

Fundamentos de programación: Tipos e instrucciones I Página 200

 Datos locales: declarados en el cuerpo de la función


Datos auxiliares que utiliza la función (puede no haber)
 Parámetros: declarados en la cabecera de la función
Datos de entrada de la función (puede no haber)
Ambos son de uso exclusivo de la función y no se conocen fuera
double f(int x, int y) {
// Declaración de datos locales:
double resultado;

// Instrucciones:
resultado = 2 * pow(x, 5) + sqrt(pow(x, 3)
/ pow(y, 2)) / abs(x * y) ‐ cos(y);
Luis Hernández Yáñez

return resultado; // Devolución del resultado
}

Fundamentos de programación: Tipos e instrucciones I Página 201


Llamada a una función con parámetros
Nombre(Argumentos)
Al llamar a la función:
— Tantos argumentos entre los paréntesis como parámetros

— Orden de declaración de los parámetros

— Cada argumento: mismo tipo que su parámetro

— Cada argumento: expresión válida (se pasa el resultado)

Se copian los valores resultantes de las expresiones


en los correspondientes parámetros
Llamadas a la función: en expresiones de otras funciones
Luis Hernández Yáñez

int valor = f(2, 3);

Fundamentos de programación: Tipos e instrucciones I Página 202

Se copian los argumentos en los parámetros


int funcion(int x, double a) { Memoria
...  i 124
} d 3.0
...
int main() {
int i = 124;
...
double  d = 3; x 124
funcion(i, 33 * d); a 99.0
... ...
Luis Hernández Yáñez

return 0; // main() devuelve 0 al S.O.
}
Los argumentos no se modifican

Fundamentos de programación: Tipos e instrucciones I Página 203


La función ha de devolver un resultado
La función termina su ejecución devolviendo un resultado
La instrucción return (sólo una en cada función)
— Devuelve el dato que se pone a continuación (tipo de la función)

— Termina la ejecución de la función

El dato devuelto sustituye a la llamada de la función:


int cuad(int x) { int main() {
return x * x; cout << 2 * cuad(16);
x = x * x;
Luis Hernández Yáñez

} return 0; 256
Esta instrucción }
no se ejecutará nunca

Fundamentos de programación: Tipos e instrucciones I Página 204

¿Qué funciones hay en el programa?


Colocaremos las funciones después de main()

¿Son correctas las llamadas a funciones del programa?


— ¿Existe la función?

— ¿Concuerdan los argumentos con los parámetros?

 Prototipos tras las inclusiones de bibliotecas


Prototipo de función: Cabecera de la función terminada en ;
double f(int x, int y);
int funcion(int x, double a) 
Luis Hernández Yáñez

int cuad(int x);


... main() es la única función
que no hay que prototipar

Fundamentos de programación: Tipos e instrucciones I Página 205


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

// Prototipos de las funciones (excepto main())
bool par(int num);
bool letra(char car);
int suma(int num);
double formula(int x, int y);

int main() {
int numero, sum, x, y;
char caracter;
double f;
cout << "Entero: ";
cin >> numero;
Luis Hernández Yáñez

if (par(numero)) {
cout << "Par";
}
...

Fundamentos de programación: Tipos e instrucciones I Página 206

else {
cout << "Impar";
}
cout << endl;
if (numero > 1) {
cout << "Sumatorio de 1 a " << numero << ": "
<< suma(numero) << endl;
}
cout << "Carácter: ";
cin >> caracter;
if (!letra(caracter)) {
cout << "no ";
}
cout << "es una letra" << endl;
cout << "f(x, y) = " << formula(x, y) << endl;
// Los argumentos pueden llamarse igual o no que los parámetros
Luis Hernández Yáñez

return 0;
}
...

Fundamentos de programación: Tipos e instrucciones I Página 207


// Implementación de las funciones propias

bool par(int num) {
bool esPar;

if (num % 2 == 0) {
esPar = true;
}
else {
esPar = false;
}

return esPar;
}
Luis Hernández Yáñez

...

Fundamentos de programación: Tipos e instrucciones I Página 208

bool letra(char car) {


bool esLetra;
if ((car >= 'a') && (car <= 'z') || (car >= 'A') && (car <= 'Z')) {
esLetra = true;
}
else {
esLetra = false;
}
return esLetra;
}

int suma(int num) {


int sum = 0, i = 1;
while (i < num) {
sum = sum + i;
i++;
Luis Hernández Yáñez

}
return sum;
}
...

Fundamentos de programación: Tipos e instrucciones I Página 209


funciones.cpp

double formula(int x, int y) {


double f;

f = 2 * pow(x, 5) + sqrt(pow(x, 3) / pow(y, 2)) 
/ abs(x * y) ‐ cos(y);

return f;
}
Luis Hernández Yáñez

Fundamentos de programación: Tipos e instrucciones I Página 210

Licencia CC (Creative Commons)


Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Tipos e instrucciones I Página 211


Fundamentos de la programación

2A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

int 214
float 216
Notación científica 217
double 218
char 220
bool 221
string 222
Literales con especificación de tipo 223
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I (Anexo)
Números enteros

Intervalo de valores:
‐2147483648 .. 2147483647
Bytes de memoria: 4* (*) Depende de la máquina 01
4 bytes es lo más habitual 02
Literales: Se puede saber cuántos 03

1363, ‐12, 010 , 0x1A se usan con la función 04


sizeof(int) 05
06
+ 07
0..9 08
‐ 0..9 09
. . .
Notación octal 0 0..7
Luis Hernández Yáñez

Notación hexadecimal x 0..9,A..F

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 214

Números enteros

Números en notación octal (base 8: dígitos entre 0 y 7):


‐010 = ‐8 en notación decimal
10 = 1 x 81 + 0 x 80 = 1 x 8 + 0
0423 = 275 en notación decimal
423 = 4 x 82 + 2 x 81 + 3 x 80 = 4 x 64 + 2 x 8 + 3 = 256
+ 16 +3

Números en notación hexadecimal (base 16):


Dígitos posibles: 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A, B, C, D, E, F
0x1F = 31 en notación decimal
1F = 1 x 161 + F x 160 = 1 x 16 + 15
0xAD = 173 en notación decimal
Luis Hernández Yáñez

AD = A x 161 + D x 160 = 10 x 16 + 13 = 160 + 13

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 215


Números reales (con decimales)

Intervalo de valores:
+/‐ 1.18e‐38 .. 3.40e+38
Bytes de memoria: 4* (*) sizeof(float) 01

Punto flotante. Precisión: 7 dígitos


02
03
Literales (punto decimal): 04

 Notación normal: 134.45, ‐1.1764 05


06
07
+
08
0..9 . 0..9 09
‐ . . .

 Notación científica: 1.4E2, ‐5.23e‐02


Luis Hernández Yáñez

+ +
0..9 . 0..9 e,E 0..9
‐ ‐

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 216

Siempre un número (con o sin signo) con un solo dígito de parte


entera, seguido del exponente (potencia de 10):
‐5.23e‐2  ‐5,23 x 10‐2  ‐0,0523
1.11e2  1,11 x 102  111,0
7.4523e‐04  7,4523 x 10‐4  0,00074523
‐3.3333e+06  ‐3,3333 x 106  ‐3.333.300
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 217


Números reales (con decimales)

Intervalo de valores:
+/‐ 2.23e‐308 .. 1.79e+308
Bytes de memoria: 8* (*) sizeof(double) 01

Punto flotante. Precisión: 15 dígitos


02
03
Literales (punto decimal): 04

 Notación normal: 134.45, ‐1.1764 05


06
07
+
08
0..9 . 0..9 09
‐ . . .

 Notación científica: 1.4E2, ‐5.23e‐02


Luis Hernández Yáñez

+ +
0..9 . 0..9 e,E 0..9
‐ ‐

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 218

Caracteres
Intervalo de valores:
Juego de caracteres (ASCII)
Bytes de memoria: 1 (FC) 01

Literales: 02
03
'a', '%', '\t' 04
Constantes de barra invertida: 05

(O secuencias de escape) 06

Para caracteres de control 07


08
'\t' = tabulador, '\n' = salto de línea, … 09
. . .
\ n, t, v, b, r, f, a, \
' '
Luis Hernández Yáñez

Carácter

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 219


Juego de caracteres ASCII:
American Standard Code for Information Interchange (1963)
Caracteres con códigos entre 0 y 127 (7 bits)
— Caracteres de control:
Códigos del 0 al 31 y 127
Tabulación, salto de línea,...
— Caracteres imprimibles:
Códigos del 32 al 126
Juego de caracteres ASCII extendido (8 bits):
ISO‐8859‐1
Multitud de codificaciones:
+ Códigos entre 128 y 255 EBCDIC, UNICODE, UTF‐8, ...
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 220

Valores lógicos

Sólo dos valores posibles:


— Verdadero (true)

— Falso (false) 01
02
Bytes de memoria: 1 (FC) 03

Literales: 04
05
true, false 06

En realidad, cualquier número 07

distinto de 0 es equivalente a true 08


09
y el número 0 es equivalente a false . . .
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 221


Cadenas de caracteres

"Hola", "Introduce el numerador: ", "X142FG5TX?%A"

" Carácter "

Secuencias de caracteres
Se asigna la memoria que se necesita para la secuencia concreta
Requieren la biblioteca string con el espacio de nombres std:
#include <string>
using namespace std;

¡Ojo!
Luis Hernández Yáñez

Las comillas tipográficas (apertura/cierre) “…” te darán problemas


al compilar. Asegúrate de utilizar comillas rectas: "…"

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 222

Por defecto un literal entero se considera un dato int


— long int: 35L, 1546l
— unsigned int: 35U, 1546u
— unsigned long int: 35UL, 1546ul
Por defecto un literal real se considera un dato double
— float: 1.35F, 15.46f
— long double: 1.35L, 15.46l

Abreviaturas para modificadores de tipos


short  short int
long  long int
Es preferible evitar el uso de tales abreviaturas:
Luis Hernández Yáñez

Minimizar la cantidad de información a recordar


sobre el lenguaje

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 223


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Tipos e instrucciones I (Anexo) Página 224


Fundamentos de la programación

3
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Tipos, valores y variables 227 El bucle for 321


Conversión de tipos 232 Bucles anidados 331
Tipos declarados por el usuario 236 Ámbito y visibilidad 339
Tipos enumerados 238 Secuencias 349
Entrada/Salida Recorrido de secuencias 355
con archivos de texto 248 Secuencias calculadas 363
Lectura de archivos de texto 253 Búsqueda en secuencias 370
Escritura en archivos de texto 266 Arrays de datos simples 374
Flujo de ejecución 272 Uso de variables arrays 379
Selección simple 276 Recorrido de arrays 382
Operadores lógicos 282 Búsqueda en arrays 387
Anidamiento de if 286 Arrays no completos 393
Condiciones 290
Selección múltiple 293
La escala if‐else‐if 295
Luis Hernández Yáñez

La instrucción switch 302
Repetición 313
El bucle while 316

Fundamentos de la programación: Tipos e instrucciones II
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 227

Tipo
Conjunto de valores con sus posibles operaciones

Valor
Conjunto de bits interpretados como de un tipo concreto

Variable (o constante)
Cierta memoria con nombre para valores de un tipo

Declaración
Instrucción que identifica un nombre

Definición
Declaración que asigna memoria a una variable o constante
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 228


Memoria suficiente para su tipo de valores
short int i = 3; i 3

int j = 9; j 9

char c = 'a'; c a

double x = 1.5; x 1.5

El significado de los bits depende del tipo de la variable:


00000000 00000000 00000000 01111000
Interpretado como int es el entero 120
Luis Hernández Yáñez

Interpretado como char (sólo 01111000) es el carácter 'x'

Fundamentos de la programación: Tipos e instrucciones II Página 229

 Simples
 Estándar: int, float, double, char, bool
Conjunto de valores predeterminado

 Definidos por el usuario: enumerados


Conjunto de valores definido por el programador

 Estructurados (Tema 5)
 Colecciones homogéneas: arrays
Todos los elementos de la colección de un mismo tipo

 Colecciones heterogéneas: estructuras


Luis Hernández Yáñez

Elementos de la colección de tipos distintos

Fundamentos de programación: Tipos e instrucciones II Página 230


Con sus posibles modificadores:
[unsigned] [short] int
long long int long int  int
float
[long] double
char
bool
Definición de variables:
tipo nombre [ = expresión] [, ...];
Luis Hernández Yáñez

Definición de constantes con nombre:


const tipo nombre = expresión;

Fundamentos de la programación: Tipos e instrucciones II Página 231


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 232


Promoción de tipos
Dos operandos de tipos distintos:
El valor del tipo menor se promociona al tipo mayor
short int i = 3; long double
int j = 2;

Promoción
double
double a = 1.5, b; float
long int
b = a + i * j;
int
short int
b = a + 3 * 2;
Valor 3 short int (2 bytes)  int (4 bytes)
Luis Hernández Yáñez

b = 1.5 + 6;
Valor 6 int (4 bytes)  double (8 bytes)

Fundamentos de la programación: Tipos e instrucciones II Página 233

Conversión segura: long double


De un tipo menor a un tipo mayor double
short int  int  long int  ... float
long int
Conversión no segura:
int
De un tipo mayor a un tipo menor short int
int entero = 1234;
char caracter;
caracter = entero; // Conversión no segura
Menor memoria: Pérdida de información en la conversión
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 234


Fuerzan una conversión de tipo:
tipo(expresión)
El valor resultante de la expresión se trata como un valor del tipo

int a = 3, b = 2;
cout << a / b;         // Muestra 1 (división entera)
cout << double(a) / b; // Muestra 1.5 (división real)

Tienen la mayor prioridad


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 235


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 236


Describimos los valores de las variables del tipo
typedef descripción nombre_de_tipo;

Identificador válido

Nombres de tipos propios:


t minúscula seguida de una o varias palabras capitalizadas
Los colorearemos en naranja, para remarcar que son tipos

typedef descripción tMiTipo;
typedef descripción tMoneda;
typedef descripción tTiposDeCalificacion;
Luis Hernández Yáñez

Declaración de tipo frente a definición de variable

Fundamentos de la programación: Tipos e instrucciones II Página 237


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 238


Enumeración del conjunto de valores posibles para las variables:
enum { símbolo1, símbolo2, ..., símboloN }

enum { Identificador }

enum { centimo, dos_centimos, cinco_centimos,
diez_centimos, veinte_centimos,
medio_euro, euro }
Valores literales que pueden tomar las variables (en amarillo)
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 239

Mejoran la legibilidad

typedef descripción nombre_de_tipo;
Elegimos un nombre para el tipo: tMoneda descripción
typedef enum { centimo, dos_centimos, cinco_centimos,
diez_centimos, veinte_centimos,
medio_euro, euro } tMoneda;

En el ámbito de la declaración, se reconoce un nuevo tipo tMoneda


tMoneda moneda1, moneda2;
Cada variable de ese tipo contendrá alguno de los símbolos
moneda1 = dos_centimos;
moneda2 = euro;
Luis Hernández Yáñez

moneda1 dos_centimos

moneda2 euro
(Internamente se usan enteros)

Fundamentos de la programación: Tipos e instrucciones II Página 240


typedef enum { enero, febrero, marzo, abril, mayo,
junio, julio, agosto, septiembre, octubre,
noviembre, diciembre } tMes;
tMes mes;
Lectura de la variable mes:
cin >> mes;
Se espera un valor entero
No se puede escribir directamente enero o junio

Y si se escribe la variable en la pantalla:


cout << mes;
Se verá un número entero
Luis Hernández Yáñez

 Código de entrada/salida específico

Fundamentos de la programación: Tipos e instrucciones II Página 241

typedef enum { enero, febrero, marzo, abril, mayo, junio, julio,
agosto, septiembre, octubre, noviembre, diciembre } tMes;

int op;
cout << " 1 ‐ Enero" << endl;
cout << " 2 ‐ Febrero" << endl;
cout << " 3 ‐ Marzo" << endl;
cout << " 4 ‐ Abril" << endl;
cout << " 5 ‐ Mayo" << endl;
cout << " 6 ‐ Junio" << endl;
cout << " 7 ‐ Julio" << endl;
cout << " 8 ‐ Agosto" << endl;
cout << " 9 ‐ Septiembre" << endl;
cout << "10 ‐ Octubre" << endl;
cout << "11 ‐ Noviembre" << endl;
Luis Hernández Yáñez

cout << "12 ‐ Diciembre" << endl;


cout << "Numero de mes: ";
cin >> op;
tMes mes = tMes(op ‐ 1);

Fundamentos de la programación: Tipos e instrucciones II Página 242


typedef enum { enero, febrero, marzo, abril, mayo, junio, julio,
agosto, septiembre, octubre, noviembre, diciembre } tMes;

if (mes == enero) {
cout << "enero";
}
if (mes == febrero) {
cout << "febrero";
}
if (mes == marzo) {
cout << "marzo";
}
...
if (mes == diciembre) {
cout << "diciembre";
Luis Hernández Yáñez

También podemos utilizar una instrucción switch

Fundamentos de la programación: Tipos e instrucciones II Página 243

Conjunto de valores ordenado (posición en la enumeración)


typedef enum { lunes, martes, miercoles, jueves,
viernes, sabado, domingo } tDiaSemana;

tDiaSemana dia; lunes < martes < miercoles < jueves


... < viernes < sabado < domingo
if (dia == jueves)...
bool noLaborable = (dia >= sabado);

No admiten operadores de incremento y decremento


Emulación con moldes:
int i = int(dia); // ¡dia no ha de valer domingo!
Luis Hernández Yáñez

i++;
dia = tDiaSemana(i);

Fundamentos de la programación: Tipos e instrucciones II Página 244


#include <iostream> Si los tipos se usan en varias funciones,
using namespace std; los declaramos antes de los prototipos

typedef enum { enero, febrero, marzo, abril, mayo,
junio, julio, agosto, septiembre, octubre,
noviembre, diciembre } tMes;

typedef enum { lunes, martes, miercoles, jueves,


viernes, sabado, domingo } tDiaSemana;

string cadMes(tMes mes);


string cadDia(tDiaSemana dia);

int main() {
tDiaSemana hoy = lunes;
Luis Hernández Yáñez

int dia = 21;


tMes mes = octubre;
int anio = 2013;
...
Fundamentos de la programación: Tipos e instrucciones II Página 245

// Mostramos la fecha
cout << "Hoy es: " << cadDia(hoy) << " " << dia
<< " de " << cadMes(mes) << " de " << anio
<< endl;

cout << "Pasada la medianoche..." << endl;
dia++;
int i = int(hoy);
i++;
hoy = tDiaSemana(i);

// Mostramos la fecha
cout << "Hoy es: " << cadDia(hoy) << " " << dia
<< " de " << cadMes(mes) << " de " << anio
<< endl;
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 246


fechas.cpp

string cadMes(tMes mes) { string cadDia(tDiaSemana dia);


string cad; string cad;

if (mes == enero) { if (dia == lunes) {


cad = "enero"; cad = "lunes";
} }
if (mes == febrero) { if (dia == martes) {
cad = "febrero"; cad = "martes";
} }
... ...
if (mes == diciembre) { if (dia == domingo) {
cad = "diciembre"; cad = "domingo";
} }

return cad; return cad;


Luis Hernández Yáñez

} }

Fundamentos de la programación: Tipos e instrucciones II Página 247


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 248


Datos del programa: en la memoria principal (volátil)
Medios (dispositivos) de almacenamiento permanente:
— Discos magnéticos fijos (internos) o portátiles (externos)

— Cintas magnéticas

— Discos ópticos (CD, DVD, BlueRay)

— Memorias USB

Mantienen la información en archivos


Secuencias de datos
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 249

Archivo de texto: secuencia de caracteres


T o t a l : 1 2 3 . 4  A …

Archivo binario: contiene una secuencia de códigos binarios


A0 25 2F 04 D6 FF 00 27 6C CA 49 07 5F A4 …

(Códigos representados en notación hexadecimal)

Los archivos se manejan en los programas por medio de flujos


Archivos de texto: flujos de texto
Similar a la E/S por consola

(Más adelante veremos el uso de archivos binarios)


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 250


Textos dispuestos en sucesivas líneas
Carácter de fin de línea entre línea y línea (Intro)
Posiblemente varios datos en cada línea
Ejemplo: Compras de los clientes
En cada línea, NIF del cliente, unidades compradas, precio
unitario y descripción de producto, separados por espacio
12345678F 2 123.95 Reproductor de DVD↲
00112233A 1 218.4 Disco portátil↲
32143567J 3 32 Memoria USB 16Gb↲
76329845H 1 134.5 Modem ADSL↲
Luis Hernández Yáñez

...
Normalmente terminan con un dato especial (centinela)
Por ejemplo, un NIF que sea X
Fundamentos de la programación: Tipos e instrucciones II Página 251

#include <fstream>

 Lectura del archivo: flujo de entrada


 Escritura en el archivo: flujo de salida
No podemos leer y escribir en un mismo flujo

Un flujo de texto se puede utilizar para lectura o para escritura:


— Flujos (archivos) de entrada: variables de tipo ifstream
— Flujos (archivos) de salida : variables de tipo ofstream

Biblioteca fstream (sin espacio de nombres)


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 252


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 253

Flujos de texto de entrada ifstream

Para leer de un archivo de texto:


1 Declara una variable de tipo ifstream
2 Asocia la variable con el archivo de texto (apertura del archivo)
3 Realiza las operaciones de lectura
4 Desliga la variable del archivo de texto (cierre el archivo)
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 254


Apertura del archivo
Conecta la variable con el archivo de texto del dispositivo
flujo.open(cadena_literal);
¡El archivo debe existir!
ifstream archivo;
is_open():
archivo.open("abc.txt");
true si el archivo
if (archivo.is_open()) ...
se ha podido abrir
false en caso contrario
Cierre del archivo
Desconecta la variable del archivo de texto del dispositivo
flujo.close();
Luis Hernández Yáñez

archivo.close();

Fundamentos de la programación: Tipos e instrucciones II Página 255

Operaciones de lectura
 Extractor (>>) archivo >> variable;
Salta primero los espacios en blanco (espacio, tab, Intro, ...)
Datos numéricos: lee hasta el primer carácter no válido
Cadenas (string): lee hasta el siguiente espacio en blanco
 archivo.get(c)
Lee el siguiente carácter en la variable c, sea el que sea
 getline(archivo, cadena)
Lee en la cadena todos los caracteres que queden en la línea
Incluidos los espacios en blanco
Luis Hernández Yáñez

Hasta el siguiente salto de línea (descartándolo)


Con los archivos no tiene efecto la función sync()

Fundamentos de la programación: Tipos e instrucciones II Página 256


¿Qué debo leer?
 Un número
Usa el extractor archivo >> num;

 Un carácter (sea el que sea)


Usa la función get() archivo.get(c);

 Una cadena sin espacios


Usa el extractor archivo >> cad;

 Una cadena posiblemente con espacios


Usa la función getline() getline(archivo, cad);
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 257

¿Dónde queda pendiente la entrada?


 Número leído con el extractor
En el primer carácter no válido (inc. espacios en blanco)

 Carácter leído con get()


En el siguiente carácter (inc. espacios en blanco)

 Una cadena leída con el extractor


En el siguiente espacio en blanco

 Una cadena leída con la función getline()


Al principio de la siguiente línea
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 258


string nif, producto;
int unidades;
double precio;
char aux;

1 ifstream archivo; 7
6
Flujo de entrada
2 archivo.open("compras.txt"); // Apertura 5
archivo
4

3 archivo >> nif >> unidades >> precio; 3
2
getline(archivo, producto); 1

Programa
4 archivo.close(); // Cierre
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 259

archivo >> nif;
archivo >> unidades;
archivo >> precio;
getline(archivo, producto);

getline() no salta espacios


12345678F 2 123.95 Reproductor de DVD
El extractor salta los espacios

nif 12345678F unidades 2


Luis Hernández Yáñez

producto Reproductor de DVD precio 123.95

Espacio

Fundamentos de la programación: Tipos e instrucciones II Página 260


archivo >> nif;
archivo >> unidades;
archivo >> precio;
archivo.get(aux); // Salta el espacio en blanco
getline(archivo, producto);

12345678F 2 123.95 Reproductor de DVD
Leemos el espacio
(no hacemos nada con él)

nif 12345678F unidades 2


Luis Hernández Yáñez

producto Reproductor de DVD precio 123.95

Sin espacio

Fundamentos de la programación: Tipos e instrucciones II Página 261

Cada línea, datos de una compra


Mostrar el total de cada compra
unidades x precio más IVA (21%)
Final: "X" como NIF
Bucle de procesamiento:
 Cada paso del bucle (ciclo) procesa una línea (compra)
 Podemos usar las mismas variables en cada ciclo

Leer primer NIF


Mientras el NIF no sea X:
Leer unidades, precio y descripción
Luis Hernández Yáñez

Calcular y mostrar el total


Leer el siguiente NIF

Fundamentos de la programación: Tipos e instrucciones II Página 262


#include <iostream> leer.cpp
#include <string>
using namespace std;
#include <fstream>
#include <iomanip> // Formato de salida

int main() {
const int IVA = 21;
string nif, producto;
int unidades;
double precio, neto, total, iva;
char aux;
ifstream archivo;
int contador = 0;
Luis Hernández Yáñez

archivo.open("compras.txt"); // Apertura
...

Fundamentos de la programación: Tipos e instrucciones II Página 263

if (archivo.is_open()) { // Existe el archivo
archivo >> nif; // Primer NIF
while (nif != "X") {
archivo >> unidades >> precio;
archivo.get(aux); // Salta el espacio
getline(archivo, producto);
contador++;
neto = unidades * precio;
iva = neto * IVA / 100;
total = neto + iva;
cout << "Compra " << contador << ".‐" << endl;
cout << "   " << producto << ": " << unidades
<< " x " << fixed << setprecision(2)
<< precio << " = " << neto << " ‐ I.V.A.: "
Luis Hernández Yáñez

<< iva << " ‐ Total: " << total << endl;


archivo >> nif; // Siguiente NIF
} ...

Fundamentos de la programación: Tipos e instrucciones II Página 264


archivo.close(); // Cierre
}
else {
cout << "ERROR: No se ha podido abrir el archivo"
<< endl;
}
return 0;
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 265


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 266


Flujos de texto de salida ofstream

Para crear un archivo de texto y escribir en él:


1 Declara una variable de tipo ofstream
2 Asocia la variable con el archivo de texto (crea el archivo)
3 Realiza las escrituras por medio del operador << (insertor)
4 Desliga la variable del archivo de texto (cierra el archivo)

¡Atención!
Si el archivo ya existe, se borra todo lo que hubiera
Luis Hernández Yáñez

¡Atención!
Si no se cierra el archivo se puede perder información

Fundamentos de la programación: Tipos e instrucciones II Página 267

int valor = 999;
Programa
2
1 ofstream archivo; 1

Flujo de salida
2 archivo.open("output.txt"); // Apertura ! archivo
a
l
3 archivo << 'X' << " Hola! " << 123.45 o

<< endl << valor << "Bye!"; H

X
4 archivo.close(); // Cierre
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 268


escribir.cpp

#include <iostream>
#include <string>
using namespace std;
#include <fstream>

int main() {
string nif, producto;
int unidades;
double precio;
char aux;
ofstream archivo;

archivo.open("output.txt"); // Apertura (creación)
Luis Hernández Yáñez

cout << "NIF del cliente (X para terminar): ";
cin >> nif;
...

Fundamentos de la programación: Tipos e instrucciones II Página 269

while (nif != "X") {


// Queda pendiente el Intro anterior...
cin.get(aux); // Leemos el Intro
cout << "Producto: ";
getline(cin, producto);
cout << "Unidades: ";
cin >> unidades;
cout << "Precio: ";
cin >> precio;
// Escribimos los datos en una línea del archivo...
// Con un espacio de separación entre ellos
archivo << nif << " " << unidades << " "
<< precio << " " << producto << endl;
cout << "NIF del cliente (X para terminar): ";
Luis Hernández Yáñez

cin >> nif;
}
...

Fundamentos de la programación: Tipos e instrucciones II Página 270


// Escribimos el centinela final...
archivo << "X";
archivo.close(); // Cierre del archivo

return 0;
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 271


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 272


{
double oper1, oper2, prod;

Instrucción 1 cout << "Operando 1: ";
Flujo de ejecución

Instrucción 2 cin >> oper1;

Instrucción 3 cout << "Operando 2: ";

...

cout << "Producto: " << prod;
Luis Hernández Yáñez

Instrucción N return 0;

Fundamentos de la programación: Tipos e instrucciones II Página 273

Uno entre dos o más caminos de ejecución


Selección simple (2 caminos) Selección múltiple (> 2 caminos)

true

true false false


Condición true

false
Instrucción T Instrucción F
true

false

if true
if‐else‐if
Luis Hernández Yáñez

false
switch
Diagramas de flujo

Fundamentos de la programación: Tipos e instrucciones II Página 274


Repetir la ejecución de una o más instrucciones
Acumular, procesar colecciones, ...

Inicialización

Sí No
¿Iterar?

Código
Luis Hernández Yáñez

while for

Fundamentos de la programación: Tipos e instrucciones II Página 275


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 276


La instrucción if
if (condición) { true false
Condición
códigoT
} BloqueT BloqueF
Opcional
[else {
códigoF
}]
condición: expresión bool
Cláusula else opcional
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 277

signo.cpp

int num;
cin >> num;
if (num < 0) {
cout << "Negativo";
} cin >> num;
else {
true false
cout << "Positivo"; num < 0
}
cout << "Negativo"; cout << "Positivo";
cout << endl;
Luis Hernández Yáñez

cout << endl;

Fundamentos de la programación: Tipos e instrucciones II Página 278


int num;
cin >> num; 1_29
129
if (num < 0) { _
Positivo
cout << "Negativo"; _
}
else {
cout << "Positivo"; cin >> num;
}
false
cout << endl;
num < 0 num 129
?

cout << "Positivo";
Luis Hernández Yáñez

cout << endl;

Fundamentos de la programación: Tipos e instrucciones II Página 279

int num;
cin >> num; ‐5
if (num < 0) { Negativo
cout << "Negativo"; _
}
else {
cout << "Positivo"; cin >> num;
}
cout << endl; true
num < 0 num ‐5
?

cout << "Negativo";
Luis Hernández Yáñez

cout << endl;

Fundamentos de la programación: Tipos e instrucciones II Página 280


división.cpp

División entre dos números protegida frente a intento de división por 0


#include <iostream>
using namespace std;

int main() {
double numerador, denominador, resultado;
cout << "Numerador: "; 
cin >> numerador;
cout << "Denominador: "; 
cin >> denominador;
if (denominador == 0) {
cout << "Imposible dividir entre 0!";
}
else {
resultado = numerador / denominador;
Luis Hernández Yáñez

cout << "Resultado: " << resultado << endl;
}
return 0;
}
Fundamentos de la programación: Tipos e instrucciones II Página 281
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 282


Se aplican a valores bool (condiciones)
El resultado es de tipo bool

Operadores (prioridad)
! NO Monario
...
&& Y Binario
!
|| O Binario
* / %
+ ‐
< <= > >=
== !=
Luis Hernández Yáñez

&&
||

Fundamentos de la programación: Tipos e instrucciones II Página 283

! && true false || true false


true false true true false true true true
false true false false false false true false

NO (Not) Y (And) O (Or)

bool cond1, cond2, resultado;
int a = 2, b = 3, c = 4;
resultado = !(a < 5);        // !(2 < 5)  !true  false
cond1 = (a * b + c) >= 12;   // 10 >= 12  false
cond2 = (a * (b + c)) >= 12; // 14 >= 12  true
Luis Hernández Yáñez

resultado = cond1 && cond2;  // false && true  false
resultado = cond1 || cond2;  // false || true  true

Fundamentos de la programación: Tipos e instrucciones II Página 284


condiciones.cpp

#include <iostream>
using namespace std;

int main()
{
int num;
cout << "Introduce un número entre 1 y 10: ";
cin >> num;
if ((num >= 1) && (num <= 10)) {
cout << "Número dentro del intervalo de valores válidos";
}
else {
cout << "Número no válido!";
}
return 0; Condiciones equivalentes
((num >= 1) && (num <= 10))
Luis Hernández Yáñez

}
((num > 0) && (num < 11))
¡Encierra las condiciones ((num >= 1) && (num < 11))
simples entre paréntesis! ((num > 0) && (num <= 10))

Fundamentos de la programación: Tipos e instrucciones II Página 285


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 286


diasmes.cpp

int mes, anio, dias;
cout << "Número de mes: "; 
cin >> mes;
cout << "Año: "; 
cin >> anio;
if (mes == 2) {
if (bisiesto(mes, anio)) {
dias = 29;
}
else {
dias = 28;
}
}
else {
if ((mes == 1) || (mes == 3) || (mes == 5) || (mes == 7) 
|| (mes == 8) || (mes == 10) || (mes == 12)) {
dias = 31;
Luis Hernández Yáñez

}
else {
dias = 30;
}
}
Fundamentos de la programación: Tipos e instrucciones II Página 287

Calendario Gregoriano: bisiesto si divisible por 4, excepto el último


de cada siglo (divisible por 100), salvo que sea divisible por 400
bool bisiesto(int mes, int anio) {
bool esBisiesto;
if ((anio % 4) == 0) { // Divisible por 4
if (((anio % 100) == 0) && ((anio % 400) != 0)) {
// Pero último de siglo y no múltiplo de 400
esBisiesto = false;
}
else {
esBisiesto = true; // Año bisiesto
}
}
else {
Luis Hernández Yáñez

esBisiesto = false;
}
return esBisiesto;
}

Fundamentos de la programación: Tipos e instrucciones II Página 288


Cada else se asocia al if anterior más cercano sin asociar (mismo bloque)
if (condición1) { 
if (condición2) {...} Una mala sangría puede confundir
if (x > 0) {
else {...}
if (y > 0) {...}
}
else {...}
else {
if (condición3) {
if (condición4) {...} if (x > 0) {
if (condición5) {...} if (y > 0) {...}
else {...} else {...}

}
Luis Hernández Yáñez

else { ...

La sangría ayuda a asociar los else con sus if

Fundamentos de la programación: Tipos e instrucciones II Página 289


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 290


• Condición simple: Expresión lógica (true/false)
Sin operadores lógicos
num < 0
car == 'a' Compatibilidad con el lenguaje C:
isalpha(car) 0 es equivalente a false
12 Cualquier valor distinto de 0 es equivalente a true

• Condición compuesta:
Combinación de condiciones simples y operadores lógicos
!isalpha(car)
(num < 0) || (car == 'a')
(num < 0) && ((car == 'a') || !isalpha(car))
Luis Hernández Yáñez

No confundas el operador de igualdad (==)


con el operador de asignación (=).

Fundamentos de la programación: Tipos e instrucciones II Página 291

Shortcut Boolean Evaluation


true || X  true
(n == 0) || (x >= 1.0 / n)
Si n es 0: ¿división por cero? (segunda condición)
Como la primera sería true: ¡no se evalúa la segunda!

false && X  false


(n != 0) && (x < 1.0 / n)
Si n es 0: ¿división por cero? (segunda condición)
Luis Hernández Yáñez

Como la primera sería false: ¡no se evalúa la segunda!

Fundamentos de la programación: Tipos e instrucciones II Página 292


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 293

true

false
true

false
true

false

true

false
Luis Hernández Yáñez

if‐else‐if
switch

Fundamentos de la programación: Tipos e instrucciones II Página 294


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 295

Ejemplo:
Calificación (en letras)
de un estudiante en base
a su nota numérica (0‐10) == 10
true
cout << "MH"
false
true
>= 9 cout << "SB"
Si nota == 10 entonces MH false
true
si no, si nota >= 9 entonces SB >= 7 cout << "NT"
false
si no, si nota >= 7 entonces NT >= 5
true
cout << "AP"
si no, si nota >= 5 entonces AP false
cout << "SS"
si no SS
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 296


nota.cpp

double nota; double nota;


cin >> nota; cin >> nota;
if (nota == 10) {
cout << "MH"; if (nota == 10) {
} cout << "MH";
else { }
if (nota >= 9) {
cout << "SB"; else if (nota >= 9) { 
} cout << "SB";
else { }
if (nota >= 7) {
cout << "NT";  else if (nota >= 7) {
} cout << "NT";
else { }
if (nota >= 5) {
cout << "AP"; else if (nota >= 5) {
} cout << "AP";
else { }
Luis Hernández Yáñez

cout << "SS";
} else {
} cout << "SS";
} }
}
Fundamentos de la programación: Tipos e instrucciones II Página 297

¡Cuidado con el orden de las condiciones!


double nota;
cin >> nota;


if (nota < 5) { cout << "SS"; }
else if (nota < 7) { cout << "AP"; }
else if (nota < 9) { cout << "NT"; }
else if (nota < 10) { cout << "SB"; }
else { cout << "MH"; }

double nota; ¡No se ejecutan nunca!


cin >> nota;


if (nota >= 5) { cout << "AP"; }
else if (nota >= 7) { cout << "NT"; }
Luis Hernández Yáñez

else if (nota >= 9) { cout << "SB"; }
else if (nota == 10) { cout << "MH"; }
else { cout << "SS"; }
Sólo muestra AP o SS
Fundamentos de la programación: Tipos e instrucciones II Página 298
Simplificación de las condiciones

0 SS 5 AP 7 NT 9 SB 10
if (nota == 10) { cout << "MH"; } MH
else if ((nota < 10) && (nota >= 9)) { cout << "SB"; }
else if ((nota < 9) && (nota >= 7)) { cout << "NT"; }
else if ((nota < 7) && (nota >= 5)) { cout << "AP"; }
else if (nota < 5)  { cout << "SS"; }
Siempre true: ramas else
Si no es 10, es menor que 10
if (nota == 10) { cout << "MH"; } Si no es >= 9, es menor que 9
else if (nota >= 9) {cout << "SB"; } Si no es >= 7, es menor que 7
Luis Hernández Yáñez

else if (nota >= 7) { cout << "NT"; } …
else if (nota >= 5) { cout << "AP"; } true && X  X
else { cout << "SS"; }

Fundamentos de la programación: Tipos e instrucciones II Página 299

nivel.cpp

#include <iostream> Si num == 4 entonces Muy alto


using namespace std; Si num == 3 entonces Alto
int main() {
int num; Si num == 2 entonces Medio
cout << "Introduce el nivel: "; Si num == 1 entonces Bajo
cin >> num;
if (num == 4) {
cout << "Muy alto" << endl;
}
else if (num == 3) {
cout << "Alto" << endl;
}
else if (num == 2) {
cout << "Medio" << endl;
}
else if (num == 1) {
cout << "Bajo" << endl;
}
Luis Hernández Yáñez

else {
cout << "Valor no válido" << endl;
}
return 0;
}
Fundamentos de la programación: Tipos e instrucciones II Página 300
if (num == 4) { cout << "Muy alto" << endl; }
else if (num == 3) { cout << "Alto" << endl; }
else if (num == 2) { cout << "Medio" << endl; }
else if (num == 1) { cout << "Bajo" << endl; }
else cout << "Valor no válido" << endl; }

if (num == 4) cout << "Muy alto";
else if (num == 3) cout << "Alto";
else if (num == 2) cout << "Medio";
else if (num == 1) cout << "Bajo";
Luis Hernández Yáñez

else cout << "Valor no válido";
cout << endl;

Fundamentos de la programación: Tipos e instrucciones II Página 301


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 302


Selección entre valores posibles de una expresión
switch (expresión) { case constanteN:
case constante1: {
{ códigoN
código1 }
} [break;]
[break;] [default:
case constante2: {
{ códigoDefault
código2 }]
} }
[break;]
Luis Hernández Yáñez

...

Fundamentos de la programación: Tipos e instrucciones II Página 303

nivel2.cpp

switch (num) {
case 4:
Si num == 4  Muy alto
{ Si num == 3  Alto
}
cout << "Muy alto"; Si num == 2  Medio
break; Si num == 1  Bajo
case 3:
{
cout << "Alto";
}
break;
case 2:
{
cout << "Medio";
}
break;
case 1:
{
cout << "Bajo";
}
break;
Luis Hernández Yáñez

default:
{
cout << "Valor no válido";
}
}

Fundamentos de la programación: Tipos e instrucciones II Página 304


Interrumpe el switch; continúa en la instrucción que le siga
switch (num) {
...
case 3:
{
cout << "Alto";
Num: 3
}
Alto
break;
case 2:
{
cout << "Medio";
}
break;
Luis Hernández Yáñez

...
}

Fundamentos de la programación: Tipos e instrucciones II Página 305

switch (num) {
...
case 3:
{
cout << "Alto";
}
case 2:
{ Num: 3
cout << "Medio"; Alto
} Medio
case 1: Bajo
{ Valor no válido
cout << "Bajo";
}
default:
Luis Hernández Yáñez

{
cout << "Valor no válido";
}
}
Fundamentos de la programación: Tipos e instrucciones II Página 306
true
num==4 Muy alto
break;
false
Sin break;
true
num==3 Alto
break;
false
Sin break;
true
num==2 Medio
break;
false
Sin break;
true
num==1 Bajo break;
false Sin break;

No válido
Luis Hernández Yáñez

default

Fundamentos de la programación: Tipos e instrucciones II Página 307

int menu() { 1 ‐ Nuevo cliente


int op = ‐1; // Cualquiera no válida 2 ‐ Editar cliente
3 ‐ Baja cliente
4 ‐ Ver cliente
while ((op < 0) || (op > 4)) {
0 ‐ Salir
cout << "1 ‐ Nuevo cliente" << endl; Opción: 5
cout << "2 ‐ Editar cliente" << endl; ¡Opción no válida!
cout << "3 ‐ Baja cliente" << endl; 1 ‐ Nuevo cliente
2 ‐ Editar cliente
cout << "4 ‐ Ver cliente" << endl; 3 ‐ Baja cliente
cout << "0 ‐ Salir" << endl; 4 ‐ Ver cliente
cout << "Opción: "; 0 ‐ Salir
Opción: 3
cin >> op;

if ((op < 0) || (op > 4)) {


cout << "¡Opción no válida!" << endl;
}
Luis Hernández Yáñez

return op;
}

Fundamentos de la programación: Tipos e instrucciones II Página 308


int opcion;
...
opcion = menu();
switch (opcion) {
case 1:
{
cout << "En la opción 1..." << endl;
}
break;
case 2:
{
cout << "En la opción 2..." << endl;
}
break;
case 3:
{
cout << "En la opción 3..." << endl;
}
break;
case 4:
Luis Hernández Yáñez

{
cout << "En la opción 4..." << endl;
} // En la última no necesitamos break
}

Fundamentos de la programación: Tipos e instrucciones II Página 309

int opcion;
...
opcion = menu();
while (opcion != 0) {
switch (opcion) {
case 1:
{
cout << "En la opción 1..." << endl;
}
break;
...
case 4:
{
cout << "En la opción 4..." << endl;
}
} // switch
...
Luis Hernández Yáñez

opcion = menu();
} // while

Fundamentos de la programación: Tipos e instrucciones II Página 310


nota2.cpp

int nota; // Sin decimales case 7:


cout << "Nota (0‐10): "; case 8:
cin >> nota; {
switch (nota) { cout << "Notable";
case 0: }
case 1: break; // 7 u 8: NT
case 2: case 9:
case 3: case 10:
case 4: {
{ cout << "Sobresaliente";
cout << "Suspenso"; }
} break; // 9 o 10: SB
break; // De 0 a 4: SS default:
case 5: {
case 6: cout << "¡No válida!";
{ }
Luis Hernández Yáñez

cout << "Aprobado"; }
}
break; // 5 o 6: AP

Fundamentos de la programación: Tipos e instrucciones II Página 311

typedef enum { enero, febrero, marzo, abril, mayo, junio,
julio, agosto, septiembre, octubre, noviembre, diciembre }
tMes;
tMes mes;
...
switch (mes) {
case enero:
{
cout << "enero";
}
break; 
case febrero:
{
cout << "febrero";
}
break;
...
Luis Hernández Yáñez

case diciembre:
{
cout << "diciembre";
}
}
Fundamentos de la programación: Tipos e instrucciones II Página 312
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 313

Inicialización

Sí No
¿Iterar?

Cuerpo

Bucles while y for


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 314


 Número de iteraciones condicionado (recorrido variable):
— Bucle while
while (condición) cuerpo
Ejecuta el cuerpo mientras la condición sea true
— Bucle do‐while
Comprueba la condición al final (lo veremos más adelante)

 Número de iteraciones prefijado (recorrido fijo):


— Bucle for
for (inicialización; condición; paso) cuerpo
Luis Hernández Yáñez

Ejecuta el cuerpo mientras la condición sea true


Se usa una variable contadora entera

Fundamentos de la programación: Tipos e instrucciones II Página 315


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 316


while.cpp

Mientras la condición sea cierta, ejecuta el cuerpo


while (condición) {
cuerpo Condición al principio del bucle
}

int i = 1; // Inicialización de la variable i
while (i <= 100) {
cout << i << endl;
i++;
}
Muestra los números del 1 al 100
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 317

int i = 1;
true false
while (i <= 100) { Condición
cout << i << endl;  Cuerpo
i++;
}
i = 1

i 100
101
99
?3
2
1
4 _
1
true false 2
i <= 100
3

cout << i << endl;
99
100
Luis Hernández Yáñez

i++ _

Fundamentos de la programación: Tipos e instrucciones II Página 318


¿Y si la condición es falsa al comenzar?
No se ejecuta el cuerpo del bucle ninguna vez
int op;
cout << "Introduce la opción: ";
cin >> op;
while ((op < 0) || (op > 4)) {
cout << "¡No válida! Inténtalo otra vez" << endl;
cout << "Introduce la opción: ";
cin >> op;
}
Si el usuario introduce un número entre 0 y 4:
No se ejecuta el cuerpo del bucle
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 319

primero.cpp

Primer entero cuyo cuadrado es mayor que 1.000


#include <iostream> ¡Ejecuta el programa para
using namespace std; saber cuál es ese número!
int main() {
int num = 1; Empezamos en 1
while (num * num <= 1000) {
num++; Incrementamos en 1
}

cout << "1er. entero con cuadrado mayor que 1.000: "
<< num << endl;
Luis Hernández Yáñez

return 0;
}
Recorre la secuencia de números 1, 2, 3, 4, 5, ...
Fundamentos de la programación: Tipos e instrucciones II Página 320
sumamedia.cpp

#include <iostream>
using namespace std; Recorre la secuencia
int main() { de números introducidos
double num, suma = 0, media = 0;
int cont = 0;
cout << "Introduce un número (0 para terminar): ";
cin >> num; Leemos el primero
while (num != 0) { // 0 para terminar
suma = suma + num;
cont++;
cout << "Introduce un número (0 para terminar): ";
cin >> num; Leemos el siguiente
}
if (cont > 0) {
media = suma / cont;
}
Luis Hernández Yáñez

cout << "Suma = " << suma << endl;
cout << "Media = " << media << endl;
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 321


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 322


Número de iteraciones prefijado
Variable contadora que determina el número de iteraciones:
for ([int] var = ini; condición; paso) cuerpo
La condición compara el valor de var con un valor final
El paso incrementa o decrementa el valor de var
El valor de var debe ir aproximándose al valor final
for (int i = 1; i <= 100; i++)... 1, 2, 3, 4, 5, ..., 100
for (int i = 100; i >= 1; i‐‐)... 100, 99, 98, 97, ..., 1

Tantos ciclos como valores toma la variable contadora


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 323

for (inicialización; condición; paso) cuerpo
for (int i = 1; i <= 100; i++) {
cout << i;
}
i = 1

true false
i <= 100

cout << i;

i++
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 324


for1.cpp

for (int i = 1; i <= 100; i++) {
cout << i << endl;
}

i 2?3
100
101
1
1
... i = 1 2
3
true false
i <= 100
99
cout << i << endl; 100
_
i++
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 325

La variable contadora for2.cpp

El paso no tiene porqué ir de uno en uno:


for (int i = 1; i <= 100; i = i + 2)
cout << i << endl;
Este bucle for muestra los números impares de 1 a 99

Muy importante
El cuerpo del bucle NUNCA debe alterar el valor del contador

Garantía de terminación
Todo bucle debe terminar su ejecución
Luis Hernández Yáñez

Bucles for: la variable contadora debe converger al valor final

Fundamentos de la programación: Tipos e instrucciones II Página 326


suma.cpp

#include <iostream>
using namespace std;
long long int suma(int n);
int main() {
int num;
cout << "Número final: "; 
cin >> num;
if (num > 0) { // El número debe ser positivo
cout << "La suma de los números entre 1 y " 
<< num << " es: " << suma(num);
}
return 0;
}
long long int suma(int n) {
long long int total = 0;
Luis Hernández Yáñez

for (int i = 1; i <= n; i++) { 
total = total + i;
} Recorre la secuencia de números
return total; 1, 2, 3, 4, 5, ..., n
}
Fundamentos de la programación: Tipos e instrucciones II Página 327

¿Incremento/decremento prefijo o postfijo?


Es indiferente
Estos dos bucles producen el mismo resultado:
for (int i = 1; i <= 100; i++) ...
for (int i = 1; i <= 100; ++i) ...

Bucles infinitos
for (int i = 1; i <= 100; i‐‐) ...
1 0 ‐1 ‐2 ‐3 ‐4 ‐5 ‐6 ‐7 ‐8 ‐9 ‐10 ‐11 ...
Luis Hernández Yáñez

Cada vez más lejos del valor final (100)

Es un error de diseño/programación

Fundamentos de la programación: Tipos e instrucciones II Página 328


Declarada en el propio bucle
for (int i = 1; ...)
Sólo se conoce en el cuerpo del bucle (su ámbito)
No se puede usar en instrucciones que sigan al bucle

Declarada antes del bucle


int i;
for (i = 1; ...)
Se conoce en el cuerpo del bucle y después del mismo
Ámbito externo al bucle
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 329

Los bucles for se pueden reescribir como bucles condicionados


for (int i = 1; i <= 100; i++) cuerpo
Es equivalente a:
int i = 1;
while (i <= 100) {
cuerpo
i++;
}
La inversa no es siempre posible:
int i;
cin >> i; ¿Bucle for equivalente?
while (i != 0) {
Luis Hernández Yáñez

cuerpo ¡No sabemos cuántos números


cin >> i; introducirá el usuario!
}

Fundamentos de la programación: Tipos e instrucciones II Página 330


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 331

Un bucle for en el cuerpo de otro bucle for


Cada uno con su propia variable contadora:
for (int i = 1; i <= 100; i++) { i j
for (int j = 1; j <= 5; j++) { 1 1
cuerpo 1 2
1 3
} 1 4
} 1 5
Para cada valor de i 2 1
2 2
el valor de j varía entre 1 y 5
2 3
j varía más rápido que i 2 4
2 5
Luis Hernández Yáñez

3 1
...

Fundamentos de la programación: Tipos e instrucciones II Página 332


tablas.cpp

#include <iostream>
using namespace std; 1 x  1 =   1
#include <iomanip> 1 x  2 =   2
1 x  3 =   3
1 x  4 =   4
int main() {
...
for (int i = 1; i <= 10; i++) { 1 x 10 =  10
for (int j = 1; j <= 10; j++) { 2 x  1 =   2
cout << setw(2) << i << " x " 2 x  2 =   4
<< setw(2) << j << " = " ...
10 x  7 =  70
<< setw(3) << i * j << endl; 10 x  8 =  80
} 10 x  9 =  90
} 10 x 10 = 100
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 333

tablas2.cpp

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

int main() {
for (int i = 1; i <= 10; i++) {
cout << "Tabla del " << i << endl;
cout << "‐‐‐‐‐‐‐‐‐‐‐‐‐‐" << endl;
for (int j = 1; j <= 10; j++) {
cout << setw(2) << i << " x "
<< setw(2) << j << " = "
<< setw(3) << i * j << endl;
}
cout << endl;
Luis Hernández Yáñez

return 0;
}
Fundamentos de la programación: Tipos e instrucciones II Página 334
menú.cpp

#include <iostream>
using namespace std;
#include <iomanip>
int menu(); // 1: Tablas de multiplicación; 2: Sumatorio
long long int suma(int n); // Sumatorio
int main() {
int opcion = menu();
while (opcion != 0) {
switch (opcion) {
case 1:
{
for (int i = 1; i <= 10; i++) {
for (int j = 1; j <= 10; j++) {
cout << setw(2) << i << " x "
<< setw(2) << j << " = "
Luis Hernández Yáñez

<< setw(3) << i * j << endl;
}
}
}
break; ...
Fundamentos de la programación: Tipos e instrucciones II Página 335

case 2:
{
int num = 0;
while (num <= 0) {
cout << "Hasta (positivo)? "; 
cin >> num;
}
cout << "La suma de los números del 1 al "
<< num << " es: " << suma(num) << endl;
}
} // switch
opcion = menu();
} // while (opcion != 0)
return 0;
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 336


int menu() {
int op = ‐1;
while ((op < 0) || (op > 2)) {
cout << "1 ‐ Tablas de multiplicar" << endl;
cout << "2 ‐ Sumatorio" << endl;
cout << "0 ‐ Salir" << endl;
cout << "Opción: " << endl;
cin >> op;
if ((op < 0) || (op > 2)) {
cout << "¡Opción no válida!" << endl;
}
}
return op;
}
long long int suma(int n) {
long long int total = 0;
for (int i = 1; i <= n; i++) { 
Luis Hernández Yáñez

total = total + i;
}
return total;
}
Fundamentos de la programación: Tipos e instrucciones II Página 337

while (opcion != 0) {
...
for (int i = 1; i <= 10; i++) {
for (int j = 1; j <= 10; j++) {
...
}
}
while (num <= 0) {
...
} suma()
for (int i = 1; i <= n; i++) {
...
}
while ((op < 0) || (op > 2)) {
Luis Hernández Yáñez

...
} menu()
}
Fundamentos de la programación: Tipos e instrucciones II Página 338
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 339

Cada bloque crea un nuevo ámbito:


int main() {
double d = ‐1, suma = 0; 3 ámbitos anidados
int cont = 0;
while (d != 0) {
cin >> d;
if (d != 0) {
suma = suma + d;
cont++;
}
}
cout << "Suma = " << suma << endl;
Luis Hernández Yáñez

cout << "Media = " << suma / cont << endl;
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 340


Un identificador se conoce
en el ámbito en el que está declarado
(a partir de su instrucción de declaración)
y en los subámbitos posteriores
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 341

int main() {
double d; Ámbito de la variable d
if (...) {
int cont = 0;
for (int i = 0; i <= 10; i++) {
...
}
}
char c;
if (...) {
double x;
...
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 342


int main() {
double d;
if (...) {
int cont = 0; Ámbito de la variable cont
for (int i = 0; i <= 10; i++) {
...
}
}
char c;
if (...) {
double x;
...
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 343

int main() {
double d;
if (...) {
int cont = 0;
for (int i = 0; i <= 10; i++) {
...
} Ámbito de la variable i
}
char c;
if (...) {
double x;
...
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 344


int main() {
double d;
if (...) {
int cont = 0;
for (int i = 0; i <= 10; i++) {
...
}
}
char c;
if (...) { Ámbito de la variable c
double x;
...
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 345

int main() {
double d;
if (...) {
int cont = 0;
for (int i = 0; i <= 10; i++) {
...
}
}
char c;
if (...) {
double x;
... Ámbito de la variable x
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 346


Si en un subámbito se declara
un identificador con idéntico nombre
que uno ya declarado en el ámbito,
el del subámbito oculta al del ámbito
(no es visible)
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 347

int main() {
int i, x; Oculta , en su ámbito, a la i anterior
if (...) {
int i = 0; Oculta , en su ámbito, a la i anterior
for(int i = 0; i <= 10; i++) {
...
}
}
char c;
if (...) {
double x; Oculta , en su ámbito, a la x anterior
...
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II Página 348


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 349

Sucesión de elementos de un mismo tipo que se acceden linealmente

elemento secuencia
secuencia o

(Secuencia vacía) elemento

1 34 12 26 4 87 184 52
Comienza en un primer elemento (si no está vacía)
A cada elemento le sigue otra secuencia (vacía, si es el último)
Acceso secuencial (lineal)
Se comienza siempre accediendo al primer elemento
Desde un elemento sólo se puede acceder a su elemento siguiente
Luis Hernández Yáñez

(sucesor), si es que existe


Todos los elementos, de un mismo tipo

Fundamentos de la programación: Tipos e instrucciones II Página 350


No tratamos secuencias infinitas: siempre hay un último elemento
 Secuencias explícitas:
— Sucesión de datos de un dispositivo (teclado, disco, sensor, ...)
 Secuencias calculadas:
— Fórmula de recurrencia que determina el elemento siguiente
 Listas (más adelante)

Secuencias explícitas que manejaremos:


Datos introducidos por el teclado o leídos de un archivo
Con un elemento especial al final de la secuencia (centinela)
Luis Hernández Yáñez

1 34 12 26 4 87 184 52 ‐1

Fundamentos de la programación: Tipos e instrucciones II Página 351

 Secuencia explícita leída de archivo:


— Detectar la marca de final de archivo (Eof ‐ End of file)

— Detectar un valor centinela al final

 Secuencia explícita leída del teclado:


— Preguntar al usuario si quiere introducir un nuevo dato

— Preguntar al usuario primero cuántos datos va a introducir

— Detectar un valor centinela al final

Valor centinela:
Valor especial al final que no puede darse en la secuencia
(Secuencia de números positivos  centinela: cualquier negativo)
Luis Hernández Yáñez

12 4 37 23 8 19 83 63 2 35 17 76 15 ‐1

Fundamentos de la programación: Tipos e instrucciones II Página 352


Debe haber algún valor que no sea un elemento válido
Secuencias numéricas:
Si se permite cualquier número, no hay centinela posible
Cadenas de caracteres:
¿Caracteres especiales (no imprimibles)?

En realidad el valor centinela es parte de la secuencia,


pero su significado es especial y no se procesa como el resto
Significa que se ha alcanzado el final de la secuencia
(Incluso aunque haya elementos posteriores)
Último elemento
Luis Hernández Yáñez

12 4 37 23 8 19 83 63 ‐1 35 17 76 15

No se procesan

Fundamentos de la programación: Tipos e instrucciones II Página 353

Tratamiento de los elementos uno a uno desde el primero


Recorrido
Un mismo tratamiento para todos los elementos de la secuencia
Ej.‐ Mostrar los elementos de una secuencia, sumar los números
de una secuencia, ¿par o impar cada número de una secuencia?, ...
Termina al llegar al final de la secuencia

Búsqueda
Recorrido de la secuencia hasta encontrar un elemento buscado
Ej.‐ Localizar el primer número que sea mayor que 1.000
Termina al localizar el primer elemento que cumple la condición
Luis Hernández Yáñez

o al llegar al final de la secuencia (no encontrado)

Fundamentos de la programación: Tipos e instrucciones II Página 354


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 355

Un mismo tratamiento a todos los elementos


Inicialización
Mientras no se llegue al final de la secuencia:
Obtener el siguiente elemento
Procesar el elemento
Finalización

Al empezar se obtiene el primer elemento de la secuencia


En los siguientes pasos del bucle se van obteniendo
los siguientes elementos de la secuencia
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 356


Inicialización

true
¿Al final?
false No sabemos cuántos
elementos hay
Obtener elemento
 No podemos
Procesar elemento implementar con for
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Tipos e instrucciones II Página 357

Implementación con while


Inicialización
Inicialización
Obtener el primer elemento Obtener 1º

Mientras no sea el centinela:


Procesar el elemento true
¿Centinela?
Obtener el siguiente elemento false
Finalización
Procesar elemento

Obtener siguiente
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Tipos e instrucciones II Página 358


Secuencia de números positivos
Siempre se realiza al menos una lectura
Centinela: ‐1

double d, suma = 0; Inicialización


cout << "Valor (‐1 termina): ";
Primer elemento
cin >> d;
while (d != ‐1) { Mientras no el centinela
suma = suma + d; Procesar elemento
cout << "Valor (‐1 termina): ";
Siguiente elemento
cin >> d;
Luis Hernández Yáñez

}
cout << "Suma = " << suma << endl; Finalización

Fundamentos de la programación: Tipos e instrucciones II Página 359

Longitud de una secuencia de caracteres longitud.cpp

Centinela: carácter punto (.)

int longitud() {
int l = 0;
char c;
cout << "Texto terminado en punto: ";
cin >> c;          // Obtener primer carácter
while (c != '.') { // Mientras no el centinela
l++;            // Procesar
cin >> c;       // Obtener siguiente carácter
}
Luis Hernández Yáñez

return l;
}

Fundamentos de la programación: Tipos e instrucciones II Página 360


¿Cuántas veces aparece un carácter en una cadena?
Centinela: asterisco (*) cont.cpp
char buscado, c;
int cont = 0;
cout << "Carácter a buscar: ";
cin >> buscado;
cout << "Cadena: ";
cin >> c; Primer elemento
while (c != '*') { Mientras no el centinela
if (c == buscado) {
cont++; Procesar elemento
}
cin >> c; Siguiente elemento
Luis Hernández Yáñez

}
cout << buscado << " aparece " << cont
<< " veces.";
Fundamentos de la programación: Tipos e instrucciones II Página 361

Suma de los números de la secuencia suma2.cpp

Centinela: 0
int sumaSecuencia() {
double d, suma = 0;
ifstream archivo; // Archivo de entrada (lectura)
archivo.open("datos.txt");
if (archivo.is_open()) {
archivo >> d; // Obtener el primero
while (d != 0) { // Mientras no sea el centinela
suma = suma + d; // Procesar el dato
archivo >> d; // Obtener el siguiente
}
Luis Hernández Yáñez

archivo.close();
}
return suma;
}
Fundamentos de la programación: Tipos e instrucciones II Página 362
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 363

sumatorio.cpp

Recurrencia: ei+1 = ei + 1 e1 = 1
1  2  3  4  5  6  7  8  ...
Suma de los números de la secuencia calculada:
int main() {
int num;
cout << "N = ";
cin >> num;
cout << "Sumatorio:" << suma(num);
return 0;
}
long long int suma(int n) {
int sumatorio = 0;
for (int i = 1; i <= n; i++) {
Luis Hernández Yáñez

sumatorio = sumatorio + i;
}
return sumatorio;
Último elemento de la secuencia: n
}
Fundamentos de la programación: Tipos e instrucciones II Página 364
long long int suma(int n) {
int sumatorio = 0;
for (int i = 1; i <= n; i++) {
sumatorio = sumatorio + i;
}
...
sumatorio = 0;

int i = 1;
n 5
?

sumatorio 15
10
01
3
6 false
i <= n
i 2
51
4
3
6 2  1
1
5  4  3  2  1
4  3  2  1
3  2  1
true
sumatorio += i;
Secuencia
Luis Hernández Yáñez

i = i + 1;

Fundamentos de la programación: Tipos e instrucciones II Página 365

Definición
Fi = Fi‐1 + Fi‐2
F1 = 0
F2 = 1
0 1 1 2 3 5 8 13 21 34 55 89 ... 

¿Fin de la secuencia?
Primer número de Fibonacci mayor que un número dado
Ese número de Fibonacci actúa como centinela
Si num es 50, la secuencia será:
Luis Hernández Yáñez

0 1 1 2 3 5 8 13 21 34 

Fundamentos de la programación: Tipos e instrucciones II Página 366


fibonacci.cpp

Recorrido de la secuencia calculada


int num, fib, fibMenos2 = 0, fibMenos1 = 1; // 1º y 2º
fib = fibMenos2 + fibMenos1; // Calculamos el tercero
cout << "Hasta: ";
cin >> num;
if (num >= 1) { // Ha de ser entero positivo
cout << "0 1 "; // Los dos primeros son <= num
while (fib <= num) { // Mientras no mayor que num
cout << fib << " ";
fibMenos2 = fibMenos1; // Actualizamos anteriores
fibMenos1 = fib; // para obtener...
fib = fibMenos2 + fibMenos1; // ... el siguiente
}
Luis Hernández Yáñez

}
¿Demasiados comentarios?
Para no oscurecer el código, mejor una explicación al principio

Fundamentos de la programación: Tipos e instrucciones II Página 367

El bucle calcula adecuadamente la secuencia:


while (fib <= num) {
cout << fib << " ";
fibMenos2 = fibMenos1;
fibMenos1 = fib;
fib = fibMenos2 + fibMenos1;
}
num 100
?
0 1 1 2 3 5 ...
fib 5
2
3
1
?

fibMenos1 3
1
2
?
Luis Hernández Yáñez

fibMenos2 2
1
0
?

Fundamentos de la programación: Tipos e instrucciones II Página 368


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 369

Localización del primer elemento con una propiedad


Inicialización
Mientras no se encuentre el elemento
y no se esté al final de la secuencia:
Obtener el siguiente elemento
Comprobar si el elemento satisface la condición
Finalización
(tratar el elemento encontrado o indicar que no se ha encontrado)

Elemento que se busca: satisfará una condición


Dos condiciones de terminación del bucle: se encuentra / al final
Luis Hernández Yáñez

Variable lógica que indique si se ha encontrado

Fundamentos de la programación: Tipos e instrucciones II Página 370


Localización del primer elemento con una propiedad

Inicialización / encontrado = false;

¿Al final o true


encontrado?
false

Obtener elemento

¿Encontrado?
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Tipos e instrucciones II Página 371

Implementación con while


Inicialización Inicialización
Obtener el primer elemento Obtener 1º
Mientras ni encontrado ni el centinela:
Obtener el siguiente elemento ¿Encontrado
true
Finalización (¿encontrado?) o centinela?
false

Obtener siguiente
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Tipos e instrucciones II Página 372


Primer número mayor que uno dado busca.cpp

Centinela: ‐1
double d, num;
bool encontrado = false;
cout << "Encontrar primero mayor que: ";
cin >> num;
cout << "Siguiente (‐1 para terminar): ";
cin >> d; // Obtener el primer elemento
while ((d != ‐1) && !encontrado) {
// Mientras no sea el centinela y no se encuentre
if (d > num) { // ¿Encontrado?
encontrado = true;
}
else {
Luis Hernández Yáñez

cout << "Siguiente (‐1 para terminar): ";
cin >> d; // Obtener el siguiente elemento
}
}
Fundamentos de la programación: Tipos e instrucciones II Página 373
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 374


Colecciones homogéneas
Un mismo tipo de dato para varios elementos:
 Notas de los estudiantes de una clase
 Ventas de cada día de la semana
 Temperaturas de cada día del mes
...
En lugar de declarar N variables...
vLun vMar vMie vJue vVie vSab vDom
125.40 76.95 328.80 254.62 435.00 164.29 0.00

... declaramos una tabla de N valores:


Luis Hernández Yáñez

ventas 125.40 76.95 328.80 254.62 435.00 164.29 0.00


Índices 0 1 2 3 4 5 6

Fundamentos de la programación: Tipos e instrucciones II Página 375

Estructura secuencial
Cada elemento se encuentra en una posición (índice):
 Los índices son enteros positivos
 El índice del primer elemento siempre es 0
 Los índices se incrementan de uno en uno
ventas 125.40 76.95 328.80 254.62 435.00 164.29 0.00
0 1 2 3 4 5 6
Acceso directo
A cada elemento se accede a través de su índice:
ventas[4] accede al 5º elemento (contiene el valor 435.00)
Luis Hernández Yáñez

cout << ventas[4];
ventas[4] = 442.75; Datos de un mismo tipo base:
Se usan como cualquier variable

Fundamentos de la programación: Tipos e instrucciones II Página 376


Declaración de tipos de arrays
typedef tipo_base nombre_tipo[tamaño];

Ejemplos:
typedef double tTemp[7];
typedef short int tDiasMes[12];
typedef char tVocales[5];
typedef double tVentas[31];
typedef tMoneda tCalderilla[15]; // Enumerado tMoneda

Recuerda: Adoptamos el convenio de comenzar


Luis Hernández Yáñez

los nombres de tipo con una t minúscula, seguida


de una o varias palabras, cada una con su inicial en mayúscula

Fundamentos de la programación: Tipos e instrucciones II Página 377

Declaración de variables arrays typedef


typedef
double tTemp[7];
short int tDiasMes[12];
typedef char tVocales[5];
tipo nombre; typedef double tVentas[31];

Ejemplos:
tTemp tempMax; tempMax ? ? ? ? ? ? ?
0 1 2 3 4 5 6

tDiasMes diasMes; diasMes ? ? ? ? ? ? ? ? ? ? ? ?


0 1 2 3 4 5 6 7 8 9 10 11

tVocales vocales; vocales ? ? ? ? ?


0 1 2 3 4

ventasFeb ? ? ? ? ? ? ? ? ? ? ? ? ? ... ?
tVentas ventasFeb;
Luis Hernández Yáñez

0 1 2 3 4 5 6 7 8 9 10 11 12 30

NO se inicializan los elementos automáticamente

Fundamentos de la programación: Tipos e instrucciones II Página 378


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 379

nombre[índice]
Cada elemento se accede a través de su índice (posición en el array)
tVocales vocales; typedef char tVocales[5];

vocales 'a' 'e' 'i' 'o' 'u'


0 1 2 3 4

5 elementos, índices de 0 a 4:
vocales[0]   vocales[1]   vocales[2]   vocales[3]   vocales[4]
Procesamiento de cada elemento:
Como cualquier otra variable del tipo base
cout << vocales[4];
Luis Hernández Yáñez

vocales[3] = 'o';
if (vocales[i] == 'e') ...

Fundamentos de la programación: Tipos e instrucciones II Página 380


¡IMPORTANTE!
¡No se comprueba si el índice es correcto!
¡Es responsabilidad del programador!
const int Dim = 100;
typedef double tVentas[Dim];
tVentas ventas;
Índices válidos: enteros entre 0 y Dim‐1
ventas[0]  ventas[1]  ventas[2] ... ventas[98]  ventas[99]

¿Qué es ventas[100]? ¿O ventas[‐1]? ¿O ventas[132]?


¡Memoria de alguna otra variable del programa!
Luis Hernández Yáñez

Define los tamaños de los arrays con constantes

Fundamentos de la programación: Tipos e instrucciones II Página 381


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 382


Arrays: tamaño fijo  Bucle de recorrido fijo (for)
Ejemplo: Media de un array de temperaturas
const int Dias = 7;
typedef double tTemp[Dias];
tTemp temp;
double media, total = 0; 
...
for (int i = 0; i < Dias; i++) {
total = total + temp[i];
}
media = total / Dias;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 383

12.40 10.96 8.43 11.65 13.70 13.41 14.07


0 1 2 3 4 5 6

tTemp temp;
double media, total = 0;  Memoria
... Dias 7
for (int i = 0; i < Dias; i++) { temp[0] 12.40
total = total + temp[i];
temp[1] 10.96
}
temp[2] 8.43
i = 0 temp[3] 11.65
temp[4] 13.70

true false temp[5] 13.41


i<Dias
temp[6] 14.07
media ?
total+=temp[i]
... total 23.36
31.79
84.62
12.40
0.00
43.44
Luis Hernández Yáñez

i++ i 3
70
2
4
1

Fundamentos de la programación: Tipos e instrucciones II Página 384


mediatemp.cpp

#include <iostream>
using namespace std;

const int Dias = 7;


typedef double tTemp[Dias];

double media(const tTemp temp);

int main() {
tTemp temp;
for (int i = 0; i < Dias; i++) { // Recorrido del array
cout << "Temperatura del día " << i + 1 << ": ";
cin >> temp[i];
}
cout << "Temperatura media: " << media(temp) << endl;
Luis Hernández Yáñez

return 0;
} Los usuarios usan de 1 a 7 para numerar los días
... La interfaz debe aproximarse a los usuarios,
aunque internamente se usen los índices de 0 a 6

Fundamentos de la programación: Tipos e instrucciones II Página 385

double media(const tTemp temp) {


double med, total = 0; 

for (int i = 0; i < Dias; i++) { // Recorrido del array


total = total + temp[i];
}
med = total / Dias;

return med;
}

Los arrays se pasan a las funciones como constantes


Las funciones no pueden devolver arrays
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 386


const int Cuantas = 15;
typedef enum { centimo, dos_centimos, cinco_centimos,
diez_centimos, veinte_centimos, medio_euro, euro } tMoneda;
typedef tMoneda tCalderilla[Cuantas];
string aCadena(tMoneda moneda);
// Devuelve la cadena correspondiente al valor de moneda

tCalderilla bolsillo; // Exactamente llevo Cuantas monedas
bolsillo[0] = euro;
bolsillo[1] = cinco_centimos;
bolsillo[2] = medio_euro;
bolsillo[3] = euro;
bolsillo[4] = centimo;
...
Luis Hernández Yáñez

for (int moneda = 0; moneda < Cuantas; moneda++)
cout << aCadena(bolsillo[moneda]) << endl;

Fundamentos de la programación: Tipos e instrucciones II Página 387


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 388


buscaarray.cpp

¿Qué día las ventas superaron los 1.000 €?


const int Dias = 365; // Año no bisiesto
typedef double tVentas[Dias];
int busca(const tVentas ventas) {
// Índice del primer elemento mayor que 1000 (‐1 si no hay)
bool encontrado = false; 
int ind = 0;
while ((ind < Dias) && !encontrado) { // Esquema de búsqueda
if (ventas[ind] > 1000) {
encontrado = true;
}
else {
ind++;
}
}
Luis Hernández Yáñez

if (!encontrado) {
ind = ‐1;
}
return ind;
}
Fundamentos de la programación: Tipos e instrucciones II Página 389
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 390


La capacidad de un array no puede ser alterada en la ejecución
El tamaño de un array es una decisión de diseño:
 En ocasiones será fácil (días de la semana)
 Cuando pueda variar ha de estimarse un tamaño
Ni corto ni con mucho desperdicio (posiciones sin usar)
STL (Standard Template Library) de C++:
Colecciones más eficientes cuyo tamaño puede variar
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 391

No se pueden copiar dos arrays (del mismo tipo) con asignación:


array2 = array1; // ¡¡¡ NO COPIA LOS ELEMENTOS !!!

Han de copiarse los elementos uno a uno:


for (int i = 0; i < N; i++) {
array2[i] = array1[i];
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 392


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 393

Puede que no necesitemos todas las posiciones de un array...


La dimensión del array será el máximo de elementos
Pero podremos tener menos elementos del máximo
Necesitamos un contador de elementos...
const int Max = 100;
typedef double tArray[Max];
tArray lista;
int contador = 0;
contador: indica cuántas posiciones del array se utilizan
Sólo accederemos a las posiciones entre 0 y contador‐1
Las demás posiciones no contienen información del programa
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II Página 394


lista.cpp

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

const int Max = 100;


typedef double tArray[Max];

double media(const tArray lista, int cont);

int main() {
tArray lista;
int contador = 0;
double valor, med;
ifstream archivo;
archivo.open("lista.txt");
if (archivo.is_open()) {
archivo >> valor;
Luis Hernández Yáñez

while ((valor != ‐1) && (contador < Max)) {
lista[contador] = valor;
contador++;
archivo >> valor;
} ...
Fundamentos de la programación: Tipos e instrucciones II Página 395

archivo.close();
med = media(lista, contador);
cout << "Media de los elementos de la lista: " << med << endl;
}
else {
cout << "¡No se pudo abrir el archivo!" << endl;
}

return 0;
}

double media(const tArray lista, int cont) {


double med, total = 0;
for (int ind = 0; ind < cont; ind++) {
total = total + lista[ind];
}
Sólo recorremos hasta cont‐1
Luis Hernández Yáñez

med = total / cont;
return med;
}

Fundamentos de la programación: Tipos e instrucciones II Página 396


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Tipos e instrucciones II Página 397


Fundamentos de la programación 

3A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Expresión condicional Condición ? Exp1 : Exp2


Dos alternativas
— Condición: Expresión lógica Operadores (prioridad)
++ ‐‐ (postfijos)
— Exp1 y Exp2: Expresiones Llamadas a funciones
Moldes
Si Condición se evalúa a true,
++ ‐‐ (prefijos) !
el resultado es Exp1; ‐ (cambio de signo)
si Condición se evalúa a false, * / %
el resultado es Exp2. + ‐

int a = 5, b = 3, c; < <= > >=


== !=
c = (a + b == 10) ? 2 : 3;
&&
c = (  8 == 10) ? 2 : 3; ||
Luis Hernández Yáñez

?:
c =     false ? 2 : 3; = += ‐= *= /= %=
c = 3;

Fundamentos de la programación: Tipos e instrucciones II (Anexo I) Página 399


Equivalencia con un if‐else
c = (a + b == 10) ? 2 : 3;

Es equivalente a:
if (a + b == 10) c = 2;
else c = 3;

Se pueden concatenar:
cout << (nota == 10 ? "MH" : (nota >= 9 ? "SB" : 
(nota >= 7 ? "NT" : (nota >= 5 ? "AP" : "SS"))))
Esto es equivalente a la escala if‐else‐if de la siguiente sección.
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo I) Página 400

Escala if ... else if ... equivalente


cout << (nota == 10 ? "MH" : (nota >= 9 ? "SB" : 
(nota >= 7 ? "NT" : (nota >= 5 ? "AP" : "SS"))))
Si nota == 10 entonces MH
true
si no, si nota >= 9 entonces SB == 10 "MH"
si no, si nota >= 7 entonces NT false
si no, si nota >= 5 entonces AP >= 9
true
"SB"
si no SS false
true
double nota; >= 7 "NT"
cin >> nota; false
if (nota == 10) { cout << "MH"; } true
>= 5 "AP"
else if (nota >= 9) { cout << "SB"; }
false
else if (nota >= 7) { cout << "NT"; }
Luis Hernández Yáñez

"SS"
else if (nota >= 5) { cout << "AP"; }
else { cout << "SS"; }

Fundamentos de la programación: Tipos e instrucciones II (Anexo I) Página 401


Fundamentos de la programación

3E
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Recorridos 404
Un aparcamiento 405
¿Paréntesis bien emparejados? 409
¿Dos secuencias iguales? 412
Números primos menores que N 413
Búsquedas 417
Búsqueda de un número en un archivo 419
Búsquedas en secuencias ordenadas 420
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II)
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 404

Secuencia de caracteres E y S en archivo


E = Entra un coche; S = Sale un coche
¿Cuántos coches quedan al final de la jornada?
Varios casos, cada uno en una línea y terminado en punto
Final: línea sólo con punto
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 405


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

int main() {
int coches;
char c;
bool terminar = false;
ifstream archivo;
archivo.open("parking.txt");
if (!archivo.is_open()) {
cout << "¡No se ha podido abrir el archivo!" << endl;
}
else {
// Recorrido...
archivo.close();
Luis Hernández Yáñez

}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 406

while (!terminar) {
archivo >> c;
if (c == '.') { // . como primer carácter? (centinela)
terminar = true;
}
else {
coches = 0;
while (c != '.') { // Recorrido de la secuencia
cout << c;
if (c == 'E') {
coches++;
}
else if (c == 'S') {
coches‐‐;
}
Luis Hernández Yáñez

archivo >> c;
}
...

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 407
parking.cpp

if (coches >= 0) {
cout << endl << "Quedan " << coches << " coches.";
}
else {
cout << endl << "Error: Más salidas que entradas!";
}
cout << endl;
}
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 408

Cada paréntesis, con su pareja


Secuencia de caracteres terminada en # y con parejas de paréntesis:
ab(c(de)fgh((i(jk))lmn)op)(rs)#

Contador del nivel de anidamiento:


Al encontrar '(' incrementamos – Al encontrar ')' decrementamos
Al terminar, el contador deberá tener el valor 0
Errores:
— Contador ‐1: paréntesis de cierre sin uno de apertura pendiente
abc)de(fgh(ij))#
Luis Hernández Yáñez

— Contador termina con un valor positivo


Más paréntesis de apertura que de cierre
Algún paréntesis sin cerrar: (a(b(cd(e)f)gh(i))jk#

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 409


Un error puede interrumpir el recorrido:
char c;
int anidamiento = 0, pos = 0;
bool error = false;
cin >> c;
while ((c != '#') && !error) {
pos++;
if (c == '(') {
anidamiento++;
}
else if (c == ')') {
anidamiento‐‐;
}
if (anidamiento < 0) {
error = true;
}
Luis Hernández Yáñez

if (!error) {
cin >> c;
}
}
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 410

parentesis.cpp

if (error) {
cout << "Error: cierre sin apertura (pos. " << pos 
<< ")";
}
else if (anidamiento > 0) {
cout << "Error: Apertura sin cierre";
}
else {
cout << "Correcto";
}
cout << endl;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 411


iguales.cpp

bool iguales() {
bool sonIguales = true;
double d1, d2;
ifstream sec1, sec2;
bool final = false;
sec1.open("secuencia1.txt");
sec2.open("secuencia2.txt");
sec1 >> d1;
sec2 >> d2; // Al menos estarán los centinelas (0)
while (sonIguales && !final) {
sonIguales = (d1 == d2);
final = ((d1 == 0) || (d2 == 0));
if (!final) {
sec1 >> d1;
sec2 >> d2;
}
} Cambia secuencia2.txt por secuencia3.txt
Luis Hernández Yáñez

sec1.close(); y por secuencia4.txt para comprobar otros casos


sec2.close();
return sonIguales;
}
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 412

primos.cpp

Secuencia calculada: números divisibles sólo por 1 y ellos mismos (< N)


#include <iostream>
using namespace std;
bool primo(int n);
int main() {
int num, candidato;
cout << "Entero en el que parar (>1): ";
cin >> num;
if (num > 1) {
candidato = 2; // El 1 no se considera un número primo
while (candidato < num) {
cout << candidato << " "; // Mostrar número primo
candidato++;
while (!primo(candidato)) { // Siguiente primo
candidato++;
}
Luis Hernández Yáñez

}
}
return 0;
}
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 413
bool primo(int n) {
bool esPrimo = true;

for (int i = 2; i <= n ‐ 1; i++) {


if (n % i == 0) {
esPrimo = false; // Es divisible por i
}
}

return esPrimo;
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 414

primos2.cpp

Mejoras: probar sólo impares; sólo pueden ser divisibles por impares;
no pueden ser divisibles por ninguno mayor que su mitad
candidato = 2;
cout << candidato << " "; // Mostrar el número primo 2
candidato++; // Seguimos con el 3, que es primo
while (candidato < num) {
cout << candidato << " ";   // Mostrar número primo
candidato = candidato + 2;  // Sólo probamos impares
while (!primo(candidato)) { // Siguiente número primo
candidato = candidato + 2;
}
} ...
bool primo(int n) {
bool esPrimo = true;
for (int i = 3; i <= n / 2; i = i + 2) {
Luis Hernández Yáñez

if (n % i == 0) {
esPrimo = false; // Es divisible por i
}
}...
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 415
primos3.cpp

Otra mejora más: Paramos al encontrar el primer divisor

bool primo(int n) {


bool esPrimo = true;

int i = 3;
while ((i <= n / 2) && esPrimo) {
if (n % i == 0) {
esPrimo = false;
}
i = i + 2;
}

return esPrimo;
}
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 416


Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 417


#include <iostream> buscaarch.cpp
using namespace std;
#include <fstream>

int busca(int n);


// Devuelve la línea en la que se encuentra o ‐1 si no está

int main() {
int num, linea;

cout << "Valor a localizar: ";
cin >> num;
linea = busca(num);
if (linea != ‐1) {
cout << "Encontrado (línea " << linea << ")" << endl;
}
else {
Luis Hernández Yáñez

cout << "No encontrado" << endl;
}
return 0;
}

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 418

int busca(int n) {


int i, linea = 0;
bool encontrado = false;
ifstream archivo;
archivo.open("enteros.txt");
if (!archivo.is_open()) {
linea = ‐1;
}
else {
archivo >> i;
while ((i != 0) && !encontrado) {
linea++;
if (i == n) {
encontrado = true;
}
archivo >> i; Centinela
}
if (!encontrado) {
linea = ‐1;
Luis Hernández Yáñez

}
archivo.close();
}
return linea;
}
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 419
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 420

Secuencia ordenada de menor a mayor: buscaord.cpp


paramos al encontrar uno mayor o igual al buscado
Los que resten serán seguro mayores: ¡no puede estar el buscado!
cout << "Valor a localizar: ";
cin >> num;
archivo >> i;
while ((i != 0) && (i < num)) {
cont++;
archivo >> i;
}
if (i == num) {
cout << "Encontrado (pos.: " << cont << ")";
}
else {
Luis Hernández Yáñez

cout << "No encontrado";
}
cout << endl;
archivo.close();
Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 421
Si el elemento está: procesamiento similar a secuencias desordenadas

2 5 9 15 16 24 41 73 78 82 123 153 159 ...

archivo >> i;

num 9
?

(i != 0)  false
&& (i < num)

i 9
5
?
2 true

cont++;
archivo >> i;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 422

Si el elemento no está: evitamos buscar en el resto de la secuencia

No se procesa
2 5 9 15 16 24 41 73 78 82 123 153 159 ...
el resto
de la secuencia
archivo >> i;

num 10
?

(i != 0)  false
&& (i < num)

i 5
2
9
?
15 true

cont++;
archivo >> i;
Luis Hernández Yáñez

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 423


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Tipos e instrucciones II (Anexo II) Página 424


Fundamentos de la programación

4
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Diseño descendente: Tareas y subtareas  427
Subprogramas 434
Subprogramas y datos 441
Parámetros 446
Argumentos 451
Resultado de la función 467
Prototipos 473
Ejemplos completos 475
Funciones de operador 477
Diseño descendente (un ejemplo) 480
Precondiciones y postcondiciones 490
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 427

Refinamientos sucesivos
Tareas que ha de realizar un programa:
Se pueden dividir en subtareas más sencillas
Subtareas:
También se pueden dividir en otras más sencillas...

 Refinamientos sucesivos
Diseño en sucesivos pasos en los se amplía el detalle

Ejemplos:

 Dibujar
Luis Hernández Yáñez

 Mostrar la cadena HOLA MAMA en letras gigantes

Fundamentos de la programación: La abstracción procedimental Página 428


1. Dibujar

2. Dibujar 1. Dibujar
Misma tarea
3. Dibujar 2. Dibujar

2.1. Dibujar

REFINAMIENTO 2.2. Dibujar


Luis Hernández Yáñez

3. Dibujar

Fundamentos de la programación: La abstracción procedimental Página 429

1. Dibujar

2. Dibujar

2.1. Dibujar
4 tareas, pero dos de ellas son iguales
2.2. Dibujar Nos basta con saber cómo dibujar:

3. Dibujar
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 430


void dibujarCirculo()
{ ... }

void dibujarSecantes()
Dibujar 
{ ... }

void dibujarLinea()
{ ... }
Dibujar  Dibujar  Dibujar 
void dibujarTriangulo()
{
dibujarSecantes();
dibujarLinea();
Dibujar  Dibujar  }

int main() {
dibujarCirculo();
Luis Hernández Yáñez

dibujarTriangulo();
dibujarSecantes();
return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 431

Mostrar la cadena HOLA MAMA en letras gigantes

Mostrar HOLA MAMA

Mostrar HOLA Espacio en blanco Mostrar MAMA

H O L A M A

Tareas básicas
Luis Hernández Yáñez

H O L A Espacio en blanco  M

Fundamentos de la programación: La abstracción procedimental Página 432


void mostrarH() { void espaciosEnBlanco() {
cout << "*   *" << endl; cout << endl << endl << endl;
cout << "*   *" << endl; }
cout << "*****" << endl;
cout << "*   *" << endl; void mostrarM()
cout << "*   *" << endl << endl; { ...}
}
int main() {
void mostrarO() { mostrarH();
cout << "*****" << endl; mostrarO();
cout << "*   *" << endl; mostrarL();
cout << "*   *" << endl; mostrarA();
cout << "*   *" << endl; espaciosEnBlanco();
cout << "*****" << endl << endl; mostrarM();
} mostrarA();
mostrarM();
void mostrarL() mostrarA();
Luis Hernández Yáñez

{ ... }
return 0;
void mostrarA() }
{ ...}

Fundamentos de la programación: La abstracción procedimental Página 433


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 434


Subprogramas
Pequeños programas dentro de otros programas
 Unidades de ejecución independientes
 Encapsulan código y datos
 Se comunican con otros subprogramas (datos)
Subrutinas, procedimientos, funciones, acciones, ...
 Realizan tareas individuales del programa
 Funcionalidad concreta, identificable y coherente (diseño)
 Se ejecutan de principio a fin cuando se llaman (invocan)
 Terminan devolviendo el control al punto de llamada
Luis Hernández Yáñez

Aumentan el nivel de abstracción del programa


Facilitan la prueba, la depuración y el mantenimiento

Fundamentos de la programación: La abstracción procedimental Página 435

Flujo de ejecución
int main()
{
mostrarH();

mostrarO();
... 
}

void mostrarH()
{

}
...

void mostrarO()
Luis Hernández Yáñez

{
 }
...

...

Fundamentos de la programación: La abstracción procedimental Página 436


Subprogramas en C++
Forma general de un subprograma en C++:
tipo nombre(parámetros) // Cabecera
{
// Cuerpo
}

 Tipo de dato que devuelve el subprograma como resultado


 Parámetros para la comunicación con el exterior
 Cuerpo: ¡Un bloque de código!
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 437

Tipos de subprogramas
Procedimientos (acciones):
NO devuelven ningún resultado de su ejecución con return
Tipo: void
Llamada: instrucción independiente
mostrarH();

Funciones:
SÍ devuelven un resultado con la instrucción return
Tipo distinto de void
Llamada: dentro de cualquier expresión
x = 12 * y + cuadrado(20) ‐ 3;
Luis Hernández Yáñez

Se sustituye en la expresión por el valor que devuelve


¡Ya venimos utilizando funciones desde el Tema 2!

Fundamentos de la programación: La abstracción procedimental Página 438


Funciones
Subprogramas de tipo distinto de void
...
int menu() int main()
{
{
int op;
cout << "1 – Editar" << endl; ...
cout << "2 – Combinar" << endl; int opcion;
cout << "3 – Publicar" << endl; opcion = menu() ;
cout << "0 – Cancelar" << endl; ...
cout << "Elija: ";
cin >> op;
return op;
}
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 439

Procedimientos
Subprogramas de tipo void
...
void menu() int main()
{
{
int op;
cout << "1 – Editar" << endl; ...
cout << "2 – Combinar" << endl; menu();
cout << "0 – Cancelar" << endl; ...
cout << "Opción: ";
cin >> op;
if (op == 1) {
editar();
}
Luis Hernández Yáñez

else if (op == 2) {
combinar();
}
}

Fundamentos de la programación: La abstracción procedimental Página 440


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 441

De uso exclusivo del subprograma


tipo nombre(parámetros) // Cabecera
{
Declaraciones locales // Cuerpo
}

 Declaraciones locales de tipos, constantes y variables


Dentro del cuerpo del subprograma

 Parámetros declarados en la cabecera del subprograma


Comunicación del subprograma con otros subprogramas
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 442


Datos en los programas
 Datos globales: declarados fuera de todos los subprogramas
Existen durante toda la ejecución del programa
 Datos locales: declarados en algún subprograma
Existen sólo durante la ejecución del subprograma

Ámbito y visibilidad de los datos Tema 3


— Ámbito de los datos globales: resto del programa
Se conocen dentro de los subprogramas que siguen
— Ámbito de los datos locales: resto del subprograma
No se conocen fuera del subprograma
Luis Hernández Yáñez

— Visibilidad de los datos


Datos locales a un bloque ocultan otros externos homónimos

Fundamentos de la programación: La abstracción procedimental Página 443

#include <iostream>
using namespace std;

const int MAX = 100;


double ingresos; Datos globales op de proc()
es distinta
... de op de main()
void proc() {
int op;
Datos locales a proc()
double ingresos;
... Se conocen MAX (global), op (local)
} e ingresos (local que oculta la global)

int main() {
int op;
Datos locales a main()
Luis Hernández Yáñez

...
return 0; Se conocen MAX (global), op (local)
} e ingresos (global)

Fundamentos de la programación: La abstracción procedimental Página 444


Sobre el uso de datos globales en los subprogramas
NO SE DEBEN USAR datos globales en subprogramas
 ¿Necesidad de datos externos?
Define parámetros en el subprograma
Los datos externos se pasan como argumentos en la llamada
 Uso de datos globales en los subprogramas:
Riesgo de efectos laterales
Modificación inadvertida de esos datos afectando otros sitios

Excepciones:
 Constantes globales (valores inalterables)
Luis Hernández Yáñez

 Tipos globales (necesarios en varios subprogramas)

Fundamentos de la programación: La abstracción procedimental Página 445


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 446


Datos de entrada, datos de salida y datos de entrada/salida

Datos de entrada: Aceptados Subprograma


Subprograma que dado un número x
muestra en la pantalla su cuadrado: cuadrado()
5

Datos de salida: Devueltos Subprograma


Subprograma que dado un número x y (=x2)
devuelve su cuadrado: cuadrado()
5

Datos de entrada/salida:
Subprograma
Aceptados y modificados
Luis Hernández Yáñez

Subprograma que dada una variable x x


numérica la eleva al cuadrado: cuadrado()

Fundamentos de la programación: La abstracción procedimental Página 447

Declaración de parámetros
Sólo dos clases de parámetros en C++:
— Sólo de entrada (por valor)

— De salida (sólo salida o E/S) (por referencia / por variable)

Lista de parámetros formales


Entre los paréntesis de la cabecera del subprograma
tipo nombre(parámetros)

&
De salida
Luis Hernández Yáñez

parámetros tipo identificador

Fundamentos de la programación: La abstracción procedimental Página 448


Reciben copias de los argumentos usados en la llamada
int cuadrado(int num)
double potencia(double base, int exp)
void muestra(string nombre, int edad, string nif)
void proc(char c, int x, double a, bool b)
Reciben sus valores en la llamada del subprograma
Argumentos: Expresiones en general
Variables, constantes, literales, llamadas a función, operaciones
Se destruyen al terminar la ejecución del subprograma
Luis Hernández Yáñez

¡Atención! Los arrays se pasan por valor como constantes:


double media(const tArray lista)

Fundamentos de la programación: La abstracción procedimental Página 449

&
Misma identidad que la variable pasada como argumento
void incrementa(int &x)
void intercambia(double &x, double &y)
void proc(char &c, int &x, double &a, bool &b)
Reciben las variables en la llamada del subprograma: ¡Variables!
Los argumentos pueden quedar modificados
¡No usaremos parámetros por valor en las funciones!
Sólo en procedimientos
Puede haber tanto por valor como por referencia

¡Atención! Los arrays se pasan por referencia sin utilizar &


Luis Hernández Yáñez

void insertar(tArray lista, int &contador, double item)


El argumento de lista (variable tArray) quedará modificado

Fundamentos de la programación: La abstracción procedimental Página 450


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 451

nombre(argumentos)
— Tantos argumentos como parámetros y en el mismo orden
— Concordancia de tipos argumento‐parámetro
— Por valor: Expresiones válidas (se pasa el resultado)
— Por referencia: ¡Sólo variables!

Se copian los valores de las expresiones pasadas por valor


en los correspondientes parámetros

Se hacen corresponder los argumentos pasados por referencia


(variables) con sus correspondientes parámetros
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 452


Expresiones válidas con concordancia de tipo:
void proc(int x, double a)  proc(23 * 4 / 7, 13.5);

 double d = 3; 
proc(12, d);

 double d = 3; 
int i = 124; 
proc(i, 33 * d);

 double d = 3; 
int i = 124; 
proc(cuad(20) * 34 + i, i * d);
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 453

void proc(int x, double a) Memoria

{ ... } i 124
d 3.0
int main() ...
{
int i = 124;
...
double  d = 3; x 124
proc(i, 33 * d); a 99.0
... ...

return 0;
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 454


void proc(int &x, double &a) Memoria
x i 124
{ ... }
a d 3.0
int main() ...
{
int i = 124;
double  d = 3;
proc(i, d);
...

return 0;
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 455

Dadas las siguientes declaraciones:


int i;
double d;
void proc(int x, double &a);
¿Qué pasos de argumentos son correctos? ¿Por qué no?
proc(3, i, d);  Nº de argumentos ≠ Nº de pará metros
proc(i, d); 
proc(3 * i + 12, d); 
proc(i, 23);  Parámetro por referencia  ¡variable!
proc(d, i);  ¡Argumento double para parámetro int!
proc(3.5, d);  ¡Argumento double para parámetro int!
Luis Hernández Yáñez

proc(i);  Nº de argumentos ≠ Nº de pará metros

Fundamentos de la programación: La abstracción procedimental Página 456


...
void divide(int op1, int op2, int &div, int &rem) {
// Divide op1 entre op2 y devuelve el cociente y el resto
div = op1 / op2;
rem = op1 % op2;
}

int main() {
int cociente, resto;
for (int j = 1; j <= 4; j++) {
for (int i = 1; i <= 4; i++) {
divide(i, j, cociente, resto);
cout << i << " entre " << j << " da un cociente de "
<< cociente << " y un resto de " << resto << endl;
}
}
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 457

...
void divide(int op1, int op2, int &div, int &rem) {
// Divide op1 entre op2 y devuelve el cociente y el resto
div = op1 / op2;
rem = op1 % op2; Memoria
} cociente ?
resto ?
int main() {
int cociente, resto; i 1
for (int j = 1; j <= 4; j++) { j 1
for (int i = 1; i <= 4; i++) { ...
divide(i, j, cociente, resto);
...
}
}
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 458


...
void divide(int op1, int op2, int &div, int &rem) {
// Divide op1 entre op2 y devuelve el cociente y el resto
div = op1 / op2;
rem = op1 % op2; Memoria
} div cociente ?
rem resto ?
int main() {
int cociente, resto; i 1
for (int j = 1; j <= 4; j++) { j 1
for (int i = 1; i <= 4; i++) { ...
divide(i, j, cociente, resto);
...
} op1 1
}
op2 1
Luis Hernández Yáñez

return 0; ...
}

Fundamentos de la programación: La abstracción procedimental Página 459

...
void divide(int op1, int op2, int &div, int &rem) {
// Divide op1 entre op2 y devuelve el cociente y el resto
div = op1 / op2;
rem = op1 % op2; Memoria
} div cociente 1
rem resto 0
int main() {
int cociente, resto; i 1
for (int j = 1; j <= 4; j++) { j 1
for (int i = 1; i <= 4; i++) { ...
divide(i, j, cociente, resto);
...
} op1 1
}
op2 1
Luis Hernández Yáñez

return 0; ...
}

Fundamentos de la programación: La abstracción procedimental Página 460


...
void divide(int op1, int op2, int &div, int &rem) {
// Divide op1 entre op2 y devuelve el cociente y el resto
div = op1 / op2;
rem = op1 % op2; Memoria
} cociente 1
resto 0
int main() {
int cociente, resto; i 1
for (int j = 1; j <= 4; j++) { j 1
for (int i = 1; i <= 4; i++) { ...
divide(i, j, cociente, resto);
...
}
}
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 461

...
void intercambia(double &valor1, double &valor2) {
// Intercambia los valores
double tmp; // Variable local (temporal)
tmp = valor1; Memoria temporal
valor1 = valor2; del procedimiento
valor2 = tmp; tmp ?
}
...
int main() {
double num1, num2;
cout << "Valor 1: "; Memoria de main()
cin >> num1; valor1 num1 13.6
cout << "Valor 2: ";
valor2 num2 317.14
cin >> num2;
intercambia(num1, num2); ...
Luis Hernández Yáñez

cout << "Ahora el valor 1 es " << num1 
<< " y el valor 2 es " << num2 << endl;
return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 462


...
// Prototipo
void cambio(double precio, double pago, int &euros, int &cent50,
int &cent20, int &cent10, int &cent5, int &cent2, int &cent1);

int main() {
double precio, pago;
int euros, cent50, cent20, cent10, cent5, cent2, cent1;
cout << "Precio: ";
cin >> precio;
cout << "Pago: ";
cin >> pago;
cambio(precio, pago, euros, cent50, cent20, cent10, cent5, cent2,
cent1);
cout << "Cambio: " << euros << " euros, " << cent50 << " x 50c., "
<< cent20 << " x 20c., " << cent10 << " x 10c., "
<< cent5 << " x 5c., " << cent2 << " x 2c. y "
Luis Hernández Yáñez

<< cent1 << " x 1c." << endl;  

return 0;
}

Fundamentos de la programación: La abstracción procedimental Página 463

void cambio(double precio, double pago, int &euros, int &cent50,


int &cent20, int &cent10, int &cent5, int &cent2, int &cent1) {
if (pago < precio) { // Cantidad insuficiente
cout << "Error: El pago es inferior al precio" << endl;
}
else {
int cantidad = int(100.0 * (pago ‐ precio) + 0.5);
euros  = cantidad / 100;
cantidad = cambio % 100;
cent50 = cantidad / 50;
cantidad = cantidad % 50;
Explicación en el libro de
cent20 = cantidad / 20;
cantidad = cantidad % 20; Adams/Leestma/Nyhoff
cent10 = cantidad / 10;
cantidad = cantidad % 10;
cent5  = cantidad / 5;
cantidad = cantidad % 5;
Luis Hernández Yáñez

cent2  = cantidad / 2;
cent1  = cantidad % 2;
}
}

Fundamentos de la programación: La abstracción procedimental Página 464


En los subprogramas se pueden detectar errores
Errores que impiden realizar los cálculos:
void cambio(double precio, double pago, int &euros, int &cent50,
int &cent20, int &cent10, int &cent5, int &cent2, int &cent1) {
if (pago < precio) { // Cantidad insuficiente
cout << "Error: El pago es inferior al precio" << endl;
}
...
¿Debe el subprograma notificar al usuario o al programa?
 Mejor notificarlo al punto de llamada y allí decidir qué hacer
void cambio(double precio, double pago, int &euros, int &cent50,
int &cent20, int &cent10, int &cent5, int &cent2, int &cent1,
bool &error) {
if (pago < precio) { // Cantidad insuficiente
Luis Hernández Yáñez

error = true;
}
else {
error = false;
...
Fundamentos de la programación: La abstracción procedimental Página 465

cambio.cpp

Al volver de la llamada se decide qué hacer si ha habido error...


 ¿Informar al usuario?
 ¿Volver a pedir los datos?
 Etcétera
int main() {
double precio, pago;
int euros, cent50, cent20, cent10, cent5, cent2, cent1;
bool error;
cout << "Precio: ";
cin >> precio;
cout << "Pago: ";
cin >> pago;
cambio(precio, pago, euros, cent50, cent20, cent10, cent5, cent2,
cent1, error);
Luis Hernández Yáñez

if (error) {
cout << "Error: El pago es inferior al precio" << endl;
}
else {
...
Fundamentos de la programación: La abstracción procedimental Página 466
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 467

Una función ha de devolver un resultado


La función ha de terminar su ejecución devolviendo el resultado
La instrucción return:
— Devuelve el dato que se indica a continuación como resultado

— Termina la ejecución de la función

El dato devuelto sustituye a la llamada de la función en la expresión


int cuad(int x) { int main() {
return x * x; cout << 2 * cuad(16);
x = x * x;
} Esta instrucción
return 0; 256
Luis Hernández Yáñez

no se ejecutará nunca }

Fundamentos de la programación: La abstracción procedimental Página 468


factorial.cpp

Factorial (N) = 1  x  2  x  3  x ... x  (N‐2)  x  (N‐1)  x  N
long long int factorial(int n); // Prototipo

int main() {
int num;
cout << "Num: ";
cin >> num;
cout << "Factorial de " << num << ": " << factorial(num) << endl;
return 0;
}

long long int factorial(int n) {


long long int fact = 1;
if (n < 0) {
fact = 0;
}
else {
for (int i = 1; i <= n; i++) {
Luis Hernández Yáñez

fact = fact * i;


}
}
return fact;
}

Fundamentos de la programación: La abstracción procedimental Página 469

int compara(int val1, int val2) {


// ‐1 si val1 < val2, 0 si iguales, +1 si val1 > val2
if (val1 == val2) {
return 0;


}
else if (val1 < val2) {
return ‐1; ¡3 puntos de salida!
}
else {
return 1;
}
}
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 470


int compara(int val1, int val2) {
// ‐1 si val1 < val2, 0 si iguales, +1 si val1 > val2
int resultado;

if (val1 == val2) {
resultado = 0;
}
else if (val1 < val2) {
resultado = ‐1;
}
else {
resultado = 1;


}

return resultado; Punto de salida único


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 471

Procedimientos (tipo void):


— Al encontrar la llave de cierre que termina el subprograma o

— Al encontrar una instrucción return (sin resultado)

Funciones (tipo distinto de void):


— SÓLO al encontrar una instrucción return (con resultado)

Nuestros subprogramas siempre terminarán al final:


 No usaremos return en los procedimientos
 Funciones: sólo un return y estará al final
Luis Hernández Yáñez

Para facilitar la depuración y el mantenimiento,


codifica los subprogramas con un único punto de salida

Fundamentos de la programación: La abstracción procedimental Página 472


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 473

¿Dónde los ponemos? ¿Antes de main()? ¿Después de main()?


 Los pondremos después de main()

¿Son correctas las llamadas a subprogramas?


En main() o en otros subprogramas
— ¿Existe el subprograma?

— ¿Concuerdan los argumentos con los parámetros?

Deben estar los prototipos de los subprogramas antes de main()


Prototipo: cabecera del subprograma terminada en ;
void dibujarCirculo();
void mostrarM(); main() es el único subprograma
Luis Hernández Yáñez

void proc(double &a); que no hay que prototipar


int cuad(int x);
...

Fundamentos de la programación: La abstracción procedimental Página 474


intercambia.cpp

#include <iostream>
using namespace std;

void intercambia(double &valor1, double &valor2); // Prototipo

int main() {
double num1, num2;
cout << "Valor 1: "; Asegúrate de que los prototipos
cin >> num1; coincidan con las implementaciones
cout << "Valor 2: ";
cin >> num2;
intercambia(num1, num2);
cout << "Ahora el valor 1 es " << num1 
<< " y el valor 2 es " << num2 << endl;
return 0;
}

void intercambia(double &valor1, double &valor2) {
Luis Hernández Yáñez

double tmp; // Variable local (temporal)
tmp = valor1;
valor1 = valor2;
valor2 = tmp;
}

Fundamentos de la programación: La abstracción procedimental Página 475

mates.cpp

#include <iostream> long long int factorial(int n) {


using namespace std; long long int fact = 1;

// Prototipos if (n < 0) {
long long int factorial(int n); fact = 0;
int sumatorio(int n); }
else {
int main() { for (int i = 1; i <= n; i++) {
int num; fact = fact * i;
cout << "Num: "; }
cin >> num; }
cout << "Factorial de "
<< num << ": " return fact;
<< factorial(num) }
<< endl
<< "Sumatorio de 1 a " int sumatorio(int n) {
<< num << ": " int sum = 0;
<< sumatorio(num)
<< endl; for (int i = 1; i <= n; i++) {
Luis Hernández Yáñez

sum = sum + i;


return 0; }
}
return sum;
}

Fundamentos de la programación: La abstracción procedimental Página 476


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 477

Notación infija (de operador)


operandoIzquierdo operador operandoDerecho
a + b
Se ejecuta el operador con los operandos como argumentos
Los operadores se implementan como funciones:
tipo operatorsímbolo(parámetros)
Si es un operador monario sólo habrá un parámetro
Si es binario habrá dos parámetros
El símbolo es un símbolo de operador (uno o dos caracteres):
Luis Hernández Yáñez

+, ‐, *, /, ‐‐, <<, %, ...

Fundamentos de la programación: La abstracción procedimental Página 478


tMatriz suma(tMatriz a, tMatriz b);
tMatriz a, b, c;
c = suma(a, b);

tMatriz operator+(tMatriz a, tMatriz b);


tMatriz a, b, c;
c = a + b;

¡La implementación será exactamente la misma!


Luis Hernández Yáñez

Mayor aproximación al lenguaje matemático

Fundamentos de la programación: La abstracción procedimental Página 479


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 480


Especificación inicial (Paso 0).‐
Desarrollar un programa que haga operaciones de conversión
de medidas hasta que el usuario decida que no quiere hacer más

Análisis y diseño aumentando el nivel de detalle en cada paso


¿Qué operaciones de conversión?

Paso 1.‐
Desarrollar un programa que haga operaciones de conversión
de medidas hasta que el usuario decida que no quiere hacer más
 Pulgadas a centímetros
 Libras a gramos
Luis Hernández Yáñez

 Grados Fahrenheit a centígrados


 Galones a litros

Fundamentos de la programación: La abstracción procedimental Página 481

Paso 2.‐
Desarrollar un programa que muestre al usuario un menú con
cuatro operaciones de conversión de medidas:
 Pulgadas a centímetros
 Libras a gramos
 Grados Fahrenheit a centígrados
 Galones a litros
Y lea la elección del usuario y proceda con la conversión, hasta que
el usuario decida que no quiere hacer más

6 grandes tareas:
Luis Hernández Yáñez

Menú, cuatro funciones de conversión y main()

Fundamentos de la programación: La abstracción procedimental Página 482


Paso 2.‐

Conversiones

Menú Pulgadas a cm. Libras a gr. ºF a ºC Galones a l. main()


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 483

Paso 3.‐
 Menú:
Mostrar las cuatro opciones más una para salir
Validar la entrada y devolver la elegida
 Pulgadas a centímetros:
Devolver el equivalente en centímetros del valor en pulgadas
 Libras a gramos:
Devolver el equivalente en gramos del valor en libras
 Grados Fahrenheit a centígrados:
Devolver el equivalente en centígrados del valor en Fahrenheit
 Galones a litros:
Devolver el equivalente en litros del valor en galones
Luis Hernández Yáñez

 Programa principal (main())

Fundamentos de la programación: La abstracción procedimental Página 484


Paso 3.‐ Cada tarea, un subprograma

Comunicación entre los subprogramas:

Función Entrada Salida Valor devuelto


menu() ― ― int
pulgACm() double pulg ― double
lbAGr() double libras ― double
grFAGrC() double grF ― double
galALtr() double galones ― double
main() ― ― int
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 485

Paso 4.‐ Algoritmos detallados de cada subprograma  Programar


#include <iostream>
using namespace std;
// Prototipos
int menu();
double pulgACm(double pulg);
double lbAGr(double libras);
double grFAGrC(double grF);
double galALtr(double galones);

int main() {
double valor;
int op = ‐1;
while (op != 0) {
op = menu();
switch (op) {
case 1:
{
Luis Hernández Yáñez

cout << "Pulgadas: ";
cin >> valor;
cout << "Son " << pulgACm(valor) << " cm." << endl;
}
. . .
break;
Fundamentos de la programación: La abstracción procedimental Página 486
case 2:
{
cout << "Libras: ";
cin >> valor;
cout << "Son " << lbAGr(valor) << " gr." << endl;
}
break;
case 3:
{
cout << "Grados Fahrenheit: ";
cin >> valor;
cout << "Son " << grFAGrC(valor) << " ºC" << endl;
}
break;
case 4:
{
cout << "Galones: ";
cin >> valor;
cout << "Son " << galALtr(valor) << " l." << endl;
}
Luis Hernández Yáñez

break;
}
}
return 0;
} . . .
Fundamentos de la programación: La abstracción procedimental Página 487

int menu() {
int op = ‐1;

while ((op < 0) || (op > 4)) {


cout << "1 ‐ Pulgadas a Cm." << endl;
cout << "2 ‐ Libras a Gr." << endl;
cout << "3 ‐ Fahrenheit a ºC" << endl;
cout << "4 ‐ Galones a L." << endl;
cout << "0 ‐ Salir" << endl;
cout << "Elige: ";
cin >> op;
if ((op < 0) || (op > 4)) {
cout << "Opción no válida" << endl;
}
}

return op;
}
Luis Hernández Yáñez

double pulgACm(double pulg) {


const double cmPorPulg = 2.54;
return pulg * cmPorPulg;
} . . .
Fundamentos de la programación: La abstracción procedimental Página 488
conversiones.cpp

double lbAGr(double libras) {


const double grPorLb = 453.6;
return libras * grPorLb;
}

double grFAGrC(double grF) {


return ((grF ‐ 32) * 5 / 9);
}

double galALtr(double galones) {


const double ltrPorGal = 4.54609;
return galones * ltrPorGal;
}
Luis Hernández Yáñez

. . .
Fundamentos de la programación: La abstracción procedimental Página 489
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 490


Integridad de los subprogramas
Condiciones que se deben dar antes de comenzar su ejecución
 Precondiciones
 Quien llame al subprograma debe garantizar que se satisfacen

Condiciones que se darán cuando termine su ejecución


 Postcondiciones
 En el punto de llamada se pueden dar por garantizadas

Aserciones:
Condiciones que si no se cumplen interrumpen la ejecución
Luis Hernández Yáñez

Función assert()

Fundamentos de la programación: La abstracción procedimental Página 491

Precondiciones
Por ejemplo, no realizaremos conversiones de valores negativos:
double pulgACm(double pulg) {
assert(pulg > 0);

double cmPorPulg = 2.54;

return pulg * cmPorPulg;


}
La función tiene una precondición: pulg debe ser positivo

assert(pulg > 0); interrumpirá la ejecución si no es cierto


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental Página 492


Precondiciones
Es responsabilidad del punto de llamada garantizar la precondición:
int main() {
double valor;
int op = ‐1;
while (op != 0) {
op = menu();
switch (op) {
case 1:
{
cout << "Pulgadas: ";
cin >> valor;
if (valor < 0) {
cout << "¡No válido!" << endl;
Luis Hernández Yáñez

}
else { // Se cumple la precondición...
...

Fundamentos de la programación: La abstracción procedimental Página 493

Postcondiciones
Un subprograma puede garantizar condiciones al terminar:
int menu() {
int op = ‐1;
while ((op < 0) || (op > 4)) {
...
cout << "Elige: ";
cin >> op;
if ((op < 0) || (op > 4)) {
cout << "Opción no válida" << endl;
}
}
assert ((op >= 0) && (op <= 4));
Luis Hernández Yáñez

return op;
}
El subprograma debe asegurarse de que se cumpla

Fundamentos de la programación: La abstracción procedimental Página 494


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: La abstracción procedimental Página 495


Fundamentos de la programación

4A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Archivos como parámetros 498
La función main() 501
Argumentos implícitos 504
Sobrecarga de subprogramas 508
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo)
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 498

#include <iostream>
using namespace std;
#include <fstream>
void sumatorio_archivo(ifstream &arch, double &suma);
int main() {
double resultado;
ifstream archivo;
archivo.open("datos.txt");
if (!archivo.is_open()) {
cout << "ERROR DE APERTURA" << endl;
}
else {
sumatorio_archivo(archivo, resultado)
cout << "Suma = " << resultado << endl;
archivo.close();
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 499


void sumatorio_archivo(ifstream &arch, double &suma) {
double dato;

suma = 0;
arch >> dato;

while (dato != ‐1) {


suma = suma + dato;
arch >> dato;
}
}

Los archivos siempre se pasan por referencia


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 500


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 501


Comunicación con el sistema operativo
Parámetros opcionales de la función main():
int main(int argc, char *argv[])

Para obtener datos proporcionados al ejecutar el programa:


C:\>prueba cad1 cad2 cad3
Ejecuta prueba.exe con tres argumentos (cadenas)

Parámetros de main():
— argc: número de argumentos que se proporcionan

4 en el ejemplo (primero: nombre del programa con su ruta)


Luis Hernández Yáñez

— argv: array con las cadenas proporcionadas como argumentos

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 502

¿Cómo ha ido la función?


La función main() devuelve al S.O. un código de terminación
— 0: Todo OK

— Distinto de 0: ¡Ha habido un error!

Si la ejecución llega al final de la función main(), todo OK:


...
return 0; // Fin del programa
}
Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 503


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 504

Valores predeterminados para parámetros por valor


Valor por defecto para un parámetro:
Tras un = a continuación del nombre del parámetro:
void proc(int i = 1);

Si no se proporciona argumento, el parámetro toma ese valor


proc(12); i toma el valor explícito 12
proc(); i toma el valor implícito (1)

Sólo puede haber argumentos implícitos en los parámetros finales:


void p(int i, int j = 2, int k = 3); // CORRECTO
Luis Hernández Yáñez

void p(int i = 1, int j, int k = 3); // INCORRECTO

Una vez asignado un valor implícito, todos los que siguen
han de tener también valor implícito
Fundamentos de la programación: La abstracción procedimental (Anexo) Página 505
Parámetros y argumentos implícitos
void p(int i, int j = 2, int k = 3);
Se copian los argumentos en los parámetros del primero al último
 los que no tengan correspondencia tomarán los implícitos
void p(int i, int j = 2, int k = 3);
...
p(13); // i toma 13, j y k sus valores implícitos
p(5, 7); // i toma 5, j toma 7 y k su valor implícito
p(3, 9, 12); // i toma 3, j toma 9 y k toma 12
Luis Hernández Yáñez

Los argumentos implícitos se declaran en el prototipo


(preferible) o en la cabecera del subprograma, pero NO en ambos

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 506

Por defecto, signo +


Por defecto, Δ es 1

#include <iostream>
using namespace std;

double f(double x, double y, int signo = 1, double delta = 1.0);

int main() {
double x, y;
cout << "X = "; No podemos dejar signo por defecto
cin >> x; y concretar delta
cout << "Y = ";
cin >> y;
cout << "signo y delta por defecto: " << f(x, y) << endl;
cout << "signo ‐1 y delta por defecto: " << f(x, y, ‐1) << endl;
cout << "signo y delta concretos: " << f(x, y, 1, 1.25) << endl;

return 0;
}
Luis Hernández Yáñez

double f(double x, double y, int signo, double delta) {


return signo * delta * x / y;
}

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 507


Luis Hernández Yáñez

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 508

Igual nombre, distintos parámetros


Funciones o procedimientos con igual nombre y distintos parámetros:
int abs(int n);
double abs(double n);
long int abs(long int n);
Se ejecutará la función que corresponda al tipo de argumento:
abs(13)  // argumento int ‐‐> primera función
abs(‐2.3)  // argumento double ‐‐> segunda función
abs(3L)  // argumento long int ‐‐> tercera función
Luis Hernández Yáñez

Para indicar que es un literal long int, en lugar de int

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 509


inter.cpp

#include <iostream> void intercambia(char &x, char &y) {


using namespace std; char tmp;
tmp = x;
void intercambia(int &x, int &y); x = y;
void intercambia(double &x, y = tmp;
double &y); }
void intercambia(char &x, char &y);
int main() {
void intercambia(int &x, int &y) { int i1 = 3, i2 = 7;
int tmp; double d1 = 12.5, d2 = 35.9;
tmp = x; char c1 = 'a', c2 = 'b';
x = y; cout << i1 << " ‐ " << i2 << endl;
y = tmp; cout << d1 << " ‐ " << d2 << endl;
} cout << c1 << " ‐ " << c2 << endl;
intercambia(i1, i2);
void intercambia(double &x, intercambia(d1, d2);
double &y) { intercambia(c1, c2);
double tmp; cout << i1 << " ‐ " << i2 << endl;
Luis Hernández Yáñez

tmp = x; cout << d1 << " ‐ " << d2 << endl;


x = y; cout << c1 << " ‐ " << c2 << endl;
y = tmp; return 0;
} }

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 510

Licencia CC (Creative Commons)


Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: La abstracción procedimental (Anexo) Página 511


Fundamentos de la programación

5
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez/Pablo Moreno Ger


Facultad de Informática
Universidad Complutense

Tipos de datos 514
Arrays de nuevo 517
Arrays y bucles for 520
Más sobre arrays 522
Inicialización de arrays 523
Enumerados como índices 524
Paso de arrays a subprogramas 525
Implementación de listas 528
Cadenas de caracteres 531
Cadenas de caracteres de tipo string 535
Entrada/salida con string 539
Luis Hernández Yáñez/Pablo Moreno Ger

Operaciones con string 541
Estructuras 543
Estructuras dentro de estructuras 549
Arrays de estructuras 550
Arrays dentro de estructuras 551
Listas de longitud variable 552
Un ejemplo completo 558
El bucle do..while 562

Fundamentos de la programación: Tipos de datos estructurados
Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 514

Clasificación de tipos
 Simples
 Estándar: int, float, double, char, bool
Conjunto de valores predeterminado 
 Definidos por el usuario: enumerados
Conjunto de valores definido por el programador 
 Estructurados
Luis Hernández Yáñez/Pablo Moreno Ger

Colecciones homogéneas: arrays




Todos los elementos del mismo tipo
 Colecciones heterogéneas: estructuras
Los elementos pueden ser de tipos distintos

Fundamentos de la programación: Tipos de datos estructurados Página 515


Colecciones o tipos aglomerados
Agrupaciones de datos (elementos):
 Todos del mismo tipo: array o tabla
 De tipos distintos: estructura, registro o tupla
Arrays (tablas)
 Elementos organizados por posición: 0, 1, 2, 3, ...
 Acceso por índice: 0, 1, 2, 3, ...
Luis Hernández Yáñez/Pablo Moreno Ger

 Una o varias dimensiones


Estructuras (tuplas, registros)
 Elementos (campos) sin orden establecido
 Acceso por nombre

Fundamentos de la programación: Tipos de datos estructurados Página 516


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 517


Estructura secuencial
Cada elemento se encuentra en una posición (índice):
 Los índices son enteros positivos
 El índice del primer elemento siempre es 0
 Los índices se incrementan de uno en uno
ventas 125.40 76.95 328.80 254.62 435.00 164.29 0.00
0 1 2 3 4 5 6
Luis Hernández Yáñez/Pablo Moreno Ger

Acceso directo []
A cada elemento se accede a través de su índice:
ventas[4] accede al 5º elemento (contiene el valor 435.00)
cout << ventas[4];
ventas[4] = 442.75; Datos de un mismo tipo base:
Se usan como cualquier variable

Fundamentos de la programación: Tipos de datos estructurados Página 518

Declaración de tipos de arrays


const int Dimensión = ...;
typedef tipo_base tNombre[Dimensión];
Ejemplo:
const int Dias = 7;
typedef double tVentas[Dias];
Declaración de variables de tipos array: como cualquier otra
Luis Hernández Yáñez/Pablo Moreno Ger

tVentas ventas;
¡NO se inicializan los elementos automáticamente!
¡Es responsabilidad del programador usar índices válidos!
No se pueden copiar arrays directamente (array1 = array2)
Hay que copiarlos elemento a elemento

Fundamentos de la programación: Tipos de datos estructurados Página 519


Procesamiento de arrays
 Recorridos
 Búsquedas
 Ordenación etcétera...

Recorrido de arrays con bucles for


Arrays: tamaño fijo  Bucles de recorrido fijo (for)
tVentas ventas; const int Dias = 7;
Luis Hernández Yáñez/Pablo Moreno Ger

double media, total = 0; 
... typedef double tVentas[Dias];
for (int i = 0; i < Dias; i++) {
total = total + ventas[i];
}
media = total / Dias;

Fundamentos de la programación: Tipos de datos estructurados Página 520

12.40 10.96 8.43 11.65 13.70 13.41 14.07


0 1 2 3 4 5 6

tVentas ventas;
Memoria
double media, total = 0; 
Dias 7
...
ventas[0] 12.40
for (int i = 0; i < Dias; i++) {
ventas[1] 10.96
total = total + ventas[i];
} ventas[2] 8.43
ventas[3] 11.65
i = 0 ventas[4] 13.70
Luis Hernández Yáñez/Pablo Moreno Ger

ventas[5] 13.41
true false ventas[6] 14.07
i<Dias
media ?
... total 84.62
23.36
31.79
12.40
0.00
43.44
total+=ventas[i]
i 3
2
70
4
1
i++

Fundamentos de la programación: Tipos de datos estructurados Página 521


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 522

Podemos inicializar los elementos de los arrays en la declaración


Asignamos una serie de valores al array:
const int DIM = 10;
typedef int tTabla[DIM];
tTabla i = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };

Se asignan los valores por su orden:


i[0] i[1] i[2] i[3] i[4] ... i[9]
Luis Hernández Yáñez/Pablo Moreno Ger

1º 2º 3º 4º 5º  ...  10º

Si hay menos valores que elementos, los restantes se ponen a 0


tTabla i = { 0 }; // Pone todos los elementos a 0

Fundamentos de la programación: Tipos de datos estructurados Página 523


const int Colores = 3,
typedef enum { rojo, verde, azul } tRGB;
typedef int tColor[Colores];
tColor color;
...
cout << "Cantidad de rojo (0‐255): ";
cin >> color[rojo];
cout << "Cantidad de verde (0‐255): ";
cin >> color[verde];
Luis Hernández Yáñez/Pablo Moreno Ger

cout << "Cantidad de azul (0‐255): ";
cin >> color[azul];

Recuerda que internamente se asignan enteros a partir de 0


a los distintos símbolos del enumerado
rojo  0 verde  1 azul  2

Fundamentos de la programación: Tipos de datos estructurados Página 524

Simulación de paso de parámetro por referencia


Sin poner & en la declaración del parámetro
Los subprogramas reciben la dirección en memoria del array
const int Max = 10;
typedef int tTabla[Max];
void inicializa(tTabla tabla); // Sin poner &

Las modificaciones del array quedan reflejadas en el argumento


Luis Hernández Yáñez/Pablo Moreno Ger

inicializa(array);
Si inicializa() modifica algún elemento de tabla,
automáticamente queda modificado ese elemento de array

¡Son el mismo array!

Fundamentos de la programación: Tipos de datos estructurados Página 525


const int Dim = 10;
typedef int tTabla[Dim];
void inicializa(tTabla tabla); // no se usa &

void inicializa(tTabla tabla) {


for (int i = 0; i < Dim; i++)
tabla[i] = i;
}
int main() {
tTabla array;
Luis Hernández Yáñez/Pablo Moreno Ger

inicializa(array); // array queda modificado
for (int i = 0; i < Dim; i++)
cout << array[i] << " ";
...

0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Tipos de datos estructurados Página 526

¿Cómo evitar que se modifique el array?


Usando el modificador const en la declaración del parámetro:
const tTabla tabla Un array de constantes
void muestra(const tTabla tabla);
El argumento se tratará como un array de constantes

Si en el subprograma hay alguna instrucción que intente


modificar un elemento del array: error de compilación
Luis Hernández Yáñez/Pablo Moreno Ger

void muestra(const tTabla tabla) {


for (int i = 0; i < Dim; i++) {
cout << tabla[i] << " ";
// OK. Se accede, pero no se modifica
}
}

Fundamentos de la programación: Tipos de datos estructurados Página 527


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 528

Listas con un número fijo de elementos


Array con el nº de elementos como dimensión
const int NUM = 100;
typedef double tLista[NUM]; // Exactamente 100 double
tLista lista;

Recorrido de la lista:
for (int i = 0; i < NUM; i++) {
Luis Hernández Yáñez/Pablo Moreno Ger

...
Búsqueda en la lista:
while ((i < NUM) && !encontrado) { 
...

Fundamentos de la programación: Tipos de datos estructurados Página 529


Listas con un número variable de elementos
Array con un máximo de elementos + Contador de elementos
const int MAX = 100;
typedef double tLista[MAX]; // Hasta 100 elementos
tLista lista; 
int contador = 0; // Se incrementa al insertar
Recorrido de la lista:
for (int i = 0; i < contador; i++) {
Luis Hernández Yáñez/Pablo Moreno Ger

...
Búsqueda en la lista:
while ((i < contador) && !encontrado) {
...

¿Array y contador por separado?  Estructuras

Fundamentos de la programación: Tipos de datos estructurados Página 530


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 531


Arrays de caracteres
Cadenas: secuencias de caracteres de longitud variable
"Hola" "Adiós" "Supercalifragilístico" "1234 56 7"
Variables de cadena: contienen secuencias de caracteres
Se guardan en arrays de caracteres: tamaño máximo (dimensión)
No todas las posiciones del array son relevantes:
 Longitud de la cadena: número de caracteres, desde el
primero, que realmente constituyen la cadena:
Luis Hernández Yáñez/Pablo Moreno Ger

H o l a
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

Longitud actual: 4

Fundamentos de la programación: Tipos de datos estructurados Página 532

Longitud de la cadena
A d i ó s
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

Longitud: 5

S u p e r c a l i f r a g i l í s t i c o
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

Longitud: 21
Luis Hernández Yáñez/Pablo Moreno Ger

Necesidad de saber dónde terminan los caracteres relevantes:


 Mantener la longitud de la cadena como dato asociado
 Colocar un carácter de terminación al final (centinela)
A d i ó s \0
0 1 2 3 4 5 6 7 8 9 10

Fundamentos de la programación: Tipos de datos estructurados Página 533


Cadenas de caracteres en C++
Dos alternativas para el manejo de cadenas:
 Cadenas al estilo de C (terminadas en nulo)
 Tipo string

Cadenas al estilo de C Anexo del tema


 Arrays de tipo char con una longitud máxima
 Un último carácter especial al final: '\0'
Luis Hernández Yáñez/Pablo Moreno Ger

Tipo string
 Cadenas más sofisticadas
 Sin longitud máxima (gestión automática de la memoria)
 Multitud de funciones de utilidad (biblioteca string)

Fundamentos de la programación: Tipos de datos estructurados Página 534


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 535


El tipo string
 El tipo asume la responsabilidad de la gestión de memoria
 Define operadores sobrecargados (+ para concatenar)
 Cadenas más eficientes y seguras de usar
Biblioteca string
Requiere establecer el espacio de nombres a std

 Se pueden inicializar en la declaración


Luis Hernández Yáñez/Pablo Moreno Ger

 Se pueden copiar con el operador de asignación


 Se pueden concatenar con el operador +
 Multitud de funciones de utilidad

Fundamentos de la programación: Tipos de datos estructurados Página 536

string.cpp

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

int main() {
string cad1("Hola");   // inicialización
string cad2 = "amigo"; // inicialización
string cad3;
cad3 = cad1; // copia
cout << "cad3 = " << cad3 << endl;
cad3 = cad1 + " "; // concatenación
Luis Hernández Yáñez/Pablo Moreno Ger

cad3 += cad2;      // concatenación
cout << "cad3 = " << cad3 << endl;
cad1.swap(cad2);  // intercambio
cout << "cad1 = " << cad1 << endl;
cout << "cad2 = " << cad2 << endl;

return 0;
}

Fundamentos de la programación: Tipos de datos estructurados Página 537


Longitud de la cadena:
cadena.length() o cadena.size()

Se pueden comparar con los operadores relacionales:


if (cad1 <= cad2) { ...

Acceso a los caracteres de una cadena:


 Como array de caracteres: cadena[i]
Sin control de acceso a posiciones inexistentes del array
Luis Hernández Yáñez/Pablo Moreno Ger

Sólo debe usarse si se está seguro de que el índice es válido


 Función at(índice): cadena.at(i)
Error de ejecución si se accede a una posición inexistente

Fundamentos de la programación: Tipos de datos estructurados Página 538

 Se muestran en la pantalla con cout <<


 Lectura con cin >>: termina con espacio en blanco (inc. Intro)
El espacio en blanco queda pendiente
 Descartar el resto de los caracteres del búfer:
cin.sync();
 Lectura incluyendo espacios en blanco:
getline(cin, cadena)
Luis Hernández Yáñez/Pablo Moreno Ger

Guarda en la cadena los caracteres leídos hasta el fin de línea


 Lectura de archivos de texto:
Igual que de consola; sync() no tiene efecto
archivo >> cadena      getline(archivo, cadena)

Fundamentos de la programación: Tipos de datos estructurados Página 539


string2.cpp

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

int main() {
string nombre, apellidos;
cout << "Introduzca un nombre: ";
cin >> nombre;
cout << "Introduzca los apellidos: ";
cin.sync();
getline(cin, apellidos);
Luis Hernández Yáñez/Pablo Moreno Ger

cout << "Nombre completo: " << nombre << " "
<< apellidos << endl;

return 0;
}

Fundamentos de la programación: Tipos de datos estructurados Página 540

 cadena.substr(posición, longitud)
Subcadena de longitud caracteres desde posición
string cad = "abcdefg";
cout << cad.substr(2, 3); // Muestra cde
 cadena.find(subcadena)
Posición de la primera ocurrencia de subcadena en cadena
string cad = "Olala";
cout << cad.find("la"); // Muestra 1
Luis Hernández Yáñez/Pablo Moreno Ger

(Recuerda que los arrays de caracteres comienzan con el índice 0)


 cadena.rfind(subcadena)
Posición de la última ocurrencia de subcadena en cadena
string cad = "Olala";
cout << cad.rfind("la"); // Muestra 3

Fundamentos de la programación: Tipos de datos estructurados Página 541


 cadena.erase(ini, num)
Elimina num caracteres a partir de la posición ini
string cad = "abcdefgh";
cad.erase(3, 4); // cad ahora contiene "abch"
 cadena.insert(ini, cadena2)
Inserta cadena2 a partir de la posición ini
string cad = "abcdefgh";
cad.insert(3, "123"); // cad ahora contiene "abc123defgh"
Luis Hernández Yáñez/Pablo Moreno Ger

http://www.cplusplus.com/reference/string/string/

Fundamentos de la programación: Tipos de datos estructurados Página 542


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 543


Colecciones heterogéneas (tuplas, registros)
Elementos de (posiblemente) distintos tipos: campos
Campos identificados por su nombre
Información relacionada que se puede manejar como una unidad
Acceso a cada elemento por su nombre de campo (operador.)
Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 544

typedef struct {
... // declaraciones de campos (como variables)
} tTipo; // nombre de tipo ‐ ¡al final!
typedef struct {
string nombre;
string apellidos;
int edad;
string nif;
} tPersona;
Luis Hernández Yáñez/Pablo Moreno Ger

Campos:
Tipos estándar o previamente declarado

Fundamentos de la programación: Tipos de datos estructurados Página 545


tPersona persona;
Las variables de tipo tPersona contienen cuatro datos (campos):
nombre apellidos edad nif
Acceso a los campos con el operador punto (.):
persona.nombre // una cadena (string)
persona.apellidos // una cadena (string)
persona.edad // un entero (int)
persona.nif // una cadena (string)
Luis Hernández Yáñez/Pablo Moreno Ger

Podemos copiar dos estructuras directamente:


tPersona persona1, persona2;
...
persona2 = persona1;
Se copian todos los campos a la vez

Fundamentos de la programación: Tipos de datos estructurados Página 546

typedef struct {
string nombre;
string apellidos; Memoria
int edad;
string nif; Luis 
} tPersona; persona.nombre
Antonio
tPersona persona;

persona
Hernández 
persona.apellidos
Yáñez
nombre Luis Antonio
Luis Hernández Yáñez/Pablo Moreno Ger

persona.edad 22
apellidos Hernández Yáñez
persona.nif 00223344F
edad 22

nif 00223344F

Fundamentos de la programación: Tipos de datos estructurados Página 547


typedef struct {
string nombre;
string apellidos;
int edad;
string nif;
} tPersona;
tPersona persona;

Los campos no siguen ningún orden establecido


Acceso directo por nombre de campo (operador .)
Luis Hernández Yáñez/Pablo Moreno Ger

Con cada campo se puede hacer lo que permita su tipo

Las estructuras se pasan por valor (sin &)


o por referencia (con &) a los subprogramas

Fundamentos de la programación: Tipos de datos estructurados Página 548

typedef struct { typedef struct { 
string dni; ...
char letra; tNif nif;
} tNif; } tPersona;

tPersona persona; tPersona


nombre
Acceso al NIF completo: apellidos
persona.nif // Otra estructura edad
tNif
Luis Hernández Yáñez/Pablo Moreno Ger

nif
Acceso a la letra del NIF: dni
letra
persona.nif.letra

Acceso al DNI:
persona.nif.dni

Fundamentos de la programación: Tipos de datos estructurados Página 549


const int DIM = 100; tPersona personal
typedef struct { nombre
nombre
apellidos
string nombre; 0 edad
apellidos
nif
string apellidos; edad
nombre
int edad; nif apellidos
1
string nif; edad
nif
} tPersona;
nombre
typedef tPersona tArray[DIM]; 2 apellidos
edad
tArray personal; nif
Luis Hernández Yáñez/Pablo Moreno Ger

Nombre de la tercera persona:


personal[2].nombre nombre

Edad de la duodécima persona:


apellidos
DIM‐1 edad
nif
personal[11].edad
NIF de la primera persona:
personal[0].nif

Fundamentos de la programación: Tipos de datos estructurados Página 550

const int MAX = 100; lista elementos contador


typedef struct { nombre
apellidos
string nombre; 0 edad
nif
string apellidos; nombre
int edad; 1
apellidos
edad
string nif; nif

} tPersona; 2
nombre
apellidos
edad
typedef tPersona tArray[MAX]; nif

typedef struct {
tArray elementos;
Luis Hernández Yáñez/Pablo Moreno Ger

int contador; MAX‐1


nombre
apellidos
edad
} tLista; nif

tLista lista;
Nombre de la tercera persona: lista.elementos[2].nombre
Edad de la duodécima persona: lista.elementos[11].edad
NIF de la primera persona: lista.elementos[0].nif
Fundamentos de la programación: Tipos de datos estructurados Página 551
/ Pablo Moreno Ger
Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 552

Estructura que agrupe el array y el contador:


const int MAX = 10;
typedef double tArray[MAX];
typedef struct {
Elementos sin usar
tArray elementos;
(datos basura)
int contador;
} tLista;

elementos
Luis Hernández Yáñez/Pablo Moreno Ger

12.0 ‐2.2 5.4 0.0 36.2 35.0 X X X X


0 1 2 3 4 5 6 7 8 9

contador 6

Nº de elementos (y primer índice sin elemento)

Operaciones principales: inserción y eliminación de elementos

Fundamentos de la programación: Tipos de datos estructurados Página 553


Insertar un nuevo elemento en una posición
Posiciones válidas: 0 a contador

42.0
12.0 ‐2.2 5.4 0.0 36.2 35.0 X X X X
nuevo
0 1 2 3 4 5 6 7 8 9
3 6
pos

Hay que asegurarse de que haya sitio (contador < máximo)


Luis Hernández Yáñez/Pablo Moreno Ger

Operación en 3 pasos:
1.‐ Abrir hueco para el nuevo elemento (desde la posición)
2.‐ Colocar el elemento nuevo en la posición
3.‐ Incrementar el contador

Fundamentos de la programación: Tipos de datos estructurados Página 554

if (lista.contador < N) {
// Abrir hueco
for (int i = lista.contador; i > pos; i‐‐) {
lista.elementos[i] = lista.elementos[i ‐ 1];
}
// Insertar e incrementar contador
lista.elementos[pos] = nuevoElemento;
lista.contador++;
}
Luis Hernández Yáñez/Pablo Moreno Ger

42.0
12.0 ‐2.2 5.4 42.0 0.0 36.2 35.0 X X X
nuevo
0 1 2 3 4 5 6 7 8 9
3 7
pos

Fundamentos de la programación: Tipos de datos estructurados Página 555


Eliminar el elemento en una posición
Posiciones válidas: 0 a contador‐1

12.0 ‐2.2 5.4 0.0 36.2 35.0 X X X X


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

Desplazar a la izquierda desde el siguiente y decrementar el contador:


Luis Hernández Yáñez/Pablo Moreno Ger

for (int i = pos; i < lista.contador ‐ 1 ; i++) {


lista.elementos[i] = lista.elementos[i + 1];
}
lista.contador‐‐;

Fundamentos de la programación: Tipos de datos estructurados Página 556

for (int i = pos; i < lista.contador ‐ 1 ; i++) {


lista.elementos[i] = lista.elementos[i + 1];
}
lista.contador‐‐;

12.0 ‐2.2 5.4 0.0 36.2 35.0 X X X X


3 0 1 2 3 4 5 6 7 8 9
pos 6
Luis Hernández Yáñez/Pablo Moreno Ger

12.0 ‐2.2 5.4 36.2 35.0 35.0 X X X X


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

Fundamentos de la programación: Tipos de datos estructurados Página 557


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 558

Descripción
Programa que mantenga una lista de los estudiantes de una clase
De cada estudiante: nombre, apellidos, edad, NIF y nota
 Se desconoce el número total de estudiantes (máximo 100)
 La información de la lista se mantiene en un archivo clase.txt
Se carga al empezar y se guarda al finalizar
 El programa debe ofrecer estas opciones:
Luis Hernández Yáñez/Pablo Moreno Ger

— Añadir un nuevo alumno

— Eliminar un alumno existente

— Calificar a los estudiantes

— Listado de notas, identificando la mayor y la media

Fundamentos de la programación: Tipos de datos estructurados Página 559


bd.cpp

#include <iostream>
#include <string>
using namespace std;
#include <fstream>
#include <iomanip>

const int MAX = 100;


typedef struct {
string nombre;
string apellidos;
int edad;
string nif;
Luis Hernández Yáñez/Pablo Moreno Ger

double nota; Declaraciones de constantes


} tEstudiante; y tipos globales
typedef tEstudiante tArray[MAX]; Tras las bibliotecas
typedef struct {
tArray elementos;
int contador;
} tLista;

Fundamentos de la programación: Tipos de datos estructurados Página 560

// Prototipos
int menu(); // Menú del programa ‐ devuelve la opción elegida
void cargar(tLista &lista, bool &ok); // Carga del archivo
void guardar(const tLista &lista); // La guarda en el archivo
void leerEstudiante(tEstudiante &estudiante); // Lee los datos
void insertarEstudiante(tLista &lista, tEstudiante estudiante,
bool &ok); // Inserta un nuevo estudiante en la lista
void eliminarEstudiante(tLista &lista, int pos, bool &ok);
// Elimina el estudiante en esa posición
string nombreCompleto(tEstudiante estudiante);
void calificar(tLista &lista); // Notas de los estudiantes
Luis Hernández Yáñez/Pablo Moreno Ger

double mediaClase(const tLista &lista); // Nota media


int mayorNota(const tLista &lista);
// Índice del estudiante con mayor nota
void mostrarEstudiante(tEstudiante estudiante);
void listado(const tLista &lista, double media, int mayor);
// Listado de la clase

Los prototipos, después de los tipos globales

Fundamentos de la programación: Tipos de datos estructurados Página 561


Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 562

El bucle do..while
do cuerpo while (condición); Condición al final del bucle

do cuerpo while ( condición ) ;

int i = 1;
do {
cout << i << endl;
Luis Hernández Yáñez/Pablo Moreno Ger

i++;
} while (i <= 100);

El cuerpo siempre se ejecuta al menos una vez


El cuerpo es un bloque de código

Fundamentos de la programación: Tipos de datos estructurados Página 563


int i = 1;
do {
Cuerpo
cout << i << endl;
i++; true
Condición
} while (i <= 100); false

i = 1;

El cuerpo
Luis Hernández Yáñez/Pablo Moreno Ger

cout << i << endl; 
se ejecuta
i++;
al menos
true una vez
i <= 100
false

Fundamentos de la programación: Tipos de datos estructurados Página 564

¿Ha de ejecutarse al menos una vez el cuerpo del bucle?


cin >> d; // Lectura del 1º do {
while (d != 0) { cin >> d;
suma = suma + d; if (d != 0) { // ¿Final?
cont++; suma = suma + d;
cin >> d; cont++;
} }
} while (d != 0);
Luis Hernández Yáñez/Pablo Moreno Ger

cout << "Opción: ";  do { // Más simple
cin >> op; // Lectura del 1º  cout << "Opción: ";
while ((op < 0) || (op > 4)) { cin >> op;
cout << "Opción: ";  } while ((op < 0) || (op > 4));
cin >> op;
}

Fundamentos de la programación: Tipos de datos estructurados Página 565


int menu() {
int op;

do {
cout << "1 ‐ Añadir un nuevo estudiante" << endl;
cout << "2 ‐ Eliminar un estudiante" << endl;
cout << "3 ‐ Calificar a los estudiantes" << endl;
cout << "4 ‐ Listado de estudiantes" << endl;
cout << "0 ‐ Salir" << endl;
cout << "Opción: ";
cin >> op;
Luis Hernández Yáñez/Pablo Moreno Ger

} while ((op < 0) || (op > 4));

return op;
}

Fundamentos de la programación: Tipos de datos estructurados Página 566

El archivo clase.txt
Un dato en cada línea ↲


Por cada estudiante: ↲


 Nombre (cadena) ↲


 Apellidos (cadena) ↲


 Edad (entero) ↲


 NIF (cadena) ↲

Luis Hernández Yáñez/Pablo Moreno Ger


 Nota (real; ‐1 si no calificado) ↲

Termina con XXX como nombre ↲



El archivo se supone correcto

Fundamentos de la programación: Tipos de datos estructurados Página 567


Lectura de la información de un estudiante
Nombre y apellidos:
Puede haber varias palabras  getline()
Edad  extractor (>>)
NIF: Una sola palabra  extractor (>>)
Nota  extractor (>>)
Queda pendiente de leer el Intro
Luis Hernández Yáñez/Pablo Moreno Ger

Hay que saltar (leer) ese carácter con get()


Si no, en el siguiente nombre se leería una cadena vacía (Intro)
No leas directamente en la lista:
getline(archivo, lista.elementos[lista.contador].nombre);
Lee en una variable auxiliar de tipo tEstudiante

Fundamentos de la programación: Tipos de datos estructurados Página 568

void cargar(tLista &lista, bool &ok) {


tEstudiante estudiante; // Variable auxiliar para leer
ifstream archivo;
char aux;
lista.contador = 0; // Inicializamos la lista
archivo.open("clase.txt");
if (!archivo.is_open()) {
ok = false;
}
else {
ok = true;
getline(archivo, estudiante.nombre); // Leemos el primer nombre
while ((estudiante.nombre != "XXX") && (lista.contador < MAX)) {
getline(archivo, estudiante.apellidos);
archivo >> estudiante.edad;
Luis Hernández Yáñez/Pablo Moreno Ger

archivo >> estudiante.nif;
archivo >> estudiante.nota;
archivo.get(aux); // Saltamos el Intro
lista.elementos[lista.contador] = estudiante; // Al final
lista.contador++; 
getline(archivo, estudiante.nombre); // Siguiente nombre
} // Si hay más de MAX estudiantes, ignoramos el resto
archivo.close();
}
}
Fundamentos de la programación: Tipos de datos estructurados Página 569
Simplemente, un dato en cada línea y en orden:
void guardar(const tLista &lista) {
ofstream archivo;
archivo.open("clase.txt");
for (int i = 0; i < lista.contador; i++) {
archivo << lista.elementos[i].nombre << endl;
archivo << lista.elementos[i].apellidos << endl;
archivo << lista.elementos[i].edad << endl;
archivo << lista.elementos[i].nif << endl;
archivo << lista.elementos[i].nota << endl;
Luis Hernández Yáñez/Pablo Moreno Ger

}
archivo << "XXX" << endl; // Centinela final
archivo.close();
}

const tLista &lista  Referencia constante


Paso por referencia pero como constante  Paso por valor
Evita la copia del argumento en el parámetro (estructuras grandes)

Fundamentos de la programación: Tipos de datos estructurados Página 570

void leerEstudiante(tEstudiante &estudiante) {
cin.sync(); // Descartamos cualquier entrada pendiente
cout << "Nombre: ";
getline(cin, estudiante.nombre);
cout << "Apellidos: ";
getline(cin, estudiante.apellidos);
cout << "Edad: ";
cin >> estudiante.edad;
cout << "NIF: ";
cin >> estudiante.nif;
estudiante.nota = ‐1; // Sin calificar de momento
Luis Hernández Yáñez/Pablo Moreno Ger

cin.sync(); // Descartamos cualquier entrada pendiente
}

Fundamentos de la programación: Tipos de datos estructurados Página 571


void insertarEstudiante(tLista &lista, tEstudiante estudiante,
bool &ok) {

ok = true;
if (lista.contador == MAX) {
ok = false;
}
else {
lista.elementos[lista.contador] = estudiante;
// Insertamos al final
lista.contador++;
Luis Hernández Yáñez/Pablo Moreno Ger

}
}

Fundamentos de la programación: Tipos de datos estructurados Página 572

void eliminarEstudiante(tLista &lista, int pos, bool &ok) {
// Espera el índice del elemento en pos

if ((pos < 0) || (pos > lista.contador ‐ 1)) {
ok = false; // Elemento inexistente
}
else {
ok = true;
for (int i = pos; i < lista.contador ‐ 1; i++) {
lista.elementos[i] = lista.elementos[i + 1];
}
Luis Hernández Yáñez/Pablo Moreno Ger

lista.contador‐‐;
}
}

Fundamentos de la programación: Tipos de datos estructurados Página 573


string nombreCompleto(tEstudiante estudiante) {
return estudiante.nombre + " " + estudiante.apellidos;
}

void calificar(tLista &lista) {

for (int i = 0; i < lista.contador; i++) {
cout << "Nota del estudiante "
<< nombreCompleto(lista.elementos[i]) << ": ";
cin >> lista.elementos[i].nota;
}
Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Tipos de datos estructurados Página 574

double mediaClase(const tLista &lista) {


double total = 0.0;
for (int i = 0; i < lista.contador; i++) {
total = total + lista.elementos[i].nota;
}
return total / lista.contador;
}

int mayorNota(const tLista &lista) {


double max = 0;
int pos = 0;
Luis Hernández Yáñez/Pablo Moreno Ger

for (int i = 0; i < lista.contador; i++) {
if (lista.elementos[i].nota > max) {
max = lista.elementos[i].nota;
pos = i;
}
}
return pos;
}

Fundamentos de la programación: Tipos de datos estructurados Página 575


void mostrarEstudiante(tEstudiante estudiante) {
cout << setw(35) << left << nombreCompleto(estudiante);
cout << estudiante.nif << " ";
cout << setw(2) << estudiante.edad << " años ";
cout << fixed << setprecision(1) << estudiante.nota;
}

void listado(const tLista &lista, double media, int mayor) {


for (int i = 0; i < lista.contador; i++) {
cout << setw(3) << i << ": ";
mostrarEstudiante(lista.elementos[i]);
Luis Hernández Yáñez/Pablo Moreno Ger

if (i == mayor) {
cout << " <<< Mayor nota!";
}
cout << endl;
}
cout << "Media de la clase: " << fixed << setprecision(1)
<< media << endl << endl;
}

Fundamentos de la programación: Tipos de datos estructurados Página 576

int main() {
tLista lista;
tEstudiante estudiante;
bool exito;
int op, pos;

cargar(lista, exito);
if (!exito) {
cout << "No se ha podido cargar la lista!" << endl;
}
else {
do { // El bucle do evita tener que leer antes la primera opción
op = menu();
Luis Hernández Yáñez/Pablo Moreno Ger

switch (op) {
case 1:
{
leerEstudiante(estudiante);
insertarEstudiante(lista, estudiante, exito);
if (!exito) {
cout << "Lista llena: imposible insertar" << endl;
}
}
break;

Fundamentos de la programación: Tipos de datos estructurados Página 577


case 2:
{
cout << "Posición: ";
cin >> pos;
eliminarEstudiante(lista, pos ‐ 1, exito);
if (!exito) {
cout << "Elemento inexistente!" << endl;
}
}
break;
case 3:
{
calificar(lista);
}
break;
Luis Hernández Yáñez/Pablo Moreno Ger

case 4:
{
listado(lista, mediaClase(lista), mayorNota(lista));
}
}
} while (op != 0);
guardar(lista);
}
return 0;
}
Fundamentos de la programación: Tipos de datos estructurados Página 578

Licencia CC (Creative Commons)


Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
Luis Hernández Yáñez/Pablo Moreno Ger

La explotación de la obra queda limitada a usos no comerciales.


Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Tipos de datos estructurados Página 579


Fundamentos de la programación

5A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez/Pablo Moreno Ger


Facultad de Informática
Universidad Complutense

Cadenas al estilo de C 582
E/S con cadenas al estilo de C 583
La biblioteca cstring 584
Ejemplo 585
Luis Hernández Yáñez/Pablo Moreno Ger

Fundamentos de la programación: Cadenas al estilo de C (Anexo)
Arrays de caracteres terminados en nulo
const Max = 15;
typedef char tCadena[Max];
tCadena cadena = "Adiós"; // Inicialización al declarar
Siempre hay al final un carácter nulo (código ASCII 0 – '\0')
Indica que en esa posición termina la cadena (exclusive)
cadena A d i ó s \0
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14
Luis Hernández Yáñez/Pablo Moreno Ger

En el array caben MAX‐1 caracteres significativos


Longitud máxima de la variable cadena: 14
No se pueden asignar cadenas literales: cadena = "Hola";
Ni copiar cadenas directamente: cad2 = cad1;
Ni comparar con op. relacionales: if (cad1 < cad2) ...
Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 582

tCadena cadena;
cin >> cadena; // Se añade un nulo al final
Extractor: la lectura termina en el primer espacio en blanco
¡No se comprueba si se leen más caracteres de los que caben!
setw(): máximo de caracteres a colocar (incluyendo el nulo)
cin >> setw(15) >> cadena;
cin.getline(cadena_estilo_C, máx):
Para leer también los espacios en blanco y no más de máx‐1
Luis Hernández Yáñez/Pablo Moreno Ger

cin.getline(cadena, 15); // Hasta 14 caracteres

cout << cadena << endl; // El nulo no se muestra

cin.getline(cad, máx) Cadenas al estilo de C


getline(cin, cad) Cadenas de tipo string

Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 583


 strlen(cadena): longitud actual de la cadena
cout << "Longitud: " << strlen(cadena);
 strcpy(destino, origen): copia origen en destino
strcpy(cad2, cad1); strcpy(cad, "Me gusta C++");
 strcat(destino, origen): añade origen al final de destino
tCadena cad1 = "Hola", cad2 = "Adiós";
strcat(cad1, cad2); // cad1 contiene "HolaAdiós"
 strcmp(cad1, cad2): compara lexicográficamente las cadenas
Luis Hernández Yáñez/Pablo Moreno Ger

0 si son iguales, 1 si cad1 > cad2 ó ‐1 si cad1 < cad2


tCadena cad1 = "Hola", cad2 = "Adiós";
strcmp(cad1, cad2) // Devuelve 1 ("Hola" > "Adiós")
. . .
http://www.cplusplus.com/reference/clibrary/cstring/

Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 584

cadenas.cpp

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

int main() {
const int MAX = 20;
typedef char tCad[MAX];
tCad cadena = "Me gusta C++";
cout << cadena << endl;
cout << "Cadena: ";
cin >> cadena; // Lee hasta el primer espacio en blanco
Luis Hernández Yáñez/Pablo Moreno Ger

cout << cadena << endl;
cin.sync(); // Sincronizar la entrada
cout << "Cadena: ";
cin.getline(cadena, MAX);
cout << cadena << endl;
cout << "Longitud: " << strlen(cadena) << endl;
strcpy(cadena, "Hola");
...

Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 585


tCad cadena2 = " amigo";
strcat(cadena, cadena2);
cout << cadena << endl;
if (strcmp(cadena, cadena2) == 0) {
cout << "Iguales";
}
else if (strcmp(cadena, cadena2) > 0) {
cout << cadena << " es mayor que " << cadena2;
}
else {
cout << cadena << " es menor que " << cadena2;
Luis Hernández Yáñez/Pablo Moreno Ger

}
cout << endl;

return 0;
}

Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 586

Licencia CC (Creative Commons)


Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
Luis Hernández Yáñez/Pablo Moreno Ger

La explotación de la obra queda limitada a usos no comerciales.


Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Cadenas al estilo de C (Anexo) Página 587


Fundamentos de la programación

6
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez / Pablo Moreno Ger


Facultad de Informática
Universidad Complutense

Recorrido de arrays 590
Arrays completos 593
Arrays no completos con centinela 594
Arrays no completos con contador 595
Ejemplos 597
Generación de números aleatorios 601
Búsquedas en arrays 604
Arrays completos 606
Arrays no completos con centinela 607
Arrays no completos con contador 608
Ejemplo 610
Recorridos y búsquedas en cadenas 614
Más ejemplos de manejo de arrays 617
Arrays multidimensionales 630
Inicialización de arrays multidimensionales 638
Luis Hernández Yáñez

Recorrido de un array bidimensional 641
Recorrido de un array N‐dimensional 644
Búsqueda en un array multidimensional 647

Fundamentos de la programación: Recorrido y búsqueda en arrays
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 590

Esquema de recorrido
Inicialización Inicialización

Mientras no al final de la secuencia:


Obtener el siguiente elemento true
¿Al final?
Procesar el elemento
false
Finalización
Obtener elemento

Procesar elemento
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 591


Recorrido de secuencias en arrays
 Todas las posiciones ocupadas:
Tamaño del array = longitud de la secuencia
N elementos en un array de N posiciones:
Recorrer el array desde la primera posición hasta la última

 Posiciones libres al final del array:


Tamaño del array > longitud de la secuencia
 Con centinela:
Recorrer el array hasta encontrar el valor centinela
 Con contador de elementos:
Luis Hernández Yáñez

Recorrer el array hasta el índice contador – 1

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 592

Recorrido de arrays completos


Todas las posiciones del array ocupadas
const int N = 10;
typedef double tVentas[N];
tVentas ventas;
...
ventas 125.40 76.95 328.80 254.62 435.00 164.29 316.05 219.99 93.45 756.62
0 1 2 3 4 5 6 7 8 9

double elemento;
for (int i = 0; i < N; i++) {
elemento = ventas[i];
Luis Hernández Yáñez

// Procesar el elemento ...
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 593


Recorrido de arrays no completos – con centinela
No todas las posiciones del array están ocupadas
const int N = 10;
typedef double tArray[N];
tArray datos; // Datos positivos: centinela = ‐1
...
datos 125.40 76.95 328.80 254.62 435.00 164.29 316.05 ‐1.0
0 1 2 3 4 5 6 7 8 9

int i = 0;
int i = 0; double elemento;
double elemento = datos[i]; do {
while (elemento != ‐1) { elemento = datos[i];
Luis Hernández Yáñez

// Procesar el elemento ... if (elemento != ‐1) {
// Procesar el elemento...
i++; i++;
elemento = datos[i]; }
} } while (elemento != ‐1);

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 594

Recorrido de arrays no completos – con contador


Array y contador íntimamente relacionados: estructura
const int N = 10;
typedef double tArray[N];
typedef struct {
tArray elementos;
int contador;
} tLista;
Listas de elementos de longitud variable
elementos
125.40 76.95 328.80 254.62 435.00 164.29 316.05
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

contador 7

Nº de elementos (primer índice sin elemento)

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 595


Recorrido de arrays no completos – con contador
const int N = 10;
typedef double tArray[N];
typedef struct {
tArray elementos;
int contador;
} tLista;
tLista lista;
...
double elemento;
for (int i = 0; i < lista.contador; i++) {
elemento = lista.elementos[i];
// Procesar el elemento...
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 596


Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 597


fibonacci.cpp

Array con los N primeros números de Fibonacci


const int N = 50;
typedef long long int tFibonacci[N]; // 50 números
tFibonacci fib;
fib[0] = 1;
fib[1] = 1;
for (int i = 2; i < N; i++) {
fib[i] = fib[i ‐ 1] + fib[i ‐ 2];
}
for (int i = 0; i < N; i++) {
cout << fib[i] << "   ";
}
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 598

Cuenta de valores con k dígitos


Recorrer una lista de N enteros contabilizando cuántos son
de 1 dígito, cuántos de 2 dígitos, etcétera (hasta 5 dígitos)
2 arrays: array con los números y array de contadores
const int NUM = 100;
typedef int tNum[NUM]; // Exactamente 100 números
tNum numeros;
const int DIG = 5;
typedef int tDig[DIG]; // i ‐‐> números de i+1 dígitos
tDig numDig = { 0 };

numeros 123 2 46237 2345 236 11234 33 999 ... 61


Luis Hernández Yáñez

0 1 2 3 4 5 6 7 99

numDig 0 0 0 0 0 0
0 1 2 3 4 5

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 599


Cuenta de valores con k dígitos
Función que devuelve el número de dígitos de un entero:
int digitos(int dato) {
int n_digitos = 1; // Al menos tiene un dígito
// Recorremos la secuencia de dígitos...
while (dato >= 10) {
dato = dato / 10;
n_digitos++;
}
return n_digitos;
}
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 600

Generación de números pseudoaleatorios


Probemos con una secuencia de enteros generada aleatoriamente
Función rand() (cstdlib): entero aleatorio entre 0 y 32766
srand() (cstdlib): inicia la secuencia de números aleatorios
Acepta un entero que usa como semilla para iniciar la secuencia
¿Qué valor usar? Uno distinto en cada ejecución
 El instante de tiempo actual (diferente cada vez)
Función time() (ctime): segundos transcurridos desde 1970
Requiere un argumento, que en nuestro caso será NULL
srand(time(NULL)); // Inicia la secuencia
Luis Hernández Yáñez

...
numeros[0] = rand(); // Entre 0 y 32766

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 601


digitos.cpp

Cuenta de valores con k dígitos


#include <iostream>
using namespace std;
#include <cstdlib> // srand() y rand()
#include <ctime> // time()

int digitos(int dato);

int main() {
const int NUM = 100;
typedef int tNum[NUM]; // Exactamente 100 números
const int DIG = 5;
typedef int tDig[DIG];
tNum numeros;
tDig numDig = { 0 }; // Inicializa todo el array a 0
Luis Hernández Yáñez

srand(time(NULL)); // Inicia la secuencia aleatoria
...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 602

for (int i = 0; i < NUM; i++) { // Creamos la secuencia


numeros[i] = rand(); // Entre 0 y 32766
}

for (int i = 0; i < NUM; i++) {


// Recorremos la secuencia de enteros
numDig[digitos(numeros[i]) ‐ 1]++;
}

for (int i = 0; i < DIG; i++) {


// Recorremos la secuencia de contadores
cout << "De " << i + 1 << " díg. = " << numDig[i] 
<< endl;
}
return 0;
Luis Hernández Yáñez

int digitos(int dato) {


...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 603


Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 604

Esquema de búsqueda
Inicialización / encontrado = false;
Inicialización
Mientras no se encuentre el elemento
true
y no se esté al final de la secuencia: ¿Al final o
encontrado?
Obtener el siguiente elemento false

Comprobar si el elemento
Obtener elemento
satisface la condición
Finalización ¿Encontrado?
(tratar el elemento encontrado
o indicar que no se ha encontrado)
Luis Hernández Yáñez

Finalización

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 605


Todas las posiciones ocupadas const int N = 100;
typedef int tArray[N];
int buscado; tArray lista;
bool encontrado = false;
cout << "Valor a buscar: ";
cin >> buscado;
int pos = 0;
while ((pos < N) && !encontrado) {
// Mientras no se llegue al final y no encontrado
if (lista[pos] == buscado) {
encontrado = true;
}
else {
pos++;
}
Luis Hernández Yáñez

}
if (encontrado) // ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 606

Con centinela
const int N = 10;
typedef int tArray[N];
int buscado; tArray array;
cout << "Valor a buscar: "; const int centinela = ‐1;
cin >> buscado;
int pos = 0;
bool encontrado = false;
while ((array[pos] != centinela) && !encontrado) {
if (array[pos] == buscado) {
encontrado = true;
}
else {
pos++;
}
}
Luis Hernández Yáñez

if (encontrado) // ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 607


Con contador
const int N = 10;
typedef double tArray[N];
int buscado; typedef struct {
cout << "Valor a buscar: "; tArray elementos;
cin >> buscado; int contador;
int pos = 0; } tLista;
bool encontrado = false; tLista miLista;
while ((pos < miLista.contador)
&& !encontrado) {
// Mientras no al final y no encontrado
if (miLista.elementos[pos] == buscado) {
encontrado = true;
}
else {
pos++;
Luis Hernández Yáñez

}
}
if (encontrado) // ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 608

Acceso directo a cualquier posición


Acceso directo: array[posición]
Si se puede calcular la posición del elemento, su acceso es directo
typedef double tVentaMes[DIAS][SUCURSALES];
typedef struct {
tVentaMes ventas;
int dias;
} tMes;
typedef tMes tVentaAnual[MESES];
tVentaAnual anual;
Ventas del cuarto día del tercer mes en la primera sucursal:
Luis Hernández Yáñez

anual[2].ventas[3][0]

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 609


Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 610

#include <iostream> umbral.cpp
using namespace std;
#include <fstream>

const int N = 100;
typedef double tArray[N];
typedef struct {
tArray elementos;
int contador;
} tLista;

void cargar(tLista &lista, bool &ok);

int main() {
tLista lista;
bool ok;
Luis Hernández Yáñez

cargar(lista, ok);
if (!ok) {
cout << "Error: no hay archivo o demasiados datos"
<< endl;
}
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 611
else {
double umbral;
cout << "Valor umbral: "; cin >> umbral;
bool encontrado = false;
int pos = 0;
while ((pos < lista.contador) && !encontrado) {
if (lista.elementos[pos] > umbral) {
encontrado = true;
}
else {
pos++;
}
}
if (encontrado) {
cout << "Valor en pos. " << pos + 1 << " (" 
<< lista.elementos[pos] << ")" << endl;
}
else {
Luis Hernández Yáñez

cout << "¡No encontrado!" << endl;
}
}
return 0;
}
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 612

void cargar(tLista &lista, bool &ok) {


ifstream archivo;
double dato;
bool abierto = true, overflow = false;
lista.contador = 0;
archivo.open("datos.txt");
if (!archivo.is_open()) {
abierto = false;
}
else {
archivo >> dato;
while ((dato >= 0) && !overflow) {
if (lista.contador == N) {
overflow = true; // ¡Demasiados!
}
else {
lista.elementos[lista.contador] = dato;
lista.contador++;
archivo >> dato;
}
Luis Hernández Yáñez

}
archivo.close();
}
ok = abierto && !overflow;
}
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 613
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 614

inversa.cpp

Recorridos y búsquedas en cadenas de caracteres


Longitud de la cadena: size() o length()
Caso similar a los arrays con contador de elementos
Ejemplo: Recorrido de una cadena generando otra invertida
string cadena, inversa = "";
int pos;
char car;
// ... (lectura de cadena)
pos = 0;
while (pos < cadena.size()) {
// Mientras no se llegue al final de la cadena
car = cadena.at(pos);
Luis Hernández Yáñez

inversa = car + inversa; // Inserta car al principio
pos++;
} // ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 615


busca.cpp

Búsqueda de un carácter en una cadena


string cadena;
char buscado;
int pos;
bool encontrado;
// ... (lectura de cadena)
cout << "Introduce el carácter a buscar: ";
cin >> buscado;
pos = 0;
encontrado = false;
while ((pos < cadena.size()) && !encontrado) {
if (cadena.at(pos) == buscado) {
encontrado = true;
}
else {
Luis Hernández Yáñez

pos++;
}
}
if (encontrado) // ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 616


Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 617


Tipo tVector para representar secuencias de N enteros:
const int N = 10;
typedef int tVector[N];
Subprogramas:
 Dado un vector, mueve sus componentes un lugar a la derecha;
el último componente se moverá al 1er lugar
 Dado un vector, calcula y devuelve la suma de los elementos que se
encuentran en las posiciones pares del vector
 Dado un vector, encuentra y devuelve la componente mayor
 Dados dos vectores, devuelve un valor que indique si son iguales
 Dado un vector, determina si alguno de los valores almacenados en
el vector es igual a la suma del resto de los valores del mismo;
Luis Hernández Yáñez

devuelve el índice del primero encontrado o ‐1 si no se encuentra


 Dado un vector, determina si alguno de los valores almacenados
en el vector está repetido

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 618

vectores.cpp

void desplazar(tVector v) {


int aux = v[N ‐ 1];

for (int i = N ‐ 1; i > 0; i‐‐) {


v[i] = v[i ‐ 1];
}
v[0] = aux;
}

int sumaPares(const tVector v) {


int suma = 0;

for (int i = 0; i < N; i = i + 2) {


suma = suma + v[i];
}
Luis Hernández Yáñez

return suma;
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 619


int encuentraMayor(const tVector v) {
int max = v[0], posMayor = 0;
for (int i = 1; i < N; i++) {
if (v[i] > max) {
posMayor = i;
max = v[i];
}
}
return posMayor;
}

bool sonIguales(const tVector v1, const tVector v2) {


//Implementación como búsqueda del primer elemento distinto
bool encontrado = false;
int i = 0;
while ((i<N) && !encontrado) {
encontrado = (v1[i] != v2[i]);
Luis Hernández Yáñez

i++;
}
return !encontrado;
}
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 620

int compruebaSuma(const tVector v) {


// ¿Alguno igual a la suma del resto?
bool encontrado = false;
int i = 0;
int suma;
while ((i < N) && !encontrado) {
suma = 0;
for (int j = 0; j < N; j++) {
if (j != i) {
suma = suma + v[j];
}
}
encontrado = (suma == v[i]);
i++;
}
if (!encontrado) {
i = 0;
Luis Hernández Yáñez

}
return i ‐ 1;
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 621


bool hayRepetidos(const tVector v) {
bool encontrado = false;
int i = 0, j;

while ((i < N) && !encontrado) {
j = i + 1;
while ((j < N) && !encontrado) {
encontrado = (v[i] == v[j]);
j++;
}
i++;
}

return encontrado;
}
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 622

Dado un vector de N caracteres v1, en el que no hay elementos


repetidos, y otro vector de M caracteres v2, donde N ≤ M, se
quiere comprobar si todos los elementos del vector v1 están
también en el vector v2
Por ejemplo, si:
v1= 'a' 'h' 'i' 'm'
v2= 'h' 'a' 'x' 'x' 'm' 'i'
El resultado sería cierto, ya que todos los elementos de v1 están
en v2
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 623


incluidos.cpp

#include <iostream>
using namespace std;

const int N = 3;


const int M = 10;
typedef char tVector1[N];
typedef char tVector2[M];

bool esta(char dato, const tVector2 v2);


bool vectorIncluido(const tVector1 v1, const tVector2 v2);

int main() {
tVector1 v1 = { 'a', 'b', 'c' };
tVector2 v2 = { 'a', 'r', 'e', 't', 'z', 's', 'a', 'h', 'b', 'x' };
bool ok = vectorIncluido(v1, v2);
if (ok) {
cout << "OK: v1 esta incluido en v2" << endl;
}
Luis Hernández Yáñez

else {
cout << "NO: v1 no esta incluido en v2" << endl;
}
return 0;
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 624

bool esta(char dato, const tVector2 v2) {


int i = 0;
bool encontrado = (dato == v2[0]);

while (!encontrado && (i < M ‐ 1)) {


i++;
encontrado = (dato == v2[i]);
}

return encontrado;
}

bool vectorIncluido(const tVector1 v1, const tVector2 v2) {


int i = 0;
bool encontrado = esta(v1[0], v2);

while (encontrado && (i < N ‐ 1)) {
i++;
Luis Hernández Yáñez

encontrado = esta(v1[i], v2);
}

return encontrado;
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 625


Un programa que lea dos cadenas del teclado y determine si una es
un anagrama de la otra, es decir, si una cadena es una permutación
de los caracteres de la otra.
Por ejemplo, "acre" es un anagrama de "cera" y de "arce". Ten
en cuenta que puede haber letras repetidas ("carro", "llave").
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 626

anagramas.cpp

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

int buscaCaracter(string cad, char c); // Índice o ‐1 si no está

int main() {
string cad1, cad2;
bool sonAnagramas = true;
int numCar, posEnCad2;

cout << "Introduce la primera cadena: ";
getline(cin, cad1);
cout << "Introduce la segunda cadena: ";
getline(cin, cad2);
if (cad1.length() != cad2.length()) { // No son anagramas
sonAnagramas = false;
}
Luis Hernández Yáñez

else {
numCar = 0; // Contador de caracteres de la primera cadena
while (sonAnagramas && (numCar < cad1.length())) {
posEnCad2 = buscaCaracter(cad2, cad1.at(numCar));

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 627


if (posEnCad2 == ‐1) { //No se ha encontrado el caracter
sonAnagramas = false;
}
else {
cad2.erase(posEnCad2, 1);
}
numCar++;
}
}

if (sonAnagramas) {
cout << "Las palabras introducidas son anagramas" << endl;
}
else {
cout << "Las palabras introducidas NO son anagramas" << endl;
}
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 628

int buscaCaracter(string cad, char c) {


int pos = 0, lon = cad.length();
bool encontrado = false;

while ((pos < lon) && !encontrado) {
if (cad.at(pos) == c) {
encontrado = true;
}
else {
pos++;
}
}
if (!encontrado) {
pos = ‐1;
}

return pos;
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 629


Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 630

Arrays de varias dimensiones


Varios tamaños en la declaración: cada uno con sus corchetes
typedef tipo_base nombre[tamaño1][tamaño2]...[tamañoN];
Varias dimensiones, tantas como tamaños se indiquen
typedef double tMatriz[50][100];
tMatriz matriz;
Tabla bidimensional de 50 filas por 100 columnas:
0 1 2 3 ... 98 99
0 ...
1 ...
2 ...
Luis Hernández Yáñez

... ... ... ... ... ... ... ...


48 ...
49 ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 631


Arrays de varias dimensiones
typedef double tMatriz[50][100];
tMatriz matriz;
Cada elemento se localiza con dos índices, uno por dimensión
cout << matriz[2][98];

0 1 2 3 ... 98 99
0 ...
1 ...
2 ...
... ... ... ... ... ... ... ...
Luis Hernández Yáñez

48 ...
49 ...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 632

Arrays de varias dimensiones


Podemos definir tantas dimensiones como necesitemos
typedef double tMatriz[5][10][20][10];
tMatriz matriz;
Necesitaremos tantos índices como dimensiones:
cout << matriz[2][9][15][6];
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 633


Ejemplo de array bidimensional
Temperaturas mínimas y máximas
Matriz bidimensional de días y mínima/máxima:
const int MaxDias = 31;
const int MED = 2; // Nº de medidas
typedef double tTemp[MaxDias][MED]; // Día x mín./máx.
tTemp temp;
Ahora:
 temp[i][0] es la temperatura mínima del día i+1
 temp[i][1] es la temperatura máxima del día i+1
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 634

temp.cpp

int main() {
const int MaxDias = 31;
const int MED = 2; // Nº de medidas
typedef double tTemp[MaxDias][MED]; // Día x mín./máx.
tTemp temp;
double tMaxMedia = 0, tMinMedia = 0,
tMaxAbs = ‐100, tMinAbs = 100;
int dia = 0;
double max, min;
ifstream archivo;

archivo.open("temp.txt");
if (!archivo.is_open()) {
cout << "No se ha podido abrir el archivo!" << endl;
}
Luis Hernández Yáñez

else {
archivo >> min >> max;
// El archivo termina con ‐99 ‐99
...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 635


while (!((min == ‐99) && (max == ‐99)) 
&& (dia < MaxDias)) {
temp[dia][0] = min;
temp[dia][1] = max;
dia++;
archivo >> min >> max;
}
archivo.close();
for (int i = 0; i < dia; i++) {
tMinMedia = tMinMedia + temp[i][0];
if (temp[i][0] < tMinAbs) {
tMinAbs = temp[i][0];
}
tMaxMedia = tMaxMedia + temp[i][1];
if (temp[i][1] > tMaxAbs) {
Luis Hernández Yáñez

tMaxAbs = temp[i][1];
}
}
...

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 636

tMinMedia = tMinMedia / dia;


tMaxMedia = tMaxMedia / dia;
cout << "Temperaturas mínimas.‐" << endl;
cout << "   Media = " << fixed << setprecision(1) 
<< tMinMedia << " C   Mínima absoluta = "
<< setprecision(1) << tMinAbs << " C" << endl;
cout << "Temperaturas máximas.‐" << endl;
cout << "   Media = " << fixed << setprecision(1) 
<< tMaxMedia << " C   Máxima absoluta = "
<< setprecision(1) << tMaxAbs << " C" << endl;
}

return 0;
}
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 637


Podemos dar valores a los elementos de un array al declararlo
Arrays bidimensionales:
typedef int tArray[5][2];
tArray cuads = {1,1, 2,4, 3,9, 4,16, 5,25};

Se asignan en el orden en el que los elementos están en memoria

La memoria es de una dimensión: secuencia de celdas


En memoria varían más rápidamente los índices de la derecha:
cuads[0][0] cuads[0][1] cuads[1][0] cuads[1][1] cuads[2][0]...

Para cada valor del primer índice: todos los valores del segundo
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 638

Inicialización de un array bidimensional


typedef int tArray[5][2];
tArray cuads = {1,1, 2,4, 3,9, 4,16, 5,25};
Memoria 0 1

cuads[0][0] 1 0 1 1
1 2 4
cuads[0][1] 1
2 3 9
cuads[1][0] 2
3 4 16
cuads[1][1] 4
4 5 25
cuads[2][0] 3
cuads[2][1] 9 Si hay menos valores que elementos,
cuads[3][0] 4 el resto se inicializan a cero
cuads[3][1] 16 Inicialización a cero de todo el array:
Luis Hernández Yáñez

cuads[4][0] 5 int cuads[5][2] = { 0 };


cuads[4][1] 25

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 639


typedef double tMatriz[3][4][2][3];
tMatriz matriz = 
{1, 2, 3, 4, 5, 6,  Memoria
matriz[0][0][0][0] 1
7, 8, 9, 10, 11, 12};
matriz[0][0][0][1] 2
matriz[0][0][0][2] 3
matriz[0][0][1][0] 4
matriz[0][0][1][1] 5
matriz[0][0][1][2] 6
matriz[0][1][0][0] 7
matriz[0][1][0][1] 8
matriz[0][1][0][2] 9
matriz[0][1][1][0] 10
Luis Hernández Yáñez

matriz[0][1][1][1] 11
matriz[0][1][1][2] 12
matriz[0][2][0][0] 0
... 0

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 640

const int FILAS = 10;
const int COLUMNAS = 5;
typedef double tMatriz[FILAS][COLUMNAS];
tMatriz matriz;

Para cada fila (de 0 a FILAS – 1):


Para cada columna (de 0 a COLUMNAS – 1):
Procesar el elemento en [fila][columna]

for (int fila = 0; fila < FILAS; fila++) {


for (int columna = 0; columna < COLUMNAS; columna++) {
// Procesar matriz[fila][columna]
}
Luis Hernández Yáñez

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 641


Ventas de todos los meses de un año
const int Meses = 12;
const int MaxDias = 31;
typedef double tVentas[Meses][MaxDias];
tVentas ventas; // Ventas de todo el año
typedef short int tDiasMes[Meses];
tDiasMes diasMes;
inicializa(diasMes); // Nº de días de cada mes
// Pedimos las ventas de cada día del año...

for (int mes = 0; mes < Meses; mes++) {
for (int dia = 0; dia < diasMes[mes]; dia++) {
cout << "Ventas del día " << dia + 1
Luis Hernández Yáñez

<< " del mes " << mes + 1 << ": ";
cin >> ventas[mes][dia];
}
}
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 642

Ventas de todos los meses de un año


Días

0 1 2 3 4 ... 28 29 30
0 201 125 234 112 156 ... 234 543 667
1 323 231 675 325 111 ...
2 523 417 327 333 324 ... 444 367 437
3 145 845 654 212 562 ... 354 548
4 327 652 555 222 777 ... 428 999 666
Celdas no
5 854 438 824 547 175 ... 321 356
Meses utilizadas
6 654 543 353 777 437 ... 765 678 555
7 327 541 164 563 327 ... 538 159 235
8 333 327 432 249 777 ... 528 529
Luis Hernández Yáñez

9 524 583 333 100 334 ... 743 468 531


10 217 427 585 218 843 ... 777 555
11 222 666 512 400 259 ... 438 637 879

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 643


const int DIM1 = 10;
const int DIM2 = 5;
const int DIM3 = 25;
const int DIM4 = 50;

typedef double tMatriz[DIM1][DIM2][DIM3][DIM4];

tMatriz matriz;
Bucles anidados, desde la primera dimensión hasta la última:
for (int n1 = 0; n1 < DIM1; n1++) {
for (int n2 = 0; n2 < DIM2; n2++) {
for (int n3 = 0; n3 < DIM3; n3++) {
for (int n4 = 0; n4 < DIM4; n4++) {
// Procesar matriz[n1][n2][n3][n4]
}
Luis Hernández Yáñez

}
}
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 644

Ventas diarias de cuatro sucursales


Cada mes del año: ingresos de cada sucursal cada día del mes
Meses con distinto nº de días  junto con la matriz de ventas
mensual guardamos el nº de días del mes concreto  estructura
const int DIAS = 31;
const int SUCURSALES = 4;
typedef double tVentaMes[DIAS][SUCURSALES];
typedef struct {
tVentaMes ventas; anual  tVentaAnual
anual[i]  tMes
int dias;
anual[i].dias  int
} tMes;
anual[i].ventas  tVentaMes
Luis Hernández Yáñez

anual[i].ventas[j][k]  double
const int MESES = 12;
typedef tMes tVentaAnual[MESES];
tVentaAnual anual;

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 645


const int DIAS = 31;
const int SUCURSALES = 4;
Cálculo de las ventas typedef double
tVentaMes[DIAS][SUCURSALES];
de todo el año: typedef struct {
Para cada mes... tVentaMes ventas;
int dias;
Para cada día del mes... } tMes;

Para cada sucursal... const int MESES = 12;


typedef tMes tVentaAnual[MESES];
Acumular las ventas tVentaAnual anual;

double total = 0;
for (int mes = 0; mes < MESES; mes++) {
for (int dia = 0; dia < anual[mes].dias; dia++) {
for (int suc = 0; suc < SUCURSALES; suc++) {
total = total + anual[mes].ventas[dia][suc];
Luis Hernández Yáñez

}
}
}

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 646

bool encontrado = false; Primer valor > umbral


int mes = 0, dia, suc;
while ((mes < MESES) && !encontrado) {
dia = 0;
while ((dia < anual[mes].dias) && !encontrado) {
suc = 0;
while ((suc < SUCURSALES) && !encontrado) {
if (anual[mes].ventas[dia][suc] > umbral) {
encontrado = true;
}
else {
suc++;
}
}
if (!encontrado) {
dia++;
}
}
Luis Hernández Yáñez

if (!encontrado) {
mes++;
}
}
if (encontrado) { ...
Fundamentos de la programación: Recorrido y búsqueda en arrays Página 647
Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Recorrido y búsqueda en arrays Página 648


Fundamentos de la programación

7
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Algoritmos de ordenación 651
Algoritmo de ordenación por inserción 654
Ordenación de arrays por inserción 665
Algoritmo de ordenación por inserción
con intercambios 672
Claves de ordenación 680
Estabilidad de la ordenación 688
Complejidad y eficiencia 692
Ordenaciones naturales 694
Ordenación por selección directa 701
Método de la burbuja 716
Listas ordenadas 722
Búsquedas en listas ordenadas 729
Búsqueda binaria 731
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 651

Ordenación de listas
array
125.40 76.95 328.80 254.62 435.00 164.29 316.05 219.99 93.45 756.62
0 1 2 3 4 5 6 7 8 9

Algoritmo de ordenación
(de menor a mayor)

array
76.95  93.45  125.40  164.29  219.99  254.62  316.05  328.80  435.00  756.62 
0 1 2 3 4 5 6 7 8 9

array[i] <= array[i + 1]
Luis Hernández Yáñez

Mostrar los datos en orden, facilitar las búsquedas, ...


Variadas formas de hacerlo (algoritmos)

Fundamentos de la programación: Algoritmos de ordenación Página 652


Ordenación de listas
Los datos de la lista deben poderse comparar entre sí
Sentido de la ordenación:
 Ascendente (de menor a mayor)
 Descendente (de mayor a menor)
Algoritmos de ordenación básicos:
 Ordenación por inserción
 Ordenación por selección directa
 Ordenación por el método de la burbuja
Los algoritmos se basan en comparaciones e intercambios
Luis Hernández Yáñez

Hay otros algoritmos de ordenación mejores

Fundamentos de la programación: Algoritmos de ordenación Página 653


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 654


Algoritmo de ordenación por inserción
Partimos de una lista vacía
Vamos insertando cada elemento en el lugar que le corresponda

6
1 Baraja de nueve cartas numeradas del 1 al 9
3
8
2
9
Las cartas están desordenadas

4
7
Luis Hernández Yáñez

Ordenaremos de menor a mayor (ascendente)


5
Fundamentos de la programación: Algoritmos de ordenación Página 655

Algoritmo de ordenación por inserción

6
1 Colocamos el primer elemento en la lista vacía
3
8
2
9
4
7
5 Lista ordenada:

5
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 656


Algoritmo de ordenación por inserción

6
1 El 7 es mayor que todos los elementos de la lista
3
8 Lo insertamos al final
2
9
4
7
Lista ordenada:

5 7
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 657

Algoritmo de ordenación por inserción

6
1 Primer elemento (5) mayor que el nuevo (4):
3
8 Desplazamos todos una posición a la derecha
2
9
Insertamos el nuevo en la primera posición

4 Hemos insertado el elemento en su lugar

Lista ordenada:

5 5
4 7 7
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 658


Algoritmo de ordenación por inserción

6
1 9 es mayor que todos los elementos de la lista
3
8 Lo insertamos al final
2
9
Lista ordenada:

4 5 7 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 659

Algoritmo de ordenación por inserción

6
1 Primer elemento (4) mayor que el nuevo (2):
3
8 Desplazamos todos una posición a la derecha
2 Insertamos el nuevo en la primera posición

Lista ordenada:

4 4
2 5 5
7 7
9 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 660


Algoritmo de ordenación por inserción

6
1 El 9 es el primer elemento mayor que el nuevo (8):
3
8 Desplazamos desde ese hacia la derecha
Insertamos donde estaba el 9

Lista ordenada:

2 4 5 7 8
4 9 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 661

Algoritmo de ordenación por inserción

6
1 Segundo elemento (4) mayor que el nuevo (3):
3 Desplazamos desde ese hacia la derecha
Insertamos donde estaba el 4

Lista ordenada:

2 3
4 4 4
5 5
7 7
8 8
9 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 662


Algoritmo de ordenación por inserción

6
1 Primer elemento (2) mayor que el nuevo (1):
Desplazamos todos una posición a la derecha
Insertamos el nuevo en la primera posición

Lista ordenada:

2 2
1 3 3
4 4
5 5
7 7
8 8
9 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 663

Algoritmo de ordenación por inserción

6 El 7 es el primer elemento mayor que el nuevo (6):


Desplazamos desde ese hacia la derecha
Insertamos donde estaba el 7

¡¡¡ LISTA ORDENADA !!!


Lista ordenada:

1 2 3 4 5 6
7 7
8 8
9 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 664


Ordenación de arrays por inserción
El array contiene inicialmente la lista desordenada:
20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

A medida que insertamos: dos zonas en el array


Parte ya ordenada y elementos por procesar

7 14 20 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Parte ya ordenada Elementos por insertar


Luis Hernández Yáñez

Siguiente elemento a insertar en la parte ya ordenada

Fundamentos de la programación: Algoritmos de ordenación Página 665

Ordenación de arrays por inserción


Situación inicial: Lista ordenada con un solo elemento (primero)
20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Desde el segundo elemento del array hasta el último:


Localizar el primer elemento mayor en lo ya ordenado

20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

nuevo 7
Primer elemento mayor o igual: índice 0

Fundamentos de la programación: Algoritmos de ordenación Página 666


Ordenación de arrays por inserción
. . .
Desplazar a la derecha los ordenados desde ese lugar
Insertar el nuevo en la posición que queda libre

20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

nuevo 7

7 20 14 32 5 14 27 12 13 15
Luis Hernández Yáñez

0 1 2 3 4 5 6 7 8 9

nuevo 7

Fundamentos de la programación: Algoritmos de ordenación Página 667

Implementación const int N = 15;
typedef int tLista[N];
... tLista lista;
int nuevo, pos;
// Desde el segundo elemento hasta el último...
for (int i = 1; i < N; i++) {
nuevo = lista[i];
pos = 0;
while ((pos < i) && !(lista[pos] > nuevo)) {
pos++;
}
// pos: índice del primer mayor; i si no lo hay
for (int j = i; j > pos; j‐‐) {
lista[j] = lista[j ‐ 1];
Luis Hernández Yáñez

}
lista[pos] = nuevo;
}

Fundamentos de la programación: Algoritmos de ordenación Página 668


20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

i 1 pos 0 nuevo 7

20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

20 20 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

7 20 14 32 5 14 27 12 13 15
Luis Hernández Yáñez

0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 669

7 14 20 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

i 4 pos 0 nuevo 5

7 14 20 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

7 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

5
Luis Hernández Yáñez

7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 670


5 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

i 5 pos 3 nuevo 14

5 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

5 7 14 20 20 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9

14 20
Luis Hernández Yáñez

5 7 14 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 671


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 672


La inserción de cada elemento se puede realizar
con comparaciones e intercambios

Desde el segundo elemento hasta el último:


Desde la posición del nuevo elemento a insertar:
Mientras el anterior sea mayor, intercambiar

5 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

5 7 14 20 14 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

5 7 14 14 20 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 673

7 14 20 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

7 14 20 5 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

7 14 5 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

7 5 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

5 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 674


const int N = 15;
typedef int tLista[N];
... tLista lista;
int tmp, pos;
// Desde el segundo elemento hasta el último...
for (int i = 1; i < N; i++) {
pos = i;
// Mientras no al principio y anterior mayor...
while ((pos > 0) && (lista[pos ‐ 1] > lista[pos])) {
// Intercambiar...
tmp = lista[pos];
lista[pos] = lista[pos ‐ 1];
lista[pos ‐ 1] = tmp;
pos‐‐; // Posición anterior
Luis Hernández Yáñez

}
}

Fundamentos de la programación: Algoritmos de ordenación Página 675

#include <iostream> insercion.cpp
using namespace std;
#include <fstream>

const int N = 100;


typedef int tArray[N];
typedef struct { // Lista de longitud variable
tArray elementos;
int contador;
} tLista;

int main() {
tLista lista;
ifstream archivo;
Luis Hernández Yáñez

int dato, pos, tmp;
lista.contador = 0;
...

Fundamentos de la programación: Algoritmos de ordenación Página 676


archivo.open("insercion.txt");
if (!archivo.is_open()) {
cout << "Error de apertura de archivo!" << endl;
}
else {
archivo >> dato;
while ((lista.contador < N) && (dato != ‐1)) {
// Centinela ‐1 al final
lista.elementos[lista.contador] = dato;
lista.contador++;
archivo >> dato;
}
archivo.close();
// Si hay más de N ignoramos el resto
cout << "Antes de ordenar:" << endl;
Luis Hernández Yáñez

for (int i = 0; i < lista.contador; i++) {


cout << lista.elementos[i] << "  ";
}
cout << endl;      ...
Fundamentos de la programación: Algoritmos de ordenación Página 677

for (int i = 1; i < lista.contador; i++) {


pos = i;
while ((pos > 0)
&& (lista.elementos[pos‐1] > lista.elementos[pos])) 
{
tmp = lista.elementos[pos];
lista.elementos[pos] = lista.elementos[pos ‐ 1];
lista.elementos[pos ‐ 1] = tmp;
pos‐‐;
}
}
cout << "Después de ordenar:" << endl;
for (int i = 0; i < lista.contador; i++) {
cout << lista.elementos[i] << "  ";
}
Luis Hernández Yáñez

cout << endl;
}
return 0;
}
Fundamentos de la programación: Algoritmos de ordenación Página 678
Consideración de implementación
¿Operador relacional adecuado?
lista[pos ‐ 1]   ¿ > o >= ? lista[pos]
Con >= se realizan intercambios inútiles:

5 7 14 20 32 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

5 7 14 20 14 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

5 7 14 14 20 32 27 12 13 15
0 1 2 3 4 5 6 7 8 9

¡Intercambio inútil!

Fundamentos de la programación: Algoritmos de ordenación Página 679


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 680


Claves de ordenación
Elementos que son estructuras con varios campos:
const int N = 15;
typedef struct {
int codigo;
string nombre;
double sueldo;
} tDato;
typedef tDato tLista[N];
tLista lista;

Clave de ordenación:
Luis Hernández Yáñez

Campo en el que se basan las comparaciones

Fundamentos de la programación: Algoritmos de ordenación Página 681

Claves de ordenación
tDato tmp;
while ((pos > 0)
&& (lista[pos ‐ 1].nombre > lista[pos].nombre)) {
tmp = lista[pos];
lista[pos] = lista[pos ‐ 1];
lista[pos ‐ 1] = tmp;
pos‐‐;
}

Comparación: campo concreto


Intercambio: elementos completos
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 682


Claves de ordenación
Función para la comparación:
bool operator>(tDato opIzq, tDato opDer) {
return (opIzq.nombre > opDer.nombre);
}

tDato tmp;
while ((pos > 0) && (lista[pos ‐ 1] > lista[pos])) {
tmp = lista[pos];
lista[pos] = lista[pos ‐ 1];
lista[pos ‐ 1] = tmp;
pos‐‐;
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 683

claves.cpp

Claves de ordenación
#include <iostream>
#include <string>
using namespace std;
#include <fstream>
#include <iomanip>
const int N = 15;
typedef struct {
int codigo;
string nombre;
double sueldo;
} tDato;
typedef tDato tArray[N];
typedef struct {
tArray datos;
Luis Hernández Yáñez

int cont;
} tLista;
...

Fundamentos de la programación: Algoritmos de ordenación Página 684


void mostrar(tLista lista);
bool operator>(tDato opIzq, tDato opDer);
int main() {
tLista lista;
ifstream archivo;
lista.cont = 0;
archivo.open("datos.txt");
if (!archivo.is_open()) {
cout << "Error de apertura del archivo!" << endl;
}
else {
tDato dato;
archivo >> dato.codigo;
while ((lista.cont < N) && (dato.codigo != ‐1)) {
archivo >> dato.nombre >> dato.sueldo;
lista.datos[lista.cont] = dato;
Luis Hernández Yáñez

lista.cont++;
archivo >> dato.codigo;
}
archivo.close();        ...
Fundamentos de la programación: Algoritmos de ordenación Página 685

cout << "Antes de ordenar:" << endl;
mostrar(lista);
for (int i = 1; i < lista.cont; i++) {
// Desde el segundo elemento hasta el último
int pos = i;
while ((pos > 0)
&& (lista.datos[pos‐1] > lista.datos[pos])) {
tDato tmp;
tmp = lista.datos[pos];
lista.datos[pos] = lista.datos[pos ‐ 1];
lista.datos[pos ‐ 1] = tmp;
pos‐‐;
}
}
cout << "Después de ordenar:" << endl;
Luis Hernández Yáñez

mostrar(lista);
}
return 0;
} ...
Fundamentos de la programación: Algoritmos de ordenación Página 686
void mostrar(tLista lista) {
for (int i = 0; i < lista.cont; i++) {
cout << setw(10) 
<< lista.datos[i].codigo
<< setw(20) 
<< lista.datos[i].nombre 
<< setw(12) 
<< fixed
<< setprecision(2)
<< lista.datos[i].sueldo 
<< endl;
}
}

bool operator>(tDato opIzq, tDato opDer) {


Luis Hernández Yáñez

return (opIzq.nombre > opDer.nombre);


}

Cambia a codigo o sueldo para ordenar por otros campos

Fundamentos de la programación: Algoritmos de ordenación Página 687


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 688


Algoritmos de ordenación estables
Al ordenar por otra clave una lista ya ordenada,
la segunda ordenación preserva el orden de la primera
tDato: tres posibles claves de ordenación (campos)
Codigo
12345   Álvarez       120000
Nombre 11111   Benítez       100000
Sueldo 21112   Domínguez      90000
11111   Durán         120000
22222   Fernández     120000
12345   Gómez         100000
Lista ordenada por Nombre  10000   Hernández     150000
21112   Jiménez       100000
11111   Pérez          90000
12345   Sánchez        90000
Luis Hernández Yáñez

10000   Sergei        100000
33333   Tarazona      120000
12345   Turégano      100000
11111   Urpiano        90000

Fundamentos de la programación: Algoritmos de ordenación Página 689

Ordenamos ahora por el campo Codigo:

10000   Sergei        100000 10000   Hernández     150000
10000   Hernández     150000 10000   Sergei        100000
11111   Urpiano 90000 11111   Benítez 100000
11111   Benítez 100000 11111   Durán 120000
11111   Pérez 90000 11111   Pérez 90000
11111   Durán 120000 11111   Urpiano 90000
12345   Sánchez        90000 12345   Álvarez       120000
12345   Álvarez       120000 12345   Gómez         100000
12345   Turégano      100000 12345   Sánchez        90000
12345   Gómez         100000 12345   Turégano      100000
21112   Domínguez      90000 21112   Domínguez      90000
21112   Jiménez       100000 21112   Jiménez       100000
22222   Fernández     120000 22222   Fernández     120000
33333   Tarazona      120000 33333   Tarazona      120000

No estable: Estable:
Luis Hernández Yáñez

Los nombres no mantienen Los nombres mantienen


sus posiciones relativas sus posiciones relativas

Fundamentos de la programación: Algoritmos de ordenación Página 690


Ordenación por inserción
Estable siempre que utilicemos < o > Con <= o >= no es estable
Ordenamos por sueldo:
A igual sueldo, ordenado por códigos y a igual código, por nombres
10000   Hernández     150000 11111   Pérez          90000
10000   Sergei        100000 11111   Urpiano        90000
11111   Benítez       100000 12345   Sánchez        90000
11111   Durán         120000 21112   Domínguez      90000
11111   Pérez          90000 10000 Sergei        100000
11111   Urpiano        90000 11111 Benítez       100000
12345   Álvarez       120000 12345 Gómez 100000
12345   Gómez         100000 12345 Turégano 100000
12345   Sánchez        90000 21112 Jiménez       100000
12345   Turégano      100000 11111   Durán         120000
Luis Hernández Yáñez

21112   Domínguez      90000 12345   Álvarez       120000
21112   Jiménez       100000 22222   Fernández     120000
22222   Fernández     120000 33333   Tarazona      120000
33333   Tarazona      120000 10000   Hernández     150000

Fundamentos de la programación: Algoritmos de ordenación Página 691


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 692


Casos de estudio para los algoritmos de ordenación
 Lista inicialmente ordenada
5 7 12 13 14 14 15 20 27 32
0 1 2 3 4 5 6 7 8 9

 Lista inicialmente ordenada al revés


32 27 20 15 14 14 13 12 7 5
0 1 2 3 4 5 6 7 8 9

 Lista con disposición inicial aleatoria


13 20 7 14 12 32 27 14 5 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

¿Trabaja menos, más o igual la ordenación en cada caso?

Fundamentos de la programación: Algoritmos de ordenación Página 693

Ordenaciones naturales
Si el algoritmo trabaja menos cuanto más ordenada está
inicialmente la lista, se dice que la ordenación es natural
Ordenación por inserción con la lista inicialmente ordenada:
 Versión que busca el lugar primero y luego desplaza:
No hay desplazamientos; mismo número de comparaciones
Comportamiento no natural
 Versión con intercambios:
Trabaja mucho menos; basta una comparación cada vez
Comportamiento natural
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 694


Elección de un algoritmo de ordenación
¿Cómo de bueno es cada algoritmo?
¿Cuánto tarda en comparación con otros algoritmos?
Algoritmos más eficientes: los de menor complejidad
Tardan menos en realizar la misma tarea
Comparamos en orden de complejidad: O()
En función de la dimensión de la lista a ordenar: N
O() = f (N)
Operaciones que realiza el algoritmo de ordenación:
 Comparaciones
Luis Hernández Yáñez

 Intercambios
Asumimos que tardan un tiempo similar

Fundamentos de la programación: Algoritmos de ordenación Página 695

Cálculo de la complejidad
Ordenación por inserción (con intercambios):
...
for (int i = 1; i < N; i++) {
int pos = i;
while ((pos > 0) && (lista[pos ‐ 1] > lista[pos])) {
int tmp; Comparación
tmp = lista[pos];
lista[pos] = lista[pos ‐ 1];
lista[pos ‐ 1] = tmp;
pos‐‐; Intercambio
}
Luis Hernández Yáñez

}
Intercambios y comparaciones:
Tantos como ciclos realicen los correspondientes bucles

Fundamentos de la programación: Algoritmos de ordenación Página 696


Cálculo de la complejidad

... N ‐ 1 ciclos
for (int i = 1; i < N; i++) {
int pos = i; Nº variable de ciclos
while ((pos > 0) && (lista[pos ‐ 1] > lista[pos])) {
int tmp;
tmp = lista[pos];
lista[pos] = lista[pos ‐ 1];
lista[pos ‐ 1] = tmp;
pos‐‐;
}
Luis Hernández Yáñez

}
Caso en el que el while se ejecuta más: caso peor
Caso en el que se ejecuta menos: caso mejor

Fundamentos de la programación: Algoritmos de ordenación Página 697

Cálculo de la complejidad
 Caso mejor: lista inicialmente ordenada
La primera comparación falla: ningún intercambio
(N ‐ 1) * (1 comparación + 0 intercambios) = N ‐ 1  O(N)
 Caso peor: lista inicialmente ordenada al revés
Para cada pos, entre i y 1: 1 comparación y 1 intercambio
1 + 2 + 3 + 4 + ... + (N ‐ 1)
((N ‐ 1) + 1) x (N ‐ 1) / 2
N * (N ‐ 1) / 2
(N2 ‐ N) / 2  O(N2)
Notación O grande: orden de complejidad en base a N
Luis Hernández Yáñez

El término en N que más rápidamente crece al crecer N


En el caso peor, N2 crece más rápido que N  O(N2)
(Ignoramos las constantes, como 2)

Fundamentos de la programación: Algoritmos de ordenación Página 698


Ordenación por inserción (con intercambios)
 Caso mejor: O(N)
 Caso peor: O(N2)
Caso medio (distribución aleatoria de los elementos): O(N2)

Hay algoritmos de ordenación mejores


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 699

Órdenes de complejidad
O(log N) < O(N) < O(N log N) < O(N2) < O(N3) ...

N       log2 N          N2
‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐
1         0             1
2         1             4
4         2            16
8         3            64
16         4           256
32         5          1024
64         6          4096
Luis Hernández Yáñez

128         7         16384
256         8         65536
...

Fundamentos de la programación: Algoritmos de ordenación Página 700


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 701

Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 5 7 4 9 2 8 3 1 6

Lista ordenada:
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 702


Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 5 7 4 9 2 8 3 6

Lista ordenada: 1
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 703

Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 5 7 4 9 8 3 6

Lista ordenada: 1 2
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 704


Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 5 7 4 9 8 6

Lista ordenada: 1 2 3
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 705

Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 5 7 9 8 6

Lista ordenada: 1 2 3 4
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 706


Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 7 9 8 6

Lista ordenada: 1 2 3 4 5
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 707

Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 7 9 8

Lista ordenada: 1 2 3 4 5 6
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 708


Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 9 8

Lista ordenada: 1 2 3 4 5 6 7
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 709

Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada: 9

Lista ordenada: 1 2 3 4 5 6 7 8
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 710


Algoritmo de ordenación por selección directa

Seleccionar el siguiente elemento menor de los que queden

Lista desordenada:

¡¡¡ LISTA ORDENADA !!!

Lista ordenada: 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 711

Ordenación de un array por selección directa


Desde el primer elemento (i = 0) hasta el penúltimo (N‐2):
Menor elemento (en m) entre i + 1 y el último (N‐1)
Intercambiar los elementos en i y m si no son el mismo
i m

20 7 14 32 5 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

im
Sólo intercambiamos si no es la misma posición
Luis Hernández Yáñez

5 7 14 32 20 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 712


Ordenación de un array por selección directa
i m

5 7 14 32 20 14 27 12 13 15
0 1 2 3 4 5 6 7 8 9

i m

5 7 12 32 20 14 27 14 13 15
0 1 2 3 4 5 6 7 8 9

i m
Luis Hernández Yáñez

5 7 12 13 20 14 27 14 32 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 713

seleccion.cpp

Implementación const int N = 15;
typedef int tLista[N];
tLista lista;
// Desde el primer elemento hasta el penúltimo...
for (int i = 0; i < N ‐ 1; i++) {
int menor = i;
// Desde i + 1 hasta el final...
for (int j = i + 1; j < N; j++) {
if (lista[j] < lista[menor]) {
menor = j;
}
}
if (menor > i) {
int tmp;
tmp = lista[i];
Luis Hernández Yáñez

lista[i] = lista[menor];
lista[menor] = tmp;
}
}
Fundamentos de la programación: Algoritmos de ordenación Página 714
Complejidad de la ordenación por selección directa
¿Cuántas comparaciones se realizan?
Bucle externo: N ‐ 1 ciclos
Tantas comparaciones como elementos queden en la lista:
(N ‐ 1) + (N ‐ 2) + (N ‐ 3) + ... + 3 + 2 + 1 =
N x (N ‐ 1) / 2 = (N2 ‐ N) / 2  O(N2)
Mismo número de comparaciones en todos los casos
Complejidad: O(N2) Igual que el método de inserción
Algo mejor (menos intercambios; uno en cada paso)
No es estable: intercambios “a larga distancia”
Luis Hernández Yáñez

No se garantiza que se mantenga el mismo orden relativo original


Comportamiento no natural (trabaja siempre lo mismo)

Fundamentos de la programación: Algoritmos de ordenación Página 715


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 716


Algoritmo de ordenación por el método de la burbuja
Variación del método de selección directa
El elemento menor va ascendiendo hasta alcanzar su posición

9 9 9 9 1
4 4 4 1 9
3 3 1 4 4
6 1 3 3 3
Luis Hernández Yáñez

1 6 6 6 6

Fundamentos de la programación: Algoritmos de ordenación Página 717

12 32 14 5 14 7
0 1 2 3 4 5

12 32 14 5 7 14
0 1 2 3 4 5

12 32 14 5 7 14
0 1 2 3 4 5

12 32 5 14 7 14
0 1 2 3 4 5

12 5 32 14 7 14
Luis Hernández Yáñez

0 1 2 3 4 5

5 12 32 14 7 14
0 1 2 3 4 5

Fundamentos de la programación: Algoritmos de ordenación Página 718


burbuja.cpp

Ordenación de un array por el método de la burbuja


Desde el primero (i = 0), hasta el penúltimo (N ‐ 2):
Desde el último (j = N – 1), hasta i + 1:
Si elemento en j < elemento en j ‐ 1, intercambiarlos

... const int N = 10;
int tmp; typedef int tLista[N];
// Del primero al penúltimo... tLista lista;
for (int i = 0; i < N ‐ 1; i++) {
// Desde el último hasta el siguiente a i...
for (int j = N ‐ 1; j > i; j‐‐) {
if (lista[j] < lista[j ‐ 1]) {
tmp = lista[j];
lista[j] = lista[j ‐ 1];
Luis Hernández Yáñez

lista[j ‐ 1] = tmp;
}
}
}

Fundamentos de la programación: Algoritmos de ordenación Página 719

Algoritmo de ordenación por el método de la burbuja


Complejidad: O(N2)
Comportamiento no natural
Estable (mantiene el orden relativo)
Mejora:
Si en un paso del bucle exterior no ha habido intercambios:
La lista ya está ordenada (no es necesario seguir)

14      14      14      12
16      16      12      14
La lista ya está ordenada
35      12      16      16
Luis Hernández Yáñez

No hace falta seguir


12      35      35      35
50      50      50      50

Fundamentos de la programación: Algoritmos de ordenación Página 720


burbuja2.cpp

bool inter = true;
int i = 0;
// Desde el 1º hasta el penúltimo si hay intercambios...
while ((i < N ‐ 1) && inter) {
inter = false;
// Desde el último hasta el siguiente a i...
for (int j = N ‐ 1; j > i; j‐‐) {
if (lista[j] < lista[j ‐ 1]) {
int tmp;
tmp = lista[j];
lista[j] = lista[j ‐ 1];
lista[j ‐ 1] = tmp;
inter = true;
}
}
if (inter) {
Luis Hernández Yáñez

i++;
}
} Esta variación sí tiene un comportamiento natural

Fundamentos de la programación: Algoritmos de ordenación Página 721


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 722


Gestión de listas ordenadas
Casi todas las tareas se realizan igual que en listas sin orden
Operaciones que tengan en cuenta el orden:
 Inserción de un nuevo elemento: debe seguir en orden
 Búsquedas más eficientes
¿Y la carga desde archivo?
 Si los elementos se guardaron en orden: se lee igual
 Si los elementos no están ordenados en el archivo: insertar
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 723

lista.cpp

Declaraciones: Iguales que para listas sin orden


const int N = 20;

typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;

typedef tRegistro tArray[N];

typedef struct {
tArray registros;
int cont;
} tLista;
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 724


Subprogramas: Misma declaración que para listas sin orden
void mostrarDato(int pos, tRegistro registro);
void mostrar(tLista lista);
bool operator>(tRegistro opIzq, tRegistro opDer);
bool operator<(tRegistro opIzq, tRegistro opDer);
tRegistro nuevo();
void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int pos, bool &ok); // pos = 1..N
int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 725

Nuevas implementaciones:
 Operadores relacionales
 Inserción (mantener el orden)
 Búsqueda (más eficiente)
Se guarda la lista en orden, por lo que cargar() no cambia

bool operator>(tRegistro opIzq, tRegistro opDer) {


return opIzq.nombre > opDer.nombre;
}
bool operator<(tRegistro opIzq, tRegistro opDer) {
return opIzq.nombre < opDer.nombre;
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 726


void insertar(tLista &lista, tRegistro registro, bool &ok) {
ok = true;
if (lista.cont == N) {
ok = false; // lista llena
}
else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
}
// Insertamos en la posición i (primer mayor o igual)
for (int j = lista.cont; j > i; j‐‐) {
// Desplazamos una posición a la derecha
lista.registros[j] = lista.registros[j ‐ 1];
}
Luis Hernández Yáñez

lista.registros[i] = registro;
lista.cont++;
}
}

Fundamentos de la programación: Algoritmos de ordenación Página 727


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 728


Búsqueda de un elemento en una secuencia
No ordenada: recorremos hasta encontrarlo o al final
Ordenada: recorremos hasta encontrarlo o mayor / al final

5 7 12 13 14 14 15 20 27 32
0 1 2 3 4 5 6 7 8 9

Buscamos el 36: al llegar al final sabemos que no está


Buscamos el 17: al llegar al 20 ya sabemos que no está
Condiciones de terminación:
 Se llega al final
 Se encuentra el elemento buscado
Luis Hernández Yáñez

 Se encuentra uno mayor


 Mientras no al final y el valor sea menor que el buscado

Fundamentos de la programación: Algoritmos de ordenación Página 729

int buscado; const int N = 10;


cout << "Valor a buscar: "; typedef int tLista[N];
cin >> buscado; tLista lista;
int i = 0;
while ((i < N) && (lista[i] < buscado)) {
i++;
}
// Ahora, o estamos al final o lista[i] >= buscado
if (i == N) { // Al final: no se ha encontrado
cout << "No encontrado!" << endl;
}
else if (lista[i] == buscado) { // Encontrado!
cout << "Encontrado en posición " << i + 1 << endl;
}
else { // Hemos encontrado uno mayor
Luis Hernández Yáñez

cout << "No encontrado!" << endl;
}
Complejidad: O(N)

Fundamentos de la programación: Algoritmos de ordenación Página 730


Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación Página 731

Búsqueda mucho más rápida que aprovecha la ordenación


Comparar con el valor que esté en el medio de la lista:
Si es el que se busca, terminar
Si no, si es mayor, buscar en la primera mitad de la lista
Si no, si es menor, buscar en la segunda mitad de la lista
Repetir hasta encontrarlo o no quede sublista donde buscar
Buscamos el 12 Elemento mitad
5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

5 7 12 14 14 15 18 20 27 32
Luis Hernández Yáñez

0 1 2 3 4 5 6 7 8 9

5 7 12
 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación Página 732


Vamos buscando en sublistas cada vez más pequeñas (mitades)
Delimitamos el segmento de la lista donde buscar
Inicialmente tenemos toda la lista:
ini mitad fin

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

Índice del elemento en la mitad: mitad = (ini + fin) / 2

Si no se encuentra, ¿dónde seguir buscando?


Buscado < elemento en la mitad: fin = mitad ‐ 1
Luis Hernández Yáñez

Buscado > elemento en la mitad: ini = mitad + 1


Si ini > fin, no queda dónde buscar

Fundamentos de la programación: Algoritmos de ordenación Página 733

Buscamos el 12

ini mitad fin

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

12 < lista[mitad]  fin = mitad – 1
ini mitad fin

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

12 > lista[mitad]  ini = mitad + 1
ini fin
Luis Hernández Yáñez

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

mitad ¡Encontrado!
Fundamentos de la programación: Algoritmos de ordenación Página 734
Si el elemento no está, nos quedamos sin sublista: ini > fin
Para el 13: mitad
ini fin

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9

13 > lista[mitad]  ini = mitad + 1
mitad
ini
fin

5 7 12 14 14 15 18 20 27 32
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

13 < lista[mitad]  fin = mitad – 1  2
¡¡¡ ini > fin !!! No hay dónde seguir buscando  No está

Fundamentos de la programación: Algoritmos de ordenación Página 735

Implementación const int N = 10;
typedef int tLista[N];
int buscado; tLista lista;
cout << "Valor a buscar: ";
cin >> buscado;
int ini = 0, fin = N – 1, mitad;
bool encontrado = false;
while ((ini <= fin) && !encontrado) {
mitad = (ini + fin) / 2; // División entera
if (buscado == lista[mitad]) {
encontrado = true;
}
else if (buscado < lista[mitad]) {
fin = mitad ‐ 1;
}
else {
Luis Hernández Yáñez

ini = mitad + 1;
}
} // Si se ha encontrado, está en [mitad]

Fundamentos de la programación: Algoritmos de ordenación Página 736


#include <iostream> binaria.cpp
using namespace std;
#include <fstream>

const int N = 100;
typedef int tArray[N];
typedef struct {
tArray elementos;
int cont;
} tLista;

int buscar(tLista lista, int buscado);

int main() {
tLista lista;
ifstream archivo;
Luis Hernández Yáñez

int dato;
lista.cont = 0;
archivo.open("ordenados.txt"); // Existe y es correcto
archivo >> dato;
...
Fundamentos de la programación: Algoritmos de ordenación Página 737

while ((lista.cont < N) && (dato != ‐1)) {


lista.elementos[lista.cont] = dato;
lista.cont++;
archivo >> dato;
}
archivo.close();
for (int i = 0; i < lista.cont; i++) {
cout << lista.elementos[i] << "  ";
}
cout << endl;
int buscado, pos;
cout << "Valor a buscar: ";
cin >> buscado;
pos = buscar(lista, buscado);
if (pos != ‐1) {
cout << "Encontrado en la posición " << pos + 1 << endl;
}
else {
Luis Hernández Yáñez

cout << "No encontrado!" << endl;
}  
return 0;
} ...

Fundamentos de la programación: Algoritmos de ordenación Página 738


int buscar(tLista lista, int buscado) {
int pos = ‐1, ini = 0, fin = lista.cont ‐ 1, mitad;
bool encontrado = false;
while ((ini <= fin) && !encontrado) {
mitad = (ini + fin) / 2; // División entera
if (buscado == lista.elementos[mitad]) {
encontrado = true;
}
else if (buscado < lista.elementos[mitad]) {
fin = mitad ‐ 1;
}
else {
ini = mitad + 1;
}
}
if (encontrado) {
Luis Hernández Yáñez

pos = mitad;
}
return pos;
}

Fundamentos de la programación: Algoritmos de ordenación Página 739

Complejidad
¿Qué orden de complejidad tiene la búsqueda binaria?
Caso peor:
No está o se encuentra en una sublista de 1 elemento
Nº de comparaciones = Nº de mitades que podemos hacer
N / 2, N / 4, N / 8, N / 16, ..., 8, 4, 2, 1
 1, 2, 4, 8, ..., N / 16, N / 8, N / 4, N / 2
Si hacemos que N sea igual a 2k:
20, 21, 22, 23, ..., 2k‐4, 2k‐3, 2k‐2, 2k‐1
Nº de elementos de esa serie: k
Luis Hernández Yáñez

Nº de comparaciones = k N = 2k  k = log2 N
Complejidad: O(log2 N) Mucho más rápida que O(N)

Fundamentos de la programación: Algoritmos de ordenación Página 740


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Algoritmos de ordenación Página 741


Fundamentos de la programación

7A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Ordenación por intercambio 744
Mezcla de dos listas ordenadas 747
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación (Anexo)
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 744

Algoritmo de ordenación por intercambio


Variación del método de selección directa
Se intercambia el elemento de la posición que se trata en cada
momento siempre que se encuentra uno que es menor:

14 7 12 32 20 14 27 5 13 15
0 1 2 3 4 5 6 7 8 9

7 14 12 32 20 14 27 5 13 15
0 1 2 3 4 5 6 7 8 9
Luis Hernández Yáñez

5 14 12 32 20 14 27 7 13 15
0 1 2 3 4 5 6 7 8 9

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 745


intercambio.cpp

const int N = 10;
typedef int tLista[N];
tLista lista;
...
for (int i = 0; i < N ‐ 1; i++) {
// Desde el primer elemento hasta el penúltimo
for (int j = i + 1; j < N; j++) {
// Desde i+1 hasta el final
if (lista[j] < lista[i]) {
int tmp;
tmp = lista[i];
lista[i] = lista[j];
lista[j] = tmp;
}
}
Luis Hernández Yáñez

}
Igual número de comparaciones, muchos más intercambios
No es estable
Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 746
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 747


Mezcla de dos listas ordenadas en arrays
const int N = 100;
typedef struct {
int elementos[N];
int cont;
} tLista;

Un índice para cada lista, inicializados a 0 (principio de las listas)

Mientras que no lleguemos al final de alguna de las dos listas:


Elegimos el elemento menor de los que tienen los índices
Lo copiamos en la lista resultado y avanzamos su índice una posición
Luis Hernández Yáñez

Copiamos en la lista resultado los que queden en la lista no acabada

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 748

void mezcla(tLista lista1, tLista lista2, tLista &listaM) {


int pos1 = 0, pos2 = 0;
listaM.cont = 0;

while ((pos1 < lista1.cont) && (pos2 < lista2.cont)
&& (listaM.cont < N)) {
if (lista1.elementos[pos1] < lista2.elementos[pos2]) {
listaM.elementos[listaM.cont] = lista1.elementos[pos1];
pos1++;
}
else {
listaM.elementos[listaM.cont] = lista2.elementos[pos2];
pos2++;
}
listaM.cont++;
Luis Hernández Yáñez

}
...

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 749


mezcla1.cpp

// Pueden quedar datos en alguna de las listas
if (pos1 < lista1.cont) {
while ((pos1 < lista1.cont) && (listaM.cont < N)) {
listaM.elementos[listaM.cont] = lista1.elementos[pos1];
pos1++;
listaM.cont++;
}
}
else { // pos2 < lista2.cont
while ((pos2 < lista2.cont) && (listaM.cont < N)) {
listaM.elementos[listaM.cont] = lista2.elementos[pos2];
pos2++;
listaM.cont++;
}
}
}
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 750

Mezcla de dos listas ordenadas en archivos


void mezcla(string nombre1, string nombre2, string nombreM) {
// Mezcla las secuencias en los archivos nombnre1 y nombre2
// generando la secuencia mezclada en el archivo nombreM
ifstream archivo1, archivo2;
ofstream mezcla;
int dato1, dato2;

// Los archivos existen y son correctos
archivo1.open(nombre1.c_str());
archivo2.open(nombre2.c_str());
mezcla.open(nombreM.c_str());
archivo1 >> dato1;
archivo2 >> dato2;
Luis Hernández Yáñez

while ((dato1 != ‐1) && (dato2 != ‐1)) {
// Mientras quede algo en ambos archivos
...

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 751


if (dato1 < dato2) {
mezcla << dato1 << endl;
archivo1 >> dato1;
} else {
mezcla << dato2 << endl;
archivo2 >> dato2;
}
} // Uno de los dos archivos se ha acabado
if (dato1 != ‐1) { // Quedan en el primer archivo
while (dato1 != ‐1) {
mezcla << dato1 << endl;
archivo1 >> dato1;
}
}
else { // Quedan en el segundo archivo
while (dato2 != ‐1) {
Luis Hernández Yáñez

mezcla << dato2 << endl;
archivo2 >> dato2;
}
}
...
Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 752

mezcla2.cpp

archivo2.close();
archivo1.close();
mezcla << ‐1 << endl;
mezcla.close();
}
Luis Hernández Yáñez

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 753


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Algoritmos de ordenación (Anexo) Página 754


Fundamentos de la programación

8
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Programas multiarchivo y compilación separada  757
Interfaz frente a implementación 762
Uso de módulos de biblioteca 768
Ejemplo: Gestión de una lista ordenada I 770
Compilación de programas multiarchivo 778
El preprocesador 780
Cada cosa en su módulo 782
Ejemplo: Gestión de una lista ordenada II 784
El problema de las inclusiones múltiples 789
Compilación condicional 794
Protección frente a inclusiones múltiples 795
Ejemplo: Gestión de una lista ordenada III 796
Implementaciones alternativas 804
Espacios de nombres 808
Implementaciones alternativas 817
Luis Hernández Yáñez

Calidad y reutilización del software 827

Fundamentos de la programación: Programación modular
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 757

Programas multiarchivo
Código fuente repartido entre varios archivos (módulos)
Cada módulo con sus declaraciones y sus subprogramas
 Módulo: Unidad funcional (estructura de datos, utilidades, ...)
Lista Principal Cálculos Archivos
const int N = 10; int main() { double mean(tArray lista); bool cargar(tArray &lista, 
typedef double tArray[N]; tArray lista; string nombre);
typedef struct { bool ok; double min(tArray lists);
tArray elem; init(lista); bool guardar(tArray lista, 
int cont; cargar(lista, "bd.txt"); double max(tArray lista); string nombre);
} tArray; sort(lista);
double dato; double desv(tArray lista); bool mezclar(string arch1, 
void init(tArray &lista); cout << "Dato: "; string arch2);
cin >> dato; int minIndex(tArray lista);
void insert(tArray &lista,  insert(lista, dato, ok); int size(string nombre);
double elem, bool &ok); cout << min(lista) << endl;  int maxIndex(tArray lista);
cout << max(lista) << endl; bool exportar(string nombre);
void remove(tArray &lista,  cout << sum(lista) << endl; double sum(tArray lista);
int pos, bool &ok); guardar(lista, "bd.txt");
...
return 0;
}
Luis Hernández Yáñez

Ejecutable
Fundamentos de la programación: Programación modular Página 758
Compilación separada
Cada módulo se compila a código objeto de forma independiente
Lista lista.obj
const int N = 10; 00101110101011001010010010101
typedef double tArray[N]; 00101010010101011111010101000
typedef struct { 10100101010101010010101010101
tArray elem; 01100101010101010101010101001
int cont; 01010101010100000101010101101
} tArray; 01001010101010101000010101011
11001010101010111100110010101
void init(tArray &lista);

void insert(tArray &lista, 
01101010101010010010101001111
00101010101001010100101010010
10100101010100101000010011110
Archivos archivos.obj
double elem, bool &ok); 10010101011001010101001010100
10101010101010010101001010101 11101010110010100100101010010
bool cargar(tArray &lista, 
void remove(tArray &lista,  01000010101011100101010010100 10100101010111110101010001010
string nombre);
int pos, bool &ok); 01110101011101001101010100101 01010101010100101010101010110
... 01011111110101011001101010111 01010101010101010101010010101
bool guardar(tArray lista, 
00001001010100101010101010110 01010101000001010101011010100
string nombre);
10101010101010000101010111100
bool mezclar(string arch1,  10101010101111001100101010110
string arch2); 10101010100100101010011110010

Cálculos calculos.obj int size(string nombre);


10101010010101001010100101010
01010101001010000100111101001
01010110010101010010101001010
bool exportar(string nombre); 10101010100101010010101010100
double mean(tArray lista); 01011001010010010101001010100 00101010111001010100101000111
10101011111010101000101001010 01010111010011010101001010101
double min(tArray lists); 10101010010101010101011001010 11111101010110011010101110000
10101010101010101001010101010 10010101001010101010101101111
double max(tArray lista); 10100000101010101101010010101
01010101000010101011110010101
Luis Hernández Yáñez

double desv(tArray lista); 01010111100110010101011010101


01010010010101001111001010101
int minIndex(tArray lista); 01001010100101010010101001010
10100101000010011110100101010
int maxIndex(tArray lista); 11001010101001010100101010101
01010010101001010101010000101
double sum(tArray lista); 01011100101010010100011101010
11101001101010100101010111111
10101011001101010111000010010
10100101010101010110001111010

Fundamentos de la programación: Programación modular Página 759

Compilación separada
Al compilar el programa principal, se adjuntan los módulos compilados
Principal
Módulos del programa int main() {
tArray lista;
Bibliotecas del sistema
bool ok;
init(lista);
cargar(lista, "bd.txt");
sort(lista);
lista.obj double dato; iostream.obj
cout << "Dato: ";
cin >> dato;
insert(lista, dato, ok);
calculos.obj cout << min(lista) << endl; 
cout << max(lista) << endl;
fstream.obj
cout << sum(lista) << endl;
guardar(lista, "bd.txt");

archivos.obj ... }
return 0; math.obj ...
Luis Hernández Yáñez

Ejecutable

Fundamentos de la programación: Programación modular Página 760


Compilación separada
¡Sólo los archivos fuente modificados necesitan ser recompilados!
Principal

lista.cpp main.cpp

COMPILACIÓN

lista.obj iostream.obj

calculos.obj
 fstream.obj

archivos.obj 
main.obj
math.obj
... ...
ENLACE
Luis Hernández Yáñez

Ejecutable
Fundamentos de la programación: Programación modular Página 761
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 762


Creación de módulos de biblioteca
Código de un programa de un único archivo:
 Definiciones de constantes
 Declaraciones de tipos de datos
 Prototipos de los subprogramas
 Implementación de los subprogramas
 Implementación de la función main()
Constantes, tipos y prototipos indican cómo se usa: Interfaz
 Estructura de datos con los subprogramas que la gestionan
 Conjunto de utilidades (subprogramas) de uso general
Luis Hernández Yáñez

 Etcétera
+ Implementación de los subprogramas (cómo se hace)

Fundamentos de la programación: Programación modular Página 763

Creación de módulos de biblioteca


Interfaz: Definiciones/declaraciones de datos y prototipos
¡Todo lo que el usuario de la unidad funcional necesita saber!
Implementación: Código de los subprogramas que hacen el trabajo
No hay que conocerlo para usarlo: ¡Seguro que es correcto!
Interfaz e implementación en dos archivos separados:
 Cabecera: Definiciones/declaraciones de datos y prototipos
 Implementación: Implementación de los subprogramas.
Archivo de cabecera: extensión .h
Mismo nombre
Archivo de implementación: extensión .cpp
Luis Hernández Yáñez

Repartimos el código entre ambos archivos (lista.h/lista.cpp)

Fundamentos de la programación: Programación modular Página 764


Creación de módulos de biblioteca
Interfaz frente a implementación
lista.h lista.cpp
const int N = 10; #include "lista.h"
Módulo
typedef double tArray[N];
typedef struct {
tArray elem;
void init(tArray &lista) {
lista.cont = 0;
Unidad
int cont;
} tArray;
}

void insert(tArray &lista, 
Biblioteca
void init(tArray &lista); double elem, bool &ok) {
if (lista.cont == N) {
void insert(tArray &lista,  ok false;
double elem, bool &ok); }
else {
void remove(tArray &lista,  ...
int pos, bool &ok);
...

Si otro módulo quiere usar algo de esa biblioteca:


Debe incluir el archivo de cabecera
Luis Hernández Yáñez

main.cpp
#include "lista.h"
Los nombres de archivos de cabecera
... propios (no del sistema) se encierran
entre dobles comillas, no entre ángulos

Fundamentos de la programación: Programación modular Página 765

Creación de módulos de biblioteca lista.h


const int N = 10;

Interfaz
typedef double tArray[N];
typedef struct {
tArray elem;
int cont;
} tArray;

Archivo de cabecera (.h): todo lo que necesita void init(tArray &lista);

void insert(tArray &lista, 

conocer otro módulo (o programa principal) double elem, bool &ok);

void remove(tArray &lista, 

que quiera utilizar sus servicios (subprogramas)


int pos, bool &ok);
...

La directiva #include añade las declaraciones del archivo


de cabecera en el código del módulo (preprocesamiento):
main.cpp main.cpp
Preprocesador
#include "lista.h" const int N = 10;
typedef double tArray[N];
... typedef struct {
tArray elem;
int cont;

Todo lo que se necesita saber para


} tArray;
Luis Hernández Yáñez

void init(tArray &lista);

comprobar si el código de main.cpp void insert(tArray &lista, double elem, 


bool &ok);

hace un uso correcto de la lista void remove(tArray &lista, int pos, 


bool &ok);
(declaraciones y llamadas) ...

Fundamentos de la programación: Programación modular Página 766


Creación de módulos de biblioteca lista.cpp

Implementación #include "lista.h"

void init(tArray &lista) {
lista.cont = 0;

Compilar el módulo significa compilar


}

void insert(tArray &lista, 
double elem, bool &ok) {

su archivo de implementación (.cpp)


if (lista.cont == N) {
ok false;
}
else {
...

También necesita conocer sus propias declaraciones:


lista.cpp
#include "lista.h" lista.obj
...
00101110101011001010010010101
00101010010101011111010101000

Al compilar el módulo se genera el código objeto


10100101010101010010101010101
01100101010101010101010101001
01010101010100000101010101101
01001010101010101000010101011
11001010101010111100110010101

Si no se modifica no hay necesidad de recompilar


01101010101010010010101001111
00101010101001010100101010010
10100101010100101000010011110
10010101011001010101001010100
10101010101010010101001010101

Código que usa el módulo:


Luis Hernández Yáñez

01000010101011100101010010100
01110101011101001101010100101
01011111110101011001101010111
00001001010100101010101010110

 Necesita sólo el archivo de cabecera para compilar


 Se adjunta el código objeto del módulo durante el enlace
Fundamentos de la programación: Programación modular Página 767
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 768


Uso de módulos de biblioteca
Ejemplo: Gestión de una lista ordenada (Tema 7)
Todo lo que tenga que ver con la lista estará en su propio módulo
Ahora el código estará repartido en tres archivos:
 lista.h: archivo de cabecera del módulo de lista
 lista.cpp: implementación del módulo de lista
 bd.cpp: programa principal que usa la lista
Tanto lista.cpp como bd.cpp deben incluir al principio lista.h
Módulo propio: dobles comillas en la directiva #include
#include "lista.h"
Luis Hernández Yáñez

Archivos de cabecera de bibliotecas del sistema: entre ángulos


Y no tienen necesariamente que llevar extensión .h

Fundamentos de la programación: Programación modular Página 769

Archivo de cabecera lista.h

Módulo: Gestión de una lista ordenada I


#include <string>
using namespace std;

const int N = 100;


typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;
typedef tRegistro tArray[N];
typedef struct {
tArray registros;
int cont;
Luis Hernández Yáñez

} tLista;
const string BD = "bd.txt";
...
¡Documenta bien el código!

Fundamentos de la programación: Programación modular Página 770


void mostrar(int pos, tRegistro registro);
void mostrar(const tLista &lista);
bool operator>(tRegistro opIzq, tRegistro opDer);
bool operator<(tRegistro opIzq, tRegistro opDer);
tRegistro nuevo();
void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int pos, bool &ok); // pos = 1..N
int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);

Cada prototipo, con un comentario que explique su utilidad/uso


Luis Hernández Yáñez

(Aquí se omiten por cuestión de espacio)

Fundamentos de la programación: Programación modular Página 771

Implementación lista.cpp

Módulo: Gestión de una lista ordenada I


#include <iostream>
#include <string>
using namespace std;
#include <fstream>
#include <iomanip>
#include "lista.h"

tRegistro nuevo() {
tRegistro registro;
cout << "Introduce el código: ";
cin >> registro.codigo;
cout << "Introduce el nombre: ";
cin >> registro.nombre;
Luis Hernández Yáñez

cout << "Introduce el sueldo: ";
cin >> registro.sueldo;
return registro;
} ...

Fundamentos de la programación: Programación modular Página 772
void insertar(tLista &lista, tRegistro registro, bool &ok) {
ok = true;
if (lista.cont == N) {
ok = false; // Lista llena
}
else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
}
// Insertamos en la posición i
for (int j = lista.cont; j > i; j‐‐) {
// Desplazamos a la derecha
lista.registros[j] = lista.registros[j ‐ 1];
}
Luis Hernández Yáñez

lista.registros[i] = registro;
lista.cont++;
}
} ...

Fundamentos de la programación: Programación modular Página 773

void eliminar(tLista &lista, int pos, bool &ok) { // pos = 1..
ok = true;
if ((pos < 1) || (pos > lista.cont)) {
ok = false; // Posición inexistente
}
else {
pos‐‐; // Pasamos a índice del array
for (int i = pos + 1; i < lista.cont; i++) {
// Desplazamos a la izquierda
lista.registros[i ‐ 1] = lista.registros[i];
}
lista.cont‐‐;
}
}
Luis Hernández Yáñez

...

Fundamentos de la programación: Programación modular Página 774
Programa principal bd.cpp

Módulo: Gestión de una lista ordenada I


#include <iostream>
using namespace std;
#include "lista.h"

int menu();

int main() {
tLista lista;
bool ok;
int op, pos;
cargar(lista, ok);
if (!ok) {
cout << "No se ha podido abrir el archivo!" << endl;
}
Luis Hernández Yáñez

else {
do {
mostrar(lista);
op = menu(); ...

Fundamentos de la programación: Programación modular Página 775

if (op == 1) {
tRegistro registro = nuevo();
insertar(lista, registro, ok);
if (!ok) {
cout << "Error: Lista llena!" << endl;
}
}
else if (op == 2) {
cout << "Posición: ";
cin >> pos;
eliminar(lista, pos, ok);
if (!ok) {
cout << "Error: Posicion inexistente!" << endl;
}
}
else if (op == 3) {
string nombre;
cin.sync();
Luis Hernández Yáñez

cout << "Nombre: ";
cin >> nombre;
int pos = buscar(lista, nombre);
...

Fundamentos de la programación: Programación modular Página 776
if (pos == ‐1) {
cout << "No se ha encontrado!" << endl;
}
else {
cout << "Encontrado en la posición " << pos << endl;
}
}
} while (op != 0);
guardar(lista);
}
return 0;
}

int menu() {
cout << endl;
cout << "1 ‐ Insertar" << endl;
cout << "2 ‐ Eliminar" << endl;
cout << "3 ‐ Buscar" << endl;
Luis Hernández Yáñez

cout << "0 ‐ Salir" << endl;


int op;
do {
...

Fundamentos de la programación: Programación modular Página 777
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 778


G++
Archivos de cabecera e implementación en la misma carpeta
Listamos todos los .cpp en la orden g++:
D:\FP\Tema08>g++ ‐o bd.exe lista.cpp bd.cpp
Recuerda que sólo se compilan los .cpp

Visual C++/Studio
Archivos de cabecera e implementación en grupos distintos:

A los archivos de cabecera


los llama de encabezado
Luis Hernández Yáñez

Con Depurar ‐> Generar solución
se compilan todos los .cpp

Fundamentos de la programación: Programación modular Página 779


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 780


Directivas: #...
Antes de compilar se pone en marcha el preprocesador
Interpreta las directivas y genera un único archivo temporal con
todo el código del módulo o programa
Como en la inclusión (directiva #include):
#include "lista.h"
#include <string> #include <string>
using namespace std; int menu(); using namespace std;

const int N = 100; ... const int N = 100;

typedef struct { typedef struct {
int codigo; int codigo;
string nombre; string nombre;
double sueldo; double sueldo;
} tRegistro; } tRegistro;

typedef tRegistro typedef tRegistro
tArray[N]; tArray[N];
Luis Hernández Yáñez

typedef struct { typedef struct {
tArray registros; tArray registros;
int cont; int cont;
} tLista; } tLista;
... ...

int menu();
...

Fundamentos de la programación: Programación modular Página 781


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 782


Distribuir la funcionalidad del programa en módulos
Encapsulación de un conjunto de subprogramas relacionados:
 Por la estructura de datos sobre la que trabajan
 Subprogramas de utilidad
A menudo las estructuras de datos contienen otras estructuras:
const int N = 100;
typedef struct {
int codigo; Lista de registros:
string nombre;
 Estructura tRegistro
double sueldo;
} tRegistro;  Estructura tLista
typedef tRegistro tArray[N]; (contiene tRegistro)
Luis Hernández Yáñez

typedef struct {
Cada estructura, en su módulo
tArray registros;
int cont;
} tLista;

Fundamentos de la programación: Programación modular Página 783

Cabecera registro.h

Gestión de una lista ordenada II


#include <string>
using namespace std;

typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;

tRegistro nuevo();
bool operator>(tRegistro opIzq, tRegistro opDer);
bool operator<(tRegistro opIzq, tRegistro opDer);
void mostrar(int pos, tRegistro registro);
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 784


Implementación registro.cpp

Gestión de una lista ordenada II


#include <iostream>
#include <string>
using namespace std;
#include <iomanip>
#include "registro.h"

tRegistro nuevo() {
tRegistro registro;
cout << "Introduce el código: ";
cin >> registro.codigo;
cout << "Introduce el nombre: ";
cin >> registro.nombre;
cout << "Introduce el sueldo: ";
cin >> registro.sueldo;
return registro;
}
Luis Hernández Yáñez

bool operator>(tRegistro opIzq, tRegistro opDer) {


return opIzq.nombre > opDer.nombre;
} ...

Fundamentos de la programación: Programación modular Página 785

Cabecera lista2.h

Gestión de una lista ordenada II


#include <string>
using namespace std;
#include "registro.h"

const int N = 100;


typedef tRegistro tArray[N];
typedef struct {
tArray registros;
int cont;
} tLista;
const string BD = "bd.txt";

void insertar(tLista &lista, tRegistro registro, bool &ok);


void eliminar(tLista &lista, int pos, bool &ok); // pos = 1..N
int buscar(tLista lista, string nombre);
Luis Hernández Yáñez

void mostrar(const tLista &lista);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);

Fundamentos de la programación: Programación modular Página 786


Implementación lista2.cpp

Gestión de una lista ordenada II


#include <iostream>
using namespace std;
#include <fstream>
#include "lista2.h"

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) {
ok = false; // Lista llena
}
else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
}
// Insertamos en la posición i
Luis Hernández Yáñez

for (int j = lista.cont; j > i; j‐‐) {// Desplazar a la derecha


lista.registros[j] = lista.registros[j ‐ 1];
}
...

Fundamentos de la programación: Programación modular Página 787

bd2.cpp

Gestión de una lista ordenada II


#include <iostream>
using namespace std;
#include "registro.h"
#include "lista2.h"

int menu();

int main() {
tLista lista;
¡No intentes compilar este ejemplo!
bool ok; Tiene errores
int op, pos;

cargar(lista, ok);
if (!ok) {
cout << "No se pudo abrir el archivo!" << endl;
}
Luis Hernández Yáñez

else {
do {
mostrar(lista);
op = menu();
...
Fundamentos de la programación: Programación modular Página 788
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 789

Gestión de una lista ordenada II


2 módulos y el programa principal: bd2.cpp

...
registro.h #include "registro.h"
#include "lista2.h"
#include <string>
...
...

registro.cpp

...
#include "registro.h"
... lista2.h

...
#include "registro.h"
...

lista2.cpp
Luis Hernández Yáñez

...
Incluye... #include "lista2.h"
...

Fundamentos de la programación: Programación modular Página 790


Gestión de una lista ordenada II
Preprocesamiento de #include: #include <string>
#include <iostream> using namespace std;
using namespace std;
typedef struct {
#include "registro.h" ...
} tRegistro;
#include "lista2.h" ...

int menu(); #include <string>
#include <string> using namespace std;
... using namespace std;
#include "registro.h" typedef struct {
...
const int N = 100; } tRegistro;
typedef tRegistro tArray[N]; ...
typedef struct {
Luis Hernández Yáñez

tArray registros;
int cont;
} tLista;
...

Fundamentos de la programación: Programación modular Página 791

Gestión de una lista ordenada II


Preprocesamiento de #include:
Sustituido
#include <iostream>
using namespace std;
#include <string>
#include <string> using namespace std;
using namespace std; #include <string>
using namespace std;
typedef struct {
... typedef struct {
} tRegistro; ...
... } tRegistro;
...
#include "lista2.h"
const int N = 100;
int menu(); typedef tRegistro tArray[N];
typedef struct {
Luis Hernández Yáñez

... tArray registros;


int cont;
} tLista;
...

Fundamentos de la programación: Programación modular Página 792


Gestión de una lista ordenada II
#include <iostream>
using namespace std; const int N = 100;
typedef tRegistro tArray[N];
#include <string> typedef struct {
using namespace std; tArray registros;
int cont;
typedef struct { } tLista;
... ...
} tRegistro;
... int menu();

#include <string> ...
using namespace std;
#include <string>
using namespace std;

typedef struct {
¡Identificador duplicado!
Luis Hernández Yáñez

...
} tRegistro;
...

Fundamentos de la programación: Programación modular Página 793

Compilación condicional
Directivas #ifdef, #ifndef, #else y #endif
Se usan en conjunción con la directiva #define
#define X #define X
#ifdef X #ifndef X
... // Código if ... // Código if
[#else [#else
... // Código else ... // Código else
] ]
#endif #endif
La directiva #define define un símbolo (identificador)
Izquierda: se compilará el “Código if” y no el “Código else”
Luis Hernández Yáñez

Derecha: al revés, o nada si no hay else


Las cláusulas else son opcionales

Fundamentos de la programación: Programación modular Página 794


Protección frente a inclusiones múltiples
lista2.cpp y bd2.cpp incluyen registro.h
 ¡Identificadores duplicados!
Cada módulo debe incluirse una y sólo una vez
Protección frente a inclusiones múltiples:
#ifndef X
#define X El símbolo X debe ser único
... // Módulo para cada módulo de la aplicación
#endif
La primera vez no está definido el símbolo X: se incluye y define
Las siguientes veces el símbolo X ya está definido: no se incluye
Luis Hernández Yáñez

Símbolo X: nombre del archivo con _ en lugar de .


registro_h, lista2_h, ...

Fundamentos de la programación: Programación modular Página 795

Cabecera registrofin.h

Gestión de una lista ordenada III


#ifndef registrofin_h
#define registrofin_h
#include <string>
using namespace std;

typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;

tRegistro nuevo();
Luis Hernández Yáñez

bool operator>(tRegistro opIzq, tRegistro opDer);


bool operator<(tRegistro opIzq, tRegistro opDer);
void mostrar(int pos, tRegistro registro);
#endif

Fundamentos de la programación: Programación modular Página 796


Implementación registrofin.cpp

Gestión de una lista ordenada III


#include <iostream>
#include <string>
using namespace std;
#include <iomanip>
#include "registrofin.h"

tRegistro nuevo() {
tRegistro registro;
cout << "Introduce el código: ";
cin >> registro.codigo;
cout << "Introduce el nombre: ";
cin >> registro.nombre;
cout << "Introduce el sueldo: ";
cin >> registro.sueldo;
return registro;
}
Luis Hernández Yáñez

bool operator>(tRegistro opIzq, tRegistro opDer) {


return opIzq.nombre > opDer.nombre;
} ...

Fundamentos de la programación: Programación modular Página 797

Cabecera listafin.h

Gestión de una lista ordenada III


#ifndef listafin_h
#define listafin_h
#include <string>
using namespace std;
#include "registrofin.h"

const int N = 100;


typedef tRegistro tArray[N];
typedef struct {
tArray registros;
int cont;
} tLista;
const string BD = "bd.txt";
void mostrar(const tLista &lista);
void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int pos, bool &ok); // pos = 1..N
Luis Hernández Yáñez

int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);
#endif

Fundamentos de la programación: Programación modular Página 798


Implementación listafin.cpp

Gestión de una lista ordenada III


#include <iostream>
using namespace std;
#include <fstream>
#include "listafin.h"

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) {
ok = false; // lista llena
}
else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
}
// Insertamos en la posición i
Luis Hernández Yáñez

for (int j = lista.cont; j > i; j‐‐) {
// Desplazamos a la derecha
lista.registros[j] = lista.registros[j ‐ 1];
}
...
Fundamentos de la programación: Programación modular Página 799

bdfin.cpp

Gestión de una lista ordenada III


#include <iostream>
using namespace std;
#include "registrofin.h"
#include "listafin.h"

int menu();

int main() {
tLista lista;
bool ok; ¡Ahora ya puedes compilarlo!
int op, pos;

cargar(lista, ok);
if (!ok) {
cout << "No se pudo abrir el archivo!" << endl;
}
Luis Hernández Yáñez

else {
do {
mostrar(lista);
op = menu();
...
Fundamentos de la programación: Programación modular Página 800
Gestión de una lista ordenada III
Preprocesamiento de #include en bdfin.cpp:
#include <iostream> #ifndef registrofin_h
using namespace std;
#define registrofin_h
#include <string>
#include "registrofin.h" using namespace std;

#include "listafin.h" typedef struct {


...
int menu(); } tRegistro;
...
...

registrofin_h no se ha definido todavía


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 801

Gestión de una lista ordenada III


Preprocesamiento de #include en bdfin.cpp:
#include <iostream>
using namespace std;
#ifndef listafin_h
#define listafin_h
#define registrofin_h #include <string>
#include <string> using namespace std;
using namespace std; #include "registrofin.h"

typedef struct { const int N = 100;


... typedef tRegistro tArray[N];
typedef struct {
} tRegistro;
tArray registros;
... int cont;
} tLista;
#include "listafin.h" ...
Luis Hernández Yáñez

int menu();

... listafin_h no se ha definido todavía

Fundamentos de la programación: Programación modular Página 802


Gestión de una lista ordenada III
Preprocesamiento de #include en bdfin.cpp:
#include <iostream>
using namespace std;
#define registrofin_h
#include <string>
using namespace std;

typedef struct { #ifndef registrofin_h


... #define registrofin_h
} tRegistro; #include <string>
... using namespace std;

#define listafin_h typedef struct {


#include <string> ...
using namespace std; } tRegistro;
Luis Hernández Yáñez

#include "registrofin.h" ...

...
int menu(); ¡registrofin_h ya está definido!
...

Fundamentos de la programación: Programación modular Página 803


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 804


Misma interfaz, implementación alternativa
#include <string> lista.h
using namespace std;
#include "registrofin.h"

const int N = 100;


typedef tRegistro tArray[N];
typedef struct {
tArray registros;
Lista int cont; Lista
} tLista;
ordenada no ordenada
void insertar(tLista &lista, tRegistro registro, bool &ok);

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) { void insertar(tLista &lista, tRegistro registro, bool &ok) {
ok = false; // Lista llena ok = true;
} if (lista.cont == N) {
else { ok = false; // Lista llena
int i = 0; }
else {
while ((i < lista.cont) && (lista.registros[i] < registro)) {
Luis Hernández Yáñez

i++; lista.registros[lista.cont] = registro;
} lista.cont++;
// Insertamos en la posición i }
for (int j = lista.cont; j > i; j‐‐) { }
// Desplazamos a la derecha
lista.registros[j] = lista.registros[j ‐ 1];
...

Fundamentos de la programación: Programación modular Página 805

Misma interfaz, implementación alternativa


listaDES.cpp: Lista no ordenada
...
#include "lista.h"

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
listaORD.cpp: Lista ordenada if (lista.cont == N) {
ok = false; // Lista llena
...
}
#include "lista.h"
else {
lista.registros[lista.cont] = registro;
void insertar(tLista &lista, tRegistro registro, bool &ok) {
ok = true; lista.cont++;
if (lista.cont == N) { }
ok = false; // Lista llena }
} ...
else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
}
// Insertamos en la posición i
Luis Hernández Yáñez

for (int j = lista.cont; j > i; j‐‐) {
// Desplazamos a la derecha
lista.registros[j] = lista.registros[j ‐ 1];
}
lista.registros[i] = registro;
...

Fundamentos de la programación: Programación modular Página 806


Misma interfaz, implementación alternativa
Al compilar, incluimos un archivo de implementación u otro:
¿Programa con lista ordenada o con lista desordenada?
g++ ‐o programa.exe registrofin.cpp listaORD.cpp ...
Incluye la implementación de la lista con ordenación
g++ ‐o programa.exe registrofin.cpp listaDES.cpp ...
Incluye la implementación de la lista sin ordenación
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 807


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 808


Agrupaciones lógicas de declaraciones
Espacio de nombres: agrupación de declaraciones
(tipos, datos, subprogramas) bajo un nombre distintivo
Forma de un espacio de nombres:
namespace nombre {
// Declaraciones
}
Por ejemplo:
namespace miEspacio {
int i;
double d;
Luis Hernández Yáñez

}
Variables i y d declaradas en el espacio de nombres miEspacio

Fundamentos de la programación: Programación modular Página 809

Acceso a miembros de un espacio de nombres


Operador de resolución de ámbito (::)
Acceso a las variables del espacio de nombres miEspacio:
Nombre del espacio y operador de resolución de ámbito
miEspacio::i
miEspacio::d
Puede haber entidades con el mismo identificador en distintos
módulos o en ámbitos distintos de un mismo módulo
Cada declaración en un espacio de nombres distinto:
namespace primero { namespace segundo {
int x = 5; double x = 3.1416;
Luis Hernández Yáñez

} }
Ahora se distingue entre primero::x y segundo::x

Fundamentos de la programación: Programación modular Página 810


using
Introduce un nombre de un espacio de nombres en el ámbito actual:
#include <iostream> 
using namespace std; 
namespace primero { 
int x = 5; 
int y = 10; 
}  5
namespace segundo {  2.7183
double x = 3.1416; 
double y = 2.7183;  10
}  3.1416
int main() { 
using primero::x; 
using segundo::y; 
cout << x << endl; // x es primero::x
Luis Hernández Yáñez

cout << y << endl; // y es segundo::y
cout << primero::y << endl; // espacio explícito
cout << segundo::x << endl; // espacio explícito
return 0; 
}
Fundamentos de la programación: Programación modular Página 811

using namespace
Introduce todos los nombres de un espacio en el ámbito actual:
#include <iostream> 
using namespace std; 
namespace primero { 
int x = 5;  5
int y = 10; 
}  using [namespace] 10
namespace segundo {  sólo tiene efecto 3.1416
double x = 3.1416;  en el bloque
double y = 2.7183;  en que se encuentra 2.7183

int main() { 
using namespace primero; 
cout << x << endl; // x es primero::x
cout << y << endl; // y es primero::y
Luis Hernández Yáñez

cout << segundo::x << endl; // espacio explícito
cout << segundo::y << endl; // espacio explícito
return 0; 
}

Fundamentos de la programación: Programación modular Página 812


#ifndef listaEN_h
#define listaEN_h
#include "registrofin.h"

namespace ord { // Lista ordenada


const int N = 100;
typedef tRegistro tArray[N];
typedef struct {
tArray registros;
int cont;
} tLista;
const string BD = "bd.txt";
void mostrar(const tLista &lista);
void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int pos, bool &ok); // 1..N
int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
Luis Hernández Yáñez

void guardar(tLista lista);
} // namespace

#endif

Fundamentos de la programación: Programación modular Página 813

Implementación
#include <iostream>
#include <fstream>
using namespace std;
#include "listaEN.h"

void ord::insertar(tLista &lista, tRegistro registro, bool &ok) {


// ...
}

void ord::eliminar(tLista &lista, int pos, bool &ok) {
// ...
}

int ord::buscar(tLista lista, string nombre) {
// ...
Luis Hernández Yáñez

...

Fundamentos de la programación: Programación modular Página 814


Uso del espacio de nombres
Quien utilice listaEN.h debe poner el nombre del espacio:
#include <iostream>
using namespace std;
#include "registrofin.h"
#include "listaEN.h"

int menu();

int main() {
ord::tLista lista;
bool ok;
ord::cargar(lista, ok);
if (!ok) {
cout << "No se pudo abrir el archivo!" << endl;
}
Luis Hernández Yáñez

else {
ord::mostrar(lista);
...
O usar una instrucción using namespace ord;
Fundamentos de la programación: Programación modular Página 815

Uso del espacio de nombres


#include <iostream>
using namespace std;
#include "registrofin.h"
#include "listaEN.h"
using namespace ord;

int menu();

int main() {
tLista lista;
bool ok;
cargar(lista, ok);
if (!ok) {
cout << "No se pudo abrir el archivo!" << endl;
Luis Hernández Yáñez

}
else {
mostrar(lista);
...

Fundamentos de la programación: Programación modular Página 816


Implementaciones alternativas
Distintos espacios de nombres para distintas implementaciones
¿Lista ordenada o lista desordenada?
namespace ord { // Lista ordenada
const int N = 100;
typedef tRegistro tArray[N];
...
void mostrar(const tLista &lista);
void insertar(tLista &lista, tRegistro registro, bool &ok);
...
} // namespace
namespace des { // Lista desordenada
const int N = 100;
typedef tRegistro tArray[N];
...
Luis Hernández Yáñez

void mostrar(const tLista &lista);


void insertar(tLista &lista, tRegistro registro, bool &ok);
...
} // namespace

Fundamentos de la programación: Programación modular Página 817

Cabecera listaEN.h

Implementaciones alternativas
Todo lo común puede estar fuera de la estructura namespace:
#ifndef listaEN_H
#define listaEN_H

#include "registrofin.h"

const int N = 100;

typedef tRegistro tArray[N];


typedef struct {
tArray registros;
int cont;
} tLista;
Luis Hernández Yáñez

void mostrar(const tLista &lista);
void eliminar(tLista &lista, int pos, bool &ok); // pos = 1..N

...

Fundamentos de la programación: Programación modular Página 818


namespace ord { // Lista ordenada
const string BD = "bd.txt";
void insertar(tLista &lista, tRegistro registro, bool &ok);
int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);
} // namespace

namespace des { // Lista desordenada


const string BD = "bddes.txt";
void insertar(tLista &lista, tRegistro registro, bool &ok);
int buscar(tLista lista, string nombre);
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);
} // namespace
Luis Hernández Yáñez

#endif cargar() y guardar() se distinguen porque usan


su propia BD, pero se implementan exactamente igual

Fundamentos de la programación: Programación modular Página 819

listaEN.cpp

#include <iostream>
using namespace std;
#include <fstream>
#include "listaEN.h"

// IMPLEMENTACIÓN DE LOS SUBPROGRAMAS COMUNES
void eliminar(tLista &lista, int pos, bool &ok) { // ...
}

void mostrar(const tLista &lista) { // ...


}

// IMPLEMENTACIÓN DE LOS SUBPROGRAMAS DEL ESPACIO DE NOMBRES ord
void ord::insertar(tLista &lista, tRegistro registro, bool &ok) {
ok = true;
if (lista.cont == N) {
ok = false; // Lista llena
}
Luis Hernández Yáñez

else {
int i = 0;
while ((i < lista.cont) && (lista.registros[i] < registro)) {
i++;
} ...

Fundamentos de la programación: Programación modular Página 820


for (int j = lista.cont; j > i; j‐‐) {
lista.registros[j] = lista.registros[j ‐ 1];
}
lista.registros[i] = registro;
lista.cont++;
}
}

int ord::buscar(tLista lista, string nombre) {


int ini = 0, fin = lista.cont ‐ 1, mitad;
bool encontrado = false;
while ((ini <= fin) && !encontrado) {
mitad = (ini + fin) / 2;
if (nombre == lista.registros[mitad].nombre) {
encontrado = true;
}
else if (nombre < lista.registros[mitad].nombre) {
fin = mitad ‐ 1;
Luis Hernández Yáñez

}
else {
ini = mitad + 1;
}
} ...

Fundamentos de la programación: Programación modular Página 821

if (encontrado) {
mitad++;
}
else {
mitad = ‐1;
}
return mitad;
}

void ord::cargar(tLista &lista, bool &ok) { // ...


}

void ord::guardar(tLista lista) { // ...


}
...
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 822


// IMPLEMENTACIÓN DE LOS SUBPROGRAMAS DEL ESPACIO DE NOMBRES des

void des::insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) {
ok = false; // Lista llena
}
else {
lista.registros[lista.cont] = registro;
lista.cont++;
}
}

int des::buscar(tLista lista, string nombre) {


int pos = 0;
bool encontrado = false;
while ((pos < lista.cont) && !encontrado) {
if (nombre == lista.registros[pos].nombre) {
encontrado = true;
Luis Hernández Yáñez

}
else {
pos++;
}
} ...

Fundamentos de la programación: Programación modular Página 823

if (encontrado) {
pos++;
}
else {
pos = ‐1;
}
return pos;
}

void des::cargar(tLista &lista, bool &ok) { // ...


}

void des::guardar(tLista lista) { // ...


}
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 824


bdEN.cpp

Programa principal
#include <iostream>
using namespace std;
#include "registrofin.h"
#include "listaEN.h"
using namespace ord;

int menu();

int main() {
tLista lista;
bool ok;
...
tRegistro registro = nuevo();
insertar(lista, registro, ok);
if (!ok) {
Luis Hernández Yáñez

...

Fundamentos de la programación: Programación modular Página 825

bdEN.cpp

Programa principal
#include <iostream>
using namespace std;
#include "registrofin.h"
#include "listaEN.h"
using namespace des;

int menu();

int main() {
tLista lista;
bool ok;
...
tRegistro registro = nuevo();
insertar(lista, registro, ok);
if (!ok) {
Luis Hernández Yáñez

...

Fundamentos de la programación: Programación modular Página 826


Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 827

Software de calidad
El software debe ser desarrollado con buenas prácticas de
ingeniería del software que aseguren un buen nivel de calidad
Los distintos módulos de la aplicación deben ser probados
exhaustivamente, tanto de forma independiente como en su
relación con los demás módulos
La prueba y depuración es muy importante y todos los equipos
deberán seguir buenas pautas para asegurar la calidad
Los módulos deben ser igualmente bien documentados, de
forma que otros desarrolladores puedan aprovecharlos
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 828


Prueba exhaustiva
El software debe ser probado exhaustivamente
Debemos intentar descubrir todos los errores posible
Los errores deben ser depurados, corrigiendo el código
Pruebas sobre listas:
 Lista inicialmente vacía
 Lista inicialmente llena
 Lista con un número intermedio de elementos
 Archivo no existente
Etcétera...
Luis Hernández Yáñez

Se han de probar todas las opciones/situaciones del programa


En las clases prácticas veremos cómo se depura el software

Fundamentos de la programación: Programación modular Página 829

No reinventemos la rueda
Desarrollar el software pensando en su posible reutilización
Un software de calidad debe poder ser fácilmente reutilizado
Nuestros módulos deben ser fácilmente usados y modificados

Por ejemplo: Nueva aplicación que gestione una lista de longitud


variable de registros con NIF, nombre, apellidos y edad
Partiremos de los módulos registro y lista existentes
Las modificaciones básicamente afectarán al módulo registro
Luis Hernández Yáñez

Fundamentos de la programación: Programación modular Página 830


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Programación modular Página 831


Fundamentos de la programación

8A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

ventas.cpp

#include <iostream> string nombre;


#include <string> double precio;
using namespace std; int unidades;
} tProducto;
const int NCLI = 100;
const int NPROD = 200; typedef struct {
const int NVENTAS = 3000; tProducto productos[NPROD];
int cont;
typedef struct { } tListaProductos;
int id_cli;
string nif; typedef struct {
string nombre; int id;
string telefono; int id_prod;
} tCliente; int id_cli;
int unidades;
typedef struct { } tVenta;
tCliente clientes[NCLI];
int cont; typedef struct {
} tListaClientes; tVenta ventas[NVENTAS];
Luis Hernández Yáñez

int cont;
typedef struct { } tListaVentas;
int id_prod;
string codigo; ...

Fundamentos de la programación: Ejemplo de modularización Página 833


tCliente nuevoCliente();
bool valida(tCliente cliente); // Función interna
bool operator<(tCliente opIzq, tCliente opDer); // Por NIF
void mostrar(tCliente cliente);
void inicializar(tListaClientes &lista);
void cargar(tListaClientes &lista);
void insertar(tListaClientes &lista, tCliente cliente, bool &ok);
void buscar(const tListaClientes &lista, string nif, tCliente &cliente, bool &ok);
void eliminar(tListaClientes &lista, string nif, bool &ok);
void mostrar(const tListaClientes &lista);
tProducto nuevoProducto();
bool valida(tProducto producto); // Función interna
bool operator<(tProducto opIzq, tProducto opDer); // Por código
void mostrar(tProducto producto);
void inicializar(tListaProductos &lista);
void cargar(tListaProductos &lista);
void insertar(tListaProductos &lista, tProducto producto, bool &ok);
Luis Hernández Yáñez

void buscar(const tListaProductos &lista, string codigo, tProducto &producto,


bool &ok);
void eliminar(tListaProductos &lista, string codigo, bool &ok);
...

Fundamentos de la programación: Ejemplo de modularización Página 834

void mostrar(const tListaProductos &lista);


tVenta nuevaVenta(int id_prod, int id_cli, int unidades);
bool valida(tVenta venta); // Función interna
void mostrar(tVenta venta, const tListaClientes &clientes,
const tListaProductos &productos);
void inicializar(tListaVentas &lista);
void cargar(tListaVentas &lista);
void insertar(tListaVentas &lista, tVenta venta, bool &ok);
void buscar(const tListaVentas &lista, int id, tVenta &venta, bool &ok);
void eliminar(tListaVentas &lista, int id, bool &ok);
void ventasPorClientes(const tListaVentas &lista);
void ventasPorProductos(const tListaVentas &lista);
double totalVentas(const tListaVentas &ventas, string nif, 
const tListaClientes &clientes, 
const tListaProductos &productos);
void stock(const tListaVentas &ventas, const tListaClientes &clientes, 
const tListaProductos &productos);
Luis Hernández Yáñez

int menu();

int main() {
...

Fundamentos de la programación: Ejemplo de modularización Página 835


#include <iostream> string nombre;
#include <string> double precio;
using namespace std; int unidades;
} tProducto;
const int NCLI = 100; Lista de productos
const int NPROD = 200; typedef struct {
const int NVENTAS = 3000; tProducto productos[NPROD];
int cont;
typedef struct { Cliente } tListaProductos;
int id_cli;
string nif; typedef struct { Venta
string nombre; int id;
string telefono; int id_prod;
} tCliente; int id_cli;
Lista de clientes int unidades;
typedef struct { } tVenta;
Lista de ventas
tCliente clientes[NCLI];
int cont; typedef struct {
} tListaClientes; tVenta ventas[NVENTAS];
Luis Hernández Yáñez

int cont;
typedef struct { Producto } tListaVentas;
int id_prod;
string codigo; ...

Fundamentos de la programación: Ejemplo de modularización Página 836

tCliente nuevoCliente(); Cliente


bool valida(tCliente cliente); // Función interna
bool operator<(tCliente opIzq, tCliente opDer); // Por NIF
void mostrar(tCliente cliente); Lista de clientes
void inicializar(tListaClientes &lista);
void cargar(tListaClientes &lista);
void insertar(tListaClientes &lista, tCliente cliente, bool &ok);
void buscar(const tListaClientes &lista, string nif, tCliente &cliente,
bool &ok);
void eliminar(tListaClientes &lista, string nif, bool &ok);
void mostrar(const tListaClientes &lista);
tProducto nuevoProducto();
bool valida(tProducto producto); // Función interna Producto
bool operator<(tProducto opIzq, tProducto opDer); // Por código
Luis Hernández Yáñez

void mostrar(tProducto producto);
...

Fundamentos de la programación: Ejemplo de modularización Página 837


Lista de productos
void inicializar(tListaProductos &lista);
void cargar(tListaProductos &lista);
void insertar(tListaProductos &lista, tProducto producto, bool &ok);
void buscar(const tListaProductos &lista, string codigo, tProducto &producto,
bool &ok);
void eliminar(tListaProductos &lista, string codigo, bool &ok);
void mostrar(const tListaProductos &lista);
tVenta nuevaVenta(int id_prod, int id_cli, int unidades);
bool valida(tVenta venta); // Función interna
void mostrar(tVenta venta, const tListaClientes &clientes, Venta
const tListaProductos &productos);
...
Luis Hernández Yáñez

Fundamentos de la programación: Ejemplo de modularización Página 838

Lista de ventas
void inicializar(tListaVentas &lista);
void cargar(tListaVentas &lista);
void insertar(tListaVentas &lista, tVenta venta, bool &ok);
void buscar(const tListaVentas &lista, int id, tVenta &venta, bool &ok);
void eliminar(tListaVentas &lista, int id, bool &ok);
void ventasPorClientes(const tListaVentas &lista);
void ventasPorProductos(const tListaVentas &lista);
double totalVentas(const tListaVentas &ventas, string nif, 
const tListaClientes &clientes, 
const tListaProductos &productos);
void stock(const tListaVentas &ventas, const tListaClientes &clientes, 
const tListaProductos &productos);
int menu();
Luis Hernández Yáñez

int main() {
...

Fundamentos de la programación: Ejemplo de modularización Página 839


 Cliente: cliente.h y cliente.cpp
 Lista de clientes: listaclientes.h y listaclientes.cpp
 Producto: producto.h y producto.cpp
 Lista de productos: listaproductos.h y listaproductos.cpp
 Venta: venta.h y venta.cpp
 Lista de ventas: listaventas.h y listaventas.cpp
 Programa principal: main.cpp
Distribución del código en los módulos:
 Declaraciones de tipos y datos en el archivo de cabecera (.h)
 Prototipos en el archivo de cabecera (.h) (excepto los de los
subprogramas privados –internos–, que irán en el .cpp)
Luis Hernández Yáñez

 Implementación de los subprogramas en el .cpp

Fundamentos de la programación: Ejemplo de modularización Página 840

Ventas
main.cpp

Cliente Lista de clientes Venta Lista de ventas


cliente.h listaclientes.h venta.h listaventas.h
cliente.cpp listaclientes.cpp venta.cpp listaventas.cpp

Producto Lista de productos


producto.h listaproductos.h
Luis Hernández Yáñez

producto.cpp listaproductos.cpp

Fundamentos de la programación: Ejemplo de modularización Página 841


Inclusiones (además de otras bibliotecas del sistema)
typedef struct {
int id_cli;
string nif;
string nombre; cliente.h string
string telefono;
} tCliente;

const int NCLI = 100;

typedef struct { listaclientes.h cliente.h


tCliente clientes[NCLI];
int cont; string
} tListaClientes;
Luis Hernández Yáñez

void buscar(const tListaClientes &lista, string nif, tCliente


&cliente, bool &ok);

Fundamentos de la programación: Ejemplo de modularización Página 842

typedef struct {
int id_prod;
string codigo;
string nombre; producto.h string
double precio;
int unidades;
} tProducto;

const int NPROD = 200;


listaproductos.h producto.h
typedef struct {
tProducto productos[NPROD]; string
int cont;
} tListaProductos;
Luis Hernández Yáñez

void buscar(const tListaProductos &lista, string codigo, tProducto


&producto, bool &ok);

Fundamentos de la programación: Ejemplo de modularización Página 843


typedef struct {
int id;
int id_prod; venta.h listaclientes.h
int id_cli;
int unidades; listaproductos.h
} tVenta;

void mostrar(tVenta venta, const tListaClientes &clientes,


const tListaProductos &productos);

const int NVENTAS = 3000;


listaventas.h venta.h
typedef struct {
tVenta ventas[NVENTAS]; listaclientes.h
int cont;
} tListaVentas; listaproductos.h
Luis Hernández Yáñez

double totalVentas(const tListaVentas &ventas, string nif, 


const tListaClientes &clientes, 
const tListaProductos &productos);

Fundamentos de la programación: Ejemplo de modularización Página 844

#ifndef cliente_h
#define cliente_h

#include <string>
using namespace std;

typedef struct {
int id_cli;
string nif;
string nombre;
string telefono;
} tCliente;

tCliente nuevoCliente();
bool operator<(tCliente opIzq, tCliente opDer); // Por NIF
Luis Hernández Yáñez

void mostrar(tCliente cliente);

#endif

Fundamentos de la programación: Ejemplo de modularización Página 845


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Ejemplo de modularización Página 846


Fundamentos de la programación

9
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Direcciones de memoria y punteros 849
Operadores de punteros 854
Punteros y direcciones válidas 864
Punteros no inicializados 866
Un valor seguro: NULL 867
Copia y comparación de punteros 868
Tipos puntero 873
Punteros a estructuras 875
Punteros a constantes y punteros constantes 877
Punteros y paso de parámetros 879
Punteros y arrays 883
Memoria y datos del programa 886
Memoria dinámica 891
Punteros y datos dinámicos 895
Gestión de la memoria 907
Luis Hernández Yáñez

Errores comunes 911
Arrays de datos dinámicos 916
Arrays dinámicos 928

Fundamentos de la programación: Punteros y memoria dinámica
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 849

Los datos en la memoria


Todo dato se almacena en memoria:
Varios bytes a partir de una dirección 0F03:1A37 ...
i 0F03:1A38 00000000
int i = 5; 0F03:1A39 00000000
0F03:1A3A 00000000
Dirección base
0F03:1A3B 00000101
0F03:1A3C ...

El dato (i) se accede a partir de su dirección base (0F03:1A38)


Dirección de la primera celda de memoria utilizada por el dato
Luis Hernández Yáñez

El tipo del dato (int) indica cuántos bytes (4) requiere el dato:
00000000 00000000 00000000 00000101  5
(La codificación de los datos puede ser diferente; y la de las direcciones también)

Fundamentos de la programación: Punteros y memoria dinámica Página 850


Los punteros contienen direcciones de memoria
Un puntero sirve para acceder a través de él a otro dato
El valor del puntero es la dirección de memoria base de otro dato
...
Indirección:
i 0F03:1A38 00
Acceso indirecto a un dato
0F03:1A39 00
0F03:1A3A 00 punt punt apunta a i
0F03:1A3B 05
... i 5
punt 0F07:0417 0F
0F07:0418 03
Luis Hernández Yáñez

0F07:0419 1A ¿De qué tipo es el dato apuntado?


0F07:041A 38 ¿Cuántas celdas ocupa?
... ¿Cómo se interpretan los 0/1?

Fundamentos de la programación: Punteros y memoria dinámica Página 851

Los punteros contienen direcciones de memoria


¿De qué tipo es el dato apuntado?
La variable a la que apunta un puntero será de un tipo concreto
¿Cuánto ocupa? ¿Cómo se interpreta?
El tipo de variable apuntado se establece al declarar el puntero:
tipo *nombre;
El puntero nombre apuntará a una variable del tipo indicado
El asterisco (*) indica que es un puntero a datos de ese tipo
int *punt; // punt inicialmente contiene una dirección
// que no es válida (no apunta a nada)
Luis Hernández Yáñez

El puntero punt apuntará a una variable entera (int)


int i; // Dato entero  vs. int *punt; // Puntero a entero

Fundamentos de la programación: Punteros y memoria dinámica Página 852


Los punteros contienen direcciones de memoria
Las variables puntero tampoco se inicializan automáticamente
Al declararlas sin inicializador contienen direcciones no válidas
int *punt; // punt inicialmente contiene una dirección
// que no es válida (no apunta a nada)
Un puntero puede apuntar a cualquier dato de su tipo base
Un puntero no tiene por qué apuntar necesariamente a un dato
(puede no apuntar a nada: valor NULL)
¿Para qué sirven los punteros?
 Para implementar el paso de parámetros por referencia
Luis Hernández Yáñez

 Para manejar datos dinámicos


(Datos que se crean y destruyen durante la ejecución)
 Para implementar los arrays
Fundamentos de la programación: Punteros y memoria dinámica Página 853
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 854


&
Obtener la dirección de memoria de ...
Operador monario y prefijo
& devuelve la dirección de memoria base del dato al que precede
int i;
cout << &i; // Muestra la dirección de memoria de i
Un puntero puede recibir la dirección de datos de su tipo base
punt
int i;
int *punt; i
punt = &i; // punt contiene la dirección de i
Ahora punt ya contiene una dirección de memoria válida
Luis Hernández Yáñez

punt apunta a (contiene la dirección de) la variable i (int)

Fundamentos de la programación: Punteros y memoria dinámica Página 855

&
Obtener la dirección de memoria de ...
...

i 0F03:1A38
int i, j;
0F03:1A39
... 0F03:1A3A
int *punt; 0F03:1A3B

j 0F03:1A3C

0F03:1A3D

0F03:1A3E

0F03:1A3F

...

punt 0F07:0417
Luis Hernández Yáñez

0F07:0418
0F07:0419
0F07:041A

...

Fundamentos de la programación: Punteros y memoria dinámica Página 856


&
Obtener la dirección de memoria de ...
...

i 0F03:1A38 00
int i, j;
0F03:1A39 00
... 0F03:1A3A 00
int *punt; 0F03:1A3B 05

... j 0F03:1A3C

0F03:1A3D
i = 5;
0F03:1A3E

0F03:1A3F

...

punt 0F07:0417
Luis Hernández Yáñez

0F07:0418
0F07:0419
i 5
0F07:041A

...

Fundamentos de la programación: Punteros y memoria dinámica Página 857

&
Obtener la dirección de memoria de ...
...

i 0F03:1A38 00
int i, j;
0F03:1A39 00
... 0F03:1A3A 00
int *punt; 0F03:1A3B 05

... j 0F03:1A3C

0F03:1A3D
i = 5;
0F03:1A3E
punt = &i;
0F03:1A3F

...

punt punt 0F07:0417 0F


Luis Hernández Yáñez

0F07:0418 03

i 0F07:0419 1A
5
0F07:041A 38
...

Fundamentos de la programación: Punteros y memoria dinámica Página 858


*
Obtener lo que hay en la dirección ...
Operador monario y prefijo
* accede a lo que hay en la dirección de memoria a la que precede
Permite acceder a un dato a través un puntero que lo apunte:
punt = &i;
cout << *punt; // Muestra lo que hay en la dirección punt
*punt: lo que hay en la dirección que contiene el puntero punt

punt contiene la dirección de memoria de la variable i


*punt accede al contenido de esa variable i
Acceso indirecto al valor de i
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 859

*
Obtener lo que hay en la dirección ...
...

i 0F03:1A38 00
int i, j;
0F03:1A39 00
... 0F03:1A3A 00
int *punt; 0F03:1A3B 05

... j 0F03:1A3C

0F03:1A3D
i = 5;
0F03:1A3E
punt = &i;
0F03:1A3F
j = *punt; ...

punt 0F07:0417 0F
Luis Hernández Yáñez

0F07:0418 03
0F07:0419 1A
punt:
0F07:041A 38
...

Fundamentos de la programación: Punteros y memoria dinámica Página 860


*
Obtener lo que hay en la dirección ...
...

i 0F03:1A38 00
int i, j;
0F03:1A39 00
... 0F03:1A3A 00
Direccionamiento
int *punt; indirecto  0F03:1A3B 05

... (indirección) j 0F03:1A3C


Se accede al dato i 0F03:1A3D
i = 5;
de forma indirecta 0F03:1A3E
punt = &i;
0F03:1A3F
j = *punt; ...

punt 0F07:0417 0F
Luis Hernández Yáñez

0F07:0418 03
0F07:0419 1A
*punt:
0F07:041A 38
...

Fundamentos de la programación: Punteros y memoria dinámica Página 861

*
Obtener lo que hay en la dirección ...
...

i 0F03:1A38 00
int i, j;
0F03:1A39 00
... 0F03:1A3A 00
int *punt; 0F03:1A3B 05

... j 0F03:1A3C 00
0F03:1A3D 00
i = 5;
0F03:1A3E 00
punt = &i;
0F03:1A3F 05
j = *punt; ...

punt 0F07:0417 0F
Luis Hernández Yáñez

0F07:0418 03
0F07:0419 1A
0F07:041A 38
...

Fundamentos de la programación: Punteros y memoria dinámica Página 862


punteros.cpp

Ejemplo de uso de punteros


#include <iostream>
using namespace std;

int main() {
int i = 5;
int j = 13;
int *punt;
punt = &i;
cout << *punt << endl; // Muestra el valor de i
punt = &j;
cout << *punt << endl; // Ahora muestra el valor de j
int *otro = &i;
cout << *otro + *punt << endl; // i + j
int k = *punt;
Luis Hernández Yáñez

cout << k << endl; // Mismo valor que j

return 0;
}

Fundamentos de la programación: Punteros y memoria dinámica Página 863


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 864


Todo puntero ha de tener una dirección válida
Un puntero sólo debe ser utilizado si tiene una dirección válida
Un puntero NO contiene una dirección válida tras ser definido
Un puntero obtiene una dirección válida:
 Asignando la dirección de otro dato (operador &)
 Asignando otro puntero (mismo tipo base) que ya sea válido
 Asignando el valor NULL (puntero nulo, no apunta a nada)

int i;
int *q; // q no tiene aún una dirección válida
int *p = &i; // p toma una dirección válida
Luis Hernández Yáñez

q = p; // ahora q ya tiene una dirección válida
q = NULL; // otra dirección válida para q

Fundamentos de la programación: Punteros y memoria dinámica Página 865

Punteros que apuntan a saber qué...


Un puntero no inicializado contiene una dirección desconocida
int *punt; // No inicializado
*punt = 12; // ¿A qué dato se está asignando el valor?
¿Dirección de la zona de datos del programa?
¡Podemos modificar inadvertidamente un dato del programa!
¿Dirección de la zona de código del programa?
¡Podemos modificar el código del propio programa!
¿Dirección de la zona de código del sistema operativo?
¡Podemos modificar el código del propio S.O.!
Luis Hernández Yáñez

 Consecuencias imprevisibles (cuelgue)


(Los S.O. modernos protegen bien la memoria)

Fundamentos de la programación: Punteros y memoria dinámica Página 866


Punteros que no apuntan a nada
Inicializando los punteros a NULL podemos detectar errores:
int *punt = NULL;
... punt X
*punt = 13;
punt ha sido inicializado a NULL: ¡No apunta a nada!

Si no apunta a nada, ¿¿¿qué significa *punt??? No tiene sentido

 ERROR: ¡Acceso a un dato a través de un puntero nulo!

Error de ejecución, lo que ciertamente no es bueno


Luis Hernández Yáñez

Pero sabemos cuál ha sido el problema, lo que es mucho


Sabemos dónde y qué buscar para depurar

Fundamentos de la programación: Punteros y memoria dinámica Página 867


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 868


Apuntando al mismo dato
Al copiar un puntero en otro, ambos apuntarán al mismo dato:
int x = 5;
int *punt1 = NULL; // punt1 no apunta a nada
int *punt2 = &x; // punt2 apunta a la variable x

punt1 X punt2
Luis Hernández Yáñez

x 5

Fundamentos de la programación: Punteros y memoria dinámica Página 869

Apuntando al mismo dato


Al copiar un puntero en otro, ambos apuntarán al mismo dato:
int x = 5;
int *punt1 = NULL; // punt1 no apunta a nada
int *punt2 = &x; // punt2 apunta a la variable x
punt1 = punt2; // ambos apuntan a la variable x

punt1 punt2
Luis Hernández Yáñez

x 5

Fundamentos de la programación: Punteros y memoria dinámica Página 870


Apuntando al mismo dato
Al copiar un puntero en otro, ambos apuntarán al mismo dato:
int x = 5;
int *punt1 = NULL; // punt1 no apunta a nada
int *punt2 = &x; // punt2 apunta a la variable x
punt1 = punt2; // ambos apuntan a la variable x
*punt1 = 8;

punt1 punt2

Al dato x ahora se puede


Luis Hernández Yáñez

acceder de tres formas:


x 8
5
x  *punt1  *punt2

Fundamentos de la programación: Punteros y memoria dinámica Página 871

¿Apuntan al mismo dato?


Operadores relacionales == y !=:
int x = 5;
int *punt1 = NULL;
int *punt2 = &x;
...
if (punt1 == punt2) {
cout << "Apuntan al mismo dato" << endl;
}
else {
cout << "No apuntan al mismo dato" << endl;
}
Luis Hernández Yáñez

Sólo se pueden comparar punteros con el mismo tipo base

Fundamentos de la programación: Punteros y memoria dinámica Página 872


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 873

tipos.cpp

Declaración de tipos puntero


Declaramos tipos para los punteros con distintos tipos base:
typedef int *tIntPtr;
typedef char *tCharPtr;
typedef double *tDoublePtr;
int entero = 5;
tIntPtr puntI = &entero;
char caracter = 'C';
tCharPtr puntC = &caracter;
double real = 5.23;
tDoublePtr puntD = &real;
cout << *puntI << "  " << *puntC << "  " << *puntD << endl;
Luis Hernández Yáñez

Con *puntero podemos hacer lo que con otros datos del tipo base

Fundamentos de la programación: Punteros y memoria dinámica Página 874


Acceso a estructuras a través de punteros
Los punteros pueden apuntar también a estructuras:
typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;
tRegistro registro;
typedef tRegistro *tRegistroPtr;
tRegistroPtr puntero = &registro;

Operador flecha (‐>):


Acceso a los campos a través de un puntero sin usar el operador *
Luis Hernández Yáñez

puntero‐>codigo puntero‐>nombre    puntero‐>sueldo
puntero‐>...  (*puntero)....

Fundamentos de la programación: Punteros y memoria dinámica Página 875

structPtr.cpp

Acceso a estructuras a través de punteros


typedef struct {
int codigo;
string nombre;
double sueldo;
} tRegistro;
tRegistro registro;
typedef tRegistro *tRegistroPtr;
tRegistroPtr puntero = &registro;
registro.codigo = 12345;
registro.nombre = "Javier";
registro.sueldo = 95000;
cout << puntero‐>codigo << "  " << puntero‐>nombre 
<< "  " << puntero‐>sueldo << endl;
Luis Hernández Yáñez

puntero‐>codigo  (*puntero).codigo  *puntero.codigo


puntero sería una estructura con campo codigo de tipo puntero

Fundamentos de la programación: Punteros y memoria dinámica Página 876


Punteros a constantes y punteros constantes
El efecto del modificador de acceso const depende de su sitio:
const tipo *puntero; Puntero a una constante
tipo *const puntero; Puntero constante

Punteros a constantes:
typedef const int *tIntCtePtr; // Puntero a constante
int entero1 = 5, entero2 = 13;
tIntCtePtr punt_a_cte = &entero1;

(*punt_a_cte)++; // ERROR: ¡Dato no modificable!
Luis Hernández Yáñez

punt_a_cte = &entero2; // OK: El puntero no es cte.

Fundamentos de la programación: Punteros y memoria dinámica Página 877

constPtr.cpp

Punteros a constantes y punteros constantes


El efecto del modificador de acceso const depende de su sitio:
const tipo *puntero; Puntero a una constante
tipo *const puntero; Puntero constante

Punteros constantes:
typedef int *const tIntPtrCte; // Puntero constante
int entero1 = 5, entero2 = 13;
tIntPtrCte punt_cte = &entero1;

(*punt_cte)++; // OK: El puntero no apunta a cte.
Luis Hernández Yáñez

punt_cte = &entero2; // ERROR: ¡Puntero constante!

Fundamentos de la programación: Punteros y memoria dinámica Página 878


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 879

param.cpp

Paso de parámetros por referencia o variable


En el lenguaje C no hay mecanismo de paso por referencia (&)
Sólo se pueden pasar parámetros por valor
¿Cómo se simula el paso por referencia? Por medio de punteros:
void incrementa(int *punt);

void incrementa(int *punt) {


(*punt)++;
}
Paso por valor:
... El argumento (el puntero) no cambia
int entero = 5;
Aquello a lo que apunta (el entero)
incrementa(&entero);
Luis Hernández Yáñez

SÍ puede cambiar
cout << entero << endl;
Mostrará 6 en la consola

Fundamentos de la programación: Punteros y memoria dinámica Página 880


Paso de parámetros por referencia o variable
int entero = 5;
incrementa(&entero); entero 5

punt recibe la dirección de entero

void incrementa(int *punt) { punt


(*punt)++;
}
entero 6
Luis Hernández Yáñez

cout << entero << endl; entero 6

Fundamentos de la programación: Punteros y memoria dinámica Página 881

Paso de parámetros por referencia o variable


¿Cuál es el equivalente en C a este prototipo de C++?
void foo(int &param1, double &param2, char &param3);

Prototipo equivalente:
void foo(int *param1, double *param2, char *param3);

void foo(int *param1, double *param2, char *param3) {


// Al primer argumento se accede con *param1
// Al segundo argumento se accede con *param2
// Al tercer argumento se accede con *param3
}
¿Cómo se llamaría?
Luis Hernández Yáñez

int entero; double real; char caracter;


//...
foo(&entero, &real, &caracter);

Fundamentos de la programación: Punteros y memoria dinámica Página 882


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 883

Una íntima relación


Variable array  Puntero al primer elemento del array
Así, si tenemos:
int dias[12] = 
{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
Entonces:
cout << *dias << endl;
Muestra 31 en la consola, el primer elemento del array
¡Un nombre de array es un puntero constante!
Siempre apunta al primer elemento (no se puede modificar)
Luis Hernández Yáñez

Acceso a los elementos del array:


Por índice o con aritmética de punteros (Anexo)

Fundamentos de la programación: Punteros y memoria dinámica Página 884


Paso de arrays a subprogramas
¡Esto explica por qué no usamos & con los parámetros array!
El nombre del array es un puntero: ya es un paso por referencia
Prototipos equivalentes para parámetros array:
const int N = ...;
void cuadrado(int arr[N]);
void cuadrado(int arr[], int size); // Array no delimitado
void cuadrado(int *arr, int size); // Puntero
Arrays no delimitados y punteros: se necesita la dimensión
Elementos: se acceden con índice (arr[i]) o con puntero (*arr)
Una función sólo puede devolver un array en forma de puntero:
Luis Hernández Yáñez

intPtr inicializar();

Fundamentos de la programación: Punteros y memoria dinámica Página 885


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 886


Regiones de la memoria
El sistema operativo distingue varias regiones en la memoria:

Pila (Stack) Datos locales

Montón (Heap) Datos dinámicos

Datos globales

Código del 
Memoria principal
programa
Luis Hernández Yáñez

S.O.

Fundamentos de la programación: Punteros y memoria dinámica Página 887

Memoria principal Pila Datos locales

Datos globales del programa: Montón Datos dinámicos

Declarados fuera Datos globales


de los subprogramas Código del programa Memoria principal

S.O.
typedef struct {
...
} tRegistro;
const int N = 1000;
typedef tRegistro tArray[N];
typedef struct {
tArray registros;
int cont;
} tLista;
Luis Hernández Yáñez

int main() {
...

Fundamentos de la programación: Punteros y memoria dinámica Página 888


La pila (stack) Pila Datos locales

Datos locales de subprogramas: Montón Datos dinámicos

Parámetros por valor Datos globales


y variables locales Código del programa Memoria principal

S.O.
void func(tLista lista, double &total)
{
tLista aux;
int i;
...

Y los punteros temporales &resultado


que apuntan a los argumentos
de los parámetros por referencia
Luis Hernández Yáñez

func(lista, resultado)

Fundamentos de la programación: Punteros y memoria dinámica Página 889

El montón (heap) Pila Datos locales

Datos dinámicos Montón Datos dinámicos

Datos que se crean y se destruyen Datos globales

durante la ejecución del programa, Código del programa Memoria principal

a medida que se necesita S.O.

Sistema de gestión de memoria dinámica (SGMD)


Cuando se necesita memoria para una variable se solicita
El SGMD reserva espacio y devuelve la dirección base
Cuando ya no se necesita más la variable, se destruye
Se libera la memoria y el SGMD cuenta de nuevo con ella
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 890


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 891

Datos dinámicos
Se crean y se destruyen durante la ejecución del programa
Se les asigna memoria del montón

Creación
Montón
Dato dinámico
Destrucción

¿Por qué utilizar memoria dinámica?


 Almacén de memoria muy grande: datos o listas de datos que
no caben en memoria principal pueden caber en el montón
Luis Hernández Yáñez

 El programa ajusta el uso de la memoria a las necesidades


de cada momento: ni le falta ni la desperdicia

Fundamentos de la programación: Punteros y memoria dinámica Página 892


¿Cuándo se asigna memoria a los datos?
 Datos globales
En memoria principal al comenzar la ejecución del programa
Existen durante toda la ejecución del programa

 Datos locales de un subprograma


En la pila al ejecutarse el subprograma
Existen sólo durante la ejecución de su subprograma

 Datos dinámicos
En el montón cuando el programa lo solicita
Luis Hernández Yáñez

Existen a voluntad del programa

Fundamentos de la programación: Punteros y memoria dinámica Página 893

Datos estáticos
 Datos declarados como de un tipo concreto:
int i;
 Se acceden directamente a través del identificador:
cout << i;

Datos dinámicos
 Datos accedidos a través de su dirección de memoria
Esa dirección de memoria debe estar el algún puntero
Los punteros son la base del SGMD
Luis Hernández Yáñez

Los datos estáticos también se pueden acceder a través de punteros


int *p = &i;
Fundamentos de la programación: Punteros y memoria dinámica Página 894
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 895

Devuelve NULL si no queda memoria suficiente


El operador new
new tipo Reserva memoria del montón para una variable del
tipo y devuelve la primera dirección de memoria
utilizada, que debe ser asignada a un puntero
int *p; // Todavía sin una dirección válida
p = new int; // Ya tiene una dirección válida
*p = 12;
La variable dinámica se accede exclusivamente por punteros
No tiene identificador asociado
int i; // i es una variable estática
int *p1, *p2;
p1 = &i; // Puntero que da acceso a la variable
Luis Hernández Yáñez

// estática i (accesible con i o con *p1)
p2 = new int; // Puntero que da acceso a una variable
// dinámica (accesible sólo a través de p2)

Fundamentos de la programación: Punteros y memoria dinámica Página 896


registros.cpp

Inicialización con el operador new


El operador new admite un valor inicial para el dato creado:
int *p;
p = new int(12);
Se crea la variable, de tipo int, y se inicializa con el valor 12
#include <iostream>
using namespace std;
#include "registro.h"

int main() {
tRegistro reg;
reg = nuevo();
Luis Hernández Yáñez

tRegistro *punt = new tRegistro(reg);


mostrar(*punt);
...

Fundamentos de la programación: Punteros y memoria dinámica Página 897

El operador delete
delete puntero; Devuelve al montón la memoria usada por
la variable dinámica apuntada por puntero
int *p;
p = new int;
*p = 12;
...
delete p; // Ya no se necesita el entero apuntado por p

¡El puntero deja de contener una dirección válida!


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 898


dinamicas.cpp

#include <iostream>
using namespace std;

int main() {
double a = 1.5; p1 Identificadores:
double *p1, *p2, *p3;
p1 = &a; 4
p2 = new double; (a, p1, p2, p3)
*p2 = *p1; a 1.5
p3 = new double;
*p3 = 123.45;
cout << *p1 << endl; p2 1.5 Variables:
cout << *p2 << endl;
cout << *p3 << endl; 6
delete p2; (+ *p2 y *p3)
Luis Hernández Yáñez

delete p3; p3 123.45

return 0; Montón (heap)


}

Fundamentos de la programación: Punteros y memoria dinámica Página 899

#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1 ?
double a = 1.5;
p2 ?
double *p1, *p2, *p3;
p3 ?
Luis Hernández Yáñez

MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 900


#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2 ?
double *p1, *p2, *p3;
p1 = &a; p3 ?
Luis Hernández Yáñez

MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 901

#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2
double *p1, *p2, *p3;
p1 = &a; p3 ?
p2 = new double;
Luis Hernández Yáñez

MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 902


#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2
double *p1, *p2, *p3;
p1 = &a; p3 ?
p2 = new double;
*p2 = *p1;
Luis Hernández Yáñez

1.5
MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 903

#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2
double *p1, *p2, *p3;
p1 = &a; p3
p2 = new double;
*p2 = *p1;
p3 = new double;
Luis Hernández Yáñez

1.5
MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 904


#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2
double *p1, *p2, *p3;
p1 = &a; p3
p2 = new double;
*p2 = *p1;
p3 = new double;
*p3 = 123.45;

123.45
Luis Hernández Yáñez

1.5
MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 905

#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2 ?
double *p1, *p2, *p3;
p1 = &a; p3
p2 = new double;
*p2 = *p1;
p3 = new double;
*p3 = 123.45;
cout << *p1 << endl;
cout << *p2 << endl;
cout << *p3 << endl;
delete p2; 123.45
Luis Hernández Yáñez

MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 906


#include <iostream>
using namespace std; PILA
a 1.5
int main() { p1
double a = 1.5;
p2 ?
double *p1, *p2, *p3;
p1 = &a; p3 ?
p2 = new double;
*p2 = *p1;
p3 = new double;
*p3 = 123.45;
cout << *p1 << endl;
cout << *p2 << endl;
cout << *p3 << endl;
delete p2;
Luis Hernández Yáñez

delete p3;

MONTÓN

Fundamentos de la programación: Punteros y memoria dinámica Página 907


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 908


La memoria se reparte entre la pila y el montón Pila

Crecen en direcciones opuestas


Al llamar a subprogramas la pila crece
Al crear datos dinámicos el montón crece
Montón

Colisión pila‐montón
Los límites de ambas regiones se encuentran
Se agota la memoria

Desbordamiento de la pila
Luis Hernández Yáñez

La pila suele tener un tamaño máximo establecido


Si se sobrepasa se agota la pila

Fundamentos de la programación: Punteros y memoria dinámica Página 909

Gestión del montón


Sistema de Gestión de Memoria Dinámica (SGMD)
Gestiona la asignación de memoria a los datos dinámicos
Localiza secciones adecuadas y sigue la pista de lo disponible
No dispone de un recolector de basura, como el lenguaje Java
¡Hay que devolver toda la memoria solicitada!
Deben ejecutarse tantos delete como new se hayan ejecutado
La memoria disponible en el montón debe ser exactamente la
misma antes y después de la ejecución del programa

Y todo dato dinámico debe tener algún acceso (puntero)


Luis Hernández Yáñez

Es un grave error perder un dato en el montón

Fundamentos de la programación: Punteros y memoria dinámica Página 910


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 911

Olvido de destrucción de un dato dinámico


...
int main() {
tRegistro *p;
p = new tRegistro;
*p = nuevo();
mostrar(*p);
Falta delete p;
return 0;
}

G++ no indicará ningún error y el programa parecerá terminar


correctamente, pero dejará memoria desperdiciada
Luis Hernández Yáñez

Visual C++ sí comprueba el uso de la memoria dinámica


y nos avisa si dejamos memoria sin liberar

Fundamentos de la programación: Punteros y memoria dinámica Página 912


Intento de destrucción de un dato inexistente
...
int main() { p1 tRegistro
tRegistro *p1 = new tRegistro;
*p1 = nuevo();
mostrar(*p1); p2
tRegistro *p2;
p2 = p1;
mostrar(*p2);
delete p1;
delete p2;

return 0;
Sólo se ha creado
Luis Hernández Yáñez

}
una variable

Fundamentos de la programación: Punteros y memoria dinámica Página 913

Pérdida de un dato dinámico


...
int main() {
tRegistro *p1, *p2;
p1 = new tRegistro(nuevo()); 
p2 = new tRegistro(nuevo());
p2 tRegistro
mostrar(*p1); 
p1 = p2;  p1 tRegistro
mostrar(*p1);

¡Perdido!
delete p1;
delete p2;
Luis Hernández Yáñez

return 0; Se pierde un dato en el montón


} Se intenta eliminar un dato ya eliminado

Fundamentos de la programación: Punteros y memoria dinámica Página 914


Intento de acceso a un dato tras su eliminación
...
int main() {
tRegistro *p;
p = new tRegistro(nuevo());

mostrar(*p);
delete p;
...
mostrar(*p); p ha dejado de apuntar
al dato dinámico destruido
return 0;  Acceso a memoria inexistente
}
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 915


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 916


Arrays de punteros a datos dinámicos
typedef struct {
int codigo;
string nombre; Los punteros ocupan
double valor; muy poco en memoria
} tRegistro; Los datos a los que apunten
typedef tRegistro *tRegPtr; estarán en el montón

const int N = 1000;


// Array de punteros a registros:
typedef tRegPtr tArray[N];
typedef struct {
tArray registros;
Luis Hernández Yáñez

int cont;
} tLista; Se crean a medida que se insertan
Se destruyen a medida que se eliminan

Fundamentos de la programación: Punteros y memoria dinámica Página 917

tLista lista;
lista.cont = 0;

lista.cont 0
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 918


tLista lista;
lista.cont = 0;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;

lista.cont 1
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 919

tLista lista;
lista.cont = 0;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;

lista.cont 2
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 920


tLista lista;
lista.cont = 0;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;
lista.registros[lista.cont] = new tRegistro(nuevo());
lista.cont++;

lista.cont 3
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 921

Los registros se acceden a través de los punteros (operador ‐>):


cout << lista.registros[0]‐>nombre;

lista.cont 3
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 922


No hay que olvidarse de devolver la memoria al montón:
for (int i = 0; i < lista.cont; i++) {
delete lista.registros[i];
}

lista.cont 3
0 1 2 3 4 5 6 998 999

lista.registros
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 923

lista.h

#ifndef lista_h
#define lista_h registro.h con el tipo puntero:
#include "registro.h"
typedef tRegistro *tRegPtr;
const int N = 1000;
const string BD = "bd.dat";
typedef tRegPtr tArray[N];
typedef struct {
tArray registros;
int cont;
} tLista;
void mostrar(const tLista &lista);
void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int code, bool &ok);
int buscar(const tLista &lista, int code);
void cargar(tLista &lista, bool &ok);
Luis Hernández Yáñez

void guardar(const tLista &lista);


void destruir(tLista &lista);

#endif

Fundamentos de la programación: Punteros y memoria dinámica Página 924


lista.cpp

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) {
ok = false;
}
else {
lista.registros[lista.cont] = new tRegistro(registro);
lista.cont++;
}
}
void eliminar(tLista &lista, int code, bool &ok) {
ok = true;
int ind = buscar(lista, code);
if (ind == ‐1) {
ok = false;
}
else {
delete lista.registros[ind];
for (int i = ind + 1; i < lista.cont; i++) {
lista.registros[i ‐ 1] = lista.registros[i];
Luis Hernández Yáñez

}
lista.cont‐‐;
}
}

Fundamentos de la programación: Punteros y memoria dinámica Página 925

int buscar(const tLista &lista, int code) {


// Devuelve el índice o ‐1 si no se ha encontrado
int ind = 0;
bool encontrado = false;
while ((ind < lista.cont) && !encontrado) {
if (lista.registros[ind]‐>codigo == code) {
encontrado = true;
}
else {
ind++;
}
if (!encontrado) {
ind = ‐1;
}
return ind;
}

void destruir(tLista &lista) {


for (int i = 0; i < lista.cont; i++) {
Luis Hernández Yáñez

delete lista.registros[i];
}
lista.cont = 0;
}
...

Fundamentos de la programación: Punteros y memoria dinámica Página 926


listadinamica.cpp

#include <iostream>
using namespace std;
#include "registro.h"
#include "lista.h"

int main() {
tLista lista;
bool ok;
cargar(lista, ok);
if (ok) {
mostrar(lista);
destruir(lista);
}

return 0;
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 927


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 928


Creación y destrucción de arrays dinámicos
Array dinámico: array que se ubica en la memoria dinámica

Creación de un array dinámico:


tipo *puntero = new tipo[dimensión];
int *p = new int[10];
Crea un array de 10 int en memoria dinámica
Los elementos se acceden a través del puntero: p[i]

Destrucción del array:


delete [] p;
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 929

#include <iostream>
using namespace std;
const int N = 10;

int main() {
int *p = new int[N];
for (int i = 0; i < N; i++) {
p[i] = i;
}
for (int i = 0; i < N; i++) {
cout << p[i] << endl;
}
delete [] p;
¡No olvides destruir el array dinámico!
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Punteros y memoria dinámica Página 930


listaAD.h

...
#include "registro.h"

const int N = 1000;

// Lista: array dinámico (puntero) y contador
typedef struct {
tRegPtr registros;
int cont;
} tLista;

...
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 931

listaAD.cpp

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
if (lista.cont == N) {
ok = false; No usamos new
}
else { Se han creado todo
lista.registros[lista.cont] = registro;
lista.cont++; el array al cargar
}
}

void eliminar(tLista &lista, int code, bool &ok) {


ok = true;
int ind = buscar(lista, code); No usamos delete
if (ind == ‐1) {
ok = false; Se destruye todo
} el array al final
else {
for (int i = ind + 1; i < lista.cont; i++) {
Luis Hernández Yáñez

lista.registros[i ‐ 1] = lista.registros[i];
}
lista.cont‐‐;
}
} ...

Fundamentos de la programación: Punteros y memoria dinámica Página 932


int buscar(tLista lista, int code) {
int ind = 0;
bool encontrado = false;
while ((ind < lista.cont) && !encontrado) {
if (lista.registros[ind].codigo == code) {
encontrado = true;
}
else {
ind++;
}
}
if (!encontrado) {
ind = ‐1;
}
return ind;
}

void destruir(tLista &lista) {


delete [] lista.registros;
Luis Hernández Yáñez

lista.cont = 0;
}
...

Fundamentos de la programación: Punteros y memoria dinámica Página 933

void cargar(tLista &lista, bool &ok) {


ifstream archivo;
char aux;
ok = true;
archivo.open(BD.c_str());
if (!archivo.is_open()) {
ok = false;
}
else {
tRegistro registro; Se crean todos a la vez
lista.cont = 0;
lista.registros = new tRegistro[N];
archivo >> registro.codigo;
while ((registro.codigo != ‐1) && (lista.cont < N)) {
archivo >> registro.valor;
archivo.get(aux); // Saltamos el espacio
getline(archivo, registro.nombre);
lista.registros[lista.cont] = registro;
lista.cont++;
Luis Hernández Yáñez

archivo >> registro.codigo;
}
archivo.close();
}
}

Fundamentos de la programación: Punteros y memoria dinámica Página 934


ejemploAD.cpp

Mismo programa principal que el del array de datos dinámicos


Pero incluyendo listaAD.h, en lugar de lista.h
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica Página 935

Array de datos dinámicos: Array de punteros a datos dinámicos


Array dinámico: Puntero a array en memoria dinámica

Array de datos dinámicos: Array dinámico:


Array de punteros Puntero a array
0 1 2 3 4 5 6 7

0 1 2 3 4 5 6 7
Luis Hernández Yáñez

Montón

Fundamentos de la programación: Punteros y memoria dinámica Página 936


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Punteros y memoria dinámica Página 937


Fundamentos de la programación

9A
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Aritmética de punteros 940
Recorrido de arrays con punteros 953
Referencias 962
Listas enlazadas 964
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo)
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 940

Operaciones aritméticas con punteros


La aritmética de punteros es una aritmética un tanto especial...
Trabaja tomando como unidad de cálculo el tamaño del tipo base
int dias[12] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
typedef int* tIntPtr;
tIntPtr punt = dias;
punt empieza apuntando al primer elemento del array:
cout << *punt << endl; // Muestra 31 (primer elemento)
punt++;
punt++ hace que punt pase a apuntar al siguiente elemento
cout << *punt << endl; // Muestra 28 (segundo elemento)
Luis Hernández Yáñez

A la dirección de memoria actual se le suman tantas unidades


como bytes (4) ocupe en memoria un dato de ese tipo (int)

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 941


int dias[12] = { 31, 28, 31, 30, 31, 30, 
...
31, 31, 30, 31, 30, 31 }; dias[0] 0F03:1A38
0F03:1A39
typedef int* tIntPtr; 0F03:1A3A 31
tIntPtr punt = dias; 0F03:1A3B
dias[1] 0F03:1A3C
0F03:1A3D
0F03:1A3E 28
0F03:1A3F
dias[2] 0F03:1A40
0F03:1A41
0F03:1A42 31
0F03:1A43
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
Luis Hernández Yáñez

punt 0F07:041B 0F
0F07:041C 03
0F07:041D 1A
0F07:041E 38
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 942

int dias[12] = { 31, 28, 31, 30, 31, 30,  ...


31, 31, 30, 31, 30, 31 }; dias[0] 0F03:1A38
0F03:1A39
typedef int* tIntPtr; 0F03:1A3A 31
tIntPtr punt = dias; 0F03:1A3B
dias[1] 0F03:1A3C
punt++; 0F03:1A3D
0F03:1A3E 28
0F03:1A3F
dias[2] 0F03:1A40
0F03:1A41
0F03:1A42 31
0F03:1A43
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
Luis Hernández Yáñez

punt 0F07:041B 0F
0F07:041C 03

punt‐‐ hace que apunte al elemento anterior


0F07:041D 1A
0F07:041E 3C
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 943


int dias[12] = { 31, 28, 31, 30, 31, 30,  ...
31, 31, 30, 31, 30, 31 }; dias[0] 0F03:1A38
0F03:1A39
typedef int* tIntPtr; 0F03:1A3A 31
tIntPtr punt = dias; 0F03:1A3B
dias[1] 0F03:1A3C
punt = punt + 2; 0F03:1A3D
0F03:1A3E 28
0F03:1A3F
dias[2] 0F03:1A40
0F03:1A41
0F03:1A42 31
0F03:1A43
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
Luis Hernández Yáñez

punt 0F07:041B 0F
0F07:041C 03
0F07:041D 1A
Restando pasamos a elementos anteriores 0F07:041E 40
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 944

int dias[12] = { 31, 28, 31, 30, 31, 30,  ...


31, 31, 30, 31, 30, 31 }; dias[0] 0F03:1A38
0F03:1A39
typedef int* tIntPtr; 0F03:1A3A 31
tIntPtr punt = dias; 0F03:1A3B
dias[1] 0F03:1A3C
punt = punt + 2; 0F03:1A3D
0F03:1A3E 28
0F03:1A3F
int num = punt ‐ dias; dias[2] 0F03:1A40
0F03:1A41
Nº de elementos entre los punteros 0F03:1A42 31
0F03:1A43
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
Luis Hernández Yáñez

punt 0F07:041B 0F
0F07:041C 03
0F07:041D 1A
0F07:041E 3C
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 945


Otro tipo base ...

short int (2 bytes)
dias[0] 0F03:1A38
0F03:1A39 31
short int dias[12] = {31, 28, 31, 30, dias[1] 0F03:1A3A
28
0F03:1A3B
31, 30, 31, 31, 30, 31, 30, 31}; dias[2] 0F03:1A3C
0F03:1A3D 31
typedef short int* tSIPtr; dias[3] 0F03:1A3E
tSIPtr punt = dias; 0F03:1A3F 30
dias[4] 0F03:1A40
0F03:1A41 31
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 38
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 946

short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
tSIPtr punt = dias; 0F03:1A3B 28
punt++; dias[2] 0F03:1A3C
31
0F03:1A3D
dias[3] 0F03:1A3E
0F03:1A3F 30
dias[4] 0F03:1A40
0F03:1A41 31
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 3A
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 947


short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
tSIPtr punt = dias; 0F03:1A3B 28
punt++; dias[2] 0F03:1A3C
0F03:1A3D 31
punt = punt + 3; dias[3] 0F03:1A3E
0F03:1A3F 30
dias[4] 0F03:1A40
0F03:1A41 31
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 40
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 948

short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
tSIPtr punt = dias; 0F03:1A3B 28
punt++; dias[2] 0F03:1A3C
0F03:1A3D 31
punt = punt + 3; dias[3] 0F03:1A3E

punt‐‐; 0F03:1A3F 30
dias[4] 0F03:1A40
0F03:1A41 31
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 3E
...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 949


short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
tSIPtr punt = dias; 0F03:1A3B 28
punt++; dias[2] 0F03:1A3C
31
0F03:1A3D
punt = punt + 3; dias[3] 0F03:1A3E

punt‐‐; 0F03:1A3F 30
dias[4] 0F03:1A40
tSIPtr punt2; 0F03:1A41 31
...
dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 3E
punt2 0F07:041F ?

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 950

short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
siPtr punt = dias; 0F03:1A3B 28
punt++; dias[2] 0F03:1A3C
0F03:1A3D 31
punt = punt + 3; dias[3] 0F03:1A3E

punt‐‐; 0F03:1A3F 30
dias[4] 0F03:1A40
tSIPtr punt2; 0F03:1A41 31
...
punt2 = dias; dias 0F07:0417 0F
0F07:0418 03
0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 3E
punt2 0F07:041F 0F

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 951


short int dias[12] = {31, 28, 31, 30,
...
31, 30, 31, 31, 30, 31, 30, 31};
dias[0] 0F03:1A38
typedef short int* tSIPtr; 0F03:1A39 31
dias[1] 0F03:1A3A
siPtr punt = dias; 28
3 0F03:1A3B

punt++; dias[2] 0F03:1A3C


0F03:1A3D 31
punt = punt + 3; dias[3] 0F03:1A3E

punt‐‐; 0F03:1A3F 30
dias[4] 0F03:1A40
tSIPtr punt2; 0F03:1A41 31
...
punt2 = dias; dias 0F07:0417 0F

cout << punt – punt2; // 3 0F07:0418 03


0F07:0419 1A
0F07:041A 38
punt 0F07:041B 0F
Luis Hernández Yáñez

0F07:041C 03
0F07:041D 1A
0F07:041E 3E
punt2 0F07:041F 0F

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 952


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 953


arraypunt.cpp

Punteros como iteradores para arrays


const int MAX = 100;
typedef int tArray[MAX];
typedef struct {
tArray elementos;
int cont;
} tLista;
typedef int* tIntPtr;
tLista lista;

Usamos un puntero como iterador para recorrer el array:


tIntPtr punt = lista.elementos;
for (int i = 0; i < lista.cont; i++) {
cout << *punt << endl;
Luis Hernández Yáñez

punt++;
}

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 954

...
intPtr punt = lista.elementos;

punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 955


...
for (int i = 0; i < lista.cont; i++) {
cout << *punt << endl;
punt++;
}

i 0 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 956

...
for (int i = 0; i < lista.cont; i++) { 4
cout << *punt << endl;
punt++;
}

i 1 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 957


...
for (int i = 0; i < lista.cont; i++) { 4
13
cout << *punt << endl;
punt++;
}

i 2 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 958

...
for (int i = 0; i < lista.cont; i++) { 4
13
cout << *punt << endl; 3
punt++;
}

i 3 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

. . .

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 959


...
for (int i = 0; i < lista.cont; i++) { 4
13
cout << *punt << endl; 3
punt++; 47
} 53
19
7

i 7 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 960

...
for (int i = 0; i < lista.cont; i++) { 4
13
cout << *punt << endl; 3
punt++; 47
} 53
19
7
48
i 8 punt

0 1 2 3 4 5 6 7 8 ... 98 99

lista.elementos 4 13 3 47 53 19 7 48

lista.cont 8
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 961


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 962

Nombres alternativos para los datos


Una referencia es una nueva forma de llamar a una variable
Nos permiten referirnos a una variable con otro identificador:
int x = 10;
int &z = x;
x y z son ahora la misma variable (comparten memoria)
Cualquier cambio en x afecta a z y cualquier cambio en z afecta a x
z = 30;
cout << x;
Las referencias se usan en el paso de parámetros por referencia
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 963


Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 964

Una implementación dinámica de listas enlazadas


Cada elemento de la lista apunta al siguiente elemento:
struct tNodo; // Declaración anticipada
typedef tNodo *tLista;
struct tNodo { reg sig
tRegistro reg; tRegistro tLista
tLista sig;
};
Una lista (tLista) es un puntero a un nodo
Si el puntero vale NULL, no apunta a ningún nodo: lista vacía
Un nodo (tNodo) es un elemento seguido de una lista
Luis Hernández Yáñez

Vacía
Lista ¡Definición recursiva!
Elemento seguido de una lista

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 965


Cada elemento de la lista en su nodo
Apuntará al siguiente elemento o a ninguno (NULL)
struct tNodo; // Declaración anticipada
typedef tNodo *tLista;
struct tNodo {
tRegistro reg;
tLista sig;
};
Además, un puntero al primer elemento (nodo) de la lista
tLista lista = NULL; // Lista vacía
Luis Hernández Yáñez

lista

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 966

struct tNodo;
typedef tNodo *tLista;
struct tNodo {
tRegistro reg;
tLista sig;
};
tLista lista = NULL; // Lista vacía
lista = new tNodo;
lista‐>reg = nuevo();
lista‐>sig = NULL;
Luis Hernández Yáñez

lista ítem1

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 967


tLista lista = NULL; // Lista vacía
lista = new tNodo;
lista‐>reg = nuevo();
lista‐>sig = NULL;
tLista p;
p = lista;

p
Luis Hernández Yáñez

lista ítem1

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 968

tLista lista = NULL; // Lista vacía
lista = new tNodo;
lista‐>reg = nuevo();
lista‐>sig = NULL;
tLista p;
p = lista;
p‐>sig = new tNodo;
p‐>sig‐>reg = nuevo();
p‐>sig‐>sig = NULL;

p
Luis Hernández Yáñez

lista ítem1 ítem2

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 969


tLista lista = NULL; // Lista vacía
lista = new tNodo;
lista‐>reg = nuevo();
lista‐>sig = NULL;
tLista p;
p = lista;
p‐>sig = new tNodo;
p‐>sig‐>reg = nuevo();
p‐>sig‐>sig = NULL;
p = p‐>sig;
p‐>sig = new tNodo;
p‐>sig‐>reg = nuevo();
p‐>sig‐>sig = NULL; p
...
Luis Hernández Yáñez

lista tRegistro tRegistro tRegistro

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 970

Usamos la memoria que necesitamos, ni más ni menos

lista tRegistro tRegistro tRegistro

Tantos elementos, tantos nodos hay en la lista


¡Pero perdemos el acceso directo!
Algunas operaciones de la lista se complican y otras no
A continuación tienes el módulo de lista implementado
como lista enlazada...
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 971


listaenlazada.h

struct tNodo;
typedef tNodo *tLista;
struct tNodo {
tRegistro reg;
tLista sig;
};

const string BD = "bd.txt";

void mostrar(tLista lista);


void insertar(tLista &lista, tRegistro registro, bool &ok);
void eliminar(tLista &lista, int code, bool &ok);
tLista buscar(tLista lista, int code); // Devuelve puntero
void cargar(tLista &lista, bool &ok);
void guardar(tLista lista);
Luis Hernández Yáñez

void destruir(tLista &lista); // Liberar la memoria dinámica

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 972

listaenlazada.cpp

void insertar(tLista &lista, tRegistro registro, bool &ok) {


ok = true;
tLista nuevo = new tNodo;
if (nuevo == NULL) {
ok = false; // No hay más memoria dinámica
}
else {
nuevo‐>reg = registro;
nuevo‐>sig = NULL;
if (lista == NULL) { // Lista vacía
lista
 lista = nuevo;
} 
nuevo
else {
tLista p = lista;
// Localizamos el último nodo...
while (p‐>sig != NULL) {
 p = p‐>sig; nuevo
Luis Hernández Yáñez

} p
p‐>sig = nuevo; 
}
lista
}
} ...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 973


void eliminar(tLista &lista, int code, bool &ok) {
ok = true;
tLista p = lista;
if (p == NULL) {
ok = false; // Lista vacía
}
else if (p‐>reg.codigo == code) { // El primero
lista = p‐>sig;
 delete p;
p
}
else { 
tLista ant = p; lista
p = p‐>sig;
bool encontrado = false;
while ((p != NULL) && !encontrado) {
if (p‐>reg.codigo == code) {
ant p
encontrado = true;
}
Luis Hernández Yáñez

else {
 ant = p;  lista
p = p‐>sig;
}
} ...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 974

if (!encontrado) {
ok = false; // No existe ese código
}
else {
ant‐>sig = p‐>sig;
delete p;
}
}
}
...
ant p

lista
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 975


tLista buscar(tLista lista, int code) {
// Devuelve un puntero al nodo, o NULL si no se encuentra
tLista p = lista;
bool encontrado = false;
while ((p != NULL) && !encontrado) {
if (p‐>reg.codigo == code) {
encontrado = true;
}
else {
p = p‐>sig;
}
}
return p;
}

void mostrar(tLista lista) {


cout << endl << "Elementos de la lista:" << endl
<< "‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐" << endl;
tLista p = lista;
Luis Hernández Yáñez

while (p != NULL) {
mostrar(p‐>reg);
p = p‐>sig;
}
} ...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 976

void cargar(tLista &lista, bool &ok) {


ifstream archivo;
char aux;
ok = true;
lista = NULL;
archivo.open(BD.c_str());
if (!archivo.is_open()) {
ok = false;
}
else {
tRegistro registro;
tLista ult = NULL;
archivo >> registro.codigo;
while (registro.codigo != ‐1) {
archivo >> registro.valor;
archivo.get(aux); // Saltamos el espacio
getline(archivo, registro.nombre);
Luis Hernández Yáñez

...

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 977


if (lista == NULL) {
lista = new tNodo;
ult = lista;
}
else {
ult‐>sig = new tNodo;
ult = ult‐>sig;
}
ult‐>reg = registro;
ult‐>sig = NULL;
archivo >> registro.codigo;
}
archivo.close();
}
return ok;
} ...
Luis Hernández Yáñez

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 978

void guardar(tLista lista) {


ofstream archivo;
archivo.open(BD);
tLista p = lista;
while (p != NULL) {
archivo << p‐>registro.codigo << " ";
archivo << p‐>registro.valor << " ";
archivo << p‐>registro.nombre << endl;
p = p‐>sig;
}
archivo.close();
}

void destruir(tLista &lista) {


tLista p;
while (lista != NULL) {
p = lista;
Luis Hernández Yáñez

lista = lista‐>sig;
delete p;
}
}

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 979


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Punteros y memoria dinámica (Anexo) Página 980


Fundamentos de la programación

10
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Concepto de recursión 983
Algoritmos recursivos 986
Funciones recursivas 987
Diseño de funciones recursivas 989
Modelo de ejecución 990
La pila del sistema 992
La pila y las llamadas a función 994
Ejecución de la función factorial() 1005
Tipos de recursión 1018
Recursión simple 1019
Recursión múltiple 1020
Recursión anidada 1022
Recursión cruzada 1026
Código del subprograma recursivo 1027
Parámetros y recursión 1032
Ejemplos de algoritmos recursivos 1034
Luis Hernández Yáñez

Búsqueda binaria 1035
Torres de Hanoi 1038
Recursión frente a iteración 1043
Estructuras de datos recursivas 1045

Fundamentos de la programación: Introducción a la recursión
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 983

Recursión (recursividad, recurrencia)


Definición recursiva: En la definición aparece lo que se define
Factorial(N) = N x Factorial(N‐1) (N >= 0)
Cada triángulo está
formado por otros
triángulos más pequeños

La cámara graba lo que graba


(http://farm1.static.flickr.com/83
/229219543_edf740535b.jpg)

(wikipedia.org)
La imagen del paquete
Luis Hernández Yáñez

aparece dentro del propio


paquete,... ¡hasta el infinito!
(wikipedia.org) Las matrioskas rusas

Fundamentos de la programación: Introducción a la recursión Página 984


Factorial(N) = N x Factorial(N‐1)
El factorial se define en función de sí mismo
Los programas no pueden manejar la recursión infinita
La definición recursiva debe adjuntar uno o más casos base
Caso base: aquel en el que no se utiliza la definición recursiva
Proporcionan puntos finales de cálculo:

N x Factorial(N‐1) si N > 0 Caso recursivo (inducción)


Factorial(N)
1 si N = 0 Caso base (o de parada)

El valor de N se va aproximando al valor del caso base (0)


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 985


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 986


Funciones recursivas
Una función puede implementar un algoritmo recursivo
La función se llamará a sí misma si no se ha llegado al caso base
1 si N = 0
Factorial(N)
N x Factorial(N‐1) si N > 0
long long int factorial(int n) {
long long int resultado;
if (n == 0) { // Caso base
resultado = 1;
}
else {
Luis Hernández Yáñez

resultado = n * factorial(n ‐ 1);


}
return resultado;
}

Fundamentos de la programación: Introducción a la recursión Página 987

factorial.cpp

Funciones recursivas
long long int factorial(int n) {
long long int resultado;
if (n == 0) { // Caso base
resultado = 1;
}
else {
resultado = n * factorial(n ‐ 1);
}
return resultado;
}
factorial(5)  5 x factorial(4)  5 x 4 x factorial(3)
 5 x 4 x 3 x factorial(2)  5 x 4 x 3 x 2 x factorial(1)
Luis Hernández Yáñez

 5 x 4 x 3 x 2 x 1 x factorial(0)  5 x 4 x 3 x 2 x 1 x 1
 120 Caso base

Fundamentos de la programación: Introducción a la recursión Página 988


Diseño de funciones recursivas
Una función recursiva debe satisfacer tres condiciones:
 Caso(s) base: Debe haber al menos un caso base de parada
 Inducción: Paso recursivo que provoca una llamada recursiva
Debe ser correcto para distintos parámetros de entrada
 Convergencia: Cada paso recursivo debe acercar a un caso base
Se describe el problema en términos de problemas más sencillos
1 si N = 0
Factorial(N)
N x Factorial(N‐1) si N > 0
Luis Hernández Yáñez

Función factorial(): tiene caso base (N = 0), siendo correcta


para N es correcta para N+1 (inducción) y se acerca cada vez
más al caso base (N‐1 está más cerca de 0 que N)

Fundamentos de la programación: Introducción a la recursión Página 989


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 990


long long int factorial(int n) {
long long int resultado;
if (n == 0) { // Caso base
resultado = 1;
}
else {
resultado = n * factorial(n ‐ 1);
}
return resultado;
}
Cada llamada recursiva fuerza una nueva ejecución de la función
Cada llamada utiliza sus propios parámetros por valor
y variables locales (n y resultado en este caso)
En las llamadas a la función se utiliza la pila del sistema para
Luis Hernández Yáñez

mantener los datos locales y la dirección de vuelta

Fundamentos de la programación: Introducción a la recursión Página 991

Regiones de memoria que distingue el sistema operativo:

Pila (Stack) Llamadas a subprogramas

Montón (Heap) Memoria dinámica (Tema 9)

Datos del programa

Código del programa Memoria principal


Luis Hernández Yáñez

S.O.

Fundamentos de la programación: Introducción a la recursión Página 992


Mantiene los datos locales de la función y la dirección de vuelta
Estructura de tipo pila: lista LIFO (last‐in first‐out)
El último que entra es el primero que sale:

Entra Entra Entra Sale


4 7 2 2

2
7 7 7
Luis Hernández Yáñez

4 4 4 4

Fundamentos de la programación: Introducción a la recursión Página 993

Datos locales y direcciones de vuelta


...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a);
...   
return b;
}
int main() {
Luis Hernández Yáñez

... Pila
Llamada a función:
<DIR1> cout << funcA(4);
Entra la dirección de vuelta
...

Fundamentos de la programación: Introducción a la recursión Página 994


Datos locales y direcciones de vuelta
...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
Entrada en la función:
int b; Se alojan los datos locales
...
<DIR2> b = funcB(a);
...   
return b;
}
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 995

Datos locales y direcciones de vuelta


...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a); Llamada a función:
Entra la dirección de vuelta
...   
return b; b
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 996


Datos locales y direcciones de vuelta
...
int funcB(int x) { Entrada en la función:
... Se alojan los datos locales
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a);
...    <DIR2>
return b; b
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 997

Datos locales y direcciones de vuelta


...
int funcB(int x) {
...
return x; Vuelta de la función:
Se eliminan los datos locales
}
int funcA(int a) {
int b;
... x
<DIR2> b = funcB(a);
...    <DIR2>
return b; b
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 998


Datos locales y direcciones de vuelta
...
int funcB(int x) {
...
return x; Vuelta de la función:
} Sale la dirección de vuelta

int funcA(int a) {


int b;
...
<DIR2> b = funcB(a);
...    <DIR2>
return b; b
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 999

Datos locales y direcciones de vuelta


...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a); La ejecución continúa
en esa dirección
...   
return b; b
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 1000


Datos locales y direcciones de vuelta
...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a);
...   
return b; Vuelta de la función: b
Se eliminan los datos locales
} a
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 1001

Datos locales y direcciones de vuelta


...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a);
...   
return b; Vuelta de la función:
Sale la dirección de vuelta
}
int main() { <DIR1>
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4);
...

Fundamentos de la programación: Introducción a la recursión Página 1002


Datos locales y direcciones de vuelta
...
int funcB(int x) {
...
return x;
}
int funcA(int a) {
int b;
...
<DIR2> b = funcB(a);
...   
return b;
}
int main() {
Luis Hernández Yáñez

... Pila
<DIR1> cout << funcA(4); La ejecución continúa
en esa dirección
...

Fundamentos de la programación: Introducción a la recursión Página 1003

Mecanismo de pila adecuado para llamadas a funciones anidadas:


Las llamadas terminan en el orden contrario a como se llaman
...
int funcC(...) {
...
}
int funcB(...) {
...
LLAMADAS
V U E LTAS

... funcC(...)
}
int funcA(...) { funcC
...
... funcB(...) funcB
}
int main() { funcA
Luis Hernández Yáñez

...
cout << funcA(...); Pila
...

Fundamentos de la programación: Introducción a la recursión Página 1004


long long int factorial(int n) {
long long int resultado;
if (n == 0) { // Caso base
resultado = 1;
}
else {
resultado = n * factorial(n ‐ 1);
}
return resultado;
}

cout << factorial(5) << endl;
Luis Hernández Yáñez

Obviaremos las direcciones de vuelta en la pila

Fundamentos de la programación: Introducción a la recursión Página 1005

factorial(5)

resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1006


factorial(5)
factorial(4)

resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1007

factorial(5)
factorial(4)
factorial(3)

resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1008


factorial(5)
factorial(4)
factorial(3)
factorial(2)

resultado = ?
n = 2
resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1009

factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
resultado = ?
n = 1
resultado = ?
n = 2
resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1010


factorial(5)
factorial(4)
factorial(3)
factorial(2) resultado = 1
factorial(1) n = 0

factorial(0) resultado = ?
n = 1
resultado = ?
n = 2
resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1011

factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0) resultado = 1
n = 1
resultado = ?
1
n = 2
resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1012


factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0)

resultado = 2
1
n = 2
1 resultado = ?
n = 3
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1013

factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0)

1
1 resultado = 6
n = 3
2
resultado = ?
n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1014


factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0)

1
1
2
resultado = 24
6 n = 4
resultado = ?
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1015

factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0)

1
1
2
6
24 resultado = 120
Luis Hernández Yáñez

n = 5

Pila

Fundamentos de la programación: Introducción a la recursión Página 1016


factorial(5)
factorial(4)
factorial(3)
factorial(2)
factorial(1)
factorial(0)

1
1
2
6
24
Luis Hernández Yáñez

120
Pila

Fundamentos de la programación: Introducción a la recursión Página 1017


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1018


Sólo hay una llamada recursiva
Ejemplo: Cálculo del factorial de un número entero positivo
long long int factorial(int n) {
long long int resultado;
if (n == 0) { // Caso base
resultado = 1;
}
else {
resultado = n * factorial(n ‐ 1);
}
return resultado;
}
Una sola llamada recursiva
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1019

Varias llamadas recursivas


Ejemplo: Cálculo de los números de Fibonacci

0 si n = 0

Fib(n) 1 si n = 1
Fib(n‐1) + Fib(n‐2) si n > 1

Dos llamadas recursivas


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1020


fibonacci.cpp

... 0 si n = 0
int main() { Fib(n) 1 si n = 1
for (int i = 0; i < 20; i++) { Fib(n‐1) + Fib(n‐2) si n > 1
cout << fibonacci(i) << endl;
}
return 0;
}

int fibonacci(int n) {


int resultado;
if (n == 0) {
resultado = 0;
}
else if (n == 1) {
resultado = 1;
}
else {
Luis Hernández Yáñez

resultado = fibonacci(n ‐ 1) + fibonacci(n ‐ 2);


}
return resultado;
}

Fundamentos de la programación: Introducción a la recursión Página 1021

En una llamada recursiva alguno de los argumentos es otra llamada


Ejemplo: Cálculo de los números de Ackermann:

n + 1 si m = 0

Ack(m, n) Ack(m‐1, 1) si m > 0 y n = 0


Ack(m‐1, Ack(m, n‐1)) si m > 0 y n > 0

Argumento que es una llamada recursiva


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1022


ackermann.cpp

n + 1 si m = 0
Números de Ackermann Ack(m, n) Ack(m‐1, 1) si m > 0 y n = 0

... Ack(m‐1, Ack(m, n‐1)) si m > 0 y n > 0


int ackermann(int m, int n) {
int resultado;
if (m == 0) {
resultado = n + 1;
}
else if (n == 0) {
resultado = ackermann(m ‐ 1, 1);
}
else {
resultado = ackermann(m ‐ 1, ackermann(m, n ‐ 1));
}
return resultado;
Luis Hernández Yáñez

}
Pruébalo con números muy bajos:
Se generan MUCHAS llamadas recursivas

Fundamentos de la programación: Introducción a la recursión Página 1023

n + 1 si m = 0
Números de Ackermann Ack(m, n) Ack(m‐1, 1) si m > 0 y n = 0
Ack(m‐1, Ack(m, n‐1)) si m > 0 y n > 0
ackermann(1, 1)
ackermann(0, ackermann(1, 0))
2
3 ackermann(0, 1)
ackermann(0, 2)
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1024


n + 1 si m = 0
Números de Ackermann Ack(m, n) Ack(m‐1, 1) si m > 0 y n = 0
Ack(m‐1, Ack(m, n‐1)) si m > 0 y n > 0
ackermann(2, 1)
ackermann(1, ackermann(2, 0))
3
5 ackermann(1, 1)
ackermann(0, ackermann(1, 0))

2
ackermann(0, 1)
3
ackermann(0, 2)
ackermann(1, 3)
ackermann(0, ackermann(1, 2))
ackermann(0, ackermann(1, 1))
4 ackermann(0, ackermann(1, 0))
5
Luis Hernández Yáñez

2
ackermann(0, 1)
3
ackermann(0, 2)
ackermann(0, 3)
ackermann(0, 4)

Fundamentos de la programación: Introducción a la recursión Página 1025


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1026


Código anterior y posterior a la llamada recursiva
{
Código anterior
Llamada recursiva
Código posterior
}
Código anterior
Se ejecuta para las distintas entradas antes que el código posterior
Código posterior
Se ejecuta para las distintas entradas tras llegarse al caso base
El código anterior se ejecuta en orden directo para las distintas
Luis Hernández Yáñez

entradas, mientras que el código posterior lo hace en orden inverso


Si no hay código anterior: recursión por delante
Si no hay código posterior: recursión por detrás

Fundamentos de la programación: Introducción a la recursión Página 1027

Código anterior y posterior a la llamada recursiva


void func(int n) {
if (n > 0) { // Caso base: n == 0
cout << "Entrando (" << n << ")" << endl; // Código anterior
func(n ‐ 1);                              // Llamada recursiva
cout << "Saliendo (" << n << ")" << endl; // Código posterior
}
}

 func(5);

El código anterior se ejecuta


para los sucesivos valores de n (5, 4, 3, ...)
Luis Hernández Yáñez

El código posterior al revés (1, 2, 3, ...)

Fundamentos de la programación: Introducción a la recursión Página 1028


directo.cpp

Recorrido de los elementos de una lista (directo)


El código anterior a la llamada procesa la lista en su orden:
...
void mostrar(tLista lista, int pos);

int main() {
tLista lista;
lista.cont = 0;
// Carga del array...
mostrar(lista, 0);

return 0;
}

void mostrar(tLista lista, int pos) {


if (pos < lista.cont) {
Luis Hernández Yáñez

cout << lista.elementos[pos] << endl;
mostrar(lista, pos + 1);
}
}

Fundamentos de la programación: Introducción a la recursión Página 1029

inverso.cpp

Recorrido de los elementos de una lista (inverso)


El código posterior procesa la lista en el orden inverso:
...
void mostrar(tLista lista, int pos);

int main() {
tLista lista;
lista.cont = 0;
// Carga del array...
mostrar(lista, 0);

return 0;
}

void mostrar(tLista lista, int pos) {


if (pos < lista.cont) {
Luis Hernández Yáñez

mostrar(lista, pos + 1);
cout << lista.elementos[pos] << endl;
}
}

Fundamentos de la programación: Introducción a la recursión Página 1030


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1031

Parámetros por valor y por referencia


Parámetros por valor: cada llamada usa los suyos propios
Parámetros por referencia: misma variable en todas las llamadas
Recogen resultados que transmiten entre las llamadas
void factorial(int n, int &fact) {
if (n == 0) {
fact = 1;
}
else {
factorial(n ‐ 1, fact);
fact = n * fact;
}
Luis Hernández Yáñez

}
Cuando n es 0, el argumento de fact toma el valor 1
Al volver se le va multiplicando por los demás n (distintos)
Fundamentos de la programación: Introducción a la recursión Página 1032
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1033

Parte el problema en subproblemas más pequeños


Aplica el mismo proceso a cada subproblema
Naturaleza recursiva (casos base: encontrado o no queda lista)
Partimos de la lista completa
Si no queda lista... terminar (lista vacía: no encontrado)
En caso contrario...
Comprobar si el elemento en la mitad es el buscado
Si es el buscado... terminar (encontrado)
Si no...
Si el buscado es menor que el elemento mitad...
Repetir con la primera mitad de la lista
Si el buscado es mayor que el elemento mitad...
Luis Hernández Yáñez

Repetir con la segunda mitad de la lista


 La repetición se consigue con las llamadas recursivas
Fundamentos de la programación: Introducción a la recursión Página 1034
Dos índices que indiquen el inicio y el final de la sublista:
int buscar(tLista lista, int buscado, int ini, int fin)
// Devuelve el índice (0, 1, ...) o ‐1 si no está

¿Cuáles son los casos base?


 Que ya no quede sublista (ini > fin)  No encontrado
 Que se encuentre el elemento

Repasa en el Tema 7 cómo funciona y cómo se implementó


iterativamente la búsqueda binaria (compárala con esta)
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1035

binaria.cpp

int buscar(tLista lista, int buscado, int ini, int fin) {


int pos = ‐1;
if (ini <= fin) {
int mitad = (ini + fin) / 2;
if (buscado == lista.elementos[mitad]) {
pos = mitad;
}
else if (buscado < lista.elementos[mitad]) {
pos = buscar(lista, buscado, ini, mitad ‐ 1);
}
else {
pos = buscar(lista, buscado, mitad + 1, fin);
}
}
return pos;
Luis Hernández Yáñez

Llamada: pos = buscar(lista, valor, 0, lista.cont ‐ 1);

Fundamentos de la programación: Introducción a la recursión Página 1036


Cuenta una leyenda que en un templo de Hanoi se dispusieron tres
pilares de diamante y en uno de ellos 64 discos de oro, de distintos
tamaños y colocados por orden de tamaño con el mayor debajo

Torre de ocho discos (wikipedia.org)

Cada monje, en su turno, debía mover un único disco de un pilar


a otro, para con el tiempo conseguir entre todos llevar la torre del
pilar inicial a uno de los otros dos; respetando una única regla:
nunca poner un disco sobre otro de menor tamaño
Luis Hernández Yáñez

Cuando lo hayan conseguido, ¡se acabará el mundo!


[Se requieren al menos 264‐1 movimientos; si se hiciera uno por segundo,
se terminaría en más de 500 mil millones de años]

Fundamentos de la programación: Introducción a la recursión Página 1037

Queremos resolver el juego en el menor número de pasos posible


¿Qué disco hay que mover en cada paso y a dónde?
Identifiquemos los elementos (torre de cuatro discos):

A B C

Cada pilar se identifica con una letra


Mover del pilar X al pilar Y:
Coger el disco superior de X y ponerlo encima de los que haya en Y
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1038


Resolución del problema en base Mover 4 discos de A a C

a problemas más pequeños


Mover N discos del pilar A al pilar C:
Mover N‐1 discos del pilar A al pilar B A B C

Mover el disco del pilar A al pilar C


Mover N‐1 discos del pilar B al pilar C
A B C

Para llevar N discos de un pilar origen a


otro destino se usa el tercero como auxiliar A B C
Mover N‐1 discos del origen al auxiliar
Mover el disco del origen al destino
Luis Hernández Yáñez

Mover N‐1 discos del auxiliar al destino A B C

Fundamentos de la programación: Introducción a la recursión Página 1039

Mover N‐1 discos se hace igual, pero Mover 3 discos de A a B

usando ahora otros origen y destino


Mover N‐1 discos del pilar A al pilar B:
Mover N‐2 discos del pilar A al pilar C A B C

Mover el disco del pilar A al pilar B


Mover N‐2 discos del pilar C al pilar B
A B C
Naturaleza recursiva de la solución

A B C
Luis Hernández Yáñez

Simulación para 4 discos (wikipedia.org)


A B C

Fundamentos de la programación: Introducción a la recursión Página 1040


hanoi.cpp

Caso base: no quedan discos que mover


...
void hanoi(int n, char origen, char destino, char auxiliar) {
if (n > 0) {
hanoi(n ‐ 1, origen, auxiliar, destino);
cout << origen << " ‐‐> " << destino << endl;
hanoi(n ‐ 1, auxiliar, destino, origen);
}
}

int main() {
int n;
cout << "Número de torres: ";
cin >> n;
hanoi(n, 'A', 'C', 'B');
Luis Hernández Yáñez

return 0;
}

Fundamentos de la programación: Introducción a la recursión Página 1041


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1042


long long int factorial(int n) { long long int factorial(int n) {
long long int fact; long long int fact = 1;

assert(n >= 0); assert(n >= 0);

if (n == 0) { for (int i = 1; i <= n; i++) {


fact = 1; fact = fact * i;
} }
else {
fact = n * factorial(n ‐ 1); return fact;
} }

return fact;
}
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1043

¿Qué es preferible?
Cualquier algoritmo recursivo tiene uno iterativo equivalente
Los recursivos son menos eficientes que los iterativos:
Sobrecarga de las llamadas a subprograma
Si hay una versión iterativa sencilla, será preferible a la recursiva
En ocasiones la versión recursiva es mucho más simple
Será preferible si no hay requisitos de rendimiento
Compara las versiones recursivas del factorial o de los números
de Fibonacci con sus equivalentes iterativas
¿Y qué tal una versión iterativa para los números de Ackermann?
Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1044


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1045

Definición recursiva de listas


Ya hemos definido de forma recursiva alguna estructura de datos:
elemento seguido de una secuencia
Secuencia
secuencia vacía (ningún elemento)
Las listas son secuencias:
elemento seguido de una lista
Lista
lista vacía (ningún elemento) (Caso base)

La lista 1, 2, 3 consiste en el elemento 1 seguido de la lista 2, 3, que,


a su vez, consiste en el elemento 2 seguido de la lista 3, que, a su vez,
Luis Hernández Yáñez

consiste en el elemento 3 seguido de la lista vacía (caso base)


Hay otras estructuras con naturaleza recursiva (p.e., los árboles)
que estudiarás en posteriores cursos
Fundamentos de la programación: Introducción a la recursión Página 1046
Procesamiento de estructuras de datos recursivas
Naturaleza recursiva de las estructuras: procesamiento recursivo
Procesar (lista):
Si lista no vacía (caso base):
Procesar el primer elemento de la lista // Código anterior
Procesar (resto(lista))
Procesar el primer elemento de la lista // Código posterior

resto(lista): sublista tras quitar el primer elemento


Luis Hernández Yáñez

Fundamentos de la programación: Introducción a la recursión Página 1047

Licencia CC (Creative Commons)


Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de la programación: Introducción a la recursión Página 1048


Fundamentos de la programación

AP
Grado en Ingeniería Informática
Grado en Ingeniería del Software
Grado en Ingeniería de Computadores

Luis Hernández Yáñez


Facultad de Informática
Universidad Complutense

Flujos 1051
Archivos binarios 1054
Tamaño de los datos: El operador sizeof() 1056
Apertura de archivos binarios 1059
Lectura de archivos binarios (acceso secuencial) 1061
Escritura en archivos binarios (acceso secuencial) 1066
Acceso directo o aleatorio 1070
Ejemplos de uso de archivos binarios 1078
Ordenación de los registros del archivo 1079
Búsqueda binaria 1085
Inserción en un archivo binario ordenado 1088
Carga de los registro de un archivo en una tabla 1092
Almacenamiento de una tabla en un archivo 1093
Luis Hernández Yáñez

Fundamentos de la programación: Archivos binarios
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1051

Flujos
Canalizan la E/S entre los dispositivos y el programa
En forma de secuencias de caracteres
La entrada puede proceder de un dispositivo o de un archivo
La salida puede dirigirse a un dispositivo o a un archivo
Siempre por medio de flujos
Dispositivos/archivos
Dispositivos/archivos de salida
de entrada

Programa
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1052


Flujos de texto y binarios
 Flujo de texto: contiene una secuencia de caracteres
T o t a l : 1 2 3 . 4  A …

 Flujo binario: contiene una secuencia de códigos binarios.


A0 25 2F 04 D6 FF 00 27 6C CA 49 07 5F A4 …
(Códigos representados en notación hexadecimal.)

Lo que signifiquen los códigos dependerá del programa que use el archivo

En ambos casos se trata de una secuencia de caracteres


En el segundo caso se interpretan como códigos binarios
Sin contemplar caracteres especiales como \n o \t
Luis Hernández Yáñez

Ya hemos usado flujos de texto para E/S por consola/archivos

Fundamentos de programación: Archivos binarios Página 1053


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1054


Codificación textual y binaria
Datos numéricos: se pueden guardar en forma textual o binaria
int dato = 124567894;
Representación como texto: caracteres '1' '2' '4' '5' '6' ...
Flujo de texto 1 2 4 5 6 7 8 9 4

9 caracteres (se guarda el código ASCII de cada uno)


Representación binaria:
00000111 01101100 11000001 01010110   Hex: 07 6C C1 56

Flujo binario 07 6C C1 56
Luis Hernández Yáñez

4 caracteres interpretados como códigos binarios

Fundamentos de programación: Archivos binarios Página 1055

El operador sizeof()
En los archivos binarios se manejan códigos binarios (bytes)
sizeof() (palabra clave): bytes que ocupa en memoria algo
Se aplica a un dato o a un tipo char  byte
const int Max = 80;
typedef char tCadena[Max];
typedef struct {
int codigo;
tCadena item;
double valor;
} tRegistro;
Luis Hernández Yáñez

const int SIZE = sizeof(tRegistro);


En un archivo binario un dato del tipo tRegistro
ocupará exactamente SIZE caracteres

Fundamentos de programación: Archivos binarios Página 1056


typedef struct {
MEMORIA
int cod;
...
double val;
reg 0F03:1A38 00
} tRegistro; 0F03:1A39 00
tRegistro reg; 0F03:1A3A 00
reg.cod
const int SIZE = sizeof(reg); (4)
0F03:1A3B 05

... 0F03:1A3C 0A
0F03:1A3D 37
Posiciones de memoria usadas  SIZE
0F03:1A3E 1C
(12)
0F03:1A3F DF
reg.val
0F03:1A40 03
(8)
0F03:1A41 92

Se guardan los SIZE bytes: 0F03:1A42 99


0F03:1A43 0E
Luis Hernández Yáñez

0F03:1A44 ...
Flujo binario
00 00 00 05 0A 37 1C DF 03 92 99 0E

Fundamentos de programación: Archivos binarios Página 1057

Por eficiencia, algunos campos de una estructura se pueden


forzar a ocupar un múltiplo del tamaño de palabra del sistema
Tamaño de palabra (4, 8, 16, ... bytes): dato más pequeño que se
lee de la memoria (aunque se usen sólo algunos de los bytes)
Así, el tamaño real de las estructuras puede ser mayor que la
simple suma de los tamaños de cada tipo
Por ejemplo:
typedef struct {
char c;
int i;
} tRegistro;
const int SIZE = sizeof(tRegistro);
Luis Hernández Yáñez

char (1 byte) + int (4 bytes) SIZE toma el valor 8 (4 + 4), no 5


char + int + double  24 bytes (8 + 8 + 8)
NOTA: El tamaño de palabra y los tamaños de los tipos dependen del sistema concreto

Fundamentos de programación: Archivos binarios Página 1058


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1059

Biblioteca fstream

Archivos binarios: tipo fstream


Apertura: función open(nombre, modo)
Nombre: char[] (función c_str() para cadenas de tipo string)
Modos de apertura del archivo:
Modo Significado
ios::app Añadir: permite seguir escribiendo a partir del final
ios::binary Binario: tratar el archivo como archivo binario
ios::in Entrada: archivo para leer de él
ios::out Salida: archivo para escribir en él
ios::trunc Truncar: borrar todo lo que haya y empezar de nuevo
Luis Hernández Yáñez

Concatenación de modos: operador | (O binaria: suma bit a bit)


archivo.open("entrada.dat", ios::in | ios::binary);

Fundamentos de programación: Archivos binarios Página 1060


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1061

archivo.read(puntero_al_búfer, número)
búfer: variable destino de los caracteres leídos
Pasado como puntero a secuencia de caracteres
Referencia (&) a la variable destino
Molde de puntero a carácter (char *)
número: cantidad de caracteres a extraer del archivo
 Operador sizeof()
Archivo abierto con los modos ios::in e ios::binary
archivo.read( (char *) &registro, sizeof(tRegistro));
Luis Hernández Yáñez

Los caracteres leídos se interpretan como códigos binarios

Fundamentos de programación: Archivos binarios Página 1062


Éxito o fallo de la lectura
Función gcount()
Nº de caracteres realmente leídos en la última operación
Si coincide con el número que se solicitaron leer: OK
Si son menos, se ha alcanzado el final del archivo: Fallo
tRegistro registro;
fstream archivo;
archivo.open("entrada.dat", ios::in | ios::binary);
archivo.read( (char *) &registro, sizeof(tRegistro));
if (archivo.gcount() < sizeof(tRegistro)) {
// Fallo en la lectura
}
Luis Hernández Yáñez

else {
// Lectura OK
...

Fundamentos de programación: Archivos binarios Página 1063

leer.cpp

#include <iostream>
using namespace std;
#include <fstream>
#include "registro.h"

int main() {
tRegistro registro;
fstream archivo;
archivo.open("bd.dat", ios::in | ios::binary);
archivo.read( (char *) &registro, SIZE);
int cuantos = archivo.gcount();
while (cuantos == SIZE) {
mostrar(registro);
archivo.read( (char *) &registro, SIZE);
cuantos = archivo.gcount();
Luis Hernández Yáñez

}
archivo.close(); ¡No olvides cerrar el archivo!
return 0;
}

Fundamentos de programación: Archivos binarios Página 1064


registro.h

El tipo tRegistro
const int Max = 80;
typedef char tCadena[Max];
typedef struct {
int codigo;
tCadena item;
double valor;
} tRegistro;
const int SIZE = sizeof(tRegistro);
¿Por qué usamos cadenas al estilo de C?
string: tamaño variable en memoria
Requieren un proceso de serialización
Luis Hernández Yáñez

Las cadenas al estilo de C siempre ocupan lo mismo en memoria

Fundamentos de programación: Archivos binarios Página 1065


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1066


archivo.write(puntero_al_búfer, número)
búfer: origen de los caracteres a escribir en el archivo
Pasado como puntero a secuencia de caracteres
Referencia (&) a la variable destino
Molde de puntero a carácter (char *)
número: cantidad de caracteres a escribir en el archivo
 Operador sizeof()
Archivo abierto con los modos ios::out e ios::binary
archivo.write( (char *) &registro, sizeof(tRegistro));
Luis Hernández Yáñez

Se escriben tantos caracteres como celdas de memoria ocupe


la variable registro

Fundamentos de programación: Archivos binarios Página 1067

escribir.cpp

#include <iostream>
using namespace std;
#include <fstream>
#include "registro.h"

int main() {
tRegistro registro;
fstream archivo;
archivo.open("bd2.dat", ios::out | ios::binary);
bool seguir = true;
while (seguir) {
cout << "Código: ";
cin.sync();
cin >> registro.codigo;
cout << "Nombre: "; 
Luis Hernández Yáñez

cin.sync();
cin.getline(registro.item, Max); // Máx: 80
...    

Fundamentos de programación: Archivos binarios Página 1068


cout << "Precio: "; 
cin.sync();
cin >> registro.valor; 
archivo.write( (char *) &registro, SIZE);
cout << "Otro [S/N]? ";
char c; 
cin >> c;
if ((c == 'n') || (c == 'N')) {
seguir = false;
}
}
archivo.close(); ¡No olvides cerrar el archivo!
(¡pérdida de datos!)
return 0;
}
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1069


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1070


Acceso secuencial: empezando en el primero pasando a siguiente
Acceso directo (también llamado aleatorio):
Para localizar registros individuales necesitamos otras rutinas:
 tellg(): lugar donde se encuentra el puntero del archivo
Siguiente posición donde se realizará una lectura o escritura
 seekg(desplazamiento, origen):
Lleva el puntero del archivo a una posición concreta:
desplazamiento caracteres desde el origen indicado
Origen:
ios::beg: principio del archivo
ios::cur: posición actual
Luis Hernández Yáñez

ios::end: final del archivo

Fundamentos de programación: Archivos binarios Página 1071

ios::beg ios::end
0 SIZE 2*SIZE 3*SIZE 4*SIZE 5*SIZE 6*SIZE
SIZE SIZE SIZE SIZE SIZE SIZE

tRegistro tRegistro tRegistro tRegistro tRegistro tRegistro

const int SIZE = sizeof(tRegistro);

Cada registro ocupa SIZE caracteres en el archivo


¿Cuántos registros hay en el archivo?
archivo.seekg(0, ios::end); // 0 car. desde el final ‐> final
int pos = archivo.tellg(); // Total de caracteres del archivo
Luis Hernández Yáñez

int numReg = pos / SIZE;

Fundamentos de programación: Archivos binarios Página 1072


ios::beg ios::end
0 SIZE 2*SIZE 3*SIZE 4*SIZE 5*SIZE 6*SIZE
SIZE SIZE SIZE SIZE SIZE SIZE

tRegistro tRegistro tRegistro tRegistro tRegistro tRegistro

const int SIZE = sizeof(tRegistro);

Poner el puntero del archivo en un nº de registro:


archivo.seekg((num ‐ 1) * SIZE, ios::beg);
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1073

Lecturas y escrituras
Una vez ubicado el puntero al principio de un registro,
se puede leer el registro o actualizar (escribir) el registro
Si se ubica al final, se puede añadir (escribir) un nuevo registro

Archivos binarios de lectura/escritura:


Se han de abrir con los modos ios::in, ios::out e ios::binary
archivo.open("bd.dat", ios::in | ios::out | ios::binary);
Ahora podemos tanto leer como escribir
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1074


actualizar.cpp

// Actualización de un registro
#include <iostream>
using namespace std;
#include <fstream>
#include "registro.h"

int main() {
tRegistro registro;
fstream archivo;

archivo.open("bd.dat", ios::in | ios::out | ios::binary);


archivo.seekg(0, ios::end);
int pos = archivo.tellg();
int numReg = pos / SIZE;
cout << "Número de registros: " << numReg << endl;
Luis Hernández Yáñez

int num;
cout << "Registro número? ";
cin >> num;
...

Fundamentos de programación: Archivos binarios Página 1075

if ((num > 0) && (num <= numReg)) {
archivo.seekg((num ‐ 1) * SIZE, ios::beg);
archivo.read( (char *) &registro, SIZE);
mostrar(registro);
cout << endl << "Cambiar nombre [S/N]? ";
char c;
cin.sync();
cin >> c;
if ((c == 's') || (c == 'S')) {
cout << "Nombre: ";
cin.sync();
cin.getline(registro.item, 80);
}
cout << endl << "Cambiar precio [S/N]? ";
cin.sync();
cin >> c;
Luis Hernández Yáñez

if ((c == 's') || (c == 'S')) {
cout << "Precio: ";
cin >> registro.valor;
}
...
Fundamentos de programación: Archivos binarios Página 1076
archivo.seekg((num ‐ 1) * SIZE, ios::beg);
archivo.write( (char *) &registro, SIZE);
cout << endl << "Registro actualizado:" << endl;
archivo.seekg((num ‐ 1) * SIZE, ios::beg);
archivo.read( (char *) &registro, SIZE);
mostrar(registro);
}
archivo.close();
return 0;
}
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1077


Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1078


ordenar.cpp

Mediante un acceso directo a los registros del archivo


Ordenaremos por el campo item
#include <iostream>
using namespace std;
#include <fstream>
#include <iomanip>
#include <cstring>
#include "registro.h"

const char BD[] = "lista.dat";

void mostrar();

...
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1079

void mostrar() {
fstream archivo;
tRegistro registro;  
int cuantos;

archivo.open(BD, ios::in | ios::binary);
archivo.read( (char *) &registro, SIZE);
cuantos = archivo.gcount();
while (cuantos == SIZE) {
mostrar(registro);
archivo.read( (char *) &registro, SIZE);
cuantos = archivo.gcount();
}
archivo.close();
}
Luis Hernández Yáñez

...

Fundamentos de programación: Archivos binarios Página 1080


int main() {

mostrar(); Orden inicial

fstream archivo;
archivo.open(BD, ios::in | ios::out | ios::binary);
archivo.seekg(0, ios::end);
Luis Hernández Yáñez

int pos = archivo.tellg();
int numReg = pos / SIZE;
...

Fundamentos de programación: Archivos binarios Página 1081

// Ordenamos con el método de selección directa
tRegistro regMenor, reg;
for (int i = 0; i < numReg ‐ 1; i++) {
int menor = i;
for (int j = i + 1; j < numReg; j++) {
archivo.seekg(menor * SIZE, ios::beg);
archivo.read( (char *) &regMenor, SIZE);
archivo.seekg(j * SIZE, ios::beg);
archivo.read( (char *) &reg, SIZE);
if (strcmp(reg.item, regMenor.item) < 0) {
menor = j;
}
}
... regMenor reg

menor
i j
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1082


if (menor > i) { // Intercambiamos
archivo.seekg(i * SIZE, ios::beg);
archivo.read( (char *) &reg, SIZE); 
archivo.seekg(menor * SIZE, ios::beg);
archivo.read( (char *) &regMenor, SIZE); 
archivo.seekg(i * SIZE, ios::beg);
archivo.write( (char *) &regMenor, SIZE); 
archivo.seekg(menor * SIZE, ios::beg);
archivo.write( (char *) &reg, SIZE); 
}
}
... reg regMenor

i
  menor j

Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1083

archivo.close();

cout << endl << "Tras ordenar:" << endl << endl;
mostrar();

return 0;
}
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1084


buscar.cpp

Archivo binario ordenado; por código


#include <iostream>
using namespace std;
#include <fstream>
#include "registro.h"

const char BD[] = "ord.dat";

void mostrar();

int main() {
mostrar();
tRegistro registro;
fstream archivo;
Luis Hernández Yáñez

...

Fundamentos de programación: Archivos binarios Página 1085

archivo.open(BD, ios::in | ios::binary);
archivo.seekg(0, ios::end);
int pos = archivo.tellg();
int numReg = pos / SIZE;
int buscado;
cout << "Código a buscar: ";
cin >> buscado;
int ini = 0, fin = numReg ‐ 1, mitad;
bool encontrado = false;
while ((ini <= fin) && !encontrado) {
mitad = (ini + fin) / 2;
archivo.seekg(mitad * SIZE, ios::beg);
archivo.read( (char *) &registro, SIZE);
if (buscado == registro.codigo) {
encontrado = true;
}
Luis Hernández Yáñez

else if (buscado < registro.codigo) {
fin = mitad ‐ 1;
}
...

Fundamentos de programación: Archivos binarios Página 1086


else {
ini = mitad + 1;
}
}
if (encontrado) {
int pos = mitad + 1;
cout << "Encontrado en la posición " << pos << endl;
mostrar(registro);
}
else {
cout << "No encontrado!" << endl;
}
archivo.close();

return 0;
Luis Hernández Yáñez

...

Fundamentos de programación: Archivos binarios Página 1087

insertar.cpp

Ordenado por el campo codigo


#include <iostream>
using namespace std;
#include <fstream>
#include "registro.h"

const char BD[] = "ord2.dat";

void mostrar();

int main() {
mostrar();
tRegistro nuevoRegistro = nuevo(), registro;
fstream archivo;
archivo.open(BD, ios::in | ios::out | ios::binary);
Luis Hernández Yáñez

archivo.seekg(0, ios::end);
int pos = archivo.tellg();
int numReg = pos / SIZE;
...
Fundamentos de programación: Archivos binarios Página 1088
pos = 0;
bool encontrado = false;
archivo.seekg(0, ios::beg);
while ((pos < numReg) && !encontrado) {
archivo.read( (char *) &registro, SIZE);
if (registro.codigo > nuevoRegistro.codigo) {
encontrado = true;
}
else {
pos++;
}
}
if (pos == numReg) { // Debe ir al final
archivo.seekg(0, ios::end);
archivo.write( (char *) &nuevoRegistro, SIZE);
Luis Hernández Yáñez

}
...

Fundamentos de programación: Archivos binarios Página 1089

else { // Hay que hacer hueco
for (int i = numReg ‐ 1; i >= pos; i‐‐) {
archivo.seekg(i * SIZE, ios::beg);
archivo.read( (char *) &registro, SIZE);
archivo.seekg((i + 1) * SIZE, ios::beg);
archivo.write( (char *) &registro, SIZE);
}
archivo.seekg(pos * SIZE, ios::beg);
archivo.write( (char *) &nuevoRegistro, SIZE);
}
archivo.close();

mostrar(); nuevoRegistro

return 0;    
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1090


Al principio

Por el medio
Luis Hernández Yáñez

Al final

Fundamentos de programación: Archivos binarios Página 1091

tabla.cpp

void cargar(tTabla &tabla, bool &ok) {


ok = true;
fstream archivo;
archivo.open(BD, ios::in | ios::binary);
if (!archivo.is_open()) {
ok = false;
}
else {
archivo.seekg(0, ios::end);
int pos = archivo.tellg();
int numReg = pos / SIZE;
tabla.cont = 0;
tRegistro registro;
archivo.seekg(0, ios::beg);
for (int i = 0; i < numReg; i++) {
archivo.read( (char *) &registro, SIZE);
Luis Hernández Yáñez

tabla.registros[tabla.cont] = registro;
tabla.cont++;
}
archivo.close();
}

Fundamentos de programación: Archivos binarios Página 1092
tabla.cpp

void guardar(tTabla tabla) {


fstream archivo;
archivo.open(BD, ios::out | ios::binary | ios::trunc);
for (int i = 0; i < tabla.cont; i++) {
archivo.write( (char *) &tabla.registros[i], SIZE);
}
archivo.close();
}
Luis Hernández Yáñez

Fundamentos de programación: Archivos binarios Página 1093

bd.cpp

#include <iostream>
using namespace std;
#include "registro.h"
#include "tabla.h"

int main() {
tTabla tabla;
tTabla ok;
cargar(tabla, ok);
if (!ok) {
cout << "Error al abrir el archivo!" << endl;
}
else {
mostrar(tabla);
insertar(tabla, nuevo(), ok);
mostrar(tabla);
Luis Hernández Yáñez

guardar(tabla);
}  
return 0;
}

Fundamentos de programación: Archivos binarios Página 1094


Licencia CC (Creative Commons)
Este tipo de licencias ofrecen algunos derechos a terceras personas
bajo ciertas condiciones.
Este documento tiene establecidas las siguientes:

Reconocimiento (Attribution):
En cualquier explotación de la obra autorizada por la licencia
hará falta reconocer la autoría.
No comercial (Non commercial):
La explotación de la obra queda limitada a usos no comerciales.
Compartir igual (Share alike):
La explotación autorizada incluye la creación de obras derivadas
siempre que mantengan la misma licencia al ser divulgadas.
Luis Hernández Yáñez

Pulsa en la imagen de arriba a la derecha para saber más.

Fundamentos de programación: Archivos binarios Página 1095

También podría gustarte