Flujo de salida estandar en C++

Anuncio
MIGUEL Á. TOLEDO MARTÍNEZ
CONTENIDO DE LA LECCIÓN 6
FLUJO DE SALIDA ESTANDAR EN C++
1. INTRODUCCIÓN
2. FLUJOS
2.1. Archivos de encabezado de la biblioteca iostream
2.2. Clases y objetos de flujo de entrada/salida
3. FLUJO DE SALIDA
3.1. Operador de inserción de flujo
3.1.1. Ejemplos 6.1, 6.2
3.2. Cascada del operador de inserción de flujo
3.2.1. Ejemplo 6.3
3.3. Uso de caracteres especiales
3.3.1. Ejemplos 6.4, 6.5, 6.6, 6.7, 6.8
3.3.2. Otros caracteres especiales
3.3.3. Ejemplo 6.9, 6.10
3.4. Visualizando números con cout
3.4.1. Ejemplos 6.11, 6.12
3.5. Visualizando varios valores con un cout
3.5.1. Ejemplo 6.13
3.6. Visualizando caracteres fijos con cout
3.6.1. Ejemplo 6.14
3.7. Visualizando cadena de caracteres y números con cout
3.7.1. Ejemplo 6.15, 6.16, 6.17
3.8. Utilizando cout para combinar diferentes tipos de valores
3.8.1. Ejemplos 6.18
3.9. Visualizando objetos variable con cout
3.9.1. Ejemplo 6.19
3.10. Salida de variables char *
3.10.1. Ejemplo 6.20
3.11. Salida de caracteres con la función miembro put( ); put( ) en cascada
4. E/S SIN FORMATO MEDIANTE read( ), gcount( ) y write( )
4.1. Ejemplo 6.21
5. REDIRECCIONANDO LA SALIDA DE cout
5.1. Ejemplos 6.22, 6.23
6. EXAMEN BREVE 14
7. MANIPULACIÓN DEL FLUJO
7.1. Cambio de base para números enteros: dec, oct, hex y setbase( )
7.1.1. Ejemplos 6.24, 6.25
7.2. Precisión de punto flotante
7.2.1. Ejemplos 6.26, 6.27
7.3. Anchura de campo (setw( ), width( ))
7.3.1. Ejemplos 6.28, 6.29, 6.30, 6.31, 6.32
7.4. Examen breve 15
7.5. Manipuladores definidos por el usuario
7.5.1. Ejemplo 6.33
FUNDAMENTOS – LECCIÓN 6
3
3
4
5
7
7
7
8
9
9
10
11
11
12
12
13
13
13
14
14
14
15
15
16
16
16
17
17
17
18
19
19
20
20
22
22
23
24
25
26
29
29
29
6-1
MIGUEL Á. TOLEDO MARTÍNEZ
8. ESTADOS DE FORMATO DE FLUJO
8.1. Indicadores de estado de formato
8.2. Ceros a la derecha y puntos decimales ( ios::showpoint )
8.2.1. Ejemplo 6.34
8.3. Alineación ( ios::left, ios::right, ios::internal )
8.3.1. Ejemplos 6.35, 6.36, 6.37, 6.38
8.4. Relleno ( fill, setfill )
8.4.1. Ejemplos 6.39, 6.40
8.5. Base de flujo integral ( ios::dec, ios::oct, ios::hex, ios::showbase )
8.5.1. Ejemplo 6.41
8.6. Números de punto flotante, notación científica ( ios::scientific, ios::fixed )
8.6.1. Ejemplos 6.42, 6.43
8.7. Control de mayúsculas / minúsculas ( ios::uppercase )
8.7.1. Ejemplo 6.44
8.8. Establecimiento y restablecimiento de los indicadores de formato
( flags, setiosflags, resetiosflags )
8.8.1. Ejemplo 6.45
9. EXAMEN BREVE 16
10. ESTADOS DE ERROR DE FLUJO
10.1. Ejemplo 6.46
11. ENLACE DE UN FLUJO DE SALIDA A UN FLUJO DE ENTRADA
12. CONCEPTO DE PRECISIÓN
12.1. Ejemplo 6.47
13. CONCEPTO DE SOBRE FLUJO
13.1. Ejemplo 6.48
14. UTILIZANDO cout PARA IMPRIMIR INFORMACIÓN
14.1. Ejemplo 6.49
15. LO QUE NECESITA SABER
16. PREGUNTAS Y PROBLEMAS
16.1. Preguntas
16.2. Problemas
FUNDAMENTOS – LECCIÓN 6
30
31
32
32
33
33
35
36
37
38
38
38
39
40
40
40
41
41
43
44
44
44
45
46
46
47
49
53
53
54
6-2
MIGUEL Á. TOLEDO MARTÍNEZ
LECCIÓN 6
FLUJO DE SALIDA ESTANDAR EN C++
INTRODUCCIÓN
En esta lección se verá como generar datos de salida desde el sistema. En general la
obtención de datos de entrada al sistema se conoce como lectura y la generación de datos
desde el sistema se denomina escritura. Veremos cómo escribir información en la pantalla
de su monitor y en la impresora. En otra lección, aprenderá cómo escribir información en
un archivo, asi como leer información desde el teclado y desde un archivo en disco.
Objetivo de esta lección:
•
•
•
•
•
•
•
Comprender y utilizar el flujo de salida orientada a objetos de C++.
Aprender a formatear las salidas.
Comprender la jerarquía de clases de flujo de E/S.
Aprender a enviar la salida a objetos de tipos definidos por el usuario.
Crear manipuladores de flujo definidos por el usuario.
Determinar el éxito o la falla de las operaciones de entrada/salida.
Saber como enlazar flujos de salida con flujos de entrada.
Las bibliotecas estándar de C++ proporcionan un amplio conjunto de capacidades
de entrada/salida. Esta lección trata de los flujos de E/S comunes, suficiente para resolver la
mayoría de los problemas de computación, además ofrece una panorámica de las capacidades restantes.
Las características de E/S que aquí se describen están orientadas a objetos. Deberá
encontrar interesante el ver la manera en que se implementan. Este estilo de E/S utiliza
otras características de C++, tales como referencias, sobrecarga de funciones y de operadores.
Como veremos, C++ utiliza E/S a prueba de tipos. Cada operación de E/S se realiza
automáticamente en una forma sensible con respecto al tipo de datos. Si una función de E/S
se ha definido adecuadamente para que maneje un tipo de dato en particular, entonces dicha
función manejará automáticamente ese tipo de datos. Si no hay concordancia entre el tipo
de datos y la función que maneja ese tipo de datos, el compilador envía un mensaje de
error. Por lo tanto, los datos inadecuados no pueden andar furtivamente por el sistema (como puede suceder en C –un hueco que C permite para producir algunos errores sutiles y
extraños).
Los usuarios pueden especificar la E/S tanto de tipos estándar como de tipos definidos por el usuario. Esta capacidad es una de las características más valiosas de C++.
FLUJOS
La E/S en C++ se da en flujo de bytes. Un flujo es simplemente una secuencia de
bytes. En las operaciones de entrada, los bytes fluyen desde un dispositivo (por ejemplo:
FUNDAMENTOS – LECCIÓN 6
6-3
MIGUEL Á. TOLEDO MARTÍNEZ
teclado, unidad de disco, escáner) hacia la memoria principal. En operaciones de salida los
bytes fluyen de la memoria principal hacia un dispositivo (por ejemplo: pantalla, impresora, unidad de disco, escáner).
La aplicación asocia significado a los bytes. Los bytes pueden representar: caracteres ASCII, datos sin procesar en formato interno, imágenes gráficas, voz digital , video
digital o cualquier otro tipo de información que pueda requerir una aplicación.
El trabajo de los mecanismos de E/S del sistema es mover bytes de los dispositivos
hacia la memoria y viceversa, en forma consistente y confiable. Tales transferencias involucran frecuentemente el movimiento mecánico, tal como la rotación de un disco o de una
cinta o la digitación de caracteres en un teclado. El tiempo que se llevan estas transferencias
normalmente es inmenso en comparación con el tiempo que le lleva al procesador manejar
los datos internamente. Por lo tanto, las operaciones de E/S requieren una planeación y afinación cuidadosa para asegurar el máximo desempeño.
C++ proporciona capacidades de E/S de bajo nivel y alto nivel. Las capacidades de
E/S de bajo nivel (es decir, E/S sin formato) por lo general especifican que un cierto número de bytes deberá transferirse de un dispositivo a la memoria o de la memoria a un dispositivo. En dichas transferencias el byte individual es el asunto de interés. Tales capacidades
de bajo nivel proporcionan transferencias de alta velocidad y de gran volumen, pero no son
particularmente convenientes para los usuarios.
Los usuarios prefieren una vista de nivel alto de la E/S (es decir, la E/S formateada), en donde los bytes están agrupados en unidades significativas, tales como : enteros,
números de punto flotante, caracteres, cadenas y tipos definidos por el usuario. Estas capacidades orientadas a tipo son satisfactorias para la mayor parte de la E/S, que no sea el
procesamiento de archivo de alto volumen.
ARCHIVOS DE ENCABEZADO DE LA BIBLIOTECA iostream
La biblioteca iostream de C++ proporciona cientos de capacidades de E/S. Varios
archivos de encabezado contienen partes de la interfaz a la biblioteca.
La mayoría de los programas C++ incluyen el archivo de encabezado <iostream.h>,
que contiene la información básica requerida para todas las operaciones de flujo de E/S.
Dicho archivo contiene los objetos cin, cout, cerr y clog, que corresponden respectivame nte
con el flujo de entrada y salida estándar, el flujo de errores estándar no almacenado en
búfer y el flujo de errores estándar almacenado en búfer. Se proporcionan capacidades de
E/S formateadas y sin formato.
El archivo de encabezado <iomanip.h> contiene información útil para realizar la
E/S formateada con los llamados manipuladores de flujo con parámetros.
El archivo de encabezado <fstream.h> contiene información importante para las
operaciones de procesamiento de archivos controladas por el usuario.
FUNDAMENTOS – LECCIÓN 6
6-4
MIGUEL Á. TOLEDO MARTÍNEZ
Cada implementación de C++ por lo general contiene otras bibliotecas relacionadas
con la E/S que proporcionan capacidades específicas del sistema, tales como el control de
dispositivos de propósito especial para E/S de audio y video.
CLASES Y OBJETOS DE FLUJO DE ENTRADA / SALIDA
La biblioteca iostream contiene muchas clases para manejar una amplia variedad de
operaciones de E/S. La clase istream soporta las operaciones de flujo de entrada. La clase
ostream soporta las operaciones de flujo de salida. La clase iostream soporta las operaciones de flujo tanto de entrada como de salida.
La clase istream y la clase ostream deriva mediante herencia simple a partir de la
clase base ios. La clase iostream deriva mediante herencia múltiple tanto de la clase istream como de la clase ostream. En la figura 6.1 se resumen estas relaciones de herencia.
ios
istream
ostream
iostream
Figura 6.1 Parte de la jerarquía de clases de flujo de E/S.
La sobrecarga de operadores (utilización de un operador para realizar más de una
tarea) proporciona una notación conveniente para la realización de la entrada/salida. El
operador de desplazamiento a la izquierda (<<) está sobrecargado para indicar el flujo de
salida y se denomina operador de inserción de flujo. El operador de desplazamiento a la
derecha (>>) está sobrecargado para indicar el flujo de salida y se denomina operador de
extracción de flujo. Estos operadores se utilizan con los objetos de flujo estándar cin, cout,
cerr, clog, y comúnmente con los objetos de flujo definidos por el usuario.
cin es un objeto de la clase istream, y se dice que está enlazado (o conectado) al
dispositivo de entrada estándar que normalmente es el teclado. El operador de extracción de
flujo, como se utiliza en la siguiente instrucción, causa que un valor para la variable entera
calificacion (suponiendo que calificacion ha sido declarada como int) se reciba desde cin
hacia la memoria:
cin >> calificacion;
Observe que la operación de extracción de flujo es lo suficientemente inteligente
para saber el tipo de dato que es. Suponiendo que calificacion se haya declarado adecuadamente, no se necesita especificar información adicional con dicho operador (como es el
caso de la E/S estilo C).
cout es un objeto de la clase ostream, y se dice que está enlazado al dispositivo de
salida estándar que normalmente es la pantalla. El operador de inserción de flujo, como se
FUNDAMENTOS – LECCIÓN 6
6-5
MIGUEL Á. TOLEDO MARTÍNEZ
utiliza en la siguiente instrucción, causa que al valor de la variable entera calificacion se le
dé salida desde la memoria hacia el dispositivo de salida estándar:
cout << calificacion;
Observe que el operador de inserción es lo suficientemente inteligente para saber el
tipo de calificacion (suponiendo que se haya declarado adecuadamente), por lo que no se
necesita especificar información adicional con dicho operador.
El formato general para el enunciado cout es el siguiente:
cout << elemento1 << elemento2 << elemento 3 << ... << elementoN;
Como puede observar, después de cout sigue una lista de elementos para escribirse,
los cuales están separadas por el operador de inserción de flujo <<. De esta manera, un
enunciado cout representa una secuencia o flujo de datos o que se dirige a la pantalla del
monitor. Los elementos se insertan dentro de un flujo de salida a través del operador de
inserción de flujo <<. Conforme los elementos se insertan dentro del flujo, estos fluyen a la
pantalla de su monitor, como se ilustra en la figura 6.3:
cout << elemento1 << elemento2 << elemento3 << ... << elementoN;
Elemento1
Elemento2
Elemento3
...
ElementoN
Elementos de datos del flujo de salida cout que se muestran en la pantalla
PANTALLA
Figura 6.3. Los elementos que se muestran se insertan dentro del flujo de salida cout usando el operador de
inserción de flujo <<.
Algunos sistemas acumulan información de salida en una memoria temporal (búfer
o buffer) hasta que haya suficiente información para justificar la escritura en la pantalla.
cerr es un objeto de la clase ostream, y se dice que está enlazado al dispositivo de
errores estándar. Las salidas hacia el objeto cerr no se almacena en el búfer. Esto significa
que cada inserción de flujo de cerr causa que su salida aparezca inmediatamente; esto es
adecuado para notificar al usuario algún error en forma inmediata.
clog es un objeto de la clase ostream, y también se dice que esta enlazado al dispositivo de errores estándar. Las salidas de clog se almacenan en el búfer. Esto significa que
cada inserción a clog puede causar que su salida se conserve en un búfer hasta que éste se
llene o vacíe.
FUNDAMENTOS – LECCIÓN 6
6-6
MIGUEL Á. TOLEDO MARTÍNEZ
El procesamiento de archivos de C++ utiliza las clases ifstream para realizar operaciones de entrada de archivo, ofstream para las operaciones de salida de archivo y fstream
para las operaciones de entrada/salida de archivo. La clase ifstream hereda de istream, la
clase ofstream hereda de ostream y la clase fstream hereda de iostream. En la figura 6.2 se
resumen las diversas relaciones de herencia de las clases relacionadas con la E/S. Existen
más clases, en la jerarquía de clases de flujo de E/S completa, soportadas en la mayoría de
las instalaciones, pero las clases que aquí se muestran proporcionan las capacidades que
necesitan casi todos los programadores. Si desea más información sobre el procesamiento
de archivos, vea la referencia de la biblioteca de clases de su sistema C++.
ios
istream
ifstream
ostream
iostream
ofstream
fstream
Figura 6.2 Parte de la jerarquía de clases de flujo de E/S con las principales clases para el procesamiento de archivos.
FLUJO DE SALIDA
La clase ostream de C++ proporciona la habilidad para realizar salida formateada y
sin formato. Las capacidades para la salida incluyen: salida de tipos de datos estándar con
el operador de inserción de flujo, salida de caracteres con la función miembro put( ), salida sin formato con la función miembro write( ), salida de enteros en formatos decimal, octal y hexadecimal, salida de valores de punto flotante con diversas precisiones, con puntos
decimales forzados en notación científica y en notación fija, salida de datos alineados en
campos de anchura indicados, salida de datos en campos rellenados con caracteres especificados y salida de letras mayúsculas en notación científica y notación hexadecimal.
OPERADOR DE INSERCIÓN DE FLUJO
El flujo de salida puede realizarse mediante el operador de inserción de flujo <<
sobrecargado. Dicho operador está sobrecargado para dar salida a elementos de datos de
tipos estándar: numérico, cadenas y valores de apuntador.
Ejemplo 6.1
El siguiente programa: SALUDOS1.CPP, muestra la salida de una cadena utilizando una sola instrucción de inserción de flujo.
FUNDAMENTOS – LECCIÓN 6
6-7
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: SALUDOS1.CPP, muestra la salida de una cadena utilizando
una sola instrucción de inserción de flujo.
*/
#include <iostream.h>
//Para cout
Salida :
¡Bienvenidos al INSTITUTO POLITECNICO NACIONAL!
void main( void )
{
cout << "¡Bienvenidos al INSTITUTO POLITÉCNICO NACIONAL!";
}//Fin de main( )
Ejemplo 6.2
El siguiente programa: SALUDOS2.CPP, muestra la misma salida que el programa SALUDOS1.CPP, utilizando varias instrucciones de inserción de flujo.
/* El siguiente programa: SALUDOS2.CPP, muestra la salida de una cadena utilizando
varias instrucciones de inserción de flujo.
*/
#include <iostream.h>
//Para cout
Salida :
¡Bienvenidos al INSTITUTO POLITECNICO NACIONAL!
void main( void )
{
cout << "¡Bienvenidos al ";
cout << "INSTITUTO POLITÉCNICO NACIONAL! ";
}//Fin de main( )
CASCADA DEL OPERADOR DE INSERCIÓN DE FLUJO
Los operadores << y >> sobrecargados pueden utilizarse en forma de cascada, como se muestra en el siguiente ejemplo:
Ejemplo 6.3
El siguiente programa: CASCADA.CPP, ilustra el uso del operador << sobrecargado en forma de
cascada.
/* El siguiente programa: CASCADA.CPP, ilustra el uso del operador de inserción <<
utilizado en forma de cascada.
*/
#include <iostream.h>
//Para cout
Salida :
15 más 27 es: 42
void main( void )
{
cout << "15 más 27 es: " << ( 15 + 27 ) << endl;
}//Fin de main( )
Las múltiples inserciones de flujo del programa CASCADA.CPP, se ejecutan como si hubieran sido escritas de la siguiente manera :
FUNDAMENTOS – LECCIÓN 6
6-8
MIGUEL Á. TOLEDO MARTÍNEZ
( ( ( cout << "15 más 27 es: ") << ( 15 + 27 ) ) << endl );
es decir, << asocia de izquierda a derecha. Este tipo de puesta en cascada de los operadores
de inserción de flujo se permite debido a que el operador << sobrecargado devuelve una referencia hacia el objeto operando izquierdo –es decir, cout. Por lo tanto, la expresión entre
paréntesis del extremo izquierdo
( cout << "15 más 27 es: " )
envía la cadena de caracteres y devuelve una referencia a cout. Esto permite evaluar la expresión que está entre paréntesis en la parte media como
( cout << ( 15 + 27 ) )
que envía el valor entero 100 y devuelve una referencia a cout. A continuación la expresión
entre paréntesis más a la derecha se evalúa como
( cout << endl );
que envía a la salida un salto a una nueva línea, vacía a cout y devuelve una referencia a cout. Esta última devolución ya no se utiliza.
USO DE CARACTERES ESPECIALES
La mayoría de los programas que elaborará, visualizarán varias líneas de salida.
Cuando desea avanzar el cursor al principio de una nueva línea, puede utilizar el carácter
nueva línea ( '\n' ) en el flujo de salida. Cuando se inserta en el flujo de salida, se genera un
retorno de carro/salto de línea (CRLF). C++ le proporciona dos maneras de generar una
nueva línea.
Primero, puede colocar el carácter '\n' dentro de una cadena de caracteres como se
muestra en el Ejemplo 6.4:
Ejemplo 6.4
El siguiente programa: DOSLINEAS.CPP, visualiza la salida de dos cadenas en dos líneas, utilizando la secuencia de escape '\n '.
/* El siguiente programa: DOSLINEAS.CPP, muestra en la pantalla dos cadenas de
caracteres en dos líneas, utilizando el carácter de escape '\n'.
*/
#include <iostream.h>
// Para cout
void main( void )
{
Salida:
Esta es la línea uno.
Esta es la línea dos.
cout << "Esta es la línea uno.\nEsta es la línea dos.";
} // Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-9
MIGUEL Á. TOLEDO MARTÍNEZ
Segundo, si no esta desplegando una cadena de caracteres, puede colocar el carácter
nueva línea entre apóstrofes, como se ve en el Ejemplo 6.5:
Ejemplo 6.5
El siguiente programa NVALINEA.CPP visualiza los números 1,0,0 y 1 en líneas diferentes:
/* El siguiente programa: NVALINEA.CPP, visualiza los números 1, 0, 0 y 1 en líneas
diferentes.
*/
#include <iostream.h>
Salida:
1
0
0
1
// Para cout
void main( void )
{
cout << 1 << '\n' << 0 << '\n' << 0 << '\n' << 1;
} // Fin de main( )
En vez de utilizar el carácter nueva línea, puede ut ilizar el manipulador de flujo
(ver este tema más adelante) endl (end line). El manipulador de flujo endl hace dos cosas:
primero, al igual que la secuencia de escape '\n' genera un CRLF y, segundo, a diferencia
de la secuencia de escape '\n', inmediatamente emite la salida. El manipulador de flujo
endl hace que cualquier información acumulada en la memoria temporal o búfer se escriba
inmediatamente.
Ejemplo 6.6
El siguiente programa: ENDL1.CPP, muestra el uso del manipulador de flujo endl.
/* El siguiente programa: ENDL1.CPP, muestra el uso del manipulador de flujo endl. */
#include <iostream.h>
//Para cout
Salida:
¡Bienvenidos al INSTITUTO POLITECNICO NACIONAL!
void main( void )
{
cout << "¡Bienvenidos al ";
cout << "INSTITUTO POLITÉCNICO NACIONAL!";
cout << endl;
}//Fin de main( )
Ejemplo 6.7
El siguiente programa: ENDL2.CPP, ilustra el uso de endl:
/* El siguiente programa. ENDL2.CPP, ilustra el uso del manipulador endl. */
#include <iostream.h>
void main( void )
{
// Para cout
Salida:
El lenguaje C++ es ...
bastante fácil.
cout << "El lenguaje C++ es ... " << endl << "bastante fácil.";
} // Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-10
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.8
Es posible enviar a la salida expresiones tal como se muestra en el siguiente programa: EXPRESIONES.CPP.
/* El siguiente programa: EXPRESIONES.CPP, muestra el uso de envío a la salida de valores
de expresiones.
*/
#include <iostream.h>
//Para cout
void main( void )
{
Salida:
15 más 27 es: 42
cout << "15 más 27 es: ";
//No se necesitan los paréntesis, pero se utilizan para claridad
cout << ( 15 + 27 );
//expresión
cout << '\n';
}//Fin de main( )
OTROS CARACTERES ESPECIALES
En la lección 3 página 3-24 (constantes de tipo carácter y cadena) hemos visto ya
los caracteres especiales, los repetiremos simplemente para tener una rápida referencia.
Carácter
\a
\b
\f
\n
\r
\t
\v
\\
\?
\’
\”
\0
\0OO
\xhhh
Propósito
Carácter alerta (o campana)
Carácter retroceso (backspace)
Carácter alimentación de hoja (formfeed)
Carácter nueva línea (newline)
Carácter retroceso de carro (carriage return)
(no alimenta nueva línea)
Carácter de tabulador horizontal
(horizontal tab)
Carácter de tabulador vertical
(vertical tab)
Carácter diagonal inverso (backslash)
Carácter interrogación (question mark)
Carácter apóstrofe
Carácter comillas
Carácter nulo
Valor octal tal como \007
Valor hexadecimal tal como \xFFFF
Ejemplo 6.9
El siguiente programa: ESPECIAL.CPP utiliza los caracteres especiales '\a ' (alerta) y tabulador
( '\t' ) para emitir un sonido en la bocina de la computadora y luego visualiza las palabras, Campana
Campana Campana, separadas cada una de ellas por un tabulador.
FUNDAMENTOS – LECCIÓN 6
6-11
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: ESPECIAL.CPP, utiliza los caracteres especiales '\a' (alerta)
y tabulador ( '\t') para emitir un sonido en la bocina de la computadora y luego
visualiza las palabras, Campana Campana Campana, separadas cada una por un
tabulador.
*/
#include <iostream.h>
// Para cout
Salida:
Campana Campana Campana
void main( void )
{
cout << "Campana\a\tCampana\a\tCampana\a";
} // Fin de main( )
Como antes hemos dicho, la salida no es inmediatamente emitida al dispositivo, sino
más bien se va acumulando en cierta área de la memoria (búfer o buffer) asignada a tal dispositivo. Sin embargo, cuando se encuentra con el manipulador endl o bien una operación
de lectura, el búfer es vaciado (enviar la información al dispositivo). Si desea vaciar inmediatamente el buffer asignado al dispositivo sin necesidad de dar un salto de línea se puede
utilizar la palabra flush.
Ejemplo 6.10
El siguiente programa: FLUSH.CPP, ilustra el uso del manipulador flush para vaciar el búfer.
/* El siguiente programa: FLUSH.CPP, muestra el uso del manipulador flush para
enviar inmediatamente la salida al dispositivo.
*/
#include <iostream.h>
// Para cout
Salida:
Esta línea aparece inmediatamente.
Lo mismo sucede con esta línea...
void main( void )
{
cout << "Esta línea aparece inmediatamente." << flush;
cout << "\nLo mismo sucede con esta línea..." << flush;
} // Fin de main( )
VISUALIZANDO NÚMEROS CON cout
Cuando quiera escribir información numérica fija, simplemente inserte los valores
numéricos dentro del flujo de salida cout utilizando el operador <<.
Ejemplo 6.11
El siguiente programa: 1001.CPP despliega el número 1001 en su pantalla:
/* El siguiente programa: 1001.CPP, despliega el número 1001 por pantalla. */
#include <iostream.h>
void main( void )
{
cout << 1001;
} // Fin de main( )
FUNDAMENTOS – LECCIÓN 6
// Para cout
Salida:
1001
6-12
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.12
Cuando se inserta un valor de punto flotante fijo, se obtiene el decimal fijo equivalente en la salida,
no el equivalente exponencial. El siguiente programa: FLOTANTE.CPP, usa cout para desplegar el
número 0.12345 en su pantalla.
/* El siguiente programa: FLOTANTE.CPP, despliega el número 0.12345 por pantalla. */
#include <iostream.h>
// Para cout
Salida:
0.12345
void main( void )
{
cout << 0.12345;
} // Fin de main( )
VISUALIZANDO VARIOS VALORES CON UN cout
Cuando utiliza cout, puede utilizar el operador de inserción más de una vez por
enunciado. La salida no genera ningún espacio entre los elementos.
Ejemplo 6.13
El siguiente programa: 1001BIS.CPP, usa el operador cuatro veces para desplegar el número 1001
en la pantalla.
/* El siguiente programa: 1001BIS.CPP, utiliza el operador << cuatro veces para
desplegar el número 1001 en la pantalla.
*/
#include <iostream.h>
Salida:
1001
// Para cout
void main( void )
{
cout << 1 << 0 << 0 << 1;
} // Fin de main( )
Ambas condiciones (espaciado y salida decimal) se pueden modificar con las opciones de formato
dentro del enunciado cout. El formato de salida se explicará mas adelante (Manipulación de flujo).
VISUALIZANDO CARACTERES FIJOS CON cout
Para escribir información de caracteres, deberá encerrar la información de salida
entre comillas (comillas simples para caracteres sencillos y comillas dobles para cadenas de
caracteres). Por consiguiente, el enunciado:
cout << 'A' ;
genera como salida una A.
FUNDAMENTOS – LECCIÓN 6
6-13
MIGUEL Á. TOLEDO MARTÍNEZ
El enunciado:
cout << " La vida es bella! " ;
produce como salida: ¡La vida es bella!
Ejemplo 6.14
Veamos un ejemplo en el que se muestren varios mensajes en la pantalla del monitor. Llamemos al
programa COUT.CPP:
/* El siguiente programa: COUT.CPP, utiliza el objeto de flujo de salida cout para
visualizar varios mensajes en la pantalla del monitor.
*/
#include <iostream.h>
// Para cout
void main( void )
{
cout << "Esta es la línea uno.\n";
cout << "Este texto se encuentra ";
cout << "en la línea dos.\n";
cout << "Esta es la última línea.";
} // Fin de main( )
Salida:
Esta es la línea uno.
Este texto se encuentra en la línea dos.
Esta es la última línea.
VISUALIZANDO CADENA DE CARACTERES Y NÚMEROS CON cout
Ejemplo 6.15
El siguiente programa: CVALNUM.CPP, utiliza cout para visualizar cadena de caracteres y números:
/* El siguiente programa: CVALNUM.CPP, muestra como cout visualiza cadena de caracteres
y números.
*/
Salida:
#include <iostream.h>
void main( void )
{
// Para cout
cout le permite visualizar cadenas de caracteres , números enteros
y de punto flotante.
1001
1.2345
cout << "cout le permite visualizar cadenas de caracteres, números enteros\n";
cout << "y de punto flotante.\n\n";
cout << 1001;
cout << '\n';
cout << 1.2345;
} // Fin de main( )
Ejemplo 6.16
El programa siguiente: MSTR1001.CPP visualiza una cadena de caracteres y un número utilizando
un cout:
FUNDAMENTOS – LECCIÓN 6
6-14
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: MSTR1001.CPP, visualiza una cadena de caracteres y un
número, utilizando un solo cout.
*/
#include <iostream.h>
// Para cout
Salida:
Mi número favorito es: 1001
void main( void )
{
cout << "Mi número favorito es: " << 1001;
} // Fin de main( )
Ejemplo 6.17
El programa siguiente: ENM1001.CPP, visualiza el número 1001 en medio de dos cadenas de caracteres:
/* El siguiente programa: ENM1001.CPP, visualiza el número 1001 en medio de dos
cadenas de caracteres.
*/
#include <iostream.h>
// Para cout
Salida:
El número 1001 es mi favorito
void main( void )
{
cout << "El número " << 1001 << " es mi favorito";
} // Fin de main( )
UTILIZANDO cout PARA COMBINAR DIFERENTES TIPOS DE VALORES
Ejemplo 6.18
Un objeto de flujo de salida cout le permite escribir diferentes tipos de valores, como se muestra en
el siguiente programa: UNCOUT.CPP:
/* El siguiente programa: UNCOUT.CPP, le muestra como se puede utilizar un solo cout
para escribir cadenas y números de diferentes tipos.
*/
Salida:
cout visualiza cadenas
#include <iostream.h>
void main( void )
{
// Para cout
1001
1.2345
cout << "cout visualiza cadenas " << "\n\n" << 1001 << '\n' << 1.2345;
} // Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-15
MIGUEL Á. TOLEDO MARTÍNEZ
VISUALIZANDO OBJETOS VARIABLE CON cout
Lo siguiente que deberá aprender es cómo escribir información contenida en un objeto variable. De nuevo, ésta es una tarea sencilla si se utiliza el objeto cout: simplemente
inserte el (los) identificador(es) de variables dentro del flujo cout con el operador de inserción <<. Por ejemplo, si su programa ha definido voltaje, corriente y resistencia como objetos variables, puede escribir sus respectivos valores insertándolos dentro de un flujo cout,
de la manera siguiente:
cout << voltaje << corriente << resistencia;
El enunciado anterior escribirá los valores almacenados en memoria para voltaje,
corriente y resistencia, en ese orden. El orden de salida será el mismo que el orden listado
dentro del enunciado cout. Sin embargo, no habrá espacios entre los valores. Los caracteres en blanco se deben insertar en forma separada para proporcionar espacio.
Ejemplo 6.19
El siguiente programa: OHM.CPP, utiliza la ley de Ohm que establece que el voltaje es igual al producto de la corriente por la resistencia. Su objetivo es escribir un mensaje junto con los valores de la
corriente y resistencia, así como el voltaje.
/* El siguiente programa: OHM.CPP, calcula y muestra el voltaje usando la ley
de OHM.
Salida:
*/
Este programa calculará el voltaje dada una corriente de 0.001 amperes
y una resistencia de 4700.0 ohms.
#include <iostream.h>
// Para cout
void main( void )
{
Valor de corriente
= 0.001 amperes.
Valor de resistencia
= 4700 ohms.
Valor del voltaje resultante = 4.7 volts
float voltaje
= 0.0;
float corriente = 0.001;
float resistencia = 4700.0;
// Para el voltaje calculado.
// Valor de la corriente.
// Valor de la resistencia.
// Breve descripción del programa.
cout << "Este programa calculará el voltaje dada una corriente de 0.001 "
"amperes\ny una resistencia de 4700.0 ohms." << endl << endl;
// Cálculo del voltaje utilizando la ley de Ohm.
voltaje = corriente * resistencia;
// Mostrar los resultados.
cout << "Valor de corriente
= " << corriente << " amperes.\ n"
"Valor de resistencia
= " << resistencia << " ohms.\ n"
"Valor del voltaje resultante = " << voltaje
<< " volts" << endl;
} // Fin de main ( )
SALIDA DE VARIABLES char *
En la E/S estilo C es necesario que el programador proporcione información del tipo
de datos. C++ determina automáticamente los tipos de datos –una mejora agradable sobre
C. Pero a veces esto resulta un estorbo. Por ejemplo, sabemos que una cadena de caracteres
FUNDAMENTOS – LECCIÓN 6
6-16
MIGUEL Á. TOLEDO MARTÍNEZ
es de tipo char *. Supongamos que quiere imprimir el valor de ese apuntador, es decir, la
dirección de memoria del primer carácter de dicha cadena. Pero el operador << ha sido sobrecargado para que imprima los datos de tipo char * como cadenas terminadas en nulo. La
solución es hacer una conversión de tipo mediante cast del apuntador a un tipo void * (esto
deberá hacerse para cualquier variable de apuntador que el programador desee enviar a
la salida como una dirección).
Ejemplo 6.20
El siguiente programa: DIRCHAR.CPP, muestra la impresión de una variable char *
en formatos de cadena y dirección. Observe que la dirección se imprime como número
hexadecimal (base 16). En C++ los números hexadecimales comienzan con 0x o 0X.
/* El siguiente programa: DIRCHAR.CPP, muestra en pantalla el contenido de una variable
char * en formatos de cadena y de dirección.
*/
#include <iostream.h>
//Para cout
Salida:
El valor de cadena es: INSTITUTO POLITECNICO NACIONAL
El valor de static_cast<void *>( cadena ) es: 0x169f00ba
void main( void )
{
char *cadena = "INSTITUTO POLITÉCNICO NACIONAL";
cout
<< "El valor de cadena es: " << cadena
<< "\nEl valor de static_cast< void * >( cadena ) es: "
<< static_cast< void * > ( cadena ) << endl;
}//Fin de main( )
SALIDA DE CARACTERES CON LA FUNCIÓN MIEMBRO put( ); put( ) EN CASCADA
La función miembro put( ) envía a la salida un carácter como en
cout.put('A');
que despliega una A en la pantalla. Las llamadas a put( ) pueden ponerse en cascada como
en
cout.put( 'A ').put ( '\n' );
la cual da salida a la letra A seguida de un carácter de nueva línea. Como sucede con <<, la
instrucción anterior se ejecuta de esta forma debido a que el operador punto (.) asocia de
izquierda a derecha y la función miembro put( ) devuelve una referencia al objeto mediante
el que se realizó la llamada a put( ). La función put( ) también puede invocarse mediante
una expresión de valor ASCII, como en cout.put( 65 ), lo cual también da salida a A.
E/S SIN FORMATO MEDIANTE read( ), gcount( ) y write( )
Aunque la idea de esta lección, es tratar exclusivamente con el objeto flujo de salida
de datos, trataremos brevemente el flujo de entrada de datos.
FUNDAMENTOS – LECCIÓN 6
6-17
MIGUEL Á. TOLEDO MARTÍNEZ
La entrada/salida sin formato se realiza con las funciones miembro read( ) y write( ). Cada una de ellas da entrada o envía a la salida algún número de bytes desde o hacia
un arreglo de caracteres que está en memoria. Estos bytes no tienen ningún formato. Se les
da entrada o salida simplemente como bytes sin formato. Por ejemplo, la llamada:
char buffer[] = " INSTITUTO POLITÉCNICO NACIONAL" ;
cout.write( buffer, 10 );
envía a la salida los primeros 10 bytes de buffer (incluyendo los caracteres nulos que podrían causar que terminara la salida con cout y <<). Debido a que una cadena de caracteres se
evalúa por la dirección de su primer carácter, la llamada
cout.write( " ABCDEFGHIJKLMNOPQRSTUVWXYZ " , 10 );
despliega los primeros 10 caracteres de la cadena.
La función miembro read( ) introduce un número determinado de caracteres en un
arreglo. Si se leen menos caracteres que el número indicado, se establece la bandera failbit.
Pronto veremos la manera de determinar si se ha establecido failbit.
La función miembro gcount( ) reporta cuántos caracteres ha leído la última operación de entrada.
Ejemplo 6.21
El siguiente programa: SINFORMATO.CPP, muestra las funciones miembro read( ) y gcount( ) de
istream y la función miembro write( ) de ostream. El programa utiliza read( ) para introducir 20 caracteres (a partir de una secuencia de entrada más larga) en el arreglo de caracteres buffer, se vale de
gcount( ) para determinar el número de caracteres que se introdujeron y emplea write( ) para enviar a
la salida los caracteres que están en el buffer.
/* El siguiente programa: SINFORMATO.CPP, ilustra el uso de las funciones miembro
read( ), gcount( ) y write( ).
*/
#include <iostream.h>
//Para cout y cin
void main( void )
{
const int TAMANO = 80;
char buffer[TAMANO];
Salida:
Introduzca una oración:
DABALE ARROZ A LA ZORRA EL ABAD
La oración introducida fue:
DABALE ARROZ A LA ZO
cout
<< "Introduzca una oración:\n";
cin.read(buffer, 20);
cout
<< "\nLa oración introducida fue:\n";
cout.write( buffer, cin.gcount( ) );
cout
<< endl;
}//Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-18
MIGUEL Á. TOLEDO MARTÍNEZ
REDIRECCIONANDO LA SALIDA DE cout
En realidad cout dirige la salida al dispositivo estándar (stdout) asignado al sistema
(en nuestro caso lo es la pantalla del monitor). Por lo tanto, es posible desde el sistema
operativo redireccionar la salida de cout. Por ejemplo, en el caso del programa
FLUSH.CPP, si quisiéramos que la salida de cout fuera dirigida a la impresora, entonces
desde el DOS podemos escribir el siguiente comando:
C:\> FLUSH.EXE > prn
o bien si quisiéramos enviarla a un archivo:
C:\> FLUSH.EXE > nombreArchivo
Si por alguna razón desea que la salida no sea redireccionable, por ejemplo, le agradaría que los mensajes de error que envíe al usuario aparezcan en la pantalla del monitor y
no en la impresora, entonces deberá utilizar el objeto cerr en vez del objeto cout. Este objeto envía la salida al dispositivo estándar para el manejo de errores (stderr) el cual no es
redireccionable y normalmente es la pantalla del monitor.
Ejemplo 6.22
El siguiente programa: CERR.CPP, ilustra el uso del objeto cerr.
* El siguiente programa: CERR.CPP, utiliza el flujo de salida cerr para visualizar
el mensaje "Este mensaje siempre aparece en el dispositivo estándar stderr".
*/
#include <iostream.h>
// Para cout
Salida:
Este mensaje siempre aparece en el dispotivo estándar: stderr
void main( void )
{
cerr << "Este mensaje siempre aparece en el dispotivo estándar: stderr";
} // Fin de main( )
Existe otro objeto de flujo de salida, clog, que se utiliza al igual que cerr, salvo que
la salida no la envía directamente al dispositivo asignado (como es el caso de cerr) sino que
la envía al área búfer.
FUNDAMENTOS – LECCIÓN 6
6-19
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.23
El siguiente programa: CLOG.CPP, ilustra el uso de clog.
/* El siguiente programa: CLOG.CPP, utiliza el flujo de salida clog para visualizar
el mensaje "Este mensaje NO siempre aparece inmediatamente en el dispositivo estándar: stderr".
*/
Salida:
Este mensaje NO siempre aparece inmediatamene en el dispositivo estándar: stderr
#include <iostream.h>
// Para cout
void main( void )
{
clog << "Este mensaje NO siempre aparece inmediatamente en el dispositivo estándar: stderr";
} // Fin de main( )
EXAMEN BREVE 14
MANIPULACIÓN DEL FLUJO
A la organización de la salida de un programa se le conoce con el nombre de formateo de la salida. Las formas más comunes de dar formato a la salida son las siguientes:
1.
Mediante el uso de funciones miembro de un objeto de flujo. Su sintaxis es la siguiente:
nombreObjetoInvocador.nombreFuncionMiembro(listaArgumentos );
Por ejemplo:
cout.width( 5 );
cout.fill( '*' );
cout.precision( 2 );
Nota: Estas instrucciones se estudiaran mas adelante.
2.
Mediante el uso de funciones especiales llamadas manipuladores (o modificadores), con o sin
argumentos.
Por ejemplo:
cout << flush;
cout << endl;
cout << setw( 5 ) << 12 << endl;
cout << seprecision( 2 ) << 12.325 << endl;
Nota: Estas instrucciones se estudiaran mas adelante.
3.
Mediante el uso de banderas (indicadores) como argumentos de la función miembro setf( ) del
objeto de flujo:
Por ejemplo:
cout.setf( ios::fixed );
cout.setf( ios::showpoint );
Nota: Estas instrucciones se estudiaran mas adelante.
FUNDAMENTOS – LECCIÓN 6
6-20
MIGUEL Á. TOLEDO MARTÍNEZ
a.
4.
Cualquier bandera (indicador) establecida puede desactivarse. Para desactivar una bandera usamos la función miembro unsetf( ). Por ejemplo, el siguiente enunciado hará que el
programa deje de incluir el signo de mas antes de los enteros positivos que se envían al flujo cout.
cout.unsetf( ios::showpos );
Mediante el uso de banderas (indicadores) como argumento de la función setiosflags( ).
a.
Cualquier bandera (indicador) establecida puede desactivarse. Para desactivar una bandera usamos la función resetiosflags( ). Por ejemplo, el siguiente enunciado hará que el programa deje de justificar la salida de los datos a la izquierda.
resetiosflags( ios::left );
5.
Para emplear los manipuladores es preciso incluir la siguiente directiva en el programa:
#include <iomanip.h>
Un manipulador es una función que se invoca de manera no tradicional. A su vez,
la función manipuladora invoca una función miembro. Los manipuladores se colocan después del operador de inserción <<, como si la función manipuladora fuera un elemento que
se enviara a la salida. Al igual que las funciones tradicionales, los manipuladores pueden
tener o no argumentos. Ya vimos un manipulador: endl.
El manipulador setw( ) y la función miembro width( ) hacen exactamente lo mismo.
Invocamos al modificador setw( ) escribiéndolo después del operador de inserción <<, como si lo fuéramos a enviar al flujo cout, y este a su vez invoca a la función miembro
width( ). Por ejemplo, lo que se muestra a continuación despliega los números 15, 25 y 35
usando la anchura de campo especificada:
cout << “Inicio” << setw( 4 ) << 15 << setw( 4 ) << 25 << setw( 6 ) << 30;
La instrucción anterior produce la siguiente salida:
Inicio 15 25 35
(Hay dos espacios antes del 15, dos antes del 25 y cuatro antes del 30).
El manipulador setprecision( ) hace exactamente lo mismo que la función miembro
precision( ). Sin embargo, las llamadas a setprecision( ) se escriben después del operador
de inserción <<, tal como se hace con el manipulador setw( ). Por ejemplo, lo que sigue
despliega los números que se listan empleando el número de dígitos después del punto decimal que se indica en la llamada a setprecision( ):
cout.setf( ios::fixed );
cout.setf( ios::showpoint );
cout << ‘$’ << setprecision( 2 ) << 10.3 << endl << ‘$’ << 20.5 << endl;
La instrucción anterior produce la siguiente salida salida:
$10.30
$20.50
FUNDAMENTOS – LECCIÓN 6
6-21
MIGUEL Á. TOLEDO MARTÍNEZ
Al igual que con la función miembro precision( ) cuando establecemos el número de
dígitos después del punto decimal empleando el manipulador setprecision( ) el efecto persiste hasta que otra llamada a setprecision( ) o precision( ) cambie el número de dígitos.
C++ proporciona diversos manipuladores de flujo que realizan tareas de formato.
Dichos manipuladores proporcionan capacidades tales como :
Establecimiento de:
•
•
•
•
•
•
•
•
•
•
Base 8, 10 y 16 para números enteros.
Precisiones.
La anchura de campos.
Punto decimal y ceros a la derecha para flotante y doble.
La alineación a la izquierda o derecha.
El caracter de relleno de una salida.
Los indicadores de formato (restablecimiento).
Vaciado de flujo.
Inserción de nuevas líneas en el flujo de salida y vaciado de flujo.
inserción de un carácter nulo en el flujo de salida y el salto de espacios en blanco en el flujo de
entrada.
Estas características se describen en las siguientes secciones.
CAMBIO DE BASE PARA NUMEROS ENTEROS: dec, oct, hex Y setbase( )
Los enteros normalmente se interpretan como valores en base 10. Para cambiar la
base sobre la que se interpretan los enteros en un flujo, inserte el manipulador hex para
establecer la base a hexadecimal (base 16), inserte el manipulador oct para establecer la
base a octal (base 8), inserte el manipulador de flujo dec para restablecer la base a decimal.
La base de un flujo también se puede cambiar por medio del manipulador de flujo
setbase( ), el cual toma un argumento entero de 8, 10 o 16 para establecer la base. Debido a
que setbase( ) toma un argumento, se le llama manipulador de flujo con parámetro. El uso
de setbase( ), o cualquier otro manipulador con parámetro, requiere la inclusión del archivo
de encabezado <iomanip.h>. La base permanece igual hasta que se cambia explícitamente.
Ejemplo 6.24
El siguiente programa: OCTHEX1.CPP, muestra el uso de los manipuladores de flujo hex, oct, dec y
setbase( ).
/* El siguiente programa: OCTHEX1.CPP, ilustra el uso de los manipiladores de flujo oct,
hex, dec y setbase( ).
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para oct, hex, dec y setbase( )
void main( void )
{
Salida:
Introduzca un número decimal: 456
456 en hexadecimal es
: 1c8
456 en octal es
: 710
456 en decimal es
: 456
int n;
cout
FUNDAMENTOS – LECCIÓN 6
<< "Introduzca un número entero: ";
6-22
MIGUEL Á. TOLEDO MARTÍNEZ
cin >> n;
cout
<< n << " en hexadecimal es: " << hex << n << '\n'
<< dec << n << " en octal es
: " << oct << n << '\n'
<< setbase( 10 ) << n << " en decimal es: " << n << endl;
}//Fin de main( )
Ejemplo 6.25
El siguiente programa: OCTHEX2.CPP, de igual manera ilustra el uso de los manipuladores hex,
oct, dec y setbase( ).
/* El siguiente programa. OCTHEX2.CPP, ilustra el uso de los manipuladores oct,
hex, dec, setbase( ).
*/
#include <iostream.h>
#include <iomanip.h>
// Para cout
// oct, dec, hex, setbase( )
Salida:
Octal:
12 24
Hexadecimal: a 14
Decimal:
10 20
Octal:
12
Decimal:
255
Decimal:
10
Hexadecimal: ff
void main( void )
{
// Observe como al utilizar un manipulador el mismo queda activo
Octal:
// hasta que otro manipulador lo desactive.
cout << "Octal:
"
cout << "Hexadecimal: "
cout << "Decimal:
"
cout << endl;
<< oct << 10 << ' ' << 20 << endl;
<< hex << 10 << ' ' << 20 << endl ;
<< dec << 10 << ' ' << 20 << endl;
cout << "Octal:
"
cout << "Decimal:
"
cout << "Decimal:
"
cout << "Hexadecimal: "
cout << endl;
<< oct << 10 << endl;
<< dec << 0xFF << endl;
<< dec << 012 << endl;
<< hex << 255 << endl;
12 24
Hexadecimal: a 14
Decimal:
10 20
// Los siguientes ejemplos muestran el uso del manipulador setbase( )
cout << "Octal:
" << setbase( 8 ) << 10 << ' ' << 20 << endl;
cout << "Hexadecimal: " << setbase( 16 ) << 10 << ' ' << 20 << endl;
cout << "Decimal:
" << setbase( 10 ) << 10 << ' ' << 20 << endl;
} // Fin de main( )
NOTA: Observe que cuando utiliza alguno de los manipuladores dec, oct, hex o setbase( ), el mismo
permanece activo hasta que termine su programa o utilice un manipulador diferente.
PRECISIÓN DE PUNTO FLOTANTE (precision( ), setprecision( ))
Dependiendo de cuando fue escrito su compilador (versión), la precisión se toma
como el número de cifras significativas o el número de dígitos después del punto decimal.
Así, algunos ejemplos de setprecision( 2 ) según la primera opción son:
23.
2.2e7
2.2
6.9e-1
0.00069
0.69
0.69e-4
Ejemplos en base a la segunda opción son:
23.56
FUNDAMENTOS – LECCIÓN 6
2.26e7
2.21
6-23
MIGUEL Á. TOLEDO MARTÍNEZ
En estas lecciones, cuando nos refiramos a precision, estaremos hablando de la segunda opción: número de dígitos después del punto decimal.
Podemos controlar la precisión de los números de punto flotante utilizando el manipulador de flujo setprecision( ) o la función miembro precision( ). Una llamada a cualquiera de éstos establece la precisión para todas las operaciones de salida subsecuentes hasta
cambiar nuevamente la precision. La función miembro precision( ) sin argumentos devue lve el valor de la precisión actual.
Ejemplo 6.26
El siguiente programa: PRECISION1.CPP, ilustra el uso del manipulador setprecision( )
/* El siguiente programa: PRECISION1.CPP, muestra el uso del manipulador setprecision( )
para indicar el número de dígitos decimales del valor flotante.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para setprecision( )
void main( void )
{
float valor = 1.2345;
cout << setiosflags( ios::fixed );
cout << setprecision( 0 ) << valor << endl;
cout << setprecision( 1 ) << valor << endl;
cout << setprecision( 2 ) << valor << endl;
cout << setprecision( 3 ) << valor << endl;
cout << setprecision( 4 ) << valor << endl;
cout << setprecision( 5 ) << valor << endl;
cout << setprecision( 6 ) << valor << endl;
} //Fin de main( )
Salida:
1
1.2
1.23
1.235
1.2345
1.23450
1.234500
Ejemplo 6.27
El siguiente programa: PRECISION2.CPP, utiliza la función miembro precision( ) y el manipulador
setprecision( ) para imprimir una tabla que muestra la raíz cuadrada de 2 con precisiones que varían de 0 a 9. (Ver salida a continuación).
Salida:
Raíz cuadrada de 2 con precisión 0-9.
Precisión establecida por la función miembro precision( ):
1
1.4
1.41
1.414
1.4142
1.41421
1.414214
1.4142136
1.41421356
1.414213562
FUNDAMENTOS – LECCIÓN 6
6-24
MIGUEL Á. TOLEDO MARTÍNEZ
Precision establecida por el manipulador setprecision( ):
1
1.4
1.41
1.414
1.4142
1.41421
1.414214
1.4142136
1.41421356
1.414213562
/* El siguiente programa: PRECISION2.CPP, utiliza la función miembro precision( ) y el
manipulador setprecision( ) para imprimir una tabla que muestra la raíz cuadrada de 2
con precisiones que varían de 0 a 9.
*/
#include <iostream.h>
#include <iomanip.h>
#include <math.h>
//Para cout
//Para precision( ) y setprecision( )
//Para sqrt( )
void main( void )
{
double raiz2 = sqrt( 2.0 );
int posiciones;
cout
<< setiosflags( ios::fixed )
<< "Raíz cuadrada de 2 con precision 0-9.\n"
<< "Precision establecida por la "
<< "funcion miembro precision( ):" << endl;
for( posiciones = 0; posiciones <= 9; posiciones++ )
{
cout.precision( posiciones );
cout
<< raiz2 << '\n';
}//Fin del for
cout
<< "\nPrecision establecida por el "
<< "manipulador setprecision( ):\n";
for( posiciones = 0; posiciones <= 9; posiciones++ )
cout << setprecision( posiciones ) << raiz2 << '\n';
}//Fin de main( )
ANCHURA DE CAMPO (setw( ), width( ))
La función miembro width( ) de ios establece la anchura del campo (es decir, el número de posiciones de carácter con el que un valor deberá enviarse a la salida o el número
de caracteres que deberá introducirse) y devuelve la anchura anterior. Si los valores procesados son más pequeños que la anchura de campo, se insertan caracteres de relleno como
relleno. Un valor más grande que el ancho indicado no se truncará, sino que se imprimirá el
número completo.
FUNDAMENTOS – LECCIÓN 6
6-25
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.28
El siguiente programa: ANCHURA1.CPP, muestra como se usa cout.width( numeroCaracteres ):
/* El siguiente programa: ANCHURA1.CPP, ilustra el uso de La función miembro cout.width( ). */
#include <iostream.h>
// Para cout
void main( void )
{
cout << "Mi número favorito es:";
cout.width( 3 );
cout << 1001 << endl;
cout << "Mi número favorito es:";
cout.width( 4 );
cout << 1001 << endl;
Salida:
Mi número favorito es:1001
Mi número favorito es:1001
Mi número favorito es: 1001
Mi número favorito es: 1001
cout << "Mi número favorito es:";
cout.width( 5 );
cout << 1001 << endl;
cout << "Mi número favorito es:";
cout.width( 6 );
cout << 1001 << endl;
} // Fin de main()
Nota:
Cuando utilice la función miembro width( ), debe de especificar el ancho deseado para cada valor que desee mostrar.
Ejemplo 6.29
Varios de los programas anteriores visualizan números en la pantalla. Para asegurar que los números
se muestren correctamente, el programa debe de manejar los espacios antes y después de los números. Cuando utiliza cout y cerr para visualizar la salida, sus programas pueden especificar el número
mínimo de caracteres requeridos para mostrar la salida utilizando el manipulador setw( ). Para utilizar setw( ) especifique el número mínimo de caracteres que el valor utilizará. Por ejemplo, el programa siguiente ANCHURA2.CPP utiliza el manipulador setw( ) para seleccionar anchos de 3, 4, 5 y
6 caracteres para el valor 1001.
/* El siguiente programa: ANCHURA2.CPP, ilustra el uso del manipulador setw( ). */
#include <iostream.h>
#include <iomanip.h>
// Para cout
// Para setw( )
void main( void )
{
cout << "Mi número favorito es:" <<
cout << "Mi número favorito es:" <<
cout << "Mi número favorito es:" <<
cout << "Mi número favorito es:" <<
} // Fin de main( )
Salida:
Mi número favorito es:1001
Mi número favorito es:1001
Mi número favorito es: 1001
Mi número favorito es: 1001
setw( 3 ) << 1001
setw( 4 ) << 1001
setw( 5 ) << 1001
setw( 6 ) << 1001
<< endl;
<< endl;
<< endl;
<< endl;
Nota: Cuando usa setw( ) para seleccionar un ancho, el ancho es únicamente para el siguiente valor.
Si requiere especificar un ancho para múltiples valores, requerirá utilizar setw( ) para cada
FUNDAMENTOS – LECCIÓN 6
6-26
MIGUEL Á. TOLEDO MARTÍNEZ
uno de ellos. Si el ancho es menor que el número a mostrar, C++ toma la decisión correcta de
utilizar como ancho, el número de dígitos que el número tenga.
Ejemplo 6.30
El siguiente programa: ANCHURA3.CPP, es otro ejemplo en el que se utiliza setw( ). Ver la salida a
continuación:
Salida:
NOMBRE
-- - -
DIRECCION
- - - - - - --
TELEFONO
- -- -- -
// El siguiente programa: ANCHURA3.CPP, muestra el uso del manipulador setw( ).
#include <iostream.h>
#include <iomanip.h>
// Para incluir cout
// Para incluir setw( )
void main( void )
{
// Salta cuatro líneas y muestra los encabezados
cout
<< "\n\n\n\n"
<< setw( 15 ) << "NOMBRE"
<< setw( 22 ) << "DIRECCIÓN"
<< setw( 23 ) << "TELÉFONO" << endl;
cout
<< setw( 14 ) << "----"
<< setw( 22 ) << "-------"
<< setw( 23 ) << "------"
<< endl;
} //Fin de main( )
Ejemplo 6.31
Dado un valor de corriente de 0.001 amperes y un valor de resistencia de 4700.0 ohms, escriba un
programa llamado: ANCHURA4.CPP, para calcular el voltaje usando la ley de Ohm. Escriba los valores de la corriente, resistencia y voltaje usando formato en la salida.
/* El siguiente programa: ANCHURA4.CPP, muestra los valores de la ley de Ohm en forma
tabular.
Salida:
*/
#include <iostream.h>
#include <iomanip.h>
// Para incluir cout
// Para incluir setw( )
void main( void )
{
CORRIENTE
-- - ----- 0.001
RESISTENCIA
----- --- --4700.000
VOLTAJE
- -- ---4.700
// Definición e iniciación de variables
float voltaje
= 0.0;
float corriente = 0.001;
float resistencia = 4700.0;
FUNDAMENTOS – LECCIÓN 6
6-27
MIGUEL Á. TOLEDO MARTÍNEZ
// Cálculo del voltaje
voltaje = corriente * resistencia;
// Muestra los títulos. Se prepara la salida de los valores en formato decimal,
// justificado a la izquierda y con tres decimales de precisión.
cout.setf( ios::fixed|ios::left );
cout.precision( 3 );
cout
<< "\n\n\n\n"
<< setw( 20 ) << "CORRIENTE"
<< setw( 20 ) << "RESISTENCIA"
<< setw( 20 ) << "VOLTAJE" << endl;
cout
<< setw( 20 ) << "---------"
<< setw( 20 ) << "-----------"
<< setw( 20 ) << "-------"
<< endl;
// Muestra valores
//cout.precision( 3 );
¿Qué ocurre si se cambia la precisión a 2 dec?
//cout.unsetf( ios::fixed );
¿Qué ocurre si se desactiva el formato decimal?
cout
<< setw( 20 ) << corriente
<< setw( 20 ) << resistencia
<< setw( 20 ) << voltaje
<< endl;
} //Fin de main( )
Nota: Más adelante se explicará el uso de los miembros cout.setf( ios::fixed | ios::left ), y
cout.unsetf( ios::fixed ).
Ejemplo 6.32
El siguiente programa: ANCHURA5.CPP, muestra el uso de la función miembro width( ) tanto en la
entrada como en la salida. Observe que en la entrada se leerá un máximo de un carácter menos que
la anchura, debido a que se hace la previsión para el carácter nulo que se colocará en la cadena de
entrada. Recuerde que la extracción de flujo termina cuando se encuentra un espacio en blanco que
no está al inicio. El manipulador de flujo setw( ) también puede utilizarse para establecer la anchura
de campo.
Nota: cuando se pide al usuario que introduzca los datos, éste deberá dar una línea de texto y oprimir ENTER seguido del fin de archivo (<ctrl>z en sistemas compatibles con IBM PC,
<ctrl>d en sistemas UNIX y Macintosh). Ver la salida a continuación.
Salida:
Introduzca una oración:
Nuestra vida son los ríos que van a parar a la mar.
Nues
tra
vida
son
los
ríos
que
van
a
para
r
a
la
mar.
FUNDAMENTOS – LECCIÓN 6
6-28
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: ANCHURA5.CPP, ilustra el uso de a función miembro width( ).
*/
#include <iostream.h>
//Para cout y cin
void main( void )
{
int ancho = 5;
char cadena[ 10 ];
cout << "Introduzca una oración:\n";
cin.width( 5 );
while( cin >> cadena )
{
cout.width( ancho++ );
cout << cadena << endl;
cin.width( 5 );
}//Fin del while
}//Fin del main( )
NOTA: Vale la pena que estudie este programa con detenimiento.
EXAMEN BREVE 15
MANIPULADORES DEFINIDOS POR EL USUARIO
Los usuarios pueden crear sus propios manipuladores de flujo.
Ejemplo 6.33
El siguiente programa: MANIPUSUA.CPP, muestra la creación y el uso de los manipuladores de flujo campana, retorno (retorno de carro), tabulador y finLinea. Los usuarios también pueden crear sus
propios manipuladores de flujo con parámetros –consulte los manuales de su instalación para seguir
las instrucciones sobre cómo hacer esto.
/* El siguiente programa: MANIPUSUA.CPP, ilustra el uso de los manipuladores sin parámetros:
campana, retorno (retorno de carro), tabulador y finLinea; definidos por el usuario.
*/
#include <iostream.h>
//Para cout
//Manipulador finLinea (que utiliza la secuencia de escape \n)
//y la función miembro flush)
ostream finLinea( ostream &salida )
{
return salida << '\n' << flush;
}//Fin de finLinea( )
Salida:
Prueba del manipulador tabulador:
a
b
c
Prueba de los manipuladores retorno y
campana:
-----.....
//Manipulador tabulador(que utiliza la secuencia de escape \t)
ostream tabulador( ostream &salida )
{
return salida << '\t';
}//Fin de tabulador( )
FUNDAMENTOS – LECCIÓN 6
6-29
MIGUEL Á. TOLEDO MARTÍNEZ
//Manipulador campana (que utiliza la secuencia de escape \a)
ostream campana( ostream &salida )
{
return salida << '\a';
}//Fin de campana( )
//Manipulador retorno (que utiliza la secuencia de escape \r)
ostream retorno(ostream &salida)
{
return salida << '\r';
}//Fin de retorno( )
void main( void )
{
cout
<< "Prueba del manipulador tabulador:" << finLinea
<< 'a' << tabulador << 'b' << tabulador << 'c' << finLinea
<< "Prueba de los manipuladores retorno y campana:"
<< finLinea << "..........";
cout
<< campana;
cout
<< retorno << "-----" << finLinea;
}//Fin de main( )
ESTADOS DE FORMATO DEL FLUJO
Diversos indicadores de formato (banderas de formato) especifican los tipos de
formato a realizarse durante las operaciones de E/S de flujo. Las funciones miembro setf( ),
unsetf( ) y flags( ) controlan los valores de los indicadores.
setf( ) es una función miembro invocada por el objeto cout. Es una abreviatura de set
flags, que significa establecer banderas (indicadores). A los argumentos de setf( ), como se
ve en la tabla 5.1 se conocen con el nombre de banderas. Originalmente se usó la palabra
bandera para referirse a bits individuales que servían para indicar algo, dependiendo de si
contenían 0 o 1, por analogía con una bandera real que indica algo al estar izada o arriada.
En inglés se dice que un bit se establece (set) si se coloca un 1 en él, y se despeja (clear) si
se coloca un 0 en él. De ahí que se acostumbre decir que la bandera se establece cuando se
enciende o activa. Así pues, cuando la bandera ios::showpoint (por ejemplo) está establecida (es decir, cuando es argumento de setf( )), el flujo que invocó la función setf( ) se comporta como se describe en la tabla 5.1.
La notación ios:: de una bandera significa lo siguiente:
La palabra ios es abreviatura de input/output stream e indica que el significado de
términos como fixed o showpoint es el que tiene cuando se usan con un flujo de entrada o
salida.
La notación :: significa usar el significado de lo que sigue a :: en el contexto de lo
que precede a ::.
FUNDAMENTOS – LECCIÓN 6
6-30
MIGUEL Á. TOLEDO MARTÍNEZ
INDICADORES DE ESTADO DE FORMATO
Cada uno de los indicadores de estado de formato que se muestran en la tabla 5.1 (y
algunos que no se muestran) está definido como una enumeración de la clase ios y se explican en las siguientes secciones.
Estos indicadores pueden controlarse mediante las funciones miembro flags( ),
setf( ) y unsetf( ), pero muchos programadores de C++ prefieren utilizar manipuladores de
flujo. El programador puede emplear la operación OR a nivel de bits, |, para combinar diversas operaciones en un solo valor long. Al llamar a la función miembro flags( ) para un
flujo y especificar opciones de tipo OR establece las opciones de ese flujo y devuelve un
valor long que contiene las opciones anteriores. Este valor se guarda frecuentemente para
que sea posible llamar a flags( ) mediante dicho valor a fin de restaurar las opciones previas
del flujo. Vea el ejemplo 6.45.
La función flags( ) debe especificar un valor que represente los valores de todos los
indicadores. Por otro lado, la función setf( ) de un solo argumento especifica uno o más
indicadores y los une mediante OR con los valores de indicadores existentes para formar un
nuevo estado de formato.
Tabla 5.1. Indicadores de estado de formato para setf( )
Indicador de estado de formato
ios::skipws
ios::left
ios::right
ios::internal
ios::dec
ios::oct
ios::hex
ios::showbase
ios::showpoint
ios::uppercase
FUNDAMENTOS – LECCIÓN 6
Descripción
Se salta los caracteres de espacio en blanco en un
flujo de entrada.
Alinea la salida a la izquierda de un campo. Los caracteres de relleno aparecen a la derecha, en caso
necesario.
Alinea la salida a la derecha de un campo. Los caracteres de relleno aparecen a la izquierda, en caso necesario.
Indica que el signo de un número debe estar alineado
a la izquierda en un campo y la magnitud del número
debe estar alineada a la derecha en ese mismo campo
(es decir, los caracteres de relleno aparecen entre el
signo y el número).
Especifica que los enteros deben tratarse como valores decimales (base 10).
Especifica que los enteros deben tratarse como valores octales (base 8).
Especifica que los enteros deben tratarse como valores hexadecimales (base 16).
Especifica que la base de un número debe aparecer en
la salida al inicio del número (un 0 inicial para los
octales, 0x o 0X inicial para los hexadecimales).
Especifica que los números de punto flotante deben
aparecer en la salida con un punto decimal. Esto se
utiliza normalmente con ios::fixed para garantizar un
determinado número de dígitos a la derecha del punto
decimal.
Especifica que se debe utilizar X mayúscula en 0X
antes de un entero hexadecimal y E mayúscula cuando se representa un valor de punto flotante en nota-
6-31
MIGUEL Á. TOLEDO MARTÍNEZ
ción científica.
Indicador de estado de formato
ios::scientific
Descripción
Especifica que la salida de un valor de punto flotante
debe estar en notación científica.
Especifica que la salida de un valor de punto flotante
debe estar en notación de punto fijo con un número de
dígitos específico a la derecha del punto decimal.
ios::fixed
El manipulador de flujo con parámetros setiosflags( ) realiza las mismas tareas que
la función miembro setf( ). El manipulador de flujo resetiosflags( ) realiza las mismas tareas
que la función miembro unsetf( ). Para utilizar cualquiera de estos manipuladores de flujo
asegúrese de emplear #include <iomanip.h>.
skipws indica que >> deberá saltarse los espacios en blanco en un flujo de entrada.
El comportamiento predeterminado de >> es saltarse los espacios en blanco. Para cambiar
esto, se utiliza la llamada a unsetf( ios::skipws ). También es posible utilizar el manipulador
de flujo ws para especificar que hay que brincarse los espacios en blanco.
CEROS A LA DERECHA Y PUNTOS DECIMALES ( ios::showpoint )
El indicador showpoint se establece para forzar a que un número de punto flotante
aparezca en la salida con su punto decimal y sus ceros a la derecha. Un valor de punto flotante de 79.0 se imprimirá como 79 cuando showpoint no esté establecido y como
79.000000 (o con tantos ceros a la derecha como lo especifique la precisión actual) cuando
sí esté establecido.
Ejemplo 6.34
El siguiente programa: PUNTODEC.CPP, muestra el uso de la función miembro setf( ) para establecer el indicador showpoint, a fin de controlar los ceros a la derecha y la impresión del punto decimal
para los valores de punto flotante. Ver la salida a continuación.
Salida:
Antes de establecer el indicador de ios::showpoint
9.9900 se imprime como: 9.99
9.9000 se imprime como: 9.9
9.0000 se imprime como: 9
Después de establecer el indicador de ios::showpoint
9.9900 se imprime como: 9.99000
9.9000 se imprime como: 9.90000
9.0000 se imprime como: 9.00000
/* El siguiente programa: PUNTODEC.CPP, ilustra el control de la impresión de ceros a la
derecha y puntos decimales para valores de punto flotante.
*/
#include <iostream.h>
//Para cout
void main( void )
{
cout
<< "Antes de establecer el indicador de ios::showpoint\n"
<< "9.9900 se imprime como: " << 9.9900
<< "\n9.9000 se imprime como: " << 9.9000
<< "\n9.0000 se imprime como: " << 9.0000
<< "\n\nDespués de establecer el indicador de ios::showpoint\n";
cout.setf( ios::showpoint );
FUNDAMENTOS – LECCIÓN 6
6-32
MIGUEL Á. TOLEDO MARTÍNEZ
cout
<< "9.9900 se imprime como: " << 9.9900
<< "\n9.9000 se imprime como: " << 9.9000
<< "\n9.0000 se imprime como: " << 9.0000 << endl;
}//Fin de main( )
ALINEACIÓN ( ios::left, ios::right, ios::internal )
Los indicadores left y right permiten que los campos se alineen a la izquierda con
caracteres de relleno a la derecha o se alineen a la derecha con caracteres de relleno a la
izquierda, respectivamente. La función miembro fill( ) o el manipulador de flujo con parámetros setfill( ) especifican los caracteres que van a utilizarse para relleno.
internal indica que el signo de un número (o la base cuando el indicador
ios::showbase esta establecido) deberá estar alineado a la izquierda dentro del campo, la
magnitud del número deberá estar alineada a la derecha y los espacios intermedios deberán
estar rellenos con el carácter de relleno. Los indicadores left, right e internal están contenidos en el dato miembro estático ios::adjustfield. El argumento ios::adjustfield se debe proporcionar como segundo argumento para setf( ) cuando se establecen los indicadores de
alineación left, right o internal. Esto permite que setf( ) se asegure que solamente esté establecido uno de los tres indicadores de alineación (son mutuamente excluyentes)
Ejemplo 6.35
El siguiente programa: DERIZQ1.CPP, muestra el uso de los manipuladores setw( ), setiosflags( ) y
resetiosflags( ), y las funciones setf( ) y unsetf( ), para controlar la alineación a la izquierda o a la derecha de datos enteros en un campo.Ver la salida a continuación.
Salida:
La justificación a la derecha está predeterminada:
12345
USO DE FUNCIONES MIEMBRO
Use setf( ) para establecer ios::left:
12345
Use unsetf ( ) para restablecer el predeterminado:
12345
USO DE MANIPULADORES CON PARAMETROS
Use setiosflags( ) para establecer ios::left:
12345
Use resetiosflags( ) para restablecer el predeterminado:
12345
/* El siguiente programa: DERIZQ1.CPP, muestra el uso de los manipuladores setw( ), setiosflags( )
y resetiosflags( ), y las funciones miembro setf( ) y unsetf( ), para controlar la alineación a la
izquierda o a la derecha de datos enteros en un campo.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para los manipuladores
void main( void )
{
int x = 12345;
FUNDAMENTOS – LECCIÓN 6
6-33
MIGUEL Á. TOLEDO MARTÍNEZ
cout
<< "La justificación a la derecha está predeterminada:\n"
<< setw( 10 ) << x << "\n\nUSO DE FUNCIONES MIEMBRO"
<< "\nUse setf( ) para establecer ios::left:\n" << setw( 10 );
cout.setf( ios::left, ios::adjustfield );
cout
<< x << "\ nUse unsetf( ) para restablecer el predeterminado:\n";
cout.unsetf( ios::left );
cout
<< setw( 10 ) << x
<< "\n\nUSO DE MANIPULADORES CON PARÁMETROS"
<<"\nUse setiosflags( ) para establecer ios::left:\n"
<< setw( 10 ) << setiosflags( ios::left ) << x
<< "\nUse resetiosflags( ) para restablecer el predeterminado:\n"
<< setw( 10 ) << resetiosflags( ios::left )
<< x << endl;
}//Fin de main( )
Ejemplo 6.36
El siguiente programa: DERIZQ2.CPP, muestra el uso del manipulador setiosflags( )
/* El siguiente programa: DERIZQ2.CPP, muestra el uso del manipulador setiosflags( ).
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para setiosflags()
Salida:
Este mensaje, por omisión, se justifica a la derecha.
1 2 3
Este mensaje se justifica a la izquierda.
1 2 3
void main( void )
{
cout.width( 60 );
cout << "Este mensaje, por omisión, se justifica a la derecha.\n";
cout.width( 5 );
cout << setiosflags( ios::right ) << 1;
cout.width( 5 );
cout << setiosflags( ios::right ) << 2;
cout.width( 5 );
cout << setiosflags( ios::right ) << 3 << "\n\n";
cout.width( 60 );
cout << setiosflags( ios::left ) << "Este mensaje se justifica a la izquierda.";
cout << '\n';
//Es interesante observar que ocurre si este carácter \n
//se agrega al final del mensaje anterior, en vez de
//hacerlo en forma independiente como lo hicimos.
cout.width( 5 );
cout << setiosflags( ios::left ) << 1;
cout.width( 5 );
cout << setiosflags(ios::left) << 2;
cout.width( 5 );
cout << setiosflags( ios::left ) << 3 << '\n';
} //Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-34
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.37
El siguiente programa: RESTAURA.CPP, restaura la justificación a la derecha que es el valor por
omisión, utilizando el manipulador resetiosflags( )
/* El siguiente programa: RESTAURA.CPP, utiliza el manipulador resetiosflags( )
para regresar la justificación a su valor por omisión.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para resetiosflags( ) y setiosflags( )
Salida:
5
5
void main( void )
{
cout.width( 5 );
cout << setiosflags( ios::left ) << 5 << endl;
1
cout.width( 5 );
cout << 5 << '\n' << resetiosflags( ios::left )<< endl;
cout.width( 5 );
cout << 1;
} //Fin de main( )
Ejemplo 6.38
El siguiente programa: ESPACIOINT.CPP, muestra el uso de los manipuladores de flujo setiosflags( ) y setw( ) para especificar el espacio interno. Observe el uso del indicador ios::showpos para
forzar la impresión del signo de más.
/* El siguiente programa: ESPACIOINT.CPP, muestra la visualización de un entero con espaciado
interno y forzado del signo de mas.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para setiosflags( ) y setw( )
Salida:
+ 123
void main( void )
{
cout
<< setiosflags( ios::internal | ios::showpos )
<< setw( 10 ) << 123 << endl;
}//Fin de main( )
RELLENO (fill( ), setfill( ))
La función miembro fill( ) especifica el carácter de relleno que se debe utilizar con
los campos alineados, si no se especifica ningún valor, se emplean espacios para el relleno.
La función miembro fill( ) devuelve el carácter de relleno anterior. El manipulador setfill( )
también establece el carácter de relleno.
FUNDAMENTOS – LECCIÓN 6
6-35
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.39
El siguiente programa: RELLENO1.CPP, muestra el uso de la función miembro fill( ) y del manipulador setfill( ) para controlar el establecimiento y restablecimiento del carácter de relleno. Ver salida
a continuación.
Salida:
10000 impreso como un int justificado a derecha e
izquierda y como hex con jusstificación interna.
Utilizando el caracter predeterminado de relleno (espacio):
10000
10000
0x 2710
Utilizando varios caracteres de relleno:
*****10000
10000%%%%%
0x^^^^2710
/* El siguiente programa: RELLENO1.CPP, ilustra el uso de la función miembro fill( ) y del
manipulador setfill( ) para cambiar el carácter de relleno para los campos que sean mas
grandes que los valores que se están imprimiendo.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
// setfill( )
void main( void )
{
int x = 10000;
cout
<< x << " impreso como un int justificado a derecha e\n"
<<
"izquierda y como hex con justificación interna.\n"
<< "Utilizando el carácter predeterminado de relleno (espacio): \n";
cout.setf( ios::showbase );
cout
<< setw( 10 ) << x << '\n';
cout.setf( ios::left, ios::adjustfield );
cout
<< setw( 10 ) << x << '\n';
cout.setf( ios::internal, ios::adjustfield );
cout
<< setw( 10 ) << hex << x;
cout
<< "\n\nUtilizando varios caracteres de relleno:\n";
cout.setf( ios::right, ios::adjustfield );
cout.fill( '*' );
cout << setw( 10 ) << dec << x << '\n';
cout.setf( ios::left, ios::adjustfield );
cout
<< setw( 10 ) << setfill( '%' ) << x << '\n';
cout.setf( ios::internal, ios::adjustfield );
cout
<< setw( 10 ) << setfill( '^' ) << hex << x << endl;
}//Fin de main( )
Ejemplo 6.40
El siguiente programa: PUNTOS.CPP, ilustra el uso de la función miembro fill( ) y del manipulador
setfill( )
FUNDAMENTOS – LECCIÓN 6
6-36
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: PUNTOS.CPP, muestra el uso de los miembros cout.width( ), cout.fill( )
y del manipulador setfill( ), para rellenar con puntos el lado izquierdo de la
salida.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout y cout.fill( )
//Para setfill( )
void main( void )
{
// Uso de los miembros cout.fill( ) y cout.width( )
cout.fill( '.' );
cout.width( 5 );
cout << 1 << '\n';
cout.width( 6 );
cout << 2 << '\n';
Salida:
....1
.....2
......3
cout.width( 7 );
cout << 3 << "\n\n";
....1
.....2
......3
//Uso del mipulador setfill( )
cout.width( 5 );
cout << setfill( '.' ) << 1 << '\n';
cout.width( 6 );
cout << 2 << '\n';
cout.width( 7 );
cout << 3 << '\n';
} //Fin de main( )
BASE DE FLUJO INTEGRAL (ios::dec, ios::oct, ios::hex, ios::showbase)
El miembro estático ios::basefield (que se utiliza en forma similar a ios::adjustfield
con setf( )) incluye los bits de indicador ios::oct, ios::hex e ios::dec para especificar que
los enteros deben tratarse como valores octales, hexadecimales y decimales, respectivamente. Las inserciones de flujo son decimales en forma predeterminada si ninguno de estos bits
está establecido. El valor predeterminado para las extracciones de flujo es procesar los datos en la forma en que se proporcionan –los enteros que comienzan con 0 se tratan como
valores octales, los enteros que comienzan con 0x o 0X se tratan como valores hexadecimales y todos los demás enteros se tratan como valores decimales. Una vez que se ha especificado una base particular para un flujo, todos los enteros de ese flujo se procesan con dicha
base hasta que se especifique una nueva o hasta que llegue el final del programa.
Establezca el indicador showbase para forzar la base de un valor entero que se va a
enviar a la salida. Los números decimales se envían a la salida en forma normal, los números octales se envían con un 0 inicial y los números hexadecimales se envían con un 0x
inicial o con 0X inicial (el indicador uppercase determina cuál opción se elige)
FUNDAMENTOS – LECCIÓN 6
6-37
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.41
El siguiente programa: OCTHEX3.CPP, muestra el uso del indicador showbase para forzar que un
entero se imprima en forma decimal, octal y hexadecimal.
/* El siguiente programa: OCTHEX3.CPP, muestra el uso del indicador showbase para
forzar que un entero se imprima en forma decimal, octal y hexadecimal.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para setf
void main(void)
{
Salida:
Imprimiendo números enteros precedidos por su base:
100
0144
0x64
int x = 100;
cout
<< setiosflags(ios::showbase)
<< "Imprimiendo números enteros precedidos por su base:\n"
<< x << '\n'
<< oct << x << '\n'
<< hex << x << endl;
}//Fin de main()
NÚMEROS DE PUNTO FLOTANTE, NOTACIÓN CIENTÍFICA (ios::scientific, ios::fixed)
El indicador ios::scientific y el indicador ios::fixed están contenidos en el dato
miembro estático ios::floatfield (que se utiliza en forma similar a ios::adjustfield e
ios::basefield en setf( )). Estos indicadores se utilizan para controlar el formato de salida de
los números de punto flotante. El indicador scientific se establece para forzar la salida de
un número de punto flotante en formato científico. El indicador fixed fuerza que un número
de punto flotante se despliegue con un número específico de dígitos (tal como lo especifica
la función miembro precision( )) a la derecha del punto decimal. Si no están establecidos
estos indicadores, el valor del número de punto flotante determina el formato de la salida.
La llamada cout.setf( 0, ios::floatfield ) restaura el formato predeterminado del sistema para la salida de números de punto flotante.
Ejemplo 6.42
El siguiente programa: PUNTOFLOT1.CPP, muestra el despliegue de números de punto flotante en
formatos fijo y científico utilizando setf( ) con dos argumentos con ios::floatfield. Ver salida a continuación:
Salida:
Desplegado en formato predeterminado:
0.00123457 1.946e+09
Desplegado en formato científico:
1.234567e-03 1.946000e+09
Desplegado en formato predeterminado después de unsetf( ):
0.00123457 1.946e+09
Desplegado en formato fijo:
0.001235
1946000000.000000
FUNDAMENTOS – LECCIÓN 6
6-38
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: PUNTOFLOT1.CPP, despliega valores en punto flotante en el formato
predeterminado del sistema, científico y fijo.
*/
#include <iostream.h>
//Para cout
void main( void )
{
double x = .001234567, y = 1.946e9;
cout
<< "Desplegado en formato predeterminado:\n"
<< x << '\t' << y << '\n';
cout.setf( ios::scientific, ios::floatfield );
cout
<< "Desplegado en formato científico:\n"
<< x << '\t' << y << '\n';
cout.unsetf( ios::scientific );
cout
<< "Desplegado en formato predeterminado después de unsetf( ):\n"
<< x << '\t' << y << '\n';
cout.setf( ios::fixed, ios::floatfield );
cout
<< "Desplegado en formato fijo:\n"
<< x << '\t' << y << endl;
}//Fin de main()
Ejemplo 6.43
Otra opción para visualizar valores flotantes en notación de punto decimal fijo o notación exponencial (científica), es utilizando los manipuladores setiosflags( ) y los indicadores fixed y scientific de
la clase ios. El siguiente programa: PUNTOFLOT2.CPP, muestra el uso de estos manipuladores.
/* El siguiente programa: PUNTOFLOT2.CPP, muestra el uso del manipulador setiosflags( )
y los indicadores fixed y scientific de la clase ios.
*/
#include <iostream.h>
#include <iomanip.h>
//Para cout
//Para setiosflags( )
void main( void )
{
Salida:
0.000123
1.230000e-04
float valor = 0.000123;
cout << setiosflags( ios::fixed ) << valor << endl;
cout << setiosflags( ios::scientific ) << valor << endl;
} //Fin de main( )
CONTROL DE MAYÚSCULAS / MINÚSCULAS (ios::uppercase)
El indicador ios::uppercase se establece para forzar que se envía a la salida una X o
E mayúsculas con los enteros hexadecimales o con los valores de punto flotante en notación científica, respectivamente.
FUNDAMENTOS – LECCIÓN 6
6-39
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.44
El siguiente programa: MAYÚSCULA.CPP, ilustra el uso del indicador ios::uppercase.
/* El siguiente programa: MAYUSCULA.CPP, ilustra el uso del indicador ios::uppercase */
#include <iostream.h>
#include <iomanip.h>
void main( void )
{
cout
//Para cout
//Para setiosflags( )
Salida:
Imprimiendo letras mayúsculas en los exponentes de la
notación científica y en los valores hexadecimales:
4.345e+10
75BCD15
<< setiosflags( ios::uppercase )
<< "Imprimiendo letras mayúsculas en los exponentes de la\n"
<< "notación científica y en los valores hexadecimales:\n"
<< 4.345e10 << '\n' << hex << 123456789 << endl;
}//Fin de main( )
ESTABLECIMIENTO Y RESTABLECIMIENTO DE LOS INDICADORES DE FORMATO (flags( ),
setiosflags( ), resetiosflags( ))
La función miembro flags( ) sin argumentos devuelve simplemente (como valor
long) los valores actuales de los indicadores de formato. La función miembro flags( ) con
un argumento long establece los indicadores de formato, tal como los especifica el argumento, y devuelve el valor anterior de los indicadores. Cualquier indicador de formato que
no esté especificado en el argumento de flags( ) es restablecido. Observe que los valores
iniciales de los indicadores de cada sistema pueden variar.
La función miembro setf( ) establece los indicadores de formato proporcionados en
su argumento y devuelve como un valor long los valores anteriores de la indicadores como
en:
long formatoOriginal = cout.set( ios::showpoint | ios:: showpos );
La función miembro setf( ) con dos argumentos long como en:
cout.setf( ios::left, ios::adjustfield );
Primero borra los bits de ios::adjustfield y luego establece el indicador ios::left. Esta versión de setf( ) se utiliza con los campos de bits asociados con ios::basefield (representado por ios::dec, ios::oct e ios::hex), ios::floatfield (representado por ios::scientific e
ios::fixed) e ios::adjustfield (representados por ios::left, ios::right e ios::internal).
La función miembro unsetf( ) restablece los indicadores designados y devuelve el
valor de dichos indicadores antes de restablecerlos.
Ejemplo 6.45
El siguiente programa: RESTAURA2.CPP, muestra el uso de la función miembro flags( ) para restablecer un nuevo estado de formato y guardar el estado de formato anterior, y luego restaura los valores de formato originales.
FUNDAMENTOS – LECCIÓN 6
6-40
MIGUEL Á. TOLEDO MARTÍNEZ
Salida:
El valor de la variable flags es: 8193
Imprime datos int y double en el formato original:
1000 0.0947628
El valor de la variable flags es: 4040
Imprime datos int y double en el nuevo formato
especificado utilizando la función miembro flags:
1750 9.476280e-02
El valor de la variable flags es: 8193
Imprime de nuevo datos int y double en el formato original:
1000 0.0947628
/* El siguiente programa: RESTAURA2.CPP, ilustra el uso de la función miembro flags( ). */
#include <iostream.h>
//Para cout
void main( void )
{
int i
= 1000;
double d = 0.0947628;
cout
<< "El valor de la variable flags es: "
<< cout.flags( )
<< "\nImprime datos int y double en el formato original:\n"
<< i << '\t' << d << "\n\n";
long formatoOriginal = cout.flags( ios::oct | ios::scientific );
cout
<< "El valor de la variable flags es: "
<< cout.flags( )
<< "\nImprime datos int y double en el nuevo formato\n"
<< "especificado utilizando la función miembro flags:\n"
<< i << '\t' << d << "\n\n";
cout.flags( formatoOriginal );
cout
<< "El valor de la variable flags es: "
<< cout.flags( )
<< "\nImprime de nuevo datos int y double en el formato original:\n"
<< i << '\t' << d << endl;
}//Fin de main( )
EXAMEN BREVE 16
ESTADOS DE ERROR DE FLUJO
El estado de un flujo puede probarse por medio de los bits de la clase ios –la clase
base de las clases istream, ostream e iostream que estamos utilizando para E/S.
El indicador eofbit se establece automáticamente para un flujo de entrada cuando se
encuentra el fin de archivo. Un programa puede utilizar la función miembro eof( ) para determinar si se ha encontrado el fin de archivo en un flujo. La llamada
FUNDAMENTOS – LECCIÓN 6
6-41
MIGUEL Á. TOLEDO MARTÍNEZ
cin.eof( );
devuelve true si se ha encontrado el fin de archivo en cin, y false en caso contrario.
failbit se establece para un flujo cuando sucede un error de formato en el flujo pero
no se ha perdido caracteres. La función miembro fail( ) determina si una operación de flujo
ha fallado, y por lo general es posible recuperarse de esos errores.
badbit se establece para un flujo cuando sucede un error que da por consecuencia la
pérdida de datos. La función miembro bad( ) determina si ha fallado una operación de flujo.
Tales fallas serias por lo general no son recuperables.
goodbit se establece para un flujo si ninguno de los bits eofbit, failbit o badbit se ha
establecido para dicho flujo.
La función miembro good( ) devuelve true si las funciones bad( ), fail( ) y eof( )
devuelven false. Las operaciones de E/S sólo deben realizarse en flujos buenos.
La función miembro rdstate( ) devuelve el estado de error de flujo. Por ejemplo, una
llamada a cout.rdstate devolvería el estado de flujo que luego podría probarse por medio de
una instrucción switch que examinara a ios::eofbit, ios::badbit, ios::failbit e ios::goodbit.
El medio preferido para probar el estado de un flujo es utilizar las funciones miembros
eof( ), bad( ), fail( ) y good( ), ya que su uso no requiere que el programador esté familiarizado con los bits de estado particulares.
La función miembro clear( ) se utiliza normalmente para restaurar el estado de un
flujo a bueno para que la E/S pueda continuar en ese flujo. El argumento predeterminado
para clear( ) es ios::goodbit, por lo que la instrucción:
cin.clear( );
borra o limpia a cin y establece el flujo a goodbit. La instrucción:
cin.clear( ios::failbit )
Establece, de hecho, a failbit. Tal vez el usuario desee hacer esto cuando realiza entrada desde cin con un tipo definido por el usuario y encuentra un problema. El nombre
clear( ) parece inadecuando en este contexto, pero es correcto.
Si badbit, failbit (o ambos) están establecidos, la función miembro operador! devuelve true. Si badbit, failbit (o ambos) están establecidos, la función miembro operator
void * devuelve false. Estas funciones son útiles en el procesamiento de archivos cuando se
está probando una condición verdadera /falsa en la condición de una estructura de selección o una estructura de repetición.
FUNDAMENTOS – LECCIÓN 6
6-42
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.46
El siguiente programa: EDOERROR.CPP, ilustra el uso de las funciones miembro rdstate( ), eof( ),
fail( ), bad( ), good( ) y clear( ).
Salida:
Antes de una operación de entrada de datos errónea:
cin.rdstate(): 0
cin.eof(): 0
cin.fail(): 0
cin.bad(): 0
cin.good(): 1
Se espera un número entero pero se introdujo un carácter: a
Después de una operación de entrada de datos errónea:
cin.rdstate(): 2
cin.eof(): 0
cin.fail(): 2
cin.bad(): 0
cin.good(): 0
Después de cin.clear()
cin.fail(): 0
cin.good(): 1
/* El siguiente programa: EDOERROR.CPP, ilustra el uso de la prueba de estados de error. */
#include <iostream.h>
//Para cout
void main( void )
{
int x;
cout
<< "Antes de una operación de entrada de datos errónea:"
<< "\n cin.rdstate( ): " << cin.rdstate( )
<< "\n cin.eof( ): " << cin.eof( )
<< "\n cin.fail( ): " << cin.fail( )
<< "\n cin.bad( ): " << cin.bad( )
<< "\n cin.good( ): " << cin.good( )
<< "\n\nSe espera un número entero pero se introdujo un carácter: ";
cin >> x;
cout
<< "\nDespués de una operación de entrada de datos errónea:"
<< "\n cin.rdstate( ): " << cin.rdstate( )
<< "\n cin.eof( ): " << cin.eof( )
<< "\n cin.fail( ): " << cin.fail( )
<< "\n cin.bad( ): " << cin.bad( )
<< "\n
cin.good( ): " << cin.good( ) << "\n\n";
cin.clear( );
cout
<< "Después de cin.clear( )"
<< "\ncin.fail( ): " << cin.fail( )
<< "\ncin.good( ): " << cin.good( ) << endl;
}//Fin de main( )
FUNDAMENTOS – LECCIÓN 6
6-43
MIGUEL Á. TOLEDO MARTÍNEZ
ENLACE DE UN FLUJO DE SALIDA A UN FLUJO DE ENTRADA
Las aplicaciones interactivas involucran, por lo general, a un istream para entrada y
a un ostream para salida. Cuando aparece un mensaje de petición en la pantalla, el usuario
responde dando los datos adecuados. Obviamente la petición necesita aparecer antes de que
continúe la operación de entrada. Con el almacenamiento temporal de salida, ésta aparece
solamente cuando el búfer está lleno, cuando la salida se vacía en forma explicita por requerimientos del programa, o bien en forma automática al final del programa. C++ proporciona la función miembro tie( ) para sincronizar (es decir, enlazar) la operación de un istream y un ostream para asegurarse de que la salida aparezca antes de su entrada subsecuente. La llamada:
cin.tie( &cout );
enlaza a cout (un ostream) con cin (un istream). De hecho, esta llamada particular es redundante, debido a que C++ realiza automáticamente esta operación para crear un entorno
de entrada/salida estándar para el usuario. Sin embargo, el usuario podría enlazar explícitamente otros pares de istream/ostream. Para desenlazar un flujo de entrada, flujoEntrada,
de un flujo de salida se utiliza la llamada:
flujoEntrada.tie( 0 );
CONCEPTO DE PRECISIÓN
Dentro de la computadora, los números son representados utilizando combinaciones
de UNOS y CEROS (dígitos binarios) Ya que una determinada clase tiene un número fijo
de bits, cada clase puede contener exclusivamente un cierto rango de valores. Si asigna un
valor fuera de este rango, ocurre un error de sobreflujo. Los valores de punto flotante pueden experimentar sobreflujo y de precisión insuficiente. Por ejemplo, los valores de clase
flotante, proporcionan 6 o 7 dígitos significativos. De esta manera si asigna el valor
1.234567890 a una variable de clase flotante, el valor aproximado es de 1.23456. Los valores de tipo doble, por otra parte proporcionan 14 a 15 dígitos significativos, de tal manera
que el valor anterior se almacena completamente en la memoria.
Cuando trabaje con números de punto flotante, debe de tener cuidado del hecho de
que sus valores son representados utilizando un número fijo de bits. De esta manera, es
imposible para la computadora, representar siempre un número de manera exacta. Por
ejemplo, el valor 0.4, puede que lo represente como 0.3999999, o el valor 0.1 como
0.099999.
Ejemplo 6.47
El siguiente programa, PRECISE.CPP, ilustra la diferencia entre precisión sencilla y doble precisión:
FUNDAMENTOS – LECCIÓN 6
6-44
MIGUEL Á. TOLEDO MARTÍNEZ
/* El siguiente programa: PRECISE.CPP, ilustra la diferencia entre precisión
sencilla y doble precisión.
*/
Salida:
Valor en precisión sencilla: 0.1234567910432815550
#include <iostream.h>
#include <iomanip.h>
//Para cout
Valor en doble precisión : 0.1234567890987654380
//Para setprecision( )
void main ( void )
{
float aproximado
= 0.1234567890987654321;
double masAproximado = 0.1234567890987654321;
cout.setf( ios::fixed );
cout << "Valor en precisión sencilla: " <<
setprecision( 19 ) << aproximado << endl;
cout << "Valor en doble precisión : " <<
setprecision( 19 ) << masAproximado << endl;
} //Fin de main()
Observe los valores mostrados y compárelos con los valores iniciales.
CONCEPTO DE SOBRE FLUJO
La clase de una variable define el rango de valores que puede almacenar y las operaciones que pueden realizarse con las mismas. Las variables de clase int, por ejemplo,
pueden almacenar valores en el rango de –32768 a 32767. Si asigna un valor fuera de este
rango a una variable de clase int, ocurrirá un error de sobre flujo. C++ utiliza 16 bits para
representar una variable de tipo int. El bit más significativo se utiliza para determinar el
signo de la variable. Si el bit más significativo es 0, el valor es positivo. Si es 1 el valor es
negativo. C++ utiliza, por lo tanto 15 bits para representar el valor de la variable. Para
comprender porque ocurre un sobre flujo, trabajemos a nivel bit, para lo cual considere los
siguientes valores:
0
1
2
3
4
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0000
0001
0010
0011
0100
32765
32766
32767
0111
0111
0111
1111
1111
1111
1111
1111
1111
1101
1110
1111
Si le agrega 1 al valor 32767, el resultado que espera es 32768. Sin embargo, C++
muestra el valor –32768, como se muestra a continuación:
32767
+ 1
- 32768
FUNDAMENTOS – LECCIÓN 6
0111
0000
10000
1111
0000
0000
1111
0000
0000
1111
0001
0000
6-45
MIGUEL Á. TOLEDO MARTÍNEZ
Ejemplo 6.48
El siguiente programa: OVERFLOW.CPP, ilustra como ocurre un sobreflujo:
/* El siguiente programa: OVERFLOW.CPP, ilustra el concepto de sobreflujo. */
#include <iostream.h>
//Para cout
void main (void)
{
Salida:
32767 + 1 = -32768
-32768 - 1 = 32767
int positivo = 32767;
int negativo = -32768;
cout << positivo << " + 1 = " << (positivo + 1) << endl;
cout << negativo << " - 1 = " << (negativo - 1) << endl;
} //Fin de main()
Observe como la suma de 32767 + 1 = -32768 y -32768 –1 = 32767, lo cual obviamente es un error.
Es importante hacer notar que los errores por sobreflujo no son notificados por el compilador.
UTILIZANDO cout PARA IMPRIMIR INFORMACIÓN
Todas las operaciones de salida que se han visto hasta ahora harán que la información se muestre en la pantalla de su monitor. ¿Cómo imprimir su información en la impresora? Para realizar la salida a la impresora en C++, debemos escribir el código para crear
nuestro propio objeto digamos imprimir. A continuación se muestra un objeto imprimir
que se desarrolló para la plataforma DOS:
//***********************************************************************
// Este segmento de código define imprimir como un flujo de salida
// direccionándolo al puerto de su impresora (PRN) (sólo para DOS).
//***********************************************************************
ofstream imprimir;
// Define imprimir como un flujo de archivos de salida
imprimir.open( "PRN" );
// Abre un archivo de impresión y lo direcciona a PRN
if( !imprimir )
// Se asegura que la impresora esté lista
{
cout << "Hay un problema con la impresora." << endl;
exit( 1 );
}
Este código trabajará solamente en plataformas DOS debido a la referencia específica para el archivo de impresora PRN. Por el momento no es importante que entienda el
código anterior orientado a objetos. Ya habrá tiempo. El código simple mente crea un objeto
flujo imprimir y lo liga a la impresora de su sistema. Se usará el flujo imprimir como un
flujo de salida justo como se ha usado el flujo cout. De esta manera, el formato del enunciado imprimir es como sigue:
imprimir << elemento1 << elemento2 << elementos3 << ... << elementoN;
FUNDAMENTOS – LECCIÓN 6
6-46
MIGUEL Á. TOLEDO MARTÍNEZ
Como puede observar, los elementos que se imprimirán se insertan dentro del flujo
imprimir usando el operador de inserción << de la misma forma que se insertaron en el
flujo cout. Lo que sigue es un ejemplo que muestra el uso de imprimir.
Ejemplo 6.49
El siguiente programa: IMPRIMIR.CPP, cuya información de salida es la misma que la del programa OHM.CPP (resuelto anteriormente) se muestre en la impresora y no en la pantalla del monitor:
/* El siguiente programa: IMPRIMIR.CPP, calcula e imprime el voltaje usando la
ley de Ohm.
*/
#include <iostream.h>
#include <fstream.h>
#include <process.h>
// Para incluir cout
// Para el objeto imprimir
// Para exit( )
void main(void)
{
//*********************************************************************
// Este segmento de código define print como un archivo de salida
// direccionándolo al puerto de su impresora (PRN) (sólo para DOS).
//*********************************************************************
ofstream imprimir;
// Define imprimir como un flujo de archivos de salida
imprimir.open( "PRN" );
// Abre un archivo de impresión y lo direcciona a PRN
if( !imprimir )
// Se asegura que la impresora esté lista
{
cout << "Hay un problema con la impresora.\n";
exit( 1 );
}
float voltaje
float corriente
float resistencia
= 0.0;
= 0.001;
= 4700.0;
cout << "Este programa calculará el voltaje determinado por una corriente de \n"
"0.001 amperes y una resistencia de 4700 ohms" << endl;
voltaje = corriente * resistencia;
imprimir << "Determinado un valor de corriente de " << corriente <<
" amperes y un\ n"
"valor de resistencia de " << resistencia << " ohms, el \n"
"voltaje resultante es " << voltaje << " volts" << endl;
imprimir.close( );
} // Fin de main( )
// Cierra el archivo de impresión
Ahora la salida del programa se generará en su impresora y no en la pantalla del monitor. Observe
que el código de definición de flujo imprimir se ha integrado dentro de la función main( ) del programa. En resumen, para que este código se compile en forma adecuada se incluyen los archivos de
cabecera <fstream.h> y <process.h>. Una vez que se crea el objeto flujo imprimir, simplemente úse-
FUNDAMENTOS – LECCIÓN 6
6-47
MIGUEL Á. TOLEDO MARTÍNEZ
lo como cout para escribir la información en la impresora. Esto es porque nuestro objeto imprimir ha
heredado las mismas características y comportamiento contenidos en el objeto cout. Ambos son objetos de la clase iostream estándar. Al final de main( ) encontrará el enunciado imprimir.close( ). Este enunciado es necesario para cerrar el archivo de impresión.
FUNDAMENTOS – LECCIÓN 6
6-48
MIGUEL Á. TOLEDO MARTÍNEZ
LO QUE NECESITA SABER
Antes de continuar con la siguiente lección, asegúrese de haber comprendido los
siguientes conceptos:
q Las operaciones de E/S se realizan tomando en cuenta el tipo de los datos.
q La E/S en C++ se realiza en flujos de bytes . Un flujo es simplemente una secuencia de bytes.
q Los mecanismos de E/S del sistema mueven bytes desde los dispositivos hacia memoria y viceversa, en forma eficiente y confiable .
q C++ proporciona capacidades de E/S de bajo nivel y de alto nivel. Las capacidades de E/S de
bajo nivel especifican que alguna cantidad de bytes debe transferirse desde un dispositivo hacia
memoria o desde memoria hacia un dispositivo. La E/S de alto nivel se realiza con bytes agrupados en unidades significativas, tales como enteros, números de punto flotante, caracteres, cadenas y tipos definidos por el usuario.
q C++ proporciona operaciones de E/S sin formato y formateadas. Las transferencias de E/S sin
formato son rápidas, pero procesan datos sin formato que a los usuarios se les dificulta utilizar.
La E/S formateada procesa datos en unidades significativas, pero requiere un tiempo de proceso
adicional que puede impactar negativamente las transferencias de datos de alto volumen.
q El flujo de salida cout le permite visualizar caracteres y números.
q La salida de datos es a cout (el flujo de salida estándar), que por lo general es la pantalla de la
computadora, pero cout puede conectarse a otros dispositivos.
q Su programa puede visualizar una nueva línea utilizando ‘\n’ o ‘endl’. Además, su programa
puede utilizar otros caracteres especiales como ‘\t’ para producir un tabulador.
q Los manipuladores dec, oct, hex le permiten a su programa visualizar valores en decimal, octal
y hexadecimal. Utilizando el manipulador setw( ), su programa puede controlar el ancho de su
valor de salida.
q Utilizando el flujo de salida cerr, su programa puede escribir mensajes al dispositivo estándar
de errores de su sistema operativo.
q La obtención de información hacia su computadora se llama lectura y la obtención de informa ción de su sistema hacia afuera se llama escritura.
q El enunciado C++ usado para lectura es cin y el enunciado que se usa para escribir es cout.
q Cada enunciado cout debe incluir una lista de los elementos que se van a escribir. Los elemen tos en la lista deben separarse por el operador de inserción de flujo <<.
q El enunciado cout se puede usar para escribir cualquier información variable o fija. La informa ción numérica fija se escribe simplemente listando los valores numéricos en la lista de elementos
de cout. Cuando se escribe información de caracteres fijos, la información escrita debe encerrarse entre comillas sencillas para caracteres individuales y doble comillas para cadenas de caracteres.
q Cuando escriba información variable, el identificador de la variable deberá listarse dentro del
enunciado cout. La información se escribirá en la pantalla del monitor o impresora del sistema.
q Use el objeto de flujo de salida preconstruido cout para escribir hacia la pantalla del monitor, pe ro deberá definir su propio objeto de flujo de impresión para escribir hacia la impresora.
q Para dar formato a una salida, a menudo se incluye un manipulador de E/S en la lista de elemen tos. Como un ejemplo, el manipulador de ancho de campo setw( ) se incluye antes del elemento
que se va a escribir para ajustar el ancho de campo del elemento.
q El valor del manipulador de ancho de campo especifica el número de columnas de salida que
serán asignadas para el elemento que se escriba.
q Deberá incluir el archivo de encabezado <iomanip.h> cuando use el manipulador de E/S setw( ).
Además, siempre use un diseño de pantalla para acomodar su resultado y determinar los valores
correctos de ancho de campo.
FUNDAMENTOS – LECCIÓN 6
6-49
MIGUEL Á. TOLEDO MARTÍNEZ
q La mayoría de los programas de C++ incluyen el archivo de encabezado <iostream.h> que contiene la información básica requerida por todas las operaciones de E/S de flujo.
q El archivo de encabezado <iomanip.h> contiene información para la entrada/salida formateada
con manipuladores de flujo con parámetros.
q El archivo de encabezado <fstream.h> contiene información para las operaciones de proces amiento de archivos.
q La clase istream soporta las operaciones de entrada de flujo.
q La clase ostream soporta las operaciones de s alida de flujo.
q La clase iostream soporta las operaciones de entrada y de salida de flujo.
q Las clases istream y ostream están derivadas mediante herencia simple a partir de la clase base
ios.
q La clase iostream está derivada mediante herencia múltiple a partir de las clases istream y ostream.
q El operador de desplazamiento a la izquierda (<<) está sobrecargado para designar la salida de
flujo y se le menciona como el operador de inserción de flujo.
q El operador de desplazamiento a la derecha (>>) está sobrecargado para designar la entrada de
flu jo y se le menciona como el operador de extracción de flujo.
q El objeto cin de la clase istream está enlazado al dispositivo de entrada estándar, que es normalmente el teclado.
q El objeto cout de la clase ostream está enlazado al dispositivo de salida estándar, que es normalmente la pantalla.
q El objeto cerr de la clase ostream está enlazado al dispositivo de errores estándar. Las salidas
hacia cerr no se almacenan en el búfer, y cada inserción a cerr aparece inmediatamente,
q El manipulador de flujo endl envía un carácter de nueva línea y vacía el búfer de salida.
q El compilador C++ determina automáticamente los tipos de datos para la entrada y la salida.
q Las direcciones se despliegan en forma predeterminada en formato hexadecimal.
q Para imprimir la dirección que está en una variable de apuntador hay que hacer una conversión
mediante cast del tipo de apuntador a void *.
q La función miembro put( ) envía un carácter a la salida. Las llamadas a put( ) se pueden poner
en cascada.
q La entrada de flujo se realiza mediante el operador de extracción de flujo >>. Este operador se
salta automáticamente cualquier espacio en blanco que haya en el flujo de entrada.
q El operador >> devuelve false cuando encuentra el fin de archivo dentro de un flujo.
q La extracción de flujo causa que el failbit se establezca cuando hay una entrada inadecuada, y el
badbit se establece si la operación falla.
q Es posible introducir una serie de valores utilizando la operación de extracción de flujo en un
encabezado de ciclo while. La extracción devuelve 0 cuando se encuentra el fin de archivo.
q C++ proporciona entrada/salida a prueba de tipos. Si se procesan datos inesperados con los
operadores << y >>, se establecen diversos tipos de indicadores, los cuales pueden ser probados
por el usuario para determinar si una operación de E/S ha fallado o ha sido satisfactoria.
q La E/S sin formato se realiza mediante las funciones miembro read( ) y write( ). Éstas introducen o envían a la salida algún número de bytes hacia o desde memoria iniciando en una direc ción de memoria designada. Dichos bytes se introducen o envían a la salida como bytes sin formato.
q La función miembro gcount ( ) devuelve el número de caracteres que se introdujeron mediante la
operación de read( ) sobre ese flujo.
q La función miembro read( ) introduce un número específico de caracteres en un arreglo de caracteres. failbit( ) se establece si se leen menos del número especificado de caracteres.
FUNDAMENTOS – LECCIÓN 6
6-50
MIGUEL Á. TOLEDO MARTÍNEZ
q Para cambiar la base en la que se envían los enteros a la salida, utilice el manipulador hex para
establecer la base a hexadecimal (base 16) u oct para establecer la base a octal (base 8) Utilice el
manipulador dec para restablecer la base a decimal. La base permanece igual hasta que se le
cambia explícitamente.
q El manipulador de flujo con parámetros setbase( ) también establece la base para la salida de
enteros. setbase( ) toma un argumento entero de 10, 8 o 16 para establecer la base.
q La precisión de punto flotante puede controlarse mediante el manipulador de flujo setprecision( ) o la función miembro precision( ). Ambos establecen la precisión para todas las operaciones de salida subsecuentes hasta la siguiente llamada de valor de precisión. La función mie mbro precision( ) sin argumentos devuelve el valor actual de la precisión.
q Los manipuladores con parámetros requieren la inclusión del archivo de encabezado <iomanip.h>.
q La función miembro width( ) establece la anchura del campo y devuelve la anchura anterior.
Los valores que son más pequeños que el campo se rellenan con caracteres de relleno. El establecimiento de anchura del campo se aplica solamente para la siguiente inserción o extracción;
después la anchura de campo se establece implícitarnente a 0 (para que los valores subsecuentes
se envíen a la salida con el tamaño necesario). Los valores que son más grandes que un campo se
imprimen completamente. La función width( ) sin argumentos devuelve el establecimiento de
anchura actual. El manipulador setw( ) también establece la anchura.
q Para la entrada, el manipulador de flujo setw( ) establece un tamaño de cadena máximo, y si se
recibe una cadena más grande, la línea más grande se divide en partes no mayores al tamaño indicado.
q Los usuarios pueden crear sus propios manipuladores de flujo.
q Las funciones miembro setf( ), unsetf( ) y flags( ) controlan los valores de los indicadores.
q El indicador skipws indica que >> se debe saltar los espacios en blanco en un flujo de entrada.
El manipulador de flujo ws también se salta los espacios en blanco iniciales en un flujo de entrada.
q Los indicadores de formato están definidos como una enumeración en la clase ios.
q Las funciones miembro flags( ) y setf( ) controlan los indicadores de formato, pero muchos programadores de C++ prefieren utilizar los manipuladores de flujo. La operación OR al nivel de
bits, |, se puede utilizar para combinar varias opciones en un solo valor long. Llamar a la función
miembro flags( ) para un flujo y especificar estas opciones de tipo OR establece las opciones de
ese flujo y devuelve un valor long que contiene las opciones anteriores. Este valor se guarda frecuentemente para que flags( ) pueda volverse a llamar con dicho valor a fin de restaurar las opciones anteriores del flujo.
q La función flags( ) debe especificar un valor que represente los valores totales de todos los indicadores. Por otro lado, la función setf( ) de un solo argumento especifica uno o más indicadores
y realiza la operación OR al nivel de bits con los valores de indicadores existentes para formar
un nuevo estado de formato.
q El indicador showpoint se establece para forzar que un número de punto flotante se envie a la
salida con un punto decimal y el número de dígitos significativos que se especifican en la precisión.
q Los indicadores lef t y right causan que los campos se alineen a la izquierda con caracteres de
relleno a la derecha o se alineen a la derecha con caracteres de relleno a la izquierda.
q internal indica que el signo de un número (o la base cuando el indicador ios:: showbase está establecido) deberá estar alineado a la izquierda dentro de un campo, la magnitud deberá estar alineada a la derecha y los espacios intermedios deberán rellenarse con el carácter de relleno.
q ios::adjustfield contiene los indicadores left, right e internal.
q La función miembro fil1( ) especifica el carácter de relleno que se va a utilizar con los campos
establecidos con left, right e internal (el valor predeterminado es el espacio) y devuelve el carác ter de relleno anterior. El manipulador de flujo setfi1l( ) también establece el carácter de relleno.
FUNDAMENTOS – LECCIÓN 6
6-51
MIGUEL Á. TOLEDO MARTÍNEZ
q El miembro estático ios::basefield incluye los bits oct, hex y dec para especificar que los enteros deben tratarse como valores octales, hexadecimales y decimales, respectivamente. El valor
predeterminado para la salida de números es decimal cuando no está establecido ninguno de estos bits, y las extracciones de flujo procesan los datos en la forma en que se les proporciona.
q Establezca el indicador showbase para forzar que la base de un valor entero aparezca en la salida.
q El dato miembro estático ios::floatfield contiene los indicadores scientific y fixed. Establezca el
indicador scientific para enviar un número de punto flotante a la salida en formato científico. Establezca el indicador fixed para enviar un número de punto flotante a la salida con la precisión
que se especifica en la función miembro precision.
q La llamada a cout.setf( 0, ios::floatfield ) restaura el formato predeterminado para el despliegue
de números de punto flotante.
q Establezca el indicador uppercase para forzar que se envíe a la salida una X o E mayúsculas con
los enteros hexadecimales o los valores de punto flotante en notación científica, respectivamente. Cuando el indicador ios::uppercase está establecido, causa que todas las letras de un valor
hexadecimal estén en mayúsculas.
q La función miembro flags( ) sin argumentos devuelve el valor long del establecimiento actual
de los indicadores de formato. La función miembro flags( ) con un argumento long establece los
indicadores de formato especificados por el argumento y devuelve los valores de indicadores
anteriores.
q La función miembro setf( ) establece los indicadores de formato que están en su argumento y
devuelve los valores de indicadores anteriores como un valor long.
q La función miembro setf ( long setbits, long resetbits ) limpia o borra los bits de resetbits y luego establece los bits de setbits.
q La función miembro unsetf( ) restablece los indicadores designados y devuelve el valor de los
indicadores antes de que se reactiven.
q El manipulador de flujo con parámetros setiosflags( ) realiza las mismas funciones que la función miembro flags( ).
q El manipulador de flujo con parámetros resetiosflags( ) realiza las mismas funciones que la
función mie mbro unsetf( ).
q Se puede probar el estado de un flujo por medio de los bits de la clase ios.
q eofbit se establece para un flujo de entrada cuando se encuentra el fin de archivo durante una
operación de entrada. La función miembro eof( ) se utiliza para determinar si se ha establecido el
eofbit.
q failbit se establece para un flujo cuando sucede un error de formato en el flujo pero no se han
perdido caracteres. La función miembro fail( ) determina si ha fallado una operación de flujo, y
normalmente es posible recuperarse de estos errores.
q badbit se establece para un flujo cuando sucede un error que da como resultado la pérdida de da tos. La función miembro bad( ) determina si una operación de flujo ha fallado. Tales fallas son
serias por lo general no son recuperables.
q La función miembro good devuelve true si las funciones bad( ), fail( ) y eof ( )devuelven false.
Las operaciones de E/S sólo deben realizarse con flujos buenos.
q La función miembro rdstate( ) devuelve el estado de error de un flujo.
q La función miembro clear( ) normalmente se utiliza para restaurar el estado de un flujo a bueno
para que pueda continuar la E/S en ese flujo.
q C++ proporciona la función miembro tie( ) para sincronizar las operaciones de istream y ostream para asegurar que la salida aparezca antes de las entradas subsecuentes.
FUNDAMENTOS – LECCIÓN 6
6-52
MIGUEL Á. TOLEDO MARTÍNEZ
PREGUNTAS Y PROBLEMAS
PREGUNTAS
Indique la salida para cada caso:
1.
cout << “\n\n” << endl ;
a.
b.
c.
d.
e.
f.
g.
h.
i.
2.
cout << setw( 40 ) << “HOLA” << endl;
cout << “*\n**\n***\n****\n*****\n”;
cout << setw( 12 ) << -36.2 << endl;
cout << 3.75 << endl;
cout << ‘\n’ << 1 << ‘\t’ << 2 << ‘\t’ << 3 << ‘\t’ << 4 << endl;
cout << setw( 20 ) << “Mi registro de prueba es: 97.6/n/n” << endl;
cout << “\n\t\tREGISTRO PRUEBA\t\t97.5” << endl;
cout << “\n\t\tREGISTRO PRUEBA\n\t\t97.5” << endl;
imprimir << “\t\t tREGISTRO PRUEBA\n\t\t97.5\f”;
Suponga que define una constante como sigue:
const char ESPACIO = ‘ ’;
¿Qué hará el siguiente enunciado?
cout << ‘\n’ << setw( 20 ) << ESPACIO << “HOLA” << endl;
3.
¿Qué hará el siguiente enunciado?
cout << ‘\n’ << setw( 20 ) << ‘ ’ << “HOLA” << endl;
4.
¿Cuál es la diferencia entre la salida que producen los dos enunciados siguientes?
cout << “\n#\n#\n#” << endl;
cout << “\n###” << endl;
5.
¿Qué archivo de encabezado deberá incluir para usar el manipulador de ancho de campo setw( )?
6.
Suponga que deberá generar dos páginas separadas de salida a una impresora, ¿qué debe hacer para que
avance la impresora a la segunda página una vez que se haya impreso la primera página?
Explique el significado de los siguientes objetos:
7.
8.
a.
cin
b.
c.
d.
cout
cerr
clog
Responda cada una de las siguientes preguntas:
a.
b.
c.
d.
e.
f.
g.
h.
i.
j.
k.
l.
m.
n.
o.
p.
q.
r.
Los operadores de flujo sobrecargados se definen frecuentemente como funciones ______________ de una clase.
Los bits de alineación de formato que se pueden ajustar incluyen a _______________, ____________ e ________________.
La entrada / salida en C++ sucede como ___________ de bytes.
Los manipuladores de flujo con parámetros ____________ y ____________ pueden utilizarse para establecer y restablecer
los indicadores de estado del formato.
La mayoría de los programas C++ deben incluir el archivo de encabezado _____________ que contiene la información básica que se requiere para todas las operaciones de E/S de flujo.
Las funciones miembro ____________ y __________ se utilizan para establecer y restablecer los indicadores de estado de
formato.
El archivo de encabezado __________ contiene información para realizar el formato en memoria .
Cuando se utilizan manipuladores con parámetros se debe incluir el archivo de encabezado ________.
El archivo de encabezado ____________ contiene información para el procesamiento de archivos controlado por el usuario.
El manipulador de flujo _____________ inserta un carácter de nueva línea en el flujo de salida y vacía el flujo de salida.
El archivo de encabezado _____________ se utiliza en programas que mezclan la E/S estilo C y C++.
La función miembro ________ de ostream se utiliza para realizar salida sin formato.
La clase ___________ soporta las operaciones de entrada.
Las salidas hacia el flujo de errores estándar están dirigidas hacia los objetos de flujo __________ o _______________.
La clase _________ soporta las operaciones de salida.
El símbolo para el operador de inserción de flujo es _______________.
Los cuatro objetos que corresponden a los dispositivos estándar del sistema incluyen a __________, ______________,
_______________ y ______________.
El símbolo para el operador de extracción de flujo es _______________.
FUNDAMENTOS – LECCIÓN 6
6-53
MIGUEL Á. TOLEDO MARTÍNEZ
s.
t.
u.
9.
Los manipuladores de flujo _________, ____________ y _____________ se utilizan para especificar que los enteros se deben desplegar en formato octal, hexadecimal y decimal.
La precisión predeterminada para el despliegue de valores de punto flotante es ___________.
Cuando se establece, el indicador ___________ causa que los números positivos se desplieguen con un signo más.
Indique si lo siguiente es verdadero o falso. Si la respuesta es falsa explique por qué.
a.
b.
c.
d.
e.
f.
g.
h.
i.
j.
k.
l.
m.
n.
o.
p.
q.
r.
La función miembro de flujo flags( ) con un argumento long establece la variable de estado flags a su argumento y devuelve
el valor anterior.
El operador de inserción de flujo << y el operador de extracción de flujo están sobrecargados para manejar todos los tipos de
datos estándar –incluyendo cadenas y direcciones de memoria (solamente la inserción de flujo) y todos los tipos de datos definidos por el usuario.
La función miembro de flujo flags( ) sin argumentos reactiva todos los bits de indicadores de la variable de estado flags.
El operador de extracción de flujo >> puede sobrecargarse con una función de operador que tome como argumentos una referencia a istream y una referencia a un tipo definido por el usuario, y devuelve una referencia a istream.
El manipulador de flujo ws se salta los espacios en blanco iniciales de un flujo de entrada.
El operador de inserción de flujo << puede sobrecargarse con una función de operador que tome como argumentos una referencia a istream y una referencia a un tipo definido por el usuario, y devuelve una referencia a istream.
La entrada con el operador de extracción de flujo >> siempre se salta los caracteres de espacio en blanco iniciales del flujo
de entrada.
Las características de entrada y salida se proporcionan como parte de C++.
La función miembro de flujo rdstate( ) devuelve el estado del flujo actual.
El flujo cout está normalmente conectado con la pantalla.
La función miembro de flujo good( ) devuelve true si las funciones miembro bad( ), fail( ) y eof( ) devuelven false.
El flujo cin está normalmente conectado con la pantalla.
Si sucede un error no recuperable durante una operación de flujo, la función miembro bad devolverá true.
La salida hacia cerr no se almacena en el búfer y la salida hacia clog si se almacena en el búfer.
Cuando se establece el indicador ios::showpoint, se fuerza a que los valores de punto flotante se impriman con la precisión
predeterminada de seis dígitos –siempre y cuando no se haya cambiado el valor de la precisión, en cuyo caso los valores de
punto flotante se imprimen con la precisión especificada.
La función miembro put de ostream da salida al número de caracteres especificado.
Los manipuladores de flujo dec, oct y hex solamente afectan las siguiente operación de salida de enteros.
Cuando se les da salida, de manera predeterminada las direcciones de memorias se despliegan como enteros long.
10. Identifique el error que hay en cada una de las siguientes instrucciones y explique la manera de corregirlas.
a.
cout << “El valor de x <= y es: “ << x <= y;
b.
La siguiente instrucción debe imprimir el valor entero de ‘c’.
cout << ‘c’;
c.
cout << ““Una cadena entre comillas””;
PROBLEMAS
1.
2.
3.
4.
Utilizando diseño de pantalla, escriba un programa que muestre su primer nombre en la mitad de la
pantalla del monitor .
Use el diseño de pantalla para escribir un programa que muestre su primer nombre en la esquina
superior izquierda de la pantalla usando caracteres con seis líneas de alto.
Escriba un programa que genera un rectángulo cuyo centro se localice a la mitad de la pantalla.
Construya el rectángulo de 8 líneas de alto y 20 columnas de ancho, con la letra X.
Escriba un programa que genere la siguiente salida en la mitad de la pantalla:
ESTUDIANTE
-------------------1
2
3
4
5
5.
PROMEDIA DEL S EMESTRE
----------------------------------------84.5
67.2
77.4
86.8
94.7
Para cada una de las siguientes indicaciones, escriba una sola instrucción que realice la tarea indicada.
a.
Envíe a la salida la cadena “Introduzca su nombre: ”.
FUNDAMENTOS – LECCIÓN 6
6-54
MIGUEL Á. TOLEDO MARTÍNEZ
b.
c.
d.
e.
f.
g.
h.
i.
j.
k.
l.
m.
n.
o.
p.
q.
r.
s.
6.
Muestre la salida para cada una de las siguientes instrucciones:
a.
b.
c.
d.
e.
f.
g.
h.
i.
j.
k.
l.
7.
9.
cout << “1234” << endl;
cout.width( 5 );
cout.fill( ‘*’ );
cout << 123 << endl << 123;
cout << setw( 10 ) << setfill( ‘$’ ) << 10000;
cout << setw( 8 ) << setprecision( 3 ) << 1024.987654;
cout << setiosflags( ios::showbase ) << oct << 99
<< endl << hex << 99;
cout << 100000 << endl;
<< setiosflags( ios::showpos ) << 100000;
cout << setw( 10 ) << setprecision( 2 ) <<
<< setiosflags( ios::scientific ) << 444.93738;
Escriba una instrucción para cada una de las siguientes indicaciones:
a.
b.
c.
d.
8.
Establezca un indicador que cause que el exponente en notación científica y las letras en valores hexadecimales se impriman
en mayúsculas.
Envía a la salida la dirección de la variable cadena de tipo char *.
Establezca un indicador para que los valores de punto flotante se impriman en notación científica.
Envíe a la salida la dirección de la variable enteroPtr de tipo int *.
Establezca un indicador para que cuando se envíen valores enteros a la salida, se despliegue la base entera para los valores
octales y hexadecimales.
Envíe a la salida el valor apuntado por floatPtr de tipo float *.
Utilice una función miembro de flujo para establecer el carácter de relleno a ‘*’ para imprimir en campos cuya anchura sea
mayor que la de los valores que se envían a la salida. Escriba una instrucción separada para hacer esto con un manipulador
de flujo.
Envíe a la salida los caracteres ‘O’ y ‘K’ en una instrucción mediante la función put de ostream.
Utilice la función miembro read de istream para introducir 50 caracteres en el arreglo linea de tipo char.
Utilice la función miembro gcount de istream para determinar el número de caracteres que se introducen en el arreglo de caracteres linea por medio de la última llamada a la función miembro read de istream, y envíe a la salida a ese número de caracteres utilizando la función miembro write de ostream.
Escriba instrucciones separadas para vaciar el flujo de salida por medio de una función miembro y un manip ulador de flujo.
Envíe a la salida los siguientes valores: 124, 18.376, ‘Z’, 1000000 y “cadena”.
Imprima el ajuste de precisión actual utilizando una función miembro.
Imprima 1.92, 1.925, 1.9258 con 3 dígitos de precisión utilizando un manipulador.
Imprima el entero 100 en octal, hexadecimal y decimal utilizando manipuladores de flujo.
Imprima el entero 100 en decimal, octal y hexadecimal utilizando un solo manipulador de flujo para cambiar la base.
Imprima 1234 alineado a la derecha en un campo de 10 dígitos.
Utilice las variables enteras x e y para especificar la anchura de campo y precisión que se utilizan para desplegar el valor
double 87.4573, y despliegue el valor.
Imprima el entero 40000 alineado a la izquierda en un campo de 15 dígitos.
Imprima 200 con y sin signo.
Imprima el valor decimal 100 en forma hexadecimal precedido por 0x.
Imprima 1.234 en un campo de 9 dígitos con ceros iniciales .
Escriba un programa que imprima valores de apuntador utilizando conversiones mediante cast de
todos los tipos de datos enteros. ¿Cuáles imprimen valores extraños? ¿Cuáles causan errores?
Escriba un programa para probar el resultado de la impresión del valor entero 12345 y el valor de
punto flotante 1.2345 en varios tamaños de campo. ¿Qué sucede cuando los valores se imprimen en
campos que contienen menos dígitos que los valores?
10. Escriba un programa que imprima el valor 100.453627 redondeado al dígito más cercano, a décimos, centésimos, milésimos y diezmilésimos.
11. Escriba un programa que convierta temperaturas Fahrenheit enteras, desde 0 hasta 212 grados, a
temperaturas Celsius de punto flotante con 3 dígitos de precisión. Utilice la formula
celsius = 5.0 / 9.0 * ( fahrenheit – 32 );
para realizar el cálculo. La salida debe imprimirse en dos columnas alineadas a la derecha, y las temperaturas Celsius deberán estar precedidas por un signo para los valores positivos y negativos.
12. Escriba un programa que utilice una estructura for para imprimir una tabla de valores ASCII
para los caracteres de conjunto de caracteres ASCII que van del 33 al 126. El programa deberá imprimir el valor decimal, octal y hexadecimal y el valor del carácter para cada carácter. Utilice los manipuladores de flujo dec, oct y hex para imprimir los valores enteros.
FUNDAMENTOS – LECCIÓN 6
6-55
MIGUEL Á. TOLEDO MARTÍNEZ
EXAMEN BREVE 14
1. El archivo que deberá incluir para usar cout es _____________________.
2. El operador que debe emplear para insertar información dentro del flujo cout es el operador
______________________.
3. Escriba un enunciado cout para mostrar su nombre como una cadena fija de información.
4. Escriba un enunciado cout para mostrar su nombre cuando se almacena en una cadena variable llamada nombre.
5. La secuencia de escape que debe usar para genera un CRLF es _________________.
EXAMEN BREVE 15
1. El archivo que debe incluir para usar el manejador de ancho de campo setw( ) es
_______________________.
2. Escriba un enunciado cout que muestre el valor de una variable de punto flotante llamada
numero justificado a la izquierda dentro de un ancho de campo de 10 columnas y una precisión
de 3 lugares decimales.
3. Explique la diferencia entre usar una \n en comparación con un endl dentro de un enunciado
cout.
4. ¿Qué salidas se producenn cuando se ejecutan las siguientes líneas?
cout << “*”;
cout.width( 5 );
cout << 123 << “*” << 123 <<“*” << endl;
cout << “* ” << setw( 5 ) << 123 << “ * ” << 123 << “ * ” << endl ;
5. ¿Qué salidas se producen cuando se ejecutan las siguientes líneas?
cout << “ * ” << setw( 5 ) << 123 ;
cout.setf( ios ::left ) ;
cout << “ * ” << setw( 5 ) << 123 ;
cout.setf( ios ::right ) ;
cout << “ * ” << setw( 5 ) << 123 << “ * ” << endl ;
6. ¿Qué salidas se producen cuando se ejecutan las siguientes líneas?
cout << “ * ” << setw( 5 ) << 123 << “ * ” << 123 << “ * ” << endl ;
cout.setf( ios ::showpos ) ;
cout << “ * ” << setw( 5 ) << 123 << “ * ” << 123 << “ * ” << endl ;
cout.unsetf( ios ::showpos ) ;
cout.setf( ios::left );
cout << “ * ” << setw( 5 ) << 123 << “ * ” << setw( 5 ) << 123 << “ * ” << endl ;
7. ¿Qué salidas se producen cuando se ejecuta la siguiente línea?
cout << “ * ” << setw( 3 ) << 12345 << “ * ” << endl ;
FUNDAMENTOS – LECCIÓN 6
6-56
MIGUEL Á. TOLEDO MARTÍNEZ
EXAMEN BREVE 16
1. Al formatear salidas, se utilizan las siguientes banderas con la función miembro setf( ). ¿Qué
efecto tiene cada una?
a.
b.
c.
d.
ios::fixed
ios::scientific
ios::showpoint
ios::showpos
e. ios::right
f. ios::left
RESPUESTAS EXAMEN BREVE 14
2. El archivo que debe incluir para usar cout es el archivo de cabecera iostream.h.
3. El operador que debe emplear para insertar información dentro del flujo cout es el operador
de inserción de flujo <<.
4. Lo siguiente es un enunciado cout para mostrar mi nombre como una cadena fija de información.
cout << “Miguel Angel “ << endl;
5. Lo siguie nte es un enunciado cout para mostrar mi nombre cuando se almacena en una variable
de cadena llamada nombre.
cout << nombre << endl;
6. La secuencia de escape que se debe utilizar para generar un CRLF es ‘\n’.
RESPUESTAS EXAMEN BREVE 15
1. El archivo que debe incluir para usar el manejador de ancho de campo setw( ) es el archivo de
cabecera iomanip.h.
2. El siguiente enunciado cout mostrará el valor de una variable de punto flotante llamada numero, justificado a la izquierda dentro de un ancho de campo de 10 columnas y una precisión de 3
lugares decimales:
cout.setf( ios::fixed|ios::left );
cout.precision( 3 );
cout << setw( 10 ) << numero << endl;
3. La diferencia que existe entre usar ‘\n’ en comparación con endl dentro del enunciado cout es
que la secuencia de escape ‘\n’ sólo genera una CRLF, mientras que el manejador endl genera
un CRLF y vacía la memoria temporal de flujo de salida.
4. La salida es la siguiente:
* 123*123*
* 123*123*
Cada uno de los espacios contiene exactamente dos caracteres en blanco.
5. La salida es la siguiente:
FUNDAMENTOS – LECCIÓN 6
6-57
MIGUEL Á. TOLEDO MARTÍNEZ
* 123*123 * 123*
Cada espacio tiene exactamente dos caracteres en blanco.
6. La salida es la siguiente:
* 123*123*
* +123*+123*
*123 *123 *
Sólo hay un espacio entre el “ * ” y el “ + ” en la segunda línea. Los demás huecos contienen exactamente dos caracteres en blanco cada uno.
7. La salida es la siguiente:
*12345*
Observe que se envía a la salida el entero completo aunque ello requiera más espacio que el que especifica setw( ).
RESPUESTAS EXAMEN BREVE 16
1. Las respuestas son las siguientes:
a. ios::fixed Establecer esta bandera hace que los números de punto flotante no se desplieguen en notación e, es
decir, en notación científica.Al establecer esta bandera, se desactiva ios::scientific.
b. ios::scientific Establecer esta bandera hace que los números de punto flotante se desplieguen en notación e, es
decir, en notación científica. Al establecer esta bandera, se desactiva ios::fixed.
c. ios::showpoint Establecer esta bandera hace que siempre se despliegue el punto decimal y los ceros a la derecha.
d. ios::showpos Establecer esta bandera hace que se despliegue un signo de más antes de los valores positivos enteros.
e. ios::right Establecer esta bandera hace que la siguiente salida se coloque en el extremo derecho de cualquier
campo que se establezca con la función miembro width( ). Es decir, si sobran espacios en blanco, se colocan
anates de la salida. Al establecer esta bandera, se desactiva ios::left.
ios::left Establecer esta bandera hace que la siguiente salida se coloque en el extremo izquierdo de cualquier campo que
se establezca con la función miembro width( ). Es decir, si sobran espacios en blanco, se colocan después de la salida. Al
establecer esta bandera, se desactiva ios::right.
FUNDAMENTOS – LECCIÓN 6
6-58
Descargar