Introcsharp Version095
Introcsharp Version095
Introducción a la programación
con C#
Está organizado de una forma ligeramente distinta a los libros de texto "convencionales",
procurando incluir ejercicios prácticos lo antes posible, para evitar que un exceso de teoría en
los primeros temas haga el texto pesado de seguir.
Este texto ha sido escrito por Nacho Cabanes. Si quiere conseguir la última versión, estará en
mi página web:
www.nachocabanes.com
Este texto es de libre distribución ("gratis"). Se puede distribuir a otras personas libremente,
siempre y cuando no se modifique.
Este texto se distribuye "tal cual", sin garantía de ningún tipo, implícita ni explícita. Aun así, mi
intención es que resulte útil, así que le rogaría que me comunique cualquier error que
encuentre.
Contenido
0. Conceptos básicos sobre programación _______________________________________________ 7
0.1. Lenguajes de alto nivel y de bajo nivel. _____________________________________________________ 7
0.2. Ensambladores, compiladores e intérpretes _________________________________________________ 9
0.3. Pseudocódigo __________________________________________________________________________ 9
1. Toma de contacto con C# __________________________________________________________ 11
1.1 Escribir un texto en C# _________________________________________________________________ 11
1.2. Cómo probar este programa con Mono ___________________________________________________ 13
1.3. Mostrar números enteros en pantalla _____________________________________________________ 19
1.4. Operaciones aritméticas básicas _________________________________________________________ 20
1.4.1. Orden de prioridad de los operadores ___________________________________________________ 20
1.4.2. Introducción a los problemas de desbordamiento __________________________________________ 20
1.5. Introducción a las variables: int__________________________________________________________ 21
1.5.1. Definición de variables: números enteros ________________________________________________ 21
1.5.2. Asignación de valores _______________________________________________________________ 21
1.5.3. Mostrar el valor de una variable en pantalla ______________________________________________ 22
1.6. Identificadores ________________________________________________________________________ 23
1.7. Comentarios__________________________________________________________________________ 24
1.8. Datos por el usuario: ReadLine __________________________________________________________ 25
2. Tipos de datos básicos_____________________________________________________________ 27
2.1. Tipo de datos entero y carácter __________________________________________________________ 27
2.1.1. Tipos de datos para números enteros ____________________________________________________ 27
2.1.2. Conversiones de cadena a entero _______________________________________________________ 28
2.1.3. Incremento y decremento_____________________________________________________________ 28
2.1.4. Operaciones abreviadas: +=___________________________________________________________ 29
2.2. Tipo de datos real _____________________________________________________________________ 29
2.2.1. Simple y doble precisión _____________________________________________________________ 30
2.2.2. Mostrar en pantalla números reales _____________________________________________________ 30
2.2.3. Formatear números _________________________________________________________________ 31
2.3. Tipo de datos carácter__________________________________________________________________ 33
2.3.1. Secuencias de escape: \n y otras. _______________________________________________________ 33
2.4. Toma de contacto con las cadenas de texto _________________________________________________ 35
2.5. Los valores "booleanos" ________________________________________________________________ 36
3. Estructuras de control_____________________________________________________________ 37
3.1. Estructuras alternativas ________________________________________________________________ 37
3.1.1. if________________________________________________________________________________ 37
3.1.2. if y sentencias compuestas ____________________________________________________________ 38
3.1.3. Operadores relacionales: <, <=, >, >=, ==, !=_____________________________________________ 39
3.1.4. if-else ____________________________________________________________________________ 40
3.1.5. Operadores lógicos: &&, ||, ! __________________________________________________________ 41
3.1.6. El peligro de la asignación en un "if"____________________________________________________ 42
3.1.7. Comparaciones y variables de tipo "bool" ________________________________________________ 43
3.1.8. Introducción a los diagramas de flujo ___________________________________________________ 44
3.1.9. Operador condicional: ?______________________________________________________________ 46
3.1.10. switch___________________________________________________________________________ 47
3.2. Estructuras repetitivas _________________________________________________________________ 50
3.2.1. while ____________________________________________________________________________ 50
3.2.2. do ... while ________________________________________________________________________ 51
3.2.3. for ______________________________________________________________________________ 53
3.3. Sentencia break: termina el bucle ________________________________________________________ 56
3.4. Sentencia continue: fuerza la siguiente iteración ____________________________________________ 57
0.95, de 01-jun-10. Recolocados (en distinto orden) los temas 8, 9, 10, para que el apartado
sobre "otras bibliotecas" pase a ser el último de esos tres. Añadido el apartado 10.12 sobre
servicios de red (cómo descargar una página web y comunicar dos ordenadores). Ampliado el
apartado 9.6 sobre expresiones regulares e incluido un segundo ejemplo de acceso a bases de
datos con SQLite en 10.10. Ampliado el apartado 2.2.3 para ver cómo convertir un número a
binario o hexadecimal. Añadidos tres ejercicios propuestos al apartado 4.1 y otros dos al 4.4.3
0.94, de 07-may-10. Algunas correcciones menores en algunos apartados (por ejemplo, 6.3).
Añadido el apartado 8.13 sobre Tao.SDL y el apéndice 4 sobre la instalación y uso básico de
Visual Studio (en sus versiones 2008 Express y 2010 Express).
0.93, de 10-mar-10. Algunas correcciones menores en algunos apartados (por ejemplo, 2.1, 2.2,
3.1.2, 3.1.9, 4.1.4). Añadido el apartado 6.13 sobre SharpDevelop y cómo crear programas a
partir de varios fuentes.
0.92, de 22-nov-09. Ampliado el tema 4 con un apartado sobre comparaciones de cadenas y otro
sobre métodos de ordenación. Añadidos ejercicios propuestos en los apartados 4.4.8, 5.5 (2), 5.7
(2), 5.9.1.
0.91, de 19-nov-09, que incluye unos 30 ejercicios propuestos adicionales (en los apartados 1.2,
1.4, 1.8, 3.1.4, 3.1.5, 3.1.9, 3.2.1, 3.2.2, 3.2.3, 4.1.1, 4.1.2, 4.3.2, 4.5 y 5.10) y algunos
apartados con su contenido ampliado (como el 4.4.6 y el 8.1).
0.90, de 24-may-09, como texto de apoyo para los alumnos de Programación en Lenguajes
Estructurados en el I.E.S. San Vicente. (Primera versión completa; ha habido 4 versiones previas,
incompletas, de distribución muy limitada).
Estas órdenes se le deben dar en un cierto lenguaje, que el ordenador sea capaz de
comprender.
El problema es que los lenguajes que realmente entienden los ordenadores resultan difíciles
para nosotros, porque son muy distintos de los que nosotros empleamos habitualmente para
hablar. Escribir programas en el lenguaje que utiliza internamente el ordenador (llamado
"lenguaje máquina" o "código máquina") es un trabajo duro, tanto a la hora de crear el
programa como (especialmente) en el momento de corregir algún fallo o mejorar lo que se
hizo.
Por ejemplo, un programa que simplemente guardara un valor "2" en la posición de memoria 1
de un ordenador sencillo, con una arquitectura propia de los años 80, basada en el procesador
Z80 de 8 bits, sería así en código máquina:
Prácticamente ilegible. Por eso, en la práctica se emplean lenguajes más parecidos al lenguaje
humano, llamados "lenguajes de alto nivel". Normalmente, estos son muy parecidos al idioma
inglés, aunque siguen unas reglas mucho más estrictas.
Uno de los lenguajes de alto nivel más sencillos es el lenguaje BASIC. En este lenguaje, escribir
el texto Hola en pantalla, sería tan sencillo como usar la orden
PRINT "Hola"
Otros lenguajes, como Pascal, nos obligan a ser algo más estrictos, pero a cambio hacen más
fácil descubrir errores (ya veremos por qué):
program Saludo;
begin
write('Hola');
end.
#include <stdio.h>
int main()
{
printf("Hola");
}
Revisión 0.95 – Página 7
Introducción a la programación con C#, por Nacho Cabanes
Los lenguajes de bajo nivel son más cercanos al ordenador que a los lenguajes humanos. Eso
hace que sean más difíciles de aprender y también que los fallos sean más difíciles de descubrir
y corregir, a cambio de que podemos optimizar al máximo la velocidad (si sabemos cómo), e
incluso llegar a un nivel de control del ordenador que a veces no se puede alcanzar con otros
lenguajes. Por ejemplo, escribir Hola en lenguaje ensamblador de un ordenador equipado con
el sistema operativo MsDos y con un procesador de la familia Intel x86 sería algo como
dosseg
.model small
.stack 100h
.data
hello_message db 'Hola',0dh,0ah,'$'
.code
main proc
mov ax,@data
mov ds,ax
mov ah,9
mov dx,offset hello_message
int 21h
mov ax,4C00h
int 21h
main endp
end main
Resulta bastante más difícil de seguir. Pero eso todavía no es lo que el ordenador entiende,
aunque tiene una equivalencia casi directa. Lo que el ordenador realmente es capaz de
comprender son secuencias de ceros y unos. Por ejemplo, las órdenes "mov ds, ax" y "mov ah,
9" (en cuyo significado no vamos a entrar) se convertirían a lo siguiente:
(Nota: los colores de los ejemplos anteriores son una ayuda que nos dan algunos entornos de
programación, para que nos sea más fácil descubrir errores).
Si elegimos un lenguaje de bajo nivel, como el ensamblador (en inglés Assembly, abreviado
como Asm), la traducción es sencilla, y de hacer esa traducción se encargan unas herramientas
llamadas ensambladores (en inglés Assembler).
Cuando el lenguaje que hemos empleado es de alto nivel, la traducción es más complicada, y a
veces implicará también recopilar varios fuentes distintos o incluir posibilidades que se encuen-
tran en bibliotecas que no hemos preparado nosotros. Las herramientas encargadas de todo
esto son los compiladores.
Por ejemplo, en el caso de Windows (y de MsDos), y del programa que nos saluda en lenguaje
Pascal, tendríamos un fichero fuente llamado SALUDO.PAS. Este fichero no serviría de nada en
un ordenador que no tuviera un compilador de Pascal. En cambio, después de compilarlo
obtendríamos un fichero SALUDO.EXE, capaz de funcionar en cualquier otro ordenador que
tuviera el mismo sistema operativo, aunque no tenga un compilador de Pascal instalado.
Hoy en día existe algo que parece intermedio entre un compilador y un intérprete. Existen
lenguajes que no se compilan a un ejecutable para un ordenador concreto, sino a un ejecutable
"genérico", que es capaz de funcionar en distintos tipos de ordenadores, a condición de que en
ese ordenador exista una "máquina virtual" capaz de entender esos ejecutables genéricos.
Esta es la idea que se aplica en Java: los fuentes son ficheros de texto, con extensión ".java",
que se compilan a ficheros ".class". Estos ficheros ".class" se podrían llevar a cualquier
ordenador que tenga instalada una "máquina virtual Java" (las hay para la mayoría de sistemas
operativos). Esta misma idea se sigue en el lenguaje C#, que se apoya en una máquina virtual
llamada "Dot Net Framework" (algo así como "armazón punto net").
0.3. Pseudocódigo
A pesar de que los lenguajes de alto nivel se acercan al lenguaje natural, que nosotros
empleamos, es habitual no usar ningún lenguaje de programación concreto cuando queremos
plantear los pasos necesarios para resolver un problema, sino emplear un lenguaje de
programación ficticio, no tan estricto, muchas veces escrito incluso en español. Este lenguaje
recibe el nombre de pseudocódigo.
Esa secuencia de pasos para resolver un problema es lo que se conoce como algoritmo
(realmente hay alguna condición más, por ejemplo, debe ser un número finito de pasos). Por
tanto, un programa de ordenador es un algoritmo expresado en un lenguaje de programación.
Por ejemplo, un algoritmo que controlase los pagos que se realizan en una tienda con tarjeta de
crédito, escrito en pseudocódigo, podría ser:
Ejercicios propuestos
1. Localizar en Internet el intérprete de Basic llamado Bywater Basic, en su versión para el
sistema operativo que se esté utilizando y probar el primer programa de ejemplo que
se ha visto en el apartado 0.1.
2. Localizar en Internet el compilador de Pascal llamado Free Pascal, en su versión para el
sistema operativo que se esté utilizando, instalarlo y probar el segundo programa de
ejemplo que se ha visto en el apartado 0.1.
3. Localizar un compilador de C para el sistema operativo que se esté utilizando (si es
Linux o alguna otra versión de Unix, es fácil que se encuentre ya instalado) y probar el
tercer programa de ejemplo que se ha visto en el apartado 0.1.
Se trata de un lenguaje creado por Microsoft para crear programas para su plataforma .NET,
pero estandarizado posteriormente por ECMA y por ISO, y del que existe una implementación
alternativa de "código abierto", el "proyecto Mono", que está disponible para Windows, Linux,
Mac OS X y otros sistemas operativos.
Nosotros comenzaremos por usar Mono como plataforma de desarrollo durante los primeros
temas. Cuando los conceptos básicos estén asentados, pasaremos a emplear Visual C#, de
Microsoft, que requiere un ordenador más potente pero a cambio incluye un entorno de
desarrollo muy avanzado, y está disponible también en una versión gratuita (Visual Studio
Express Edition).
La mayoría de los compiladores actuales permiten dar todos estos pasos desde un único
entorno, en el que escribimos nuestros programas, los compilamos, y los depuramos en caso
de que exista algún fallo.
Esto escribe "Hola" en la pantalla. Pero hay mucho alrededor de ese "Hola", y vamos a
comentarlo antes de proseguir, aunque muchos de los detalles se irán aclarando más adelante.
En este primer análisis, iremos de dentro hacia fuera:
Como se puede ver, mucha parte de este programa todavía es casi un "acto de fe" para
nosotros. Debemos creernos que "se debe hacer así". Poco a poco iremos detallando el por qué
de "public", de "static", de "void", de "class"... Por ahora nos limitaremos a "rellenar" el cuerpo
del programa para entender los conceptos básicos de programación.
Ejercicio propuesto: Crea un programa en C# que te salude por tu nombre (ej: "Hola,
Nacho").
http://www.mono-project.com/
En la parte superior derecha aparece el enlace para descargar ("download now"), que nos lleva
a una nueva página en la que debemos elegir la plataforma para la que queremos nuestro
Mono. Nosotros descargaremos la versión más reciente para Windows (la 1.9.1 en el momento
de escribir este texto).
Se trata de un fichero de algo más de 70 Mb. Cuando termina la descarga, hacemos doble clic
en el fichero recibido y comienza la instalación, en la que primero se nos muestra el mensaje de
bienvenida:
Revisión 0.95 – Página 13
Introducción a la programación con C#, por Nacho Cabanes
Después se nos muestra una ventana de información, en la que se nos avisa de que se va a
instalar Mono 1.9.1, junto con las librerías Gtk# para creación de interfaces de usuario y XSP
(eXtensible Server Pages, un servidor web).
A continuación se nos pregunta en qué carpeta queremos instalar. Como es habitual, se nos
propone que sea dentro de "Archivos de programa":
El siguiente paso es elegir qué componentes queremos instalar (Mono, Gtk#, XSP):
El siguiente paso es indicar en qué carpeta del menú de Inicio queremos que quede accesible:
Mono está listo para usar. En nuestro menú de Inicio deberíamos tener una nueva carpeta
llamada "Mono 1.9.1 for Windows", y dentro de ella un acceso a "Mono-1.9.1 Command
Prompt":
Primero debemos teclear nuestro fuente. Para ello podemos usar cualquier editor de texto. En
este primer fuente, usaremos simplemente el "Bloc de notas" de Windows. Para ello tecleamos:
notepad ejemplo01.cs
Aparecerá la pantalla del "Bloc de notas", junto con un aviso que nos indica que no existe ese
fichero, y que nos pregunta si deseamos crearlo. Respondemos que sí y podemos empezar a
teclear el ejemplo que habíamos visto anteriormente:
Guardamos los cambios, salimos del "Bloc de notas" y nos volvemos a encontrar en la pantalla
negra del símbolo del sistema. Nuestro fuente ya está escrito. El siguiente paso es compilarlo.
Para eso, tecleamos
mcs ejemplo01.cs
Si no se nos responde nada, quiere decir que no ha habido errores. Si todo va bien, se acaba
de crear un fichero "ejemplo01.exe". En ese caso, podríamos lanzar el programa tecleando
mono ejemplo01
Si en nuestro ordenador está instalado el "Dot Net Framework" (algo que debería ser cierto en
las últimas versiones de Windows), no debería hacer falta decir que queremos que sea Mono
quien lance nuestro programa, y podremos ejecutarlo directamente con su nombre:
ejemplo01
Nota: Si quieres un editor más potente que el Bloc de notas de Windows, puedes probar
Notepad++, que es gratuito (realmente más que eso: es de "código abierto") y podrás localizar
fácilmente en Internet. Y si prefieres un entorno desde el que puedas teclear, compilar y probar
tus programas, incluso los de gran tamaño que estén formados por varios ficheros, en el
apartado 6.13 hablaremos de SharpDevelop. Si quieres saber cosas sobre el entorno "oficial",
llamado Visual Studio, los tienes en el Apéndice 4.
Ejercicio propuesto: crea un programa que diga el resultado de sumar 118 y 56.
Operador Operación
+ Suma
- Resta, negación
* Multiplicación
/ División
% Resto de la división ("módulo")
Ejercicios propuestos:
Ejercicio propuesto: Calcular (a mano y después comprobar desde C#) el resultado de estas
operaciones:
-2 + 3 * 5
(20+5) % 6
15 + -5*6 / 10
2 + 10 / 5 * 2 - 7 % 1
El primer ejemplo nos permitía escribir "Hola". El segundo nos permitía sumar dos números que
habíamos prefijado en nuestro programa. Pero esto tampoco es "lo habitual", sino que esos
números dependerán de valores que haya tecleado el usuario o de cálculos anteriores.
Por eso necesitaremos usar variables, zonas de memoria en las que guardemos los datos con
los que vamos a trabajar y también los resultados temporales. Como primer ejemplo, vamos a
ver lo que haríamos para sumar dos números enteros que fijásemos en el programa.
El primer tipo de datos que usaremos serán números enteros (sin decimales), que se indican
con "int" (abreviatura del inglés "integer"). Después de esta palabra se indica el nombre que
tendrá la variable:
int primerNumero;
Esa orden reserva espacio para almacenar un número entero, que podrá tomar distintos
valores, y al que nos referiremos con el nombre "primerNumero".
primerNumero = 234;
O también podemos darles un valor inicial ("inicializarlas") antes de que empiece el programa,
en el mismo momento en que las definimos:
(esta línea reserva espacio para dos variables, que usaremos para almacenar números enteros;
una de ellas se llama primerNumero y tiene como valor inicial 234 y la otra se llama
segundoNumero y tiene como valor inicial 567).
Después ya podemos hacer operaciones con las variables, igual que las hacíamos con los
números:
Una vez que sabemos cómo mostrar un número en pantalla, es sencillo mostrar el valor de una
variable. Para un número hacíamos cosas como
System.Console.WriteLine(3+4);
System.Console.WriteLine(suma);
O bien, si queremos mostrar un texto además del valor de la variable, podemos indicar el texto
entre comillas, detallando con {0} en qué parte del texto queremos que aparezca el valor de la
variable, de la siguiente forma:
Si se trata de más de una variable, indicaremos todas ellas tras el texto, y detallaremos dónde
debe aparecer cada una de ellas, usando {0}, {1} y así sucesivamente:
Ya sabemos todo lo suficiente para crear nuestro programa que sume dos números usando
variables:
primerNumero = 234;
segundoNumero = 567;
suma = primerNumero + segundoNumero;
(Nos saltamos todavía los detalles de qué quieren decir "public", "class", "static" y
"void").
Main() indica donde comienza el cuerpo del programa, que se delimita entre llaves { y }
int primerNumero; reserva espacio para guardar un número entero, al que llamaremos
primerNumero.
int segundoNumero; reserva espacio para guardar otro número entero, al que
llamaremos segundoNumero.
int suma; reserva espacio para guardar un tercer número entero, al que llamaremos
suma.
primerNumero = 234; da el valor del primer número que queremos sumar
segundoNumero = 567; da el valor del segundo número que queremos sumar
suma = primerNumero + segundoNumero; halla la suma de esos dos números y la
guarda en otra variable, en vez de mostrarla directamente en pantalla.
System.Console.WriteLine("La suma de {0} y {1} es {2}", primerNumero,
segundoNumero, suma); muestra en pantalla el texto y los valores de las tres variables
(los dos números iniciales y su suma).
Ejercicio propuesto: Hacer un programa que calcule el producto de los números 121 y 132,
usando variables.
1.6. Identificadores
Estos nombres de variable (lo que se conoce como "identificadores") pueden estar formados
por letras, números o el símbolo de subrayado (_) y deben comenzar por letra o subrayado. No
deben tener espacios entre medias, y hay que recordar que las vocales acentuadas y la eñe son
problemáticas, porque no son letras "estándar" en todos los idiomas.
Tampoco podremos usar como identificadores las palabras reservadas de C#. Por ejemplo, la
palabra "int" se refiere a que cierta variable guardará un número entero, así que esa palabra
"int" no la podremos usar tampoco como nombre de variable (pero no vamos a incluir ahora
una lista de palabras reservadas de C#, ya nos iremos encontrando con ellas).
De momento, intentaremos usar nombres de variables que a nosotros nos resulten claros, y
que no parezca que puedan ser alguna orden de C#.
PrimerNumero = 0;
primernumero = 0;
o cualquier variación similar, el compilador protestará y nos dirá que no conoce esa variable,
porque la habíamos declarado como
int primerNumero;
1.7. Comentarios
Podemos escribir comentarios, que el compilador ignora, pero que pueden servir para
aclararnos cosas a nosotros. Se escriben entre /* y */:
Es conveniente escribir comentarios que aclaren la misión de las partes de nuestros programas
que puedan resultar menos claras a simple vista. Incluso suele ser aconsejable que el
programa comience con un comentario, que nos recuerde qué hace el programa sin que
necesitemos mirarlo de arriba a abajo. Un ejemplo casi exagerado:
/* Esto
es un comentario que
ocupa más de una línea
*/
También es posible declarar otro tipo de comentarios, que comienzan con doble barra y
terminan cuando se acaba la línea (estos comentarios, claramente, no podrán ocupar más de
una línea). Son los "comentarios al estilo de C++":
texto = System.Console.ReadLine();
pero eso ocurrirá en el próximo tema, cuando veamos cómo manejar textos. De momento,
nosotros sólo sabemos manipular números enteros, así que deberemos convertir ese dato a un
número entero, usando Convert.ToInt32:
primerNumero = System.Convert.ToInt32(System.Console.ReadLine());
Un ejemplo de programa que sume dos números tecleados por el usuario sería:
Va siendo hora de hacer una pequeña mejora: no es necesario repetir "System." al principio de
la mayoría de las órdenes que tienen que ver con el sistema (por ahora, las de consola y las de
conversión), si al principio del programa utilizamos "using System":
using System;
Ejercicios propuestos:
1. Multiplicar dos números tecleados por usuario.
2. El usuario tecleará dos números (x e y), y el programa deberá calcular cual es el
resultado de su división y el resto de esa división.
3. El usuario tecleará dos números (a y b), y el programa mostrará el resultado de la
operación (a+b)*(a-b) y el resultado de la operación a2-b2.
4. Sumar tres números tecleados por usuario.
5. Pedir al usuario un número y mostrar su tabla de multiplicar. Por ejemplo, si el número
es el 3, debería escribirse algo como
3x 0=0
3x 1=3
3x 2=6
…
3x 10 = 30
Empieza a ser el momento de refinar, de dar más detalles. El primer "matiz" importante que
nos hemos saltado es el tamaño de los números que podemos emplear, así como su signo
(positivo o negativo). Por ejemplo, un dato de tipo "int" puede guardar números de hasta unas
nueve cifras, tanto positivos como negativos, y ocupa 4 bytes en memoria.
Pero no es la única opción. Por ejemplo, si queremos guardar la edad de una persona, no
necesitamos usar números negativos, y nos bastaría con 3 cifras, así que es de suponer que
existirá algún tipo de datos más adecuado, que desperdicie menos memoria. También existe el
caso contrario: un banco puede necesitar manejar números con más de 9 cifras, así que un
dato "int" se les quedaría corto. Siendo estrictos, si hablamos de valores monetarios,
necesitaríamos usar decimales, pero eso lo dejamos para el siguiente apartado.
Nombre Tamaño
Del Tipo (bytes) Rango de valores
sbyte 1 -128 a 127
byte 1 0 a 255
short 2 -32768 a 32767
ushort 2 0 a 65535
int 4 -2147483648 a 2147483647
uint 4 0 a 4294967295
long 8 -9223372036854775808 a 9223372036854775807
ulong 8 0 a 18446744073709551615
Como se puede observar en esta tabla, el tipo de dato más razonable para guardar edades
sería "byte", que permite valores entre 0 y 255, y ocupa 3 bytes menos que un "int".
Ejercicios propuestos:
• Preguntar al usuario su edad, que se guardará en un "byte". A continuación, se deberá
le deberá decir que no aparenta tantos años (por ejemplo, "No aparentas 20 años").
• Pedir al usuario dos números de dos cifras ("byte"), calcular su multiplicación, que se
deberá guardar en un "ushort", y mostrar el resultado en pantalla.
• Pedir al usuario dos números enteros largos ("long") y mostrar cuanto es su suma, su
resta y su producto.
a = a + 1;
Pues bien, en C#, existe una notación más compacta para esta operación, y para la opuesta (el
decremento):
Pero esto tiene más misterio todavía del que puede parecer en un primer vistazo: podemos
distinguir entre "preincremento" y "postincremento". En C# es posible hacer asignaciones como
b = a++;
Así, si "a" valía 2, lo que esta instrucción hace es dar a "b" el valor de "a" y aumentar el valor
de "a". Por tanto, al final tenemos que b=2 y a=3 (postincremento: se incrementa "a" tras
asignar su valor).
En cambio, si escribimos
b = ++a;
y "a" valía 2, primero aumentamos "a" y luego los asignamos a "b" (preincremento), de modo
que a=3 y b=3.
Ejercicios propuestos:
Crear un programa que use tres variables x,y,z. Sus valores iniciales serán 15, -10,
2.147.483.647. Se deberá incrementar el valor de estas variables. ¿Qué valores esperas
que se obtengan? Contrástalo con el resultado obtenido por el programa.
¿Cuál sería el resultado de las siguientes operaciones? a=5; b=++a; c=a++; b=b*5;
a=a*2;
Y ya que estamos hablando de las asignaciones, hay que comentar que en C# es posible hacer
asignaciones múltiples:
a = b = c = 1;
Ejercicios propuestos:
Crear un programa que use tres variables x,y,z. Sus valores iniciales serán 15, -10, 214.
Se deberá incrementar el valor de estas variables en 12, usando el formato abreviado.
¿Qué valores esperas que se obtengan? Contrástalo con el resultado obtenido por el
programa.
¿Cuál sería el resultado de las siguientes operaciones? a=5; b=a+2; b-=3; c=-3;
c*=2; ++c; a*=b;
float x;
o bien, si queremos dar un valor inicial en el momento de definirlos (recordando que para las
cifras decimales no debemos usar una coma, sino un punto):
float x = 12.56;
using System;
}
}
Cuidado al probar este programa: aunque en el fuente debemos escribir los decimales usando
un punto, como 123.456, al poner el ejecutable en marcha parte del trabajo se le encarga al
sistema operativo, de modo que si éste sabe que en nuestro país se usa la coma para los
decimales, considere la coma el separador correcto y no el punto, como ocurre si introducimos
estos datos en la versión española de Windows XP:
ejemplo05
Introduce el primer número
23,6
Introduce el segundo número
34.2
La suma de 23,6 y 342 es 365,6
Una forma de conseguirlo es crear una cadena de texto a partir del número, usando "ToString".
A esta orden se le puede indicar un dato adicional, que es el formato numérico que queremos
usar, por ejemplo: suma.ToString("0.00")
• Un cero (0) indica una posición en la que debe aparecer un número, y se mostrará un 0
si no hay ninguno.
• Una almohadilla (#) indica una posición en la que puede aparecer un número, y no se
escribirá nada si no hay número.
• Un punto (.) indica la posición en la que deberá aparecer la coma decimal.
• Alternativamente, se pueden usar otros formatos abreviados: por ejemplo, N2 quiere
decir "con dos cifras decimales" y N5 es "con cinco cifras decimales"
using System;
Console.WriteLine( numero.ToString("N1") );
Console.WriteLine( numero.ToString("N3") );
Console.WriteLine( numero.ToString("0.0") );
Console.WriteLine( numero.ToString("0.000") );
Console.WriteLine( numero.ToString("#.#") );
Console.WriteLine( numero.ToString("#.###") );
}
}
ejemplo06
12,3
12,340
12,3
12,340
12,3
12,34
Ejercicios propuestos:
using System;
}
}
Su resultado sería:
f7
11110111
char letra;
letra = 'a';
Para leer valores desde teclado, lo podemos hacer de forma similar a los casos anteriores:
leemos toda una frase con ReadLine y convertimos a tipo "char" usando Convert.ToChar:
letra = Convert.ToChar(Console.ReadLine());
Así, un programa que de un valor inicial a una letra, la muestre, lea una nueva letra tecleada
por el usuario, y la muestre, podría ser:
using System;
letra = 'a';
Console.WriteLine("La letra es {0}", letra);
pueden escribir después de una barra invertida (\) y que nos permiten conseguir escribir esas
comillas dobles y algún otro carácter poco habitual. Por ejemplo, con \" se escribirán unas
comillas dobles, y con \' unas comillas simples, o con \n se avanzará a la línea siguiente de
pantalla.
Secuencia Significado
\a Emite un pitido
\b Retroceso (permite borrar el último carácter)
\f Avance de página (expulsa una hoja en la impresora)
\n Avanza de línea (salta a la línea siguiente)
\r Retorno de carro (va al principio de la línea)
\t Salto de tabulación horizontal
\v Salto de tabulación vertical
\' Muestra una comilla simple
\" Muestra una comilla doble
\\ Muestra una barra invertida
\0 Carácter nulo (NULL)
using System;
Juguemos mas:
otro salto
Comillas dobles: " y simples ', y barra \
Revisión 0.95 – Página 34
Introducción a la programación con C#, por Nacho Cabanes
En algunas ocasiones puede ser incómodo manipular estas secuencias de escape. Por ejemplo,
cuando usemos estructuras de directorios: c:\\datos\\ejemplos\\curso\\ejemplo1. En este caso,
se puede usar una arroba (@) antes del texto sin las barras invertidas:
ruta = @"c:\datos\ejemplos\curso\ejemplo1"
En este caso, el problema está si aparecen comillas en medio de la cadena. Para solucionarlo,
se duplican las comillas, así:
Ejercicio propuesto: Crear un programa que pida al usuario que teclee cuatro letras y las
muestre en pantalla juntas, pero en orden inverso, y entre comillas dobles. Por ejemplo si las
letras que se teclean son a, l, o, h, escribiría "hola".
Así, un ejemplo que diera un valor a un "string", lo mostrara (entre comillas, para practicar las
secuencias de escape que hemos visto en el apartado anterior) y leyera un valor tecleado por el
usuario podría ser:
using System;
Se pueden hacer muchas más operaciones sobre cadenas de texto: convertir a mayúsculas o a
minúsculas, eliminar espacios, cambiar una subcadena por otra, dividir en trozos, etc. Pero ya
volveremos a las cadenas más adelante, cuando conozcamos las estructuras de control básicas.
bool encontrado;
encontrado = true;
Este tipo de datos hará que podamos escribir de forma sencilla algunas condiciones que podrían
resultar complejas, pero eso lo veremos en el tema 3.
3. Estructuras de control
3.1. Estructuras alternativas
3.1.1. if
Vamos a ver cómo podemos comprobar si se cumplen condiciones. La primera construcción
que usaremos será "si ... entonces ...". El formato es
if (condición) sentencia;
/*---------------------------*/
/* Ejemplo en C# nº 10: */
/* ejemplo10.cs */
/* */
/* Condiciones con if */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero>0) Console.WriteLine("El número es positivo.");
}
}
Este programa pide un número al usuario. Si es positivo (mayor que 0), escribe en pantalla "El
número es positivo."; si es negativo o cero, no hace nada.
Como se ve en el ejemplo, para comprobar si un valor numérico es mayor que otro, usamos el
símbolo ">". Para ver si dos valores son iguales, usaremos dos símbolos de "igual": if
(numero==0). Las demás posibilidades las veremos algo más adelante. En todos los casos, la
condición que queremos comprobar deberá indicarse entre paréntesis.
Este programa comienza por un comentario que nos recuerda de qué se trata. Como nuestros
fuentes irán siendo cada vez más complejos, a partir de ahora incluiremos comentarios que nos
permitan recordar de un vistazo qué pretendíamos hacer.
Ejercicios propuestos:
Crear un programa que pida al usuario un número entero y diga si es par (pista: habrá
que comprobar si el resto que se obtiene al dividir entre dos es cero: if (x % 2 == 0)
…).
Crear un programa que pida al usuario dos números enteros y diga cual es el mayor de
ellos.
Crear un programa que pida al usuario dos números enteros y diga si el primero es
múltiplo del segundo (pista: igual que antes, habrá que ver si el resto de la división es
cero: a % b == 0).
/*---------------------------*/
/* Ejemplo en C# nº 11: */
/* ejemplo11.cs */
/* */
/* Condiciones con if (2) */
/* Sentencias compuestas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero > 0) {
Console.WriteLine("El número es positivo.");
Console.WriteLine("Recuerde que también puede usar negativos.");
} /* Aqui acaba el "if" */
} /* Aqui acaba "Main" */
} /* Aqui acaba "Ejemplo11" */
En este caso, si el número es positivo, se hacen dos cosas: escribir un texto y luego... ¡escribir
otro! (Claramente, en este ejemplo, esos dos "WriteLine" podrían ser uno solo, en el que los
dos textos estuvieran separados por un "\n"; más adelante iremos encontrando casos en lo que
necesitemos hacer cosas "más serias" dentro de una sentencia compuesta).
Como se ve en este ejemplo, cada nuevo "bloque" se suele escribir un poco más a la derecha
que los anteriores, para que sea fácil ver dónde comienza y termina cada sección de un
Revisión 0.95 – Página 38
Introducción a la programación con C#, por Nacho Cabanes
programa. Por ejemplo, el contenido de "Ejemplo11" está un poco más a la derecha que la
cabecera "public class Ejemplo11", y el contenido de "Main" algo más a la derecha, y la
sentencia compuesta que se debe realizar si se cumple la condición del "if" está algo más a la
derecha que la orden "if". Este "sangrado" del texto se suele llamar "escritura indentada".
Un tamaño habitual para el sangrado es de 4 espacios, aunque en este texto habitualmente
usaremos sólo dos espacios, para no llegar al margen derecho del papel con mucha facilidad.
Operador Operación
< Menor que
> Mayor que
<= Menor o igual que
>= Mayor o igual que
== Igual a
!= No igual a (distinto de)
/*---------------------------*/
/* Ejemplo en C# nº 12: */
/* ejemplo12.cs */
/* */
/* Condiciones con if (3) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero != 0) Console.WriteLine("El número no es cero.");
}
}
Ejercicios propuestos:
Crear un programa que multiplique dos números enteros de la siguiente forma: pedirá
al usuario un primer número entero. Si el número que se que teclee es 0, escribirá en
pantalla "El producto de 0 por cualquier número es 0". Si se ha tecleado un número
distinto de cero, se pedirá al usuario un segundo número y se mostrará el producto de
ambos.
Crear un programa que pida al usuario dos números reales. Si el segundo no es cero,
mostrará el resultado de dividir entre el primero y el segundo. Por el contrario, si el
segundo número es cero, escribirá “Error: No se puede dividir entre cero”.
3.1.4. if-else
Podemos indicar lo que queremos que ocurra en caso de que no se cumpla la condición, usando
la orden "else" (en caso contrario), así:
/*---------------------------*/
/* Ejemplo en C# nº 13: */
/* ejemplo13.cs */
/* */
/* Condiciones con if (4) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero > 0) Console.WriteLine("El número es positivo.");
else Console.WriteLine("El número es cero o negativo.");
}
}
Podríamos intentar evitar el uso de "else" si utilizamos un "if" a continuación de otro, así:
/*---------------------------*/
/* Ejemplo en C# nº 14: */
/* ejemplo14.cs */
/* */
/* Condiciones con if (5) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
Revisión 0.95 – Página 40
Introducción a la programación con C#, por Nacho Cabanes
Podemos enlazar los "if" usando "else", para decir "si no se cumple esta condición, mira a ver si
se cumple esta otra":
/*---------------------------*/
/* Ejemplo en C# nº 15: */
/* ejemplo15.cs */
/* */
/* Condiciones con if (6) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero > 0)
Console.WriteLine("El número es positivo.");
else
if (numero < 0)
Console.WriteLine("El número es negativo.");
else
Console.WriteLine("El número es cero.");
}
}
Ejercicio propuesto:
• Mejorar la solución a los dos ejercicios del apartado anterior, usando "else".
Operador Significado
&& Y
|| O
! No
Ejercicios propuestos:
Crear un programa que pida una letra al usuario y diga si se trata de una vocal.
Crear un programa que pida al usuario dos números enteros y diga "Uno de los
números es positivo", "Los dos números son positivos" o bien "Ninguno de los números
es positivo", según corresponda.
Crear un programa que pida al usuario tres números reales y muestre cuál es el mayor
de los tres.
Crear un programa que pida al usuario dos números enteros cortos y diga si son iguales
o, en caso contrario, cuál es el mayor de ellos.
Crear un programa que pida al usuario su nombre, y le diga "Hola" si se llama "Juan", o
bien le diga "No te conozco" si teclea otro nombre.
En algunos compiladores de lenguaje C, esto podría ser un problema serio, porque se considera
válido hacer una asignación dentro de un "if" (aunque la mayoría de compiladores modernos
nos avisarían de que quizá estemos asignando un valor por error). En el caso del lenguaje C#,
la "condición" debe ser algo cuyo resultado "verdadero" o "falso" (un dato de tipo "bool"),de
modo que obtendríamos un error "Cannot implicitly convert type 'int' to 'bool'" (no puedo
convertir un "int" a "bool", en casos como el del siguiente programa:
/*---------------------------*/
/* Ejemplo en C# nº 16: */
/* ejemplo16.cs */
/* */
/* Condiciones con if (7) */
/* comparacion incorrecta */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce un número");
numero = Convert.ToInt32(Console.ReadLine());
if (numero = 0)
Revisión 0.95 – Página 42
Introducción a la programación con C#, por Nacho Cabanes
A las variables "bool" se les da un valor casi como a cualquier otra variable, con la diferencia de
que en la parte derecha de la asignación aparece "true" , "false" o una comparación, así:
partidaTerminada = false;
partidaTerminada = (enemigos ==0) && (nivel == ultimoNivel);
if (vidas ==0) partidaTerminada = true;
Lo emplearemos a partir de ahora en los fuentes que usen condiciones un poco complejas. Un
ejemplo que pida una letra y diga si es una vocal, una cifra numérica u otro símbolo, usando
variables "bool" podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 17: */
/* ejemplo17.cs */
/* */
/* Condiciones con if (8) */
/* Variables bool */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
if (esCifra)
Console.WriteLine("Es una cifra numérica.");
else if (esVocal)
Console.WriteLine("Es una vocal.");
else
Console.WriteLine("Es una consonante u otro número.");
}
}
Para ayudarnos a centrarnos en el problema, existen notaciones gráficas, como los diagramas
de flujo, que nos permiten ver mejor qué se debe hacer y cuando.
En primer lugar, vamos a ver los 4 elementos básicos de un diagrama de flujo, y luego los
aplicaremos a un caso concreto.
El inicio o el final del programa se indica dentro de un círculo. Los procesos internos, como
realizar operaciones, se encuadran en un rectángulo. Las entradas y salidas (escrituras en
pantalla y lecturas de teclado) se indican con un paralelogramo que tenga su lados superior e
inferior horizontales, pero no tenga verticales los otros dos. Las decisiones se indican dentro de
un rombo.
El paso de aquí al correspondiente programa en lenguaje C# (el que vimos en el ejemplo 16)
debe ser casi inmediato: sabemos como leer de teclado, como escribir en pantalla, y las
decisiones serán un "if", que si se cumple ejecutará la sentencia que aparece en su salida "si" y
si no se cumple ("else") ejecutará lo que aparezca en su salida "no".
Ejercicios propuestos:
Crear el diagrama de flujo y la versión en C# de un programa que dé al usuario tres
oportunidades para adivinar un número del 1 al 10.
Crear el diagrama de flujo para el programa que pide al usuario dos números y dice si
uno de ellos es positivo, si lo son los dos o si no lo es ninguno.
Crear el diagrama de flujo para el programa que pide tres números al usuario y dice
cuál es el mayor de los tres.
y equivale a decir "si se cumple la condición, toma el valor v1; si no, toma el valor v2". Un
ejemplo de cómo podríamos usarlo sería
numeroMayor = (a>b) ? a : b;
/*---------------------------*/
/* Ejemplo en C# nº 18: */
/* ejemplo18.cs */
/* */
/* El operador condicional */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
mayor = (a>b) ? a : b;
(La orden Console.Write, empleada en el ejemplo anterior, escribe un texto sin avanzar a la
línea siguiente, de modo que el próximo texto que escribamos –o introduzcamos- quedará a
continuación de éste).
Un segundo ejemplo, que sume o reste dos números según la opción que se escoja, sería:
/*---------------------------*/
/* Ejemplo en C# nº 19: */
/* ejemplo19.cs */
/* */
/* Operador condicional - 2 */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Ejercicios propuestos:
• Crear un programa que use el operador condicional para mostrar un el valor absoluto
de un número de la siguiente forma: si el número es positivo, se mostrará tal cual; si es
negativo, se mostrará cambiado de signo.
• Crear un programa que use el operador condicional para dar a una variable llamada
"iguales" (booleana) el valor "true" si los dos números que ha tecleado el usuario son
iguales, o "false" si son distintos.
• Usar el operador condicional para calcular el mayor de dos números.
3.1.10. switch
Si queremos ver varios posibles valores, sería muy pesado tener que hacerlo con muchos
"if" seguidos o encadenados. La alternativa es la orden "switch", cuya sintaxis es
switch (expresión)
{
case valor1: sentencia1;
break;
case valor2: sentencia2;
sentencia2b;
break;
...
case valorN: sentenciaN;
break;
default:
otraSentencia;
break;
};
Es decir, se escribe tras "switch" la expresión a analizar, entre paréntesis. Después, tras varias
órdenes "case" se indica cada uno de los valores posibles. Los pasos (porque pueden ser
varios) que se deben dar si se trata de ese valor se indican a continuación, terminando con
"break". Si hay que hacer algo en caso de que no se cumpla ninguna de las condiciones, se
detalla tras "default". Si dos casos tienen que hacer lo mismo, se añade "goto case" a uno de
ellos para indicarlo.
Vamos con un ejemplo, que diga si la tecla que pulsa el usuario es una cifra numérica, un
espacio u otro símbolo:
/*---------------------------*/
/* Ejemplo en C# nº 20: */
/* ejemplo20.cs */
/* */
/* La orden "switch" (1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
switch (letra)
{
case ' ': Console.WriteLine("Espacio.");
break;
case '1': goto case '0';
case '2': goto case '0';
case '3': goto case '0';
case '4': goto case '0';
case '5': goto case '0';
case '6': goto case '0';
case '7': goto case '0';
case '8': goto case '0';
case '9': goto case '0';
case '0': Console.WriteLine("Dígito.");
break;
default: Console.WriteLine("Ni espacio ni dígito.");
break;
}
}
}
Cuidado quien venga del lenguaje C: en C se puede dejar que un caso sea manejado por el
siguiente, lo que se consigue si no se usa "break", mientras que C# siempre obliga a usar
"break" o "goto" en cada caso, para evitar errores difíciles de detectar. El fragmento anterior en
C habría sido:
switch (tecla)
{
case ' ': printf("Espacio.\n");
break;
case '1':
case '2':
case '3':
case '4':
case '5':
case '6':
case '7':
case '8':
case '9':
case '0': printf("Dígito.\n");
break;
default: printf("Ni espacio ni dígito.\n");
}
En el lenguaje C, que es más antiguo, sólo se podía usar "switch" para comprobar valores de
variables "simples" (numéricas y caracteres); en C#, que es un lenguaje más evolucionado, se
puede usar también para comprobar valores de "strings". Por ejemplo, un programa que nos
salude de forma personalizada si somos "Juan" o "Pedro" podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 21: */
/* ejemplo21.cs */
/* */
/* La orden "switch" (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Introduce tu nombre");
nombre = Console.ReadLine();
switch (nombre)
{
case "Juan": Console.WriteLine("Bienvenido, Juan.");
break;
case "Pedro": Console.WriteLine("Que tal estas, Pedro.");
break;
default: Console.WriteLine("Procede con cautela, desconocido.");
break;
}
}
}
Ejercicios propuestos:
Crear un programa que lea una letra tecleada por el usuario y diga si se trata de una
vocal, una cifra numérica o una consonante.
Crear un programa que lea una letra tecleada por el usuario y diga si se trata de un
signo de puntuación, una cifra numérica o algún otro carácter.
Repetir los dos programas anteriores, empleando "if" en lugar de "switch".
3.2.1. while
Si queremos hacer que una sección de nuestro programa se repita mientras se cumpla una
cierta condición, usaremos la orden "while". Esta orden tiene dos formatos distintos, según
comprobemos la condición al principio o al final.
while (condición)
sentencia;
Un ejemplo que nos diga si cada número que tecleemos es positivo o negativo, y que pare
cuando tecleemos el número 0, podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 22: */
/* ejemplo22.cs */
/* */
/* La orden "while" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
{
if (numero > 0) Console.WriteLine("Es positivo");
else Console.WriteLine("Es negativo");
Console.WriteLine("Teclea otro número (0 para salir): ");
numero = Convert.ToInt32(Console.ReadLine());
}
}
}
Ejercicios propuestos:
• Crear un programa que pida al usuario su contraseña. Deberá terminar cuando
introduzca como contraseña la palabra "clave", pero volvérsela a pedir tantas veces
como sea necesario.
• Crea un programa que escriba en pantalla los números del 1 al 10, usando "while".
• Crea un programa que escriba en pantalla los números pares del 26 al 10 (descen-
diendo), usando "while".
• Crear un programa calcule cuantas cifras tiene un número entero positivo (pista: se
puede hacer dividiendo varias veces entre 10).
do
sentencia;
while (condición)
Al igual que en el caso anterior, si queremos que se repitan varias órdenes (es lo habitual),
deberemos encerrarlas entre llaves.
Como ejemplo, vamos a ver cómo sería el típico programa que nos pide una clave de acceso y
no nos deja entrar hasta que tecleemos la clave correcta:
/*---------------------------*/
/* Ejemplo en C# nº 23: */
/* ejemplo23.cs */
/* */
/* La orden "do..while" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
do
{
Console.Write("Introduzca su clave numérica: ");
clave = Convert.ToInt32(Console.ReadLine());
if (clave != valida) Console.WriteLine("No válida!\n");
}
while (clave != valida);
Console.WriteLine("Aceptada.\n");
}
}
En este caso, se comprueba la condición al final, de modo que se nos preguntará la clave al
menos una vez. Mientras que la respuesta que demos no sea la correcta, se nos vuelve a
preguntar. Finalmente, cuando tecleamos la clave correcta, el ordenador escribe "Aceptada" y
termina el programa.
Si preferimos que la clave sea un texto en vez de un número, los cambios al programa son
mínimos:
/*---------------------------*/
/* Ejemplo en C# nº 24: */
/* ejemplo24.cs */
/* */
/* La orden "do..while" (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
do
{
Console.Write("Introduzca su clave: ");
clave = Console.ReadLine();
if (clave != valida) Console.WriteLine("No válida!\n");
}
while (clave != valida);
Console.WriteLine("Aceptada.\n");
}
}
Ejercicios propuestos:
• Crear un programa que pida números positivos al usuario, y vaya calculando la suma
de todos ellos (terminará cuando se teclea un número negativo o cero).
• Crea un programa que escriba en pantalla los números del 1 al 10, usando "do..while".
• Crea un programa que escriba en pantalla los números pares del 26 al 10 (descen-
diendo), usando "do..while".
• Crea un programa que pida al usuario su nombre y su contraseña, y no le permita
seguir hasta que introduzca como nombre "Pedro" y como contraseña "Peter".
3.2.3. for
Ésta es la orden que usaremos habitualmente para crear partes del programa que se repitan
un cierto número de veces. El formato de "for" es
Así, para contar del 1 al 10, tendríamos 1 como valor inicial, <=10 como condición de
repetición, y el incremento sería de 1 en 1. Por tanto, el programa quedaría:
/*---------------------------*/
/* Ejemplo en C# nº 25: */
/* ejemplo25.cs */
/* */
/* Uso básico de "for" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
int contador;
}
}
Ejercicios propuestos:
Crear un programa que muestre los números del 15 al 5, descendiendo (pista: en cada
pasada habrá que descontar 1, por ejemplo haciendo i--).
Revisión 0.95 – Página 53
Introducción a la programación con C#, por Nacho Cabanes
Crear un programa que muestre los primeros ocho números pares (pista: en cada
pasada habrá que aumentar de 2 en 2, o bien mostrar el doble del valor que hace de
contador).
En un "for", realmente, la parte que hemos llamado "Incremento" no tiene por qué incrementar
la variable, aunque ése es su uso más habitual. Es simplemente una orden que se ejecuta
cuando se termine la "Sentencia" y antes de volver a comprobar si todavía se cumple la
condición de repetición.
Un caso todavía más exagerado de algo a lo que se entra y de lo que no se sale sería la
siguiente orden:
for ( ; ; )
Los bucles "for" se pueden anidar (incluir uno dentro de otro), de modo que podríamos escribir
las tablas de multiplicar del 1 al 5 con:
/*---------------------------*/
/* Ejemplo en C# nº 26: */
/* ejemplo26.cs */
/* */
/* "for" anidados */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
En estos ejemplos que hemos visto, después de "for" había una única sentencia. Si queremos
que se hagan varias cosas, basta definirlas como un bloque (una sentencia compuesta)
encerrándolas entre llaves. Por ejemplo, si queremos mejorar el ejemplo anterior haciendo que
deje una línea en blanco entre tabla y tabla, sería:
/*---------------------------*/
/* Ejemplo en C# nº 27: */
/* ejemplo27.cs */
/* */
/* "for" anidados (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Para "contar" no necesariamente hay que usar números. Por ejemplo, podemos contar con
letras así:
/*---------------------------*/
/* Ejemplo en C# nº 28: */
/* ejemplo28.cs */
/* */
/* "for" que usa "char" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
char letra;
}
}
Si queremos contar de forma decreciente, o de dos en dos, o como nos interese, basta
indicarlo en la condición de finalización del "for" y en la parte que lo incrementa:
/*---------------------------*/
/* Ejemplo en C# nº 29: */
/* ejemplo29.cs */
/* */
/* "for" que descuenta */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
char letra;
}
}
Ejercicios propuestos:
Crear un programa que muestre las letras de la Z (mayúscula) a la A (mayúscula,
descendiendo).
Crear un programa que escriba en pantalla la tabla de multiplicar del 5.
Crear un programa que escriba en pantalla los números del 1 al 50 que sean múltiplos
de 3 (pista: habrá que recorrer todos esos números y ver si el resto de la división entre
3 resulta 0).
/*---------------------------*/
/* Ejemplo en C# nº 30: */
/* ejemplo30.cs */
/* */
/* "for" interrumpido con */
/* "break" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
int contador;
}
}
1 2 3 4
(en cuanto se llega al valor 5, se interrumpe el "for", por lo que no se alcanza el valor 10).
/*---------------------------*/
/* Ejemplo en C# nº 31: */
/* ejemplo31.cs */
/* */
/* "for" interrumpido con */
/* "continue" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
int contador;
}
}
1 2 3 4 6 7 8 9 10
Ejercicios resueltos:
¿Qué escribiría en pantalla este fragmento de código?
Respuesta: los números del 1 al 3 (se empieza en 1 y se repite mientras sea menor que
4).
Respuesta: escribe un 5, porque hay un punto y coma después del "for", de modo que
repite cuatro veces una orden vacía, y cuando termina, "i" ya tiene el valor 5.
Respuesta: escribe los números del 0 al 4, porque la condición del "continue" nunca se
llega a dar.
Respuesta: escribe 5, porque no hay llaves tras el "for", luego sólo se repite la orden
"if".
El formato de "goto" es
goto donde;
donde:
/*---------------------------*/
/* Ejemplo en C# nº 32: */
/* ejemplo32.cs */
/* */
/* "for" y "goto" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
int i, j;
}
}
i vale 0 y j vale 0.
i vale 0 y j vale 2.
i vale 0 y j vale 4.
i vale 0 y j vale 6.
i vale 0 y j vale 8.
i vale 0 y j vale 10.
i vale 0 y j vale 12.
Revisión 0.95 – Página 60
Introducción a la programación con C#, por Nacho Cabanes
Por su parte, un bucle "while" se vería como una condición que hace que algo se repita (una
flecha que vuelve hacia atrás, al punto en el que se comprobaba la condición):
Aun así, existen otras notaciones más modernas y que pueden resultar más cómodas. Sólo
comentaremos una: los diagramas de Chapin. En ellos se representa cada orden dentro de una
caja:
Pedir número n1
Pedir número n2
n1>n2?
si no
Decir "n1 es Decir "n2 es
mayor" mayor"
Y las condiciones repetitivas se indican dejando una barra a la izquierda, que marca qué es lo
que se repite, tanto si la condición se comprueba al final (do..while):
Abrir fichero
Mientras haya datos en fichero
Leer dato
Mostrar dato
Cerrar fichero
En general, nos interesará usar "while" cuando puede que la parte repetitiva no se
llegue a repetir nunca (por ejemplo: cuando leemos un fichero, si el fichero está vacío,
no habrá datos que leer).
De igual modo, "do...while" será lo adecuado cuando debamos repetir al menos una
vez (por ejemplo, para pedir una clave de acceso, se le debe preguntar al menos una
vez al usuario, o quizá más veces, si la teclea correctamente).
En cuanto a "for", es equivalente a un "while", pero la sintaxis habitual de la oren "for"
hace que sea especialmente útil cuando sabemos exactamente cuantas veces queremos
que se repita (por ejemplo: 10 veces sería "for (i=1; i<=10; i++)").
Ejercicios propuestos:
Por ejemplo, si queremos definir un grupo de números enteros, el tipo de datos que usaremos
para declararlo será "int[]". Si sabemos desde el principio cuantos datos tenemos (por ejemplo
4), les reservaremos espacio con "= new int[4]", así
Podemos acceder a cada uno de los valores individuales indicando su nombre (ejemplo) y el
número de elemento que nos interesa, pero con una precaución: se empieza a numerar desde
0, así que en el caso anterior tendríamos 4 elementos, que serían ejemplo[0], ejemplo[1],
ejemplo[2], ejemplo[3].
/*---------------------------*/
/* Ejemplo en C# nº 33: */
/* ejemplo33.cs */
/* */
/* Primer ejemplo de tablas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Ejercicios propuestos:
• Un programa que pida al usuario 4 números, los memorice (utilizando una tabla),
calcule su media aritmética y después muestre en pantalla la media y los datos
tecleados.
• Un programa que pida al usuario 5 números reales y luego los muestre en el orden
contrario al que se introdujeron.
• Un programa que pida al usuario 10 números enteros y calcule (y muestre) cuál es el
mayor de ellos.
/*---------------------------*/
/* Ejemplo en C# nº 34: */
/* ejemplo34.cs */
/* */
/* Segundo ejemplo de */
/* tablas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Ejercicios propuestos:
• Un programa que almacene en una tabla el número de días que tiene cada mes (su-
pondremos que es un año no bisiesto), pida al usuario que le indique un mes (1=enero,
12=diciembre) y muestre en pantalla el número de días que tiene ese mes.
• Un programa que almacene en una tabla el número de días que tiene cada mes (año
no bisiesto), pida al usuario que le indique un mes (ej. 2 para febrero) y un día (ej. el
día 15) y diga qué número de día es dentro del año (por ejemplo, el 15 de febrero sería
el día número 46, el 31 de diciembre sería el día 365).
El "truco" consistirá en emplear cualquiera de las estructuras repetitivas que ya hemos visto
(while, do..while, for), por ejemplo así:
/*---------------------------*/
/* Ejemplo en C# nº 35: */
/* ejemplo35.cs */
/* */
/* Tercer ejemplo de */
/* tablas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
En este caso, que sólo sumábamos 5 números, no hemos escrito mucho menos, pero si
trabajásemos con 100, 500 o 1000 números, la ganancia en comodidad sí que está clara.
/*---------------------------*/
/* Ejemplo en C# nº 36: */
/* ejemplo36.cs */
/* */
/* Cuarto ejemplo de */
/* tablas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Ejercicios propuestos:
• A partir del programa propuesto en 4.1.2, que almacenaba en una tabla el número de
días que tiene cada mes, crear otro que pida al usuario que le indique la fecha,
detallando el día (1 al 31) y el mes (1=enero, 12=diciembre), como respuesta muestre
en pantalla el número de días que quedan hasta final de año.
Revisión 0.95 – Página 67
Introducción a la programación con C#, por Nacho Cabanes
• Crear un programa que pida al usuario 10 números y luego los muestre en orden
inverso (del último que se ha introducido al primero que se introdujo).
• Un programa que pida al usuario 10 números y luego calcule y muestre cual es el
mayor de todos ellos.
• Un programa que pida al usuario 10 números, calcule su media y luego muestre los que
están por encima de la media.
• Un programa que pida 10 nombres y los memorice (pista: esta vez se trata de un array
de "string"). Después deberá pedir que se teclee un nombre y dirá si se encuentra o no
entre los 10 que se han tecleado antes. Volverá a pedir otro nombre y a decir si se
encuentra entre ellos, y así sucesivamente hasta que se teclee "fin".
• Un programa que prepare espacio para un máximo de 100 nombres. El usuario deberá
ir introduciendo un nombre cada vez, hasta que se pulse Intro sin teclear nada,
momento en el que dejarán de pedirse más nombres y se mostrará en pantalla la lista
de los nombres que se han introducido.
• Un programa que prepare espacio para un máximo de 10 nombres. Deberá mostrar al
usuario un menú que le permita realizar las siguientes operaciones:
o Añadir un dato al final de los ya existentes.
o Insertar un dato en una cierta posición (lo que quedén detrás deberán
desplazarse "a la derecha" para dejarle hueco; por ejemplo, si el array contiene
"hola", "adios" y se pide insertar "bien" en la segunda posición, el array pasará
a contener "hola", "bien", "adios".
o Borrar el dato que hay en una cierta posición (lo que estaban detrás deberán
desplazarse "a la izquierda" para que no haya huecos; por ejemplo, si el array
contiene "hola", "bien", "adios" y se pide borrar el dato de la segunda posición,
el array pasará a contener "hola", "adios"
o Mostrar los datos que contiene el array.
o Salir del programa.
En cualquier caso, si queremos indicar valores iniciales, lo haremos entre llaves, igual que si
fuera una tabla de una única dimensión.
Vamos a ver un primer ejemplo del uso con arrays de la forma [2,20], lo que podríamos llamar
el "estilo Pascal", en el usemos tanto arrays con valores prefijados, como arrays para los que
reservemos espacio con "new" y a los que demos valores más tarde:
/*---------------------------*/
/* Ejemplo en C# nº 37: */
/* ejemplo37.cs */
/* */
/* Array de dos dimensiones */
/* al estilo Pascal */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Este tipo de tablas de varias dimensiones son las que se usan también para guardar matrices,
cuando se trata de resolver problemas matemáticos más complejos que los que hemos visto
hasta ahora.
La otra forma de tener arrays multidimensionales son los "arrays de arrays", que, como ya
hemos comentado, y como veremos en este ejemplo, pueden tener elementos de distinto
tamaño. En ese caso nos puede interesar saber su longitud, para lo que podemos usar
"a.Length":
/*---------------------------*/
Revisión 0.95 – Página 69
Introducción a la programación con C#, por Nacho Cabanes
/* Ejemplo en C# nº 38: */
/* ejemplo38.cs */
/* */
/* Array de dos dimensiones */
/* al estilo C... o casi */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
0123456789
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
2 3 4 5 6 7 8 9 10 11 12 13
Ejercicios propuestos:
• Un programa que al usuario dos bloques de 10 números enteros (usando un array de
dos dimensiones). Después deberá mostrar el mayor dato que se ha introducido en
cada uno de ellos.
Revisión 0.95 – Página 70
Introducción a la programación con C#, por Nacho Cabanes
En C# (al contrario que en C), primero deberemos declarar cual va a ser la estructura de
nuestro registro, lo que no se puede hacer dentro de "Main". Más adelante, ya dentro de
"Main", podremos declarar variables de ese nuevo tipo.
Los datos que forman un "struct" pueden ser públicos o privados. Por ahora, a nosotros nos
interesará que sean accesibles desde el resto de nuestro programa, por lo que les añadiremos
delante la palabra "public" para indicar que queremos que sean públicos.
Ya desde el cuerpo del programa, para acceder a cada uno de los datos que forman el registro,
tanto si queremos leer su valor como si queremos cambiarlo, se debe indicar el nombre de la
variable y el del dato (o campo) separados por un punto:
/*---------------------------*/
/* Ejemplo en C# nº 39: */
/* ejemplo39.cs */
/* */
/* Registros (struct) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
struct tipoPersona
{
public string nombre;
public char inicial;
public int edad;
public float nota;
}
persona.nombre = "Juan";
persona.inicial = 'J';
persona.edad = 20;
persona.nota = 7.5f;
Revisión 0.95 – Página 71
Introducción a la programación con C#, por Nacho Cabanes
Nota: La notación 7.5f se usa para detallar que se trata de un número real de simple precisión
(un "float"), porque de lo contrario, 7.5 se consideraría un número de doble precisión, y al
tratar de compilar obtendríamos un mensaje de error, diciendo que no se puede convertir de
"double" a "float" sin pérdida de precisión. Al añadir la "f" al final, estamos diciendo "quiero que
éste número se tome como un float; sé que habrá una pérdida de precisión pero es aceptable
para mí”.
Ejercicios propuestos:
• Un "struct" que almacene datos de una canción en formato MP3: Artista, Título,
Duración (en segundos), Tamaño del fichero (en KB). Un programa debe pedir los
datos de una canción al usuario, almacenarlos en dicho "struct" y después mostrarlos
en pantalla.
/*---------------------------*/
/* Ejemplo en C# nº 40: */
/* ejemplo40.cs */
/* */
/* Array de struct */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
struct tipoPersona
{
public string nombre;
public char inicial;
public int edad;
public float nota;
}
persona[0].nombre = "Juan";
persona[0].inicial = 'J';
persona[0].edad = 20;
persona[0].nota = 7.5f;
Revisión 0.95 – Página 72
Introducción a la programación con C#, por Nacho Cabanes
persona[1].nombre = "Pedro";
Console.WriteLine("La edad de {0} es {1}",
persona[1].nombre, persona[1].edad);
}
}
La edad de Juan es 20
La edad de Pedro es 0
Porque cuando reservamos espacio para los elementos de un "array" usando "new", sus valores
se dejan "vacíos" (0 para los números, cadenas vacías para las cadenas de texto).
Ejercicios propuestos:
• Ampliar el programa del apartado 4.3.1, para que almacene datos de hasta 100
canciones. Deberá tener un menú que permita las opciones: añadir una nueva canción,
mostrar el título de todas las canciones, buscar la canción que contenga un cierto texto
(en el artista o en el título).
• Un programa que permita guardar datos de "imágenes" (ficheros de ordenador que
contengan fotografías o cualquier otro tipo de información gráfica). De cada imagen se
debe guardar: nombre (texto), ancho en píxeles (por ejemplo 2000), alto en píxeles
(por ejemplo, 3000), tamaño en Kb (por ejemplo 145,6). El programa debe ser capaz
de almacenar hasta 700 imágenes (deberá avisar cuando su capacidad esté llena).
Debe permitir las opciones: añadir una ficha nueva, ver todas las fichas (número y
nombre de cada imagen), buscar la ficha que tenga un cierto nombre.
Podemos encontrarnos con un registro que tenga varios datos, y que a su vez ocurra que uno
de esos datos esté formado por varios datos más sencillos. Por ejemplo, una fecha de
nacimiento podría estar formada por día, mes y año. Para hacerlo desde C#, incluiríamos un
"struct" dentro de otro, así:
/*---------------------------*/
/* Ejemplo en C# nº 41: */
/* ejemplo41.cs */
/* */
/* Registros anidados */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
Revisión 0.95 – Página 73
Introducción a la programación con C#, por Nacho Cabanes
using System;
struct fechaNacimiento
{
public int dia;
public int mes;
public int anyo;
}
struct tipoPersona
{
public string nombre;
public char inicial;
public fechaNacimiento diaDeNacimiento;
public float nota;
}
persona.nombre = "Juan";
persona.inicial = 'J';
persona.diaDeNacimiento.dia = 15;
persona.diaDeNacimiento.mes = 9;
persona.nota = 7.5f;
Console.WriteLine("{0} nació en el mes {1}",
persona.nombre, persona.diaDeNacimiento.mes);
}
}
Ejercicios propuestos:
• Ampliar el programa del primer apartado de 4.3.2, para que el campo "duración" se
almacene como minutos y segundos, usando un "struct" anidado que contenga a su vez
estos dos campos.
Así, un ejemplo que nos pidiese nuestro nombre y nos saludase usando todas estas
posibilidades podría ser:
/*---------------------------*/
Revisión 0.95 – Página 74
Introducción a la programación con C#, por Nacho Cabanes
/* Ejemplo en C# nº 42: */
/* ejemplo42.cs */
/* */
/* Cadenas de texto (1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
if (nombre == "Alberto")
Console.WriteLine("Dices que eres Alberto?");
else
Console.WriteLine("Así que no eres Alberto?");
Eso sí, las cadenas en C# no se pueden modificar letra a letra: no podemos hacer texto[0]=’a’.
Para eso habrá que usar una construcción auxiliar, que veremos más adelante.
Ejercicios propuestos:
• Un programa que te pida tu nombre y lo muestre en pantalla separando cada letra de
la siguiente con un espacio. Por ejemplo, si tu nombre es "Juan", debería aparecer en
pantalla "J u a n".
• Un programa capaz de sumar dos números enteros muy grandes (por ejemplo, de 30
cifras), que se deberán pedir como cadena de texto y analizar letra a letra.
• Un programa capaz de multiplicar dos números enteros muy grandes (por ejemplo, de
30 cifras), que se deberán pedir como cadena de texto y analizar letra a letra.
saludo = frase.Substring(0,4);
Podemos omitir el segundo número, y entonces se extraerá desde la posición indicada hasta el
final de la cadena.
Podemos añadir un segundo parámetro opcional, que es la posición a partir de la que queremos
buscar:
La búsqueda termina al final de la cadena, salvo que indiquemos que termine antes con un
tercer parámetro opcional:
De forma similar, LastIndexOf ("última posición de") indica la última aparición (es decir, busca
de derecha a izquierda).
• Insert(int posición, string subcadena): Insertar una subcadena en una cierta posición
de la cadena inicial: nombreFormal = nombre.Insert(0,"Don");
• Remove(int posición, int cantidad): Elimina una cantidad de caracteres en cierta
posición: apellidos = nombreCompleto.Remove(0,6);
• Replace(string textoASustituir, string cadenaSustituta): Sustituye una cadena (todas las
veces que aparezca) por otra: nombreCorregido = nombre.Replace("Pepe", "Jose");
/*---------------------------*/
/* Ejemplo en C# nº 43: */
/* ejemplo43.cs */
/* */
/* Cadenas de texto (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Y su resultado sería
Otra posibilidad interesante, aunque un poco más avanzada, es la de descomponer una cadena
en trozos, que estén separados por una serie de delimitadores (por ejemplo, espacios o comas).
Para ello se puede usar Split, que crea un array a partir de los fragmentos de la cadena, así:
/*---------------------------*/
/* Ejemplo en C# nº 43b: */
/* ejemplo43b.cs */
/* */
/* Cadenas de texto (2b) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Fragmento 0= uno
Fragmento 1= dos
Fragmento 2= tres
Fragmento 3= cuatro
if (frase.CompareTo("hola") > 0)
Console.WriteLine("Es mayor que hola");
/*---------------------------*/
/* Ejemplo en C# nº 43c: */
/* ejemplo43c.cs */
/* */
/* Cadenas de texto (2c) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
string frase;
}
}
Si tecleamos una palabra como "gol", que comienza por G, que alfabéticamente está antes de
la H de "hola", se nos dirá que esa palabra es menor:
Si escribimos "hOLa", que coindice con "hola" salvo por las mayúsculas, una comparación
normal nos dirá que es mayor (las mayúsculas se consideran "mayores" que las minúsculas), y
una comparación sin considerar mayúsculas o minúsculas nos dirá que coinciden:
ciertas ocasiones con los Arrays), y se pueden convertir a una cadena "convencional" usando
"ToString":
/*---------------------------*/
/* Ejemplo en C# nº 44: */
/* ejemplo44.cs */
/* */
/* Cadenas modificables */
/* con "StringBuilder" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Text; // Usaremos un System.Text.StringBuilder
string cadenaNormal;
cadenaNormal = cadenaModificable.ToString();
Console.WriteLine("Cadena normal a partir de ella: {0}",
cadenaNormal);
}
}
Ejercicios propuestos:
• Un programa que pida tu nombre, tu día de nacimiento y tu mes de nacimiento y lo
junte todo en una cadena, separando el nombre de la fecha por una coma y el día del
mes por una barra inclinada, así: "Juan, nacido el 31/12".
• Crear un juego del ahorcado, en el que un primer usuario introduzca la palabra a
adivinar, se muestre esta programa oculta con guiones (-----) y el programa acepte las
letras que introduzca el segundo usuario, cambiando los guiones por letras correctas
cada vez que acierte (por ejemplo, a---a-t-). La partida terminará cuando se acierte la
palabra por completo o el usuario agote sus 8 intentos.
/*---------------------------*/
/* Ejemplo en C# nº 45: */
/* ejemplo45.cs */
/* */
/* Ejemplo de "foreach" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Vamos a hacer un ejemplo completo que use tablas ("arrays"), registros ("struct") y que
además manipule cadenas.
La idea va a ser la siguiente: Crearemos un programa que pueda almacenar datos de hasta
1000 ficheros (archivos de ordenador). Para cada fichero, debe guardar los siguientes datos:
Nombre del fichero, Tamaño (en KB, un número de 0 a 8.000.000.000). El programa mostrará
un menú que permita al usuario las siguientes operaciones:
No debería resultar difícil. Vamos a ver directamente una de las formas en que se podría
plantear y luego comentaremos alguna de las mejoras que se podría (incluso se debería) hacer.
Una opción que podemos a tomar para resolver este problema es la de contar el número de
fichas que tenemos almacenadas, y así podremos añadir de una en una. Si tenemos 0 fichas,
deberemos almacenar la siguiente (la primera) en la posición 0; si tenemos dos fichas, serán la
0 y la 1, luego añadiremos en la posición 2; en general, si tenemos "n" fichas, añadiremos cada
nueva ficha en la posición "n". Por otra parte, para revisar todas las fichas, recorreremos desde
la posición 0 hasta la n-1, haciendo algo como
El resto del programa no es difícil: sabemos leer y comparar textos y números, comprobar
varias opciones con "switch", etc. Aun así, haremos una última consideración: hemos limitado el
número de fichas a 1000, así que, si nos piden añadir, deberíamos asegurarnos antes de que
todavía tenemos hueco disponible.
/*---------------------------*/
/* Ejemplo en C# nº 46: */
/* ejemplo46.cs */
/* */
/* Tabla con muchos struct */
/* y menu para manejarla */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
struct tipoFicha {
public string nombreFich; /* Nombre del fichero */
public long tamanyo; /* El tamaño en bytes */
};
do {
/* Menu principal */
Console.WriteLine();
Console.WriteLine("Escoja una opción:");
Console.WriteLine("1.- Añadir datos de un nuevo fichero");
Console.WriteLine("2.- Mostrar los nombres de todos los ficheros");
Console.WriteLine("3.- Mostrar ficheros que sean de mas de un cierto tamaño");
Console.WriteLine("4.- Ver datos de un fichero");
Console.WriteLine("5.- Salir");
}
}
Revisión 0.95 – Página 84
Introducción a la programación con C#, por Nacho Cabanes
Funciona, y hace todo lo que tiene que hacer, pero es mejorable. Por supuesto, en un caso real
es habitual que cada ficha tenga que guardar más información que sólo esos dos apartados de
ejemplo que hemos previsto esta vez. Si nos muestra todos los datos en pantalla y se trata de
muchos datos, puede ocurrir que aparezcan en pantalla tan rápido que no nos dé tiempo a
leerlos, así que sería deseable que parase cuando se llenase la pantalla de información (por
ejemplo, una pausa tras mostrar cada 25 datos). Por supuesto, se nos pueden ocurrir muchas
más preguntas que hacerle sobre nuestros datos. Y además, cuando salgamos del programa se
borrarán todos los datos que habíamos tecleado, pero eso es lo único "casi inevitable", porque
aún no sabemos manejar ficheros.
Ejercicios propuestos:
• Un programa que pida el nombre, el apellido y la edad de una persona, los almacene
en un "struct" y luego muestre los tres datos en una misma línea, separados por
comas.
• Un programa que pida datos de 8 personas: nombre, dia de nacimiento, mes de
nacimiento, y año de nacimiento (que se deben almacenar en una tabla de structs).
Después deberá repetir lo siguiente: preguntar un número de mes y mostrar en
pantalla los datos de las personas que cumplan los años durante ese mes. Terminará
de repetirse cuando se teclee 0 como número de mes.
• Un programa que sea capaz de almacenar los datos de 50 personas: nombre, dirección,
teléfono, edad (usando una tabla de structs). Deberá ir pidiendo los datos uno por uno,
hasta que un nombre se introduzca vacío (se pulse Intro sin teclear nada). Entonces
deberá aparecer un menú que permita:
o Mostrar la lista de todos los nombres.
o Mostrar las personas de una cierta edad.
o Mostrar las personas cuya inicial sea la que el usuario indique.
o Salir del programa
(lógicamente, este menú debe repetirse hasta que se escoja la opción de "salir").
• Mejorar la base de datos de ficheros (ejemplo 46) para que no permita introducir
tamaños incorrectos (números negativos) ni nombres de fichero vacíos.
• Ampliar la base de datos de ficheros (ejemplo 46) para que incluya una opción de
búsqueda parcial, en la que el usuario indique parte del nombre y se muestre todos los
ficheros que contienen ese fragmento (usando "IndexOf"). Esta búsqueda no debe
distinguir mayúsculas y minúsculas (con la ayuda de ToUpper o ToLower).
• Ampliar el ejercicio anterior (el que permite búsqueda parcial) para que la búsqueda
sea incremental: el usuario irá indicando letra a letra el texto que quiere buscar, y se
mostrará todos los datos que lo contienen (por ejemplo, primero los que contienen "j",
luego "ju", después "jua" y finalmente "juan").
• Ampliar la base de datos de ficheros (ejemplo 46) para que se pueda borrar un cierto
dato (habrá que "mover hacia atrás" todos los datos que había después de ese, y
disminuir el contador de la cantidad de datos que tenemos).
• Mejorar la base de datos de ficheros (ejemplo 46) para que se pueda modificar un
cierto dato a partir de su número (por ejemplo, el dato número 3). En esa modificación,
se deberá permitir al usuario pulsar Intro sin teclear nada, para indicar que no desea
modificar un cierto dato, en vez de reemplazarlo por una cadena vacía.
• Ampliar la base de datos de ficheros (ejemplo 46) para que se permita ordenar los
datos por nombre. Para ello, deberás buscar información sobre algún método de
ordenación sencillo, como el "método de burbuja" (en el siguiente apartado tienes
algunos), y aplicarlo a este caso concreto.
Existen ligeras mejoras (por ejemplo, cambiar uno de los "for" por un "while", para no repasar
todos los datos si ya estaban parcialmente ordenados), así como métodos claramente más
efectivos, pero más difíciles de programar, alguno de los cuales veremos más adelante.
Veremos tres de estos métodos simples de ordenación, primero mirando la apariencia que tiene
el algoritmo, y luego juntando los tres en un ejemplo que los pruebe:
Método de burbuja
(Intercambiar cada pareja consecutiva que no esté ordenada)
(Nota: algunos autores hacen el bucle exterior creciente y otros decreciente, así:)
Selección directa
(En cada pasada busca el menor, y lo intercambia al final de la pasada)
Nota: el símbolo "<>" se suele usar en pseudocódigo para indicar que un dato es distinto de
otro, de modo que equivale al "!=" de C#. La penúltima línea en C# saldría a ser algo como "if
(menor !=i)"
Inserción directa
(Comparar cada elemento con los anteriores -que ya están ordenados- y desplazarlo hasta su
posición correcta).
(Es mejorable, no intercambiando el dato que se mueve con cada elemento, sino sólo al final de
cada pasada, pero no entraremos en más detalles).
/*---------------------------*/
/* Ejemplo en C# */
/* ordenar.cs */
/* */
/* Ordenaciones simples */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// BURBUJA
// (Intercambiar cada pareja consecutiva que no esté ordenada)
// Para i=1 hasta n-1
// Para j=i+1 hasta n
// Si A[i] > A[j]
// Intercambiar ( A[i], A[j])
Console.WriteLine("Ordenando mediante burbuja... ");
for(i=0; i < n-1; i++)
{
foreach (int dato in datos) // Muestro datos
Console.Write("{0} ",dato);
Revisión 0.95 – Página 87
Introducción a la programación con C#, por Nacho Cabanes
Console.WriteLine();
// SELECCIÓN DIRECTA:
// (En cada pasada busca el menor, y lo intercambia al final de la pasada)
// Para i=1 hasta n-1
// menor = i
// Para j=i+1 hasta n
// Si A[j] < A[menor]
// menor = j
// Si menor <> i
// Intercambiar ( A[i], A[menor])
Console.WriteLine("\nOrdenando mediante selección directa... ");
int[] datos2 = {5, 3, 14, 20, 8, 9, 1};
for(i=0; i < n-1; i++)
{
foreach (int dato in datos2) // Muestro datos
Console.Write("{0} ",dato);
Console.WriteLine();
int menor = i;
for(j=i+1; j < n; j++)
if (datos2[j] < datos2[menor])
menor = j;
if (i != menor)
{
datoTemporal = datos2[i];
datos2[i] = datos2[menor];
datos2[menor] = datoTemporal;
}
}
Console.Write("Ordenado:");
// INSERCION DIRECTA:
// (Comparar cada elemento con los anteriores -que ya están ordenados-
// y desplazarlo hasta su posición correcta).
j = i-1;
while ((j>=0) && (datos3[j] > datos3[j+1]))
{
datoTemporal = datos3[j];
datos3[j] = datos3[j+1];
datos3[j+1] = datoTemporal;
j--;
}
}
Console.Write("Ordenado:");
}
}
Y su resultado sería:
3 5 14 20 8 9 1
3 5 14 20 8 9 1
3 5 14 20 8 9 1
3 5 8 14 20 9 1
3 5 8 9 14 20 1
Ordenado:1 3 5 8 9 14 20
Hasta ahora hemos estado pensando los pasos que deberíamos dar para resolver un cierto
problema, y hemos creado programas a partir de cada uno de esos pasos. Esto es razonable
cuando los problemas son sencillos, pero puede no ser la mejor forma de actuar cuando se
trata de algo más complicado.
A partir de ahora vamos a empezar a intentar descomponer los problemas en trozos más
pequeños, que sean más fáciles de resolver. Esto nos puede suponer varias ventajas:
• Cada "trozo de programa" independiente será más fácil de programar, al realizar una
función breve y concreta.
• El "programa principal" será más fácil de leer, porque no necesitará contener todos los
detalles de cómo se hace cada cosa.
• Podremos repartir el trabajo, para que cada persona se encargue de realizar un "trozo
de programa", y finalmente se integrará el trabajo individual de cada persona.
En C#, al igual que en C y los demás lenguajes derivados de él, todos los "trozos de programa"
son funciones, incluyendo el propio cuerpo de programa, Main. De hecho, la forma básica de
definir una función será indicando su nombre seguido de unos paréntesis vacíos, como
hacíamos con "Main", y precediéndolo por ciertas palabras reservadas, como "public static
void", cuyo significado iremos viendo muy pronto. Después, entre llaves indicaremos todos los
pasos que queremos que dé ese "trozo de programa".
Por ejemplo, podríamos crear una función llamada "saludar", que escribiera varios mensajes en
la pantalla:
Ahora desde dentro del cuerpo de nuestro programa, podríamos "llamar" a esa función:
{
saludar();
…
}
Así conseguimos que nuestro programa principal sea más fácil de leer.
Un detalle importante: tanto la función habitual "Main" como la nueva función "Saludar" serían
parte de nuestra "class", es decir, el fuente completo sería así:
/*---------------------------*/
/* Ejemplo en C# nº 47: */
/* ejemplo47.cs */
/* */
/* Funcion "saludar" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Como ejemplo más detallado, la parte principal de una agenda podría ser simplemente:
leerDatosDeFichero();
do {
mostrarMenu();
pedirOpcion();
switch( opcion ) {
case 1: buscarDatos(); break;
case 2: modificarDatos(); break;
case 3: anadirDatos(); break;
…
escribeNumeroReal(2.3f);
(recordemos que el sufijo "f" es para indicar al compilador que trate ese número como un
"float", porque de lo contrario, al ver que tiene cifras decimales, lo tomaría como "double", que
permite mayor precisión... pero a cambio nosotros tendríamos un mensaje de error en nuestro
programa, diciendo que estamos dando un dato "double" a una función que espera un "float").
/*---------------------------*/
/* Ejemplo en C# nº 48: */
/* ejemplo48.cs */
/* */
/* Funcion */
/* "escribeNumeroReal" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
x= 5.1f;
Console.WriteLine("El primer numero real es: ");
escribeNumeroReal(x);
Console.WriteLine(" y otro distinto es: ");
escribeNumeroReal(2.3f);
}
Estos datos adicionales que indicamos a la función es lo que llamaremos sus "parámetros".
Como se ve en el ejemplo, tenemos que indicar un nombre para cada parámetro (puede haber
varios) y el tipo de datos que corresponde a ese parámetro. Si hay más de un parámetro,
deberemos indicar el tipo y el nombre para cada uno de ellos, y separarlos entre comas:
Pero eso no es lo que ocurre con las funciones matemáticas que estamos acostumbrados a
manejar: sí devuelven un valor, el resultado de una operación.
De igual modo, para nosotros también será habitual que queramos que nuestra función realice
una serie de cálculos y nos "devuelva" (return, en inglés) el resultado de esos cálculos, para
poderlo usar desde cualquier otra parte de nuestro programa. Por ejemplo, podríamos crear
una función para elevar un número entero al cuadrado así:
resultado = cuadrado( 5 );
/*---------------------------*/
/* Ejemplo en C# nº 49: */
/* ejemplo49.cs */
/* */
/* Funcion "cuadrado" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
numero= 5;
resultado = cuadrado(numero);
Console.WriteLine("El cuadrado del numero {0} es {1}",
numero, resultado);
Console.WriteLine(" y el de 3 es {0}", cuadrado(3));
}
Podemos hacer una función que nos diga cual es el mayor de dos números reales así:
Ejercicios propuestos:
• Crear una función que borre la pantalla dibujando 25 líneas en blanco. No debe
devolver ningún valor.
• Crear una función que calcule el cubo de un número real (float). El resultado deberá ser
otro número real. Probar esta función para calcular el cubo de 3.2 y el de 5.
• Crear una función que calcule cual es el menor de dos números enteros. El resultado
será otro número entero.
• Crear una función llamada "signo", que reciba un número real, y devuelva un número
entero con el valor: -1 si el número es negativo, 1 si es positivo o 0 si es cero.
• Crear una función que devuelva la primera letra de una cadena de texto. Probar esta
función para calcular la primera letra de la frase "Hola".
• Crear una función que devuelva la última letra de una cadena de texto. Probar esta
función para calcular la última letra de la frase "Hola".
• Crear una función que reciba un número y muestre en pantalla el perímetro y la
superficie de un cuadrado que tenga como lado el número que se ha indicado como
parámetro.
Las variables se pueden declarar dentro de un bloque (una función), y entonces sólo ese bloque
las conocerá, no se podrán usar desde ningún otro bloque del programa. Es lo que llamaremos
"variables locales".
Por el contrario, si declaramos una variable al comienzo del programa, fuera de todos los
"bloques" de programa, será una "variable global", a la que se podrá acceder desde cualquier
parte.
Vamos a verlo con un ejemplo. Crearemos una función que calcule la potencia de un número
entero (un número elevado a otro), y el cuerpo del programa que la use.
3 elevado a 5 = 3 · 3 · 3 · 3 · 3
(multiplicamos 5 veces el 3 por sí mismo). En general, como nos pueden pedir cosas como "6
elevado a 100" (o en general números que pueden ser grandes), usaremos la orden "for" para
multiplicar tantas veces como haga falta:
/*---------------------------*/
/* Ejemplo en C# nº 50: */
/* ejemplo50.cs */
/* */
/* Ejemplo de función con */
/* variables locales */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
En este caso, las variables "temporal" e "i" son locales a la función "potencia": para "Main" no
existen. Si en "Main" intentáramos hacer i=5; obtendríamos un mensaje de error.
De igual modo, "num1" y "num2" son locales para "main": desde la función "potencia" no
podemos acceder a su valor (ni para leerlo ni para modificarlo), sólo desde "main".
En general, deberemos intentar que la mayor cantidad de variables posible sean locales (lo
ideal sería que todas lo fueran). Así hacemos que cada parte del programa trabaje con sus
propios datos, y ayudamos a evitar que un error en un trozo de programa pueda afectar al
resto. La forma correcta de pasar datos entre distintos trozos de programa es usando los
parámetros de cada función, como en el anterior ejemplo.
Ejercicios propuestos:
• Crear una función "pedirEntero", que reciba como parámetros el texto que se debe
mostrar en pantalla, el valor mínimo aceptable y el valor máximo aceptable. Deberá
pedir al usuario que introduzca el valor tantas veces como sea necesario, volvérselo a
pedir en caso de error, y devolver un valor correcto. Probarlo con un programa que
pida al usuario un año entre 1800 y 2100.
• Crear una función "escribirTablaMultiplicar", que reciba como parámetro un número
entero, y escriba la tabla de multiplicar de ese número (por ejemplo, para el 3 deberá
llegar desde 3x0=0 hasta 3x10=30).
• Crear una función "esPrimo", que reciba un número y devuelva el valor booleano "true"
si es un número primo o "false" en caso contrario.
• Crear una función que reciba una cadena y una letra, y devuelva la cantidad de veces
que dicha letra aparece en la cadena. Por ejemplo, si la cadena es "Barcelona" y la letra
es 'a', debería devolver 2 (aparece 2 veces).
• Crear una función que reciba un numero cualquiera y que devuelva como resultado la
suma de sus dígitos. Por ejemplo, si el número fuera 123 la suma sería 6.
• Crear una función que reciba una letra y un número, y escriba un "triángulo" formado
por esa letra, que tenga como anchura inicial la que se ha indicado. Por ejemplo, si la
letra es * y la anchura es 4, debería escribir
****
***
**
*
/*---------------------------*/
/* Ejemplo en C# nº 51: */
/* ejemplo51.cs */
/* */
/* Dos variables locales */
/* con el mismo nombre */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
n vale 5
Ahora n vale 5
¿Por qué? Sencillo: tenemos una variable local dentro de "duplica" y otra dentro de "main". El
hecho de que las dos tengan el mismo nombre no afecta al funcionamiento del programa,
siguen siendo distintas.
Si la variable es "global", declarada fuera de estas funciones, sí será accesible por todas ellas:
/*---------------------------*/
/* Ejemplo en C# nº 52: */
/* ejemplo52.cs */
/* */
/* Una variable global */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
static int n = 7;
Dentro de poco, hablaremos de por qué cada uno de los bloques de nuestro programa, e
incluso las "variables globales", tienen delante la palabra "static". Será cuando tratemos la
"Programación Orientada a Objetos", en el próximo tema.
/*---------------------------*/
/* Ejemplo en C# nº 53: */
/* ejemplo53.cs */
/* */
/* Modificar una variable */
/* recibida como parámetro */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
n vale 5
El valor recibido vale 5
y ahora vale 10
Ahora n vale 5
Vemos que al salir de la función, los cambios que hagamos a esa variable que se ha recibido
como parámetro no se conservan.
Esto se debe a que, si no indicamos otra cosa, los parámetros "se pasan por valor", es decir,
la función no recibe los datos originales, sino una copia de ellos. Si modificamos algo, estamos
cambiando una copia de los datos originales, no dichos datos.
Si queremos que los cambios se conserven, basta con hacer un pequeño cambio: indicar que la
variable se va a pasar "por referencia", lo que se indica usando la palabra "ref", tanto en la
declaración de la función como en la llamada, así:
/*---------------------------*/
/* Ejemplo en C# nº 54: */
/* ejemplo54.cs */
/* */
/* Modificar una variable */
/* recibida como parámetro */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
n vale 5
El valor recibido vale 5
y ahora vale 10
Ahora n vale 10
El hecho de poder modificar valores que se reciban como parámetros abre una posibilidad que
no se podría conseguir de otra forma: con "return" sólo se puede devolver un valor de una
función, pero con parámetros pasados por referencia podríamos devolver más de un dato. Por
ejemplo, podríamos crear una función que intercambiara los valores de dos variables:
La posibilidad de pasar parámetros por valor y por referencia existe en la mayoría de lenguajes
de programación. En el caso de C# existe alguna posibilidad adicional que no existe en otros
lenguajes, como los "parámetros de salida". Las veremos más adelante.
Ejercicios propuestos:
• Crear una función "intercambia", que intercambie el valor de los dos números enteros
que se le indiquen como parámetro.
• Crear una función "iniciales", que reciba una cadena como "Nacho Cabanes" y devuelva
las letras N y C (primera letra, y letra situada tras el primer espacio), usando
parámetros por referencia.
/*---------------------------*/
/* Ejemplo en C# nº 55: */
/* ejemplo55.cs */
/* */
/* Función tras Main */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Revisión 0.95 – Página 101
Introducción a la programación con C#, por Nacho Cabanes
Podemos hacer que sea realmente un poco más aleatorio si en la primera orden le indicamos
que tome como semilla el instante actual:
De hecho, una forma muy simple de obtener un número "casi al azar" entre 0 y 999 es tomar
las milésimas de segundo de la hora actual:
Vamos a ver un ejemplo, que muestre en pantalla un número al azar entre 1 y 10:
/*---------------------------*/
/* Ejemplo en C# nº 56: */
/* ejemplo56.cs */
/* */
/* Números al azar */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Ejercicios propuestos:
• Crear un programa que genere un número al azar entre 1 y 100. El usuario tendrá 6
oportunidades para acertarlo.
• Mejorar el programa del ahorcado propuesto en el apartado 4.4.8, para que la palabra
a adivinar no sea tecleado por un segundo usuario, sino que se escoja al azar de un
"array" de palabras prefijadas (por ejemplo, nombres de ciudades).
• Sin(x): Seno
• Sinh(x): Seno hiperbólico
• Sqrt(x): Raíz cuadrada
• Tan(x): Tangente
• Tanh(x): Tangente hiperbólica
Ejercicios propuestos:
• Crear un programa que halle cualquier raíz de un número. El usuario deberá indicar el
número (por ejemplo, 2) y el índice de la raíz (por ejemplo, 3 para la raíz cúbica). Pista:
hallar la raíz cúbica de 2 es lo mismo que elevar 2 a 1/3.
• Crear un programa que resuelva ecuaciones de segundo grado, del tipo ax2 + bx + c =
0 El usuario deberá introducir los valores de a, b y c. Se deberá crear una función
"raicesSegundoGrado", que recibirá como parámetros los coeficientes a, b y c, así como
las soluciones x1 y x2 (por referencia). Deberá devolver los valores de las dos
soluciones x1 y x2. Si alguna solución no existe, se devolverá como valor 100.000 para
esa solución. Pista: la solución se calcula con
x = -b ± raíz (b2 – 4·a·c) / 2·a
5.10. Recursividad
Una función recursiva es aquella que se define a partir de ella misma. Dentro de las
matemáticas tenemos varios ejemplos. Uno clásico es el "factorial de un número":
Entonces podemos escribir el factorial de un número a partir del factorial del siguiente número:
n! = n · (n-1)!
Esta es la definición recursiva del factorial, ni más ni menos. Esto, programando, se haría:
/*---------------------------*/
/* Ejemplo en C# nº 57: */
/* ejemplo57.cs */
/* */
/* Funciones recursivas: */
/* factorial */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
¿Qué utilidad tiene esto? Pues más de la que parece: muchos problemas complicados se
pueden expresar a partir de otro más sencillo. En muchos de esos casos, ese problema se
podrá expresar de forma recursiva. Más adelante veremos algún otro ejemplo.
Ejercicios propuestos:
• Crear una función que calcule el valor de elevar un número entero a otro número
entero (por ejemplo, 5 elevado a 3 = 53 = 5 ·5 ·5 = 125). Esta función se debe crear
de forma recursiva.
• Como alternativa, crear una función que calcule el valor de elevar un número entero a
otro número entero de forma NO recursiva (lo que llamaremos "de forma iterativa"),
usando la orden "for".
• Crear un programa que emplee recursividad para calcular un número de la serie
Fibonacci (en la que los dos primeros elementos valen 1, y para los restantes, cada
elemento es la suma de los dos anteriores).
• Crear un programa que emplee recursividad para calcular la suma de los elementos de
un vector.
• Crear un programa que emplee recursividad para calcular el mayor de los elementos de
un vector.
• Crear un programa que emplee recursividad para dar la vuelta a una cadena de
caracteres (por ejemplo, a partir de "Hola" devolvería "aloH").
• Crear, tanto de forma recursiva como de forma iterativa, una función diga si una
cadena de caracteres es simétrica (un palíndromo). Por ejemplo,
"DABALEARROZALAZORRAELABAD" es un palíndromo.
• Crear un programa que encuentre el máximo común divisor de dos números usando el
algoritmo de Euclides: Dados dos números enteros positivos m y n, tal que m > n, para
encontrar su máximo común divisor, es decir, el mayor entero positivo que divide a
ambos: - Dividir m por n para obtener el resto r (0 ≤ r < n) ; - Si r = 0, el MCD es n.; -
Si no, el máximo común divisor es MCD(n,r).
ls –l *.c
En este caso, la orden sería "ls", y las dos opciones (argumentos o parámetros) que le
indicamos son "-l" y "*.c".
dir *.c
Pues bien, estas opciones que se le pasan al programa se pueden leer desde C#. Se hace
indicando un parámetro especial en Main, un array de strings:
Para conocer esos parámetros lo haríamos de la misma forma que se recorre habitualmente un
array cuyo tamaño no conocemos: con un "for" que termine en la longitud ("Length") del array:
Por otra parte, si queremos que nuestro programa se interrumpa en un cierto punto,
podemos usar la orden "Environment.Exit". Su manejo habitual es algo como
Environment.Exit(1);
Es decir, entre paréntesis indicamos un cierto código, que suele ser (por convenio) un 0 si no
ha habido ningún error, u otro código distinto en caso de que sí exista algún error.
Este valor se podría comprobar desde el sistema operativo. Por ejemplo, en MsDos y Windows
se lee con "IF ERRORLEVEL", así:
Una forma alternativa de que "Main" indique errores al sistema operativo es no declarándolo
como "void", sino como "int", y empleando entonces la orden "return" cuando nos interese:
/*---------------------------*/
/* Ejemplo en C# nº 58: */
/* ejemplo58.cs */
/* */
/* Parámetros y valor de */
/* retorno de "Main" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
{
Console.WriteLine("Parámetros: {0}", args.Length);
if (args.Length == 0)
{
Console.WriteLine("Escriba algún parámetro!");
Environment.Exit(1);
}
return 0;
Ejercicios propuestos:
• Crear un programa llamado "suma", que calcule (y muestre) la suma de dos números
que se le indiquen como parámetro. Por ejemplo, si se teclea "suma 2 3" deberá
responder "5", y si se teclea "suma 2" deberá responder "no hay suficientes datos y
devolver un código de error 1.
• Crear una calculadora básica, llamada "calcula", que deberá sumar, restar, multiplicar o
dividir los dos números que se le indiquen como parámetros. Ejemplos de su uso sería
"calcula 2 + 3" o "calcula 5 * 60".
Pero no todo lo que nos rodea es tan fácil de cuadricular. Supongamos por ejemplo que
tenemos que introducir datos sobre una puerta en nuestro programa. ¿Nos limitamos a
programar los procedimientos AbrirPuerta y CerrarPuerta? Al menos, deberíamos ir a la zona
de declaración de variables, y allí guardaríamos otras datos como su tamaño, color, etc.
No está mal, pero es "antinatural": una puerta es un conjunto: no podemos separar su color de
su tamaño, o de la forma en que debemos abrirla o cerrarla. Sus características son tanto las
físicas (lo que hasta ahora llamábamos variables) como sus comportamientos en distintas
circunstancias (lo que para nosotros eran las funciones). Todo ello va unido, formando un
"objeto".
Por otra parte, si tenemos que explicar a alguien lo que es el portón de un garaje, y ese alguien
no lo ha visto nunca, pero conoce cómo es la puerta de su casa, le podemos decir "se parece a
una puerta de una casa, pero es más grande para que quepan los coches, está hecha de metal
en vez de madera...". Es decir, podemos describir unos objetos a partir de lo que conocemos de
otros.
Finalmente, conviene recordar que "abrir" no se refiere sólo a una puerta. También podemos
hablar de abrir una ventana o un libro, por ejemplo.
Con esto, hemos comentado casi sin saberlo las tres características más importantes de la
Programación Orientada a Objetos (OOP):
• Herencia: Unos objetos pueden heredar métodos y datos de otros. Esto hace más
fácil definir objetos nuevos a partir de otros que ya teníamos anteriormente (como
ocurría con el portón y la puerta) y facilitará la reescritura de los programas, pudiendo
aprovechar buena parte de los anteriores... si están bien diseñados.
Otro concepto importante es el de "clase": Una clase es un conjunto de objetos que tienen
características comunes. Por ejemplo, tanto mi puerta como la de mi vecino son puertas, es
decir, ambas son objetos que pertenecen a la clase "puerta". De igual modo, tanto un Ford
Focus como un Honda Civic o un Toyota Corolla son objetos concretos que pertenecen a la
clase "coche".
Como se puede observar, los objetos de la clase "Puerta" tendrán un ancho, un alto, un color, y
un estado (abierta o no abierta), y además se podrán abrir o cerrar (y además, nos pueden
"mostrar su estado, para comprobar que todo funciona correctamente).
Para declarar estos objetos que pertenecen a la clase "Puerta", usaremos la palabra "new",
igual que hacíamos con los "arrays":
Vamos a completar un programa de prueba que use un objeto de esta clase (una "Puerta"):
/*---------------------------*/
/* Ejemplo en C# nº 59: */
/* ejemplo59.cs */
/* */
/* Primer ejemplo de clases */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.WriteLine("Valores iniciales...");
p.MostrarEstado();
Console.WriteLine("\nVamos a abrir...");
p.Abrir();
p.MostrarEstado();
}
}
Revisión 0.95 – Página 111
Introducción a la programación con C#, por Nacho Cabanes
Este fuente ya no contiene una única clase (class), como todos nuestros ejemplos anteriores,
sino dos clases distintas:
• La clase "Puerta", que son los nuevos objetos con los que vamos a practicar.
• La clase "Ejemplo59", que representa a nuestra aplicación.
Valores iniciales...
Ancho: 0
Alto: 0
Color: 0
Abierta: False
Vamos a abrir...
Ancho: 0
Alto: 0
Color: 0
Abierta: True
Se puede ver que en C#, al contrario que en otros lenguajes, las variables que forman parte de
una clase (los "atributos") tienen un valor inicial predefinido: 0 para los números, una cadena
vacía para las cadenas de texto, o "False" para los datos booleanos.
Vemos también que se accede a los métodos y a los datos precediendo el nombre de cada uno
por el nombre de la variable y por un punto, como hacíamos con los registros (struct). Aun así,
en nuestro caso no podemos hacer directamente "p.abierta = true", por dos motivos:
• El atributo "abierta" no tiene delante la palabra "public"; por lo que no es público, sino
privado, y no será accesible desde otras clases (en nuestro caso, desde Ejemplo59).
• Los puristas de la Programación Orientada a Objetos recomiendan que no se acceda
directamente a los atributos, sino que siempre se modifiquen usando métodos
auxiliares (por ejemplo, nuestro "Abrir"), y que se lea su valor también usando una
función. Esto es lo que se conoce como "ocultación de datos". Supondrá ventajas
como que podremos cambiar los detalles internos de nuestra clase sin que afecte a su
uso.
También puede desconcertar que en "Main" aparezca la palabra "static", mientras que no lo
hace en los métodos de la clase "Puerta". Veremos este detalle un poco más adelante.
Ejercicio propuesto:
• Crear una clase llamada Persona, en el fichero "persona.cs". Esta clase deberá tener un
atributo "nombre", de tipo string. También deberá tener un método "SetNombre", de
tipo void y con un parámetro string, que permita cambiar el valor del nombre.
Finalmente, también tendrá un método "Saludar", que escribirá en pantalla "Hola, soy "
seguido de su nombre. Crear también una clase llamada PruebaPersona. Esta clase
deberá contener sólo la función Main, que creará dos objetos de tipo Persona, les
asignará un nombre y les pedirá que saluden.
En un proyecto grande, es recomendable que cada clase esté en su propio fichero fuente, de
forma que se puedan localizar con rapidez (en los que hemos hecho en el curso hasta ahora, no
era necesario, porque eran muy simples). Precisamente por eso, es interesante (pero no
obligatorio) que cada clase esté en un fichero que tenga el mismo nombre: que la clase Puerta
se encuentre en el fichero "Puerta.cs". Esta es una regla que no seguiremos en algunos de los
ejemplos del texto, por respetar la numeración consecutiva de los ejemplos, pero que sí se
debería seguir en un proyecto de mayor tamaño, formado por varias clases.
Para compilar un programa formado por varios fuentes, basta con indicar los nombres de
todos ellos. Por ejemplo, con Mono sería
En ese caso, el ejecutable obtenido tendía el nombre del primero de los fuentes (fuente1.exe).
Podemos cambiar el nombre del ejecutable con la opción "-out" de Mono:
Vamos a dividir en dos fuentes el último ejemplo y a ver cómo se compilaría. La primera clase
podría ser ésta:
/*---------------------------*/
/* Ejemplo en C# nº 59b: */
/* ejemplo59b.cs */
/* */
/* Dos clases en dos */
/* ficheros (fichero 1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
/*---------------------------*/
/* Ejemplo en C# nº 59c: */
/* ejemplo59c.cs */
/* */
/* Dos clases en dos */
/* ficheros (fichero 2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
Console.WriteLine("Valores iniciales...");
p.MostrarEstado();
Console.WriteLine("\nVamos a abrir...");
p.Abrir();
p.MostrarEstado();
}
Y lo compilaríamos con:
Aun así, para estos proyectos formados por varias clases, lo ideal es usar algún entorno más
avanzado, como SharpDevelop o VisualStudio, que permitan crear todas las clases con
comodidad, saltar de una clase a clase otra rápidamente, que marquen dentro del propio editor
la línea en la que están los errores... por eso, al final de este tema tendrás un apartado con una
introducción al uso de SharpDevelop.
Ejercicio propuesto:
• Modificar el fuente del ejercicio anterior, para dividirlo en dos ficheros: Crear una clase
llamada Persona, en el fichero "persona.cs". Esta clase deberá tener un atributo
"nombre", de tipo string. También deberá tener un método "SetNombre", de tipo void y
con un parámetro string, que permita cambiar el valor del nombre. Finalmente, también
tendrá un método "Saludar", que escribirá en pantalla "Hola, soy " seguido de su
nombre. Crear también una clase llamada PruebaPersona, en el fichero
"pruebaPersona.cs". Esta clase deberá contener sólo la función Main, que creará dos
objetos de tipo Persona, les asignará un nombre y les pedirá que saluden.
bool bloqueada;
Con "public class Porton: Puerta" indicamos que Porton debe "heredar" todo lo que ya
habíamos definido para Puerta. Por eso, no hace falta indicar nuevamente que un Portón tendrá
un cierto ancho, o un color, o que se puede abrir: todo eso lo tiene por ser un "descendiente"
de Puerta.
No tenemos por qué heredar todo; también podemos "redefinir" algo que ya existía. Por
ejemplo, nos puede interesar que "MostrarEstado" ahora nos diga también si la puerta está
bloqueada. Para eso, basta con volverlo a declarar y añadir la palabra "new" para indicar al
compilador de C# que sabemos que ya existe ese método y que sabemos seguro que lo
queremos redefinir:
Aun así, esto todavía no funciona: los atributos de una Puerta, como el "ancho" y el "alto"
estaban declarados como "privados" (es lo que se considera si no decimos los contrario), por lo
que no son accesibles desde ninguna otra clase, ni siquiera desde Porton.
La solución más razonable no es declararlos como "public", porque no queremos que sean
accesibles desde cualquier sitio. Sólo querríamos que esos datos estuvieran disponibles para
todos los tipos de Puerta, incluyendo sus "descendientes", como un Porton. Esto se puede
conseguir usando otro método de acceso: "protected". Todo lo que declaremos como
"protected" será accesible por las clases derivadas de la actual, pero por nadie más:
(Si quisiéramos dejar claro que algún elemento de una clase debe ser totalmente privado,
podemos usar la palabra "private", en vez de "public" o "protected").
Un fuente completo que declarase la clase Puerta, la clase Porton a partir de ella, y que además
contuviese un pequeño "Main" de prueba podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 60: */
/* ejemplo60.cs */
/* */
/* Segundo ejemplo de */
/* clases: herencia */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// -------------------------------
public class Puerta
{
// -------------------------------
public class Porton: Puerta
{
bool bloqueada;
// -------------------------------
public class Ejemplo60
{
Console.WriteLine("Valores iniciales...");
p.MostrarEstado();
Console.WriteLine("\nVamos a bloquear...");
p.Bloquear();
p.MostrarEstado();
Ejercicios propuestos:
• Ampliar las clases del primer ejercicio propuesto, creando un nuevo proyecto con las
siguientes características: La clase Persona no cambia. Se creará una nueva clase
PersonaInglesa, en el fichero "personaInglesa.cs". Esta clase deberá heredar las
características de la clase "Persona", y añadir un método "TomarTe", de tipo void, que
escribirá en pantalla "Estoy tomando té". Crear también una clase llamada
PruebaPersona2, en el fichero "pruebaPersona2.cs". Esta clase deberá contener sólo la
función Main, que creará dos objetos de tipo Persona y uno de tipo PersonaInglesa, les
asignará un nombre, les pedirá que saluden y pedirá a la persona inglesa que tome té.
• Ampliar las clases del segundo ejercicio propuesto, creando un nuevo proyecto con las
siguientes características: La clase Persona no cambia. La clase PersonaInglesa se
modificará para que redefina el método "Saludar", para que escriba en pantalla "Hi, I
am " seguido de su nombre. Se creará una nueva clase PersonaItaliana, en el fichero
"personaItaliana.cs". Esta clase deberá heredar las características de la clase
"Persona", pero redefinir el método "Saludar", para que escriba en pantalla "Ciao".
Crear también una clase llamada PruebaPersona3, en el fichero " pruebaPersona3.cs".
Esta clase deberá contener sólo la función Main, que creará un objeto de tipo Persona,
dos de tipo PersonaInglesa, uno de tipo PersonaItaliana, les asignará un nombre, les
pedirá que saluden y pedirá a la persona inglesa que tome té.
La metodología más extendida actualmente para diseñar estos objetos y sus interacciones
(además de otras muchas cosas) se conoce como UML (Unified Modelling Language, lenguaje
de modelado unificado). El estándar UML propone distintos tipos de diagramas para representar
los posibles "casos de uso" de una aplicación, la secuencia de acciones que se debe seguir, las
clases que la van a integrar (es lo que a nosotros nos interesa en este momento), etc.
Vamos a ver la apariencia que tendría un "diagrama de clases". En concreto, vamos a ver un
ejemplo usando ArgoUML, que es una herramienta gratuita de modelado UML, que está
creada en Java, por lo que se puede utilizar desde multitud de sistemas operativos.
Ampliando el ejemplo anterior, vamos a suponer que queremos hacer un sistema "domótico",
para automatizar ciertas funciones en una casa: apertura y cierre de ventanas y puertas,
encendido de calefacción, etc.
Este diagrama es una forma más simple de ver las clases existentes y las relaciones entre ellas.
Si generamos las clases a partir del diagrama, tendremos parte del trabajo hecho: ya "sólo" nos
quedará rellenar los detalles de métodos como "Abrir", pero el esqueleto de todas las clases ya
estará "escrito" para nosotros.
La palabra "static" delante de un atributo (una variable) de una clase, indica que es una
"variable de clase", es decir, que su valor es el mismo para todos los objetos de la clase. Por
ejemplo, si hablamos de coches convencionales, podríamos suponer que el atributo
"numeroDeRuedas" va a valer 4 para cualquier objeto que pertenezca a esa clase (cualquier
coches). Por eso, se podría declarar como "static".
De igual modo, si un método (una función) está precedido por la palabra "static", indica que es
un "método de clase", es decir, un método que se podría usar sin necesidad de declarar ningún
objeto de la clase. Por ejemplo, si queremos que se pueda usar la función "BorrarPantalla" de
una clase "Hardware" sin necesidad de crear primero un objeto perteneciente a esa clase, lo
podríamos conseguir así:
que desde dentro de "Main" (incluso perteneciente a otra clase) se usaría con el nombre de la
clase delante:
public ComienzoPartida() {
Hardware.BorrarPantalla ();
...
Desde una función "static" no se puede llamar a otras funciones que no lo sean. Por eso, como
nuestro "Main" debe ser static, deberemos siempre elegir entre:
• Que todas las demás funciones de nuestro fuente también estén declaradas como
"static", por lo que podrán ser utilizadas desde "Main".
• Declarar un objeto de la clase correspondiente, y entonces sí podremos acceder a sus
métodos desde "Main":
public LanzarJuego () {
Revisión 0.95 – Página 120
Introducción a la programación con C#, por Nacho Cabanes
Un constructor es una función especial, que se pone en marcha cuando se crea un objeto de
una clase, y se suele usar para dar esos valores iniciales, para reservar memoria si fuera
necesario, etc.
Se declara usando el mismo nombre que el de la clase, y sin ningún tipo de retorno. Por
ejemplo, un "constructor" para la clase Puerta que le diera los valores iniciales de 100 para el
ancho, 200 para el alto, etc., podría ser así:
public Puerta()
{
ancho = 100;
alto = 200;
color = 0xFFFFFF;
abierta = false;
}
Podemos tener más de un constructor, cada uno con distintos parámetros. Por ejemplo, puede
haber otro constructor que nos permita indicar el ancho y el alto:
Ahora, si declaramos un objeto de la clase puerta con "Puerta p = new Puerta();" tendrá de
ancho 100 y de alto 200, mientras que si lo declaramos con "Puerta p2 = new Puerta(90,220);"
tendrá 90 como ancho y 220 como alto.
Un programa de ejemplo que usara estos dos constructores para crear dos puertas con
características iniciales distintas podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 61: */
/* ejemplo61.cs */
/* */
Revisión 0.95 – Página 121
Introducción a la programación con C#, por Nacho Cabanes
using System;
public Puerta()
{
ancho = 100;
alto = 200;
color = 0xFFFFFF;
abierta = false;
}
{
Puerta p = new Puerta();
Puerta p2 = new Puerta(90,220);
Console.WriteLine("Valores iniciales...");
p.MostrarEstado();
Console.WriteLine("\nVamos a abrir...");
p.Abrir();
p.MostrarEstado();
Nota: al igual que existen los "constructores", también podemos indicar un "destructor" para
una clase, que se encargue de liberar la memoria que pudiéramos haber reservado en nuestra
clase (no es nuestro caso, porque aún no sabemos manejar memoria dinámica) o para cerrar
ficheros abiertos (que tampoco sabemos).
Un "destructor" se llama igual que la clase, pero precedido por el símbolo "~", no tiene tipo de
retorno, y no necesita ser "public", como ocurre en este ejemplo:
~Puerta()
{
// Liberar memoria
// Cerrar ficheros
}
puerta.Abrir ();
libro.Abrir ();
En este caso, la función "Abrir" está sobrecargada: se usa tanto para referirnos a abrir un libro
como para abrir una puerta. Se trata de dos acciones que no son exactamente iguales, que se
aplican a objetos distintos, pero que se llaman igual.
/*---------------------------*/
/* Ejemplo en C# nº 62: */
/* ejemplo62.cs */
/* */
/* Cuarto ejemplo de clases */
/* Constructores y herencia */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
public Animal()
{
Console.WriteLine("Ha nacido un animal");
}
}
// ------------------
public Perro()
{
Console.WriteLine("Ha nacido un perro");
}
}
// ------------------
public Gato()
{
Console.WriteLine("Ha nacido un gato");
}
}
// ------------------
Revisión 0.95 – Página 124
Introducción a la programación con C#, por Nacho Cabanes
public GatoSiames()
{
Console.WriteLine("Ha nacido un gato siamés");
}
}
// ------------------
Ha nacido un animal
Ha nacido un animal
Ha nacido un gato
Ha nacido un gato siamés
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un gato
Ejercicio propuesto:
• Crear un único fuente que contenga las siguientes clases:
o Una clase Trabajador, cuyo constructor escriba en pantalla "Soy un trabajador".
o Una clase Programador, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy programador".
o Una clase Analista, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy analista".
o Una clase Ingeniero, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy ingeniero".
o Una clase IngenieroInformatico, que derive de Ingeniero, cuyo constructor
escriba en pantalla "Soy ingeniero informático".
o Una clase "PruebaDeTrabajadores", que cree un objeto perteneciente a cada
una de esas clases.
En ese caso, deberemos reservar memoria primero para el array, y luego para cada uno de los
elementos. Por ejemplo, podríamos tener un array de 5 perros, que crearíamos de esta forma:
/*---------------------------*/
/* Ejemplo en C# nº 63: */
/* ejemplo63.cs */
/* */
/* Quinto ejemplo de clases */
/* Array de objetos */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
public Animal()
{
Console.WriteLine("Ha nacido un animal");
}
}
// ------------------
public Perro()
{
Console.WriteLine("Ha nacido un perro");
}
}
// ------------------
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ejercicio propuesto:
• Crea una versión ampliada del anterior ejercicio propuesto, en la que no se cree un
único objeto de cada clase, sino un array de tres objetos.
Además, existe una peculiaridad curiosa: podemos crear un array de "Animales", pero luego
indicar que unos de ellos son perros, otros gatos, etc.,
/*---------------------------*/
/* Ejemplo en C# nº 64: */
/* ejemplo64.cs */
/* */
/* Ejemplo de clases */
/* Array de objetos de */
/* varias subclases */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
public Animal()
{
Console.WriteLine("Ha nacido un animal");
}
}
// ------------------
public Perro()
{
Console.WriteLine("Ha nacido un perro");
}
}
// ------------------
public Gato()
{
Console.WriteLine("Ha nacido un gato");
}
}
// ------------------
public GatoSiames()
{
Console.WriteLine("Ha nacido un gato siamés");
}
}
// ------------------
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un gato
Ha nacido un animal
Ha nacido un gato
Ha nacido un gato siamés
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Ha nacido un perro
Ha nacido un animal
Vamos a verlo con un ejemplo, que en vez de tener constructores va a tener un único método
"Hablar", que se redefine en cada una de las clases hijas, y después comentaremos qué ocurre
al ejecutarlo:
/*---------------------------*/
/* Ejemplo en C# nº 65: */
/* ejemplo65.cs */
/* */
/* Ejemplo de clases */
/* Array de objetos de */
/* varias subclases con */
/* metodos */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
{
Console.WriteLine("Estoy comunicándome...");
}
}
// ------------------
// ------------------
// ------------------
miPerro.Hablar();
miGato.Hablar();
miAnimal.Hablar();
misAnimales[0].Hablar();
misAnimales[1].Hablar();
misAnimales[2].Hablar();
}
Guau!
Miauuu
Estoy comunicándome...
Estoy comunicándome...
Estoy comunicándome...
Estoy comunicándome...
La primera parte era de esperar: si creamos un perro, debería decir "Guau", un gato debería
decir "Miau" y un animal genérico debería comunicarse. Eso es lo que se consigue con este
fragmento:
miPerro.Hablar();
miGato.Hablar();
miAnimal.Hablar();
misAnimales[0].Hablar();
misAnimales[1].Hablar();
misAnimales[2].Hablar();
Es decir, como la clase base es "Animal", el primer elemento hace lo que corresponde a un
Animal genérico (decir "Estoy comunicándome"), a pesar de que hayamos dicho que se trata de
un Perro.
Generalmente, no será esto lo que queramos. Sería interesante no necesitar crear un array de
perros y otros de gatos, sino poder crear un array de animales, y que contuviera animales de
distintos tipos.
Para conseguir este comportamiento, debemos indicar a nuestro compilador que el método
"Hablar" que se usa en la clase Animal puede que sea redefinido por otras clases hijas, y que
en ese caso debe prevalecer lo que indiquen las clases hijas.
La forma de hacerlo es declarando ese método "Hablar" como "virtual", y empleando en las
clases hijas la palabra "override" en vez de "new", así:
/*---------------------------*/
/* Ejemplo en C# nº 66: */
/* ejemplo66.cs */
/* */
/* Ejemplo de clases */
/* Array de objetos de */
/* varias subclases con */
/* metodos virtuales */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// ------------------
// ------------------
// ------------------
miPerro.Hablar();
miGato.Hablar();
miAnimal.Hablar();
misAnimales[0].Hablar();
misAnimales[1].Hablar();
misAnimales[2].Hablar();
}
Guau!
Miauuu
Estoy comunicándome...
Guau!
Miauuu
Estoy comunicándome...
/*---------------------------*/
/* Ejemplo en C# nº 67: */
/* ejemplo67.cs */
/* */
/* Ejemplo de clases */
/* Llamar a la superclase */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// ------------------
// ------------------
// ------------------
miGato.Hablar();
Console.WriteLine(); // Linea en blanco
miGato2.Hablar();
Su resultado sería
Miauuu
Miauuu
Pfff
return nuevaMatriz;
}
Desde "Main", calcularíamos una matriz como suma de otras dos haciendo simplemente
Ejercicios propuestos:
• Desarrolla una clase "Matriz", que represente a una matriz de 3x3, con métodos para
indicar el valor que hay en una posición, leer el valor de una posición, escribir la matriz
en pantalla y sumar dos matrices.
En estos casos, un editor como el Bloc de Notas, o incluso como Notepad++, se queda corto.
Sería preferible un entorno que nos permitiera editar nuestros fuentes, compilarlos sin
necesidad de salir de él, que nos destacara las líneas que contienen errores, que nos mostrara
ayuda sobre la sintaxis de cada función, que nos permitiera depurar nuestros programas
avanzando paso a paso…
Existen entornos que nos permiten hacer todo eso, y además hacerlo gratis. El más conocido es
el Visual Studio de Microsoft, que en su versión Express incluye todo lo un programador novel
como nosotros puede necesitar. Una alternativa muy similar, pero algo más sencilla (lo que
supone que funcione más rápido en ordenadores no demasiado potentes) es SharpDevelop.
Vamos a ver las pautas básicas de manejo de SharpDevelop (que se aplicarían con muy pocos
cambios al caso de Visual Studio):
Comenzamos por descargar el fichero de instalación del entorno, desde su página oficial
(http://www.icsharpcode.net/opensource/sd/). La versión 3.1, para las versiones 2.0 y 3.5 de la
plataforma .Net, ocupa unos 15 Mb.
La instalación comenzará simplemente con hacer doble clic. Deberíamos ver una ventana
parecida a ésta:
Como es habitual, el siguiente paso será aceptar el contrato de licencia, después deberemos
decir en qué carpeta queremos instalarlo, comenzará a copiar archivos y al cabo de un instante,
tendremos un nuevo icono en nuestro escritorio:
La instalación debería ser muy sencilla en Windows Vista y superiores, pero en Windows XP
quizá necesite que instalemos la versión 3.5 de la plataforma .Net (se puede hacer
gratuitamente desde su página oficial).
Cuando lanzamos nuestro nuevo icono, veremos la pantalla principal de SharpDevelop, que nos
muestra la lista de los últimos proyectos ("soluciones") que hemos realizado, y nos permite
crear uno nuevo:
En nuestro caso, comenzaremos por crear una "Nueva solución", y se nos mostrará los tipos de
proyectos para los que se nos podría crear un esqueleto vacío que después iríamos rellenando:
De estos tipos, el único que conocemos es una "Aplicación de Consola" (en C#, claro).
Deberemos escribir también el nombre., y aparecerá un esqueleto de aplicación que nosotros
sólo tendríamos que completar:
Cuando hayamos terminado de realizar nuestros cambios, podemos compilar el programa con el
botón:
(Si la ventana de nuestro programa se cierra tan rápido que no tenemos tiempo de leerla, nos
puede interesar añadir provisionalmente una línea ReadLine() al final del fuente, para que éste
se detenga hasta que pulsemos la tecla Intro)
Así prepararíamos y lanzaríamos un programa formado por un solo fuente. Si se trata de varios
fuentes, basta con ir añadiendo nuevas clases al proyecto. Lo conseguimos pulsando el botón
derecho sobre el nombre del proyecto (en la ventana izquierda, "Proyectos") y escogiendo las
opciones Agregar / Nuevo Elemento:
Normalmente, el tipo de elemento que nos interesará será una clase, cuyo nombre deberemos
indicar:
y obtendríamos un nuevo esqueleto vacío (esta vez sin "Main"), que deberíamos completar.
Nuestro programa, que ahora estaría formado por dos clases, se compilaría y se ejecutaría de la
misma forma que cuando estaba integrado por una única clase.
Ejercicio propuesto:
• Crear un proyecto que contenga las siguientes clases (cada una en un fichero distinto):
o Una clase Trabajador, cuyo constructor escriba en pantalla "Soy un trabajador".
o Una clase Programador, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy programador".
o Una clase Analista, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy analista".
o Una clase Ingeniero, que derive de Trabajador, cuyo constructor escriba en
pantalla "Soy ingeniero".
7. Manejo de ficheros
7.1. Escritura en un fichero de texto
Para manejar ficheros, siempre deberemos realizar tres operaciones básicas:
• Abrir el fichero.
• Leer datos de él o escribir datos en él.
• Cerrar el fichero.
Eso sí, no siempre podremos realizar esas operaciones, así que además tendremos que
comprobar los posibles errores. Por ejemplo, puede ocurrir que intentemos abrir un fichero que
realmente no exista, o que queramos escribir en un dispositivo que sea sólo de lectura.
Vamos a ver un ejemplo, que cree un fichero de texto y escriba algo en él:
/*---------------------------*/
/* Ejemplo en C# nº 70 */
/* ejemplo70.cs */
/* */
/* Escritura en un fichero */
/* de texto */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO; // Para StreamWriter
fichero = File.CreateText("prueba.txt");
fichero.WriteLine("Esto es una línea");
fichero.Write("Esto es otra");
fichero.WriteLine(" y esto es continuación de la anterior");
fichero.Close();
}
• Finalmente, debemos cerrar el fichero con Close, o podría quedar algún dato sin
guardar.
Ejercicios propuestos:
• Crea un programa que vaya leyendo las frases que el usuario teclea y las guarde en un
fichero de texto llamado "registroDeUsuario.txt". Terminará cuando la frase introducida
sea "fin" (esa frase no deberá guardarse en el fichero).
/*---------------------------*/
/* Ejemplo en C# nº 71: */
/* ejemplo71.cs */
/* */
/* Lectura de un fichero de */
/* texto */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO; // Para StreamReader
fichero = File.OpenText("prueba.txt");
linea = fichero.ReadLine();
Console.WriteLine( linea );
Console.WriteLine( fichero.ReadLine() );
fichero.Close();
}
Ejercicios propuestos:
• Crea un programa que lea las tres primeras líneas del fichero creado en el apartado
anterior y las muestre en pantalla.
En un fichero de texto, la forma de saber si hemos llegado al final es intentar leer una línea, y
comprobar si el resultado ha sido "null", lo que nos indicaría que no se ha podido leer y que,
por tanto estamos en el final del fichero.
Normalmente, si queremos procesar todo un fichero, esta lectura y comprobación debería ser
repetitiva, así:
/*---------------------------*/
/* Ejemplo en C# nº 72: */
/* ejemplo72.cs */
/* */
/* Lectura de un fichero de */
/* texto */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO; // Para StreamReader
fichero = File.OpenText("prueba.txt");
do
{
linea = fichero.ReadLine();
if (linea != null)
Console.WriteLine( linea );
}
while (linea != null);
fichero.Close();
}
Ejercicios propuestos:
• Un programa que pida al usuario que teclee frases, y las almacene en el fichero
"frases.txt". Acabará cuando el usuario pulse Intro sin teclear nada. Después deberá
mostrar el contenido del fichero.
• Un programa que pregunte un nombre de fichero y muestre en pantalla el contenido de
ese fichero, haciendo una pausa después de cada 25 líneas, para que dé tiempo a
leerlo. Cuando el usuario pulse intro, se mostrarán las siguientes 25 líneas, y así hasta
que termine el fichero.
/*---------------------------*/
/* Ejemplo en C# nº 73 */
/* ejemplo73.cs */
/* */
/* Añadir en un fichero */
/* de texto */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO; // Para StreamWriter
fichero = File.CreateText("prueba2.txt");
fichero.WriteLine("Primera línea");
fichero.Close();
fichero = File.AppendText("prueba2.txt");
fichero.WriteLine("Segunda línea");
fichero.Close();
Ejercicios propuestos:
• Un programa que pida al usuario que teclee frases, y las almacene en el fichero
"registro.txt", que puede existir anteriormente (y que no deberá borrarse, sino añadir al
final de su contenido). Cada sesión acabará cuando el usuario pulse Intro sin teclear
nada.
•
Como esta sintaxis puede llegar a resultar incómoda, en C# existe una alternativa: podemos
indicar una arroba (@) justo antes de abrir las comillas, y entonces no será necesario delimitar
los caracteres de control:
Ejercicios propuestos:
• Crear un programa que pida al usuario pares de números enteros y escriba su suma
(con el formato "20 + 3 = 23") en pantalla y en un fichero llamado "sumas.txt", que se
encontrará en un subdirectorio llamado "resultados". Cada vez que se ejecute el
programa, deberá añadir los nuevos resultados a continuación de los resultados de las
ejecuciones anteriores.
Una primera solución es usar "File.Exists(nombre)", para comprobar si está, antes de intentar
abrirlo:
/*---------------------------*/
/* Ejemplo en C# nº 74: */
/* ejemplo74.cs */
/* */
/* Lectura de un fichero de */
/* texto */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
while (true) {
Console.Write( "Dime el nombre del fichero (\"fin\" para terminar): ");
nombre = Console.ReadLine();
if (nombre == "fin")
break;
if ( File.Exists(nombre) )
{
fichero = File.OpenText( nombre );
Console.WriteLine("Su primera linea es: {0}",
fichero.ReadLine() );
fichero.Close();
}
else
Console.WriteLine( "No existe!" );
}
}
Ejercicios propuestos:
• Mejorar el segundo ejercicio del apartado 7.3 (el que muestra un fichero, haciendo una
pausa cada 25 líneas) para que compruebe antes si el fichero existe, y muestre un
mensaje de aviso en caso de que no sea así.
Una forma más eficaz de comprobar si ha existido algún tipo de error es utilizar el manejo de
"excepciones" que permiten los lenguajes modernos, como C#.
La idea es la siguiente: "intentaremos" dar una serie de pasos, y al final de todos ellos indica-
remos qué pasos hay que dar en caso de que alguno no se consiga completar. Esto permite
que el programa sea más legible que la alternativa "convencional", que consistía en: intentar un
paso y comprobar errores; si todo era correcto, intentar otro paso y volver a comprobar
errores; si todo seguía siendo correcto, intentar otro nuevo paso, y así sucesivamente.
La forma de conseguirlo es dividir en dos bloques las partes de programa que puedan dar lugar
a error:
Un primer ejemplo, que mostrara todo el contenido de un fichero de texto, y que en caso de
error, se limitara a mostrar un mensaje de error y a abandonar el programa, podría ser:
/*---------------------------*/
/* Ejemplo en C# nº 75: */
/* ejemplo75.cs */
/* */
/* Excepciones y ficheros */
/* (1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
try
{
fichero = File.OpenText(nombre);
do
{
linea = fichero.ReadLine();
if (linea != null)
Console.WriteLine( linea );
}
while (linea != null);
fichero.Close();
}
catch (Exception exp)
{
Console.WriteLine("Ha habido un error: {0}", exp.Message);
return;
}
}
Pero en general, lo razonable no es interceptar "todas las excepciones a la vez", sino crear un
análisis para cada caso, que permita recuperarse del error y seguir adelante, para lo que se
suelen crear varios bloques "catch". Por ejemplo, en el caso de que queramos crear un fichero,
podemos tener excepciones como éstas:
• El fichero existe y es de sólo lectura (se lanzará una excepción del tipo "IOException").
• La ruta del fichero es demasiado larga (excepción de tipo "PathTooLongException").
• El disco puede estar lleno (IOException).
Así, dentro de cada bloque "catch" podríamos indicar una excepción más concreta, de forma
que el mensaje de aviso sea más concreto, o que podamos dar pasos más adecuados para
solucionar el problema:
/*---------------------------*/
/* Ejemplo en C# nº 76: */
/* ejemplo76.cs */
/* */
/* Excepciones y ficheros */
/* (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
try
{
fichero = File.CreateText(nombre);
fichero.WriteLine( linea );
fichero.Close();
}
catch (PathTooLongException e)
{
Console.WriteLine("Nombre demasiado largo!");
}
catch (IOException e)
{
Revisión 0.95 – Página 148
Introducción a la programación con C#, por Nacho Cabanes
Como la consola se comporta como un fichero de texto (realmente, como un fichero de entrada
y otro de salida), se puede usar "try…catch" para comprobar ciertos errores relacionados con la
entrada de datos, como cuando no se puede convertir un dato a un cierto tipo (por ejemplo, si
queremos convertir a Int32, pero el usuario ha tecleado sólo texto).
Ejercicios propuestos:
• Un programa que pida al usuario el nombre de un fichero de origen y el de un fichero
de destino, y que vuelque al segundo fichero el contenido del primero, convertido a
mayúsculas. Se debe controlar los posibles errores, como que el fichero de origen no
exista, o que el fichero de destino no se pueda crear.
• Un programa que pida al usuario un número, una operación (+, -, *, /) y un segundo
número, y muestre el resultado de la correspondiente operación. Si se teclea un dato
no numérico, el programa deberá mostrar un aviso y volver a pedirlo, en vez de
interrumpir la ejecución.
• Un programa que pida al usuario repetidamente pares de números y la operación a
realizar con ellos (+, -, *, /) y guarde en un fichero "calculadora.txt" el resultado de
dichos cálculos (con la forma "15 * 6 = 90"). Debe controlar los posibles errores, como
que los datos no sean numéricos, la división entre cero, o que el fichero no se haya
podido crear.
Como primer acercamiento, vamos a ver cómo abrir un fichero (no necesariamente de texto) y
leer el primer byte que contiene. Usaremos una clase específicamente diseñada para leer datos
de los tipos básicos existentes en C# (byte, int, float, etc.), la clase "BinaryReader":
/*---------------------------*/
/* Ejemplo en C# nº 77: */
/* ejemplo77.cs */
/* */
/* Ficheros binarios (1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
try
{
fichero = new BinaryReader(
File.Open(nombre, FileMode.Open));
unDato = fichero.ReadByte();
Console.WriteLine("El byte leido es un {0}",
unDato);
fichero.Close();
Revisión 0.95 – Página 150
Introducción a la programación con C#, por Nacho Cabanes
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
}
La forma de abrir un BinaryReader es algo más incómoda que con los ficheros de texto, porque
nos obliga a llamar a un constructor y a indicarle ciertas opciones sobre el modo de fichero (la
habitual será simplemente "abrirlo", con "FileMode.Open"), pero a cambio podemos leer
cualquier tipo de dato, no sólo texto: ReadByte lee un dato "byte", ReadInt32 lee un "int",
ReadSingle lee un "float", ReadString lee un "string", etc.
Ejercicios propuestos:
• Abrir un fichero con extensión EXE y comprobar si realmente se trata de un ejecutable,
mirando si los dos primeros bytes del fichero corresponden a una letra "M" y una letra
"Z", respectivamente.
Para eso, usaremos la clase "FileStream", más genérica. Esta clase tiene método "Read", que
nos permite leer una cierta cantidad de datos desde el fichero. Le indicaremos en qué array
guardar esos datos, a partir de qué posición del array debe introducir los datos, y qué cantidad
de datos se deben leer. Nos devuelve un valor, que es la cantidad de datos que se han podido
leer realmente (porque puede ser menos de lo que le hemos pedido, si hay un error o estamos
al final del fichero).
/*---------------------------*/
/* Ejemplo en C# nº 78: */
/* ejemplo78.cs */
/* */
/* Ficheros binarios (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
try
{
fichero = File.OpenRead(nombre);
datos = new byte[10];
int posicion = 0;
int cantidadALeer = 10;
cantidadLeida = fichero.Read(datos, posicion, cantidadALeer);
fichero.Close();
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
}
Ejercicios propuestos:
• Abrir un fichero con extensión EXE y comprobar si realmente se trata de un ejecutable,
mirando si los dos primeros bytes del fichero corresponden a una letra "M" y una letra
"Z", respectivamente. Se deben leer ambos bytes a la vez, usando un array.
Para ello, tenemos el método "Seek". A este método se le indican dos parámetros: la posición a
la que queremos saltar, y el punto desde el que queremos que se cuente esa posición (desde el
comienzo del fichero –SeekOrigin.Begin-, desde la posición actual –SeekOrigin.Current- o desde
el final del fichero –SeekOrigin.End-). La posición es un Int64, porque puede ser un número
muy grande e incluso un número negativo (si miramos desde el final del fichero, porque desde
él habrá que retroceder).
De igual modo, podemos saber en qué posición del fichero nos encontramos, consultando la
propiedad "Position", así como la longitud del fichero, mirando su propiedad "Length", como en
este ejemplo:
/*---------------------------*/
/* Ejemplo en C# nº 79: */
/* ejemplo79.cs */
/* */
/* Ficheros binarios (3) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
try
{
fichero = File.OpenRead(nombre);
datos = new byte[10];
int posicion = 0;
int cantidadALeer = 10;
cantidadLeida = fichero.Read(datos, posicion, cantidadALeer);
fichero.Length);
}
fichero.Close();
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
}
(Nota: existe una propiedad "CanSeek" que nos permite saber si el fichero que hemos abierto
permite realmente que nos movamos a unas posiciones u otras).
Vamos a ver un ejemplo que junte todo ello: crearemos un fichero, guardaremos datos, lo
leeremos para comprobar que todo es correcto, añadiremos al final, y volveremos a leer:
/*---------------------------*/
/* Ejemplo en C# nº 80: */
/* ejemplo80.cs */
/* */
/* Ficheros binarios (4): */
/* Escritura */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
byte[] datos;
nombre = "datos.dat";
datos = new byte[TAMANYO_BUFFER];
try
{
int posicion = 0;
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
}
(Nota: existe una propiedad "CanWrite" que nos permite saber si se puede escribir en el
fichero).
Si queremos que escribir datos básicos de C# (float, int, etc.) en vez de un array de bytes,
podemos usar un "BinaryWriter", que se maneja de forma similar a un "BinaryReader", con la
diferencia de que no tenemos métodos WriteByte, WriteString y similares, sino un único método
"Write", que se encarga de escribir el dato que le indiquemos, sea del tipo que sea:
/*---------------------------*/
/* Ejemplo en C# nº 81: */
/* ejemplo81.cs */
/* */
/* Ficheros binarios (5) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
Console.WriteLine("Creando fichero...");
// Primero vamos a grabar datos
try
{
ficheroSalida = new BinaryWriter(
File.Open(nombre, FileMode.Create));
unDatoByte = 1;
unDatoInt = 2;
unDatoFloat = 3.0f;
unDatoDouble = 4.0;
unDatoString = "Hola";
ficheroSalida.Write(unDatoByte);
ficheroSalida.Write(unDatoInt);
ficheroSalida.Write(unDatoFloat);
ficheroSalida.Write(unDatoDouble);
ficheroSalida.Write(unDatoString);
ficheroSalida.Close();
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
Como se puede ver en este ejemplo, también podemos usar "Seek" para movernos a un punto
u otro de un fichero si usamos un "BinaryReader", pero está un poco más escondido: no se lo
pedimos directamente a nuestro fichero, sino al "Stream" (flujo de datos) que hay por debajo:
ficheroEntrada.BaseStream.Seek(1, SeekOrigin.Begin);
El float leido es un 3
El double leido es un 4
El string leido es "Hola"
Volvamos a leer el int...
El int leido es un 2
En este caso hemos usado "FileMode.Create" para indicar que queremos crear el fichero, en vez
de abrir un fichero ya existente. Los modos de fichero que podemos emplear en un Binary-
Reader o en un BinaryWriter son los siguientes:
Ahora vamos a ver un ejemplo un poco más sofisticado y un poco más real: vamos a abrir un
fichero que sea una imagen en formato BMP y a mostrar en pantalla si está comprimido o no.
Para eso necesitamos antes saber cómo se guarda la información en un fichero BMP, pero esto
es algo fácil de localizar en Internet:
FICHEROS .BMP
Un fichero BMP está compuesto por las siguientes partes: una cabecera de
fichero, una cabecera del bitmap, una tabla de colores y los bytes que
definirán la imagen.
Con esta información nos basta para nuestro propósito: la compresión se indica en la posición
30 del fichero, es un entero de 4 bytes (lo mismo que un "int" en los sistemas operativos de 32
bits), y si es un 0 querrá decir que la imagen no está comprimida.
Como el bit menos significativo se almacena en primer lugar, nos podría bastar con leer sólo el
byte de la posición 30, para ver si vale 0, y despreciar los 3 bytes siguientes. Entonces, lo
podríamos comprobar así:
/*---------------------------*/
/* Ejemplo en C# nº 82: */
/* ejemplo82.cs */
/* */
/* Ficheros binarios (6): */
/* Ver si un BMP está */
/* comprimido */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
if (! File.Exists( nombre) )
{
Console.WriteLine("No encontrado!");
}
else
{
fichero = File.OpenRead(nombre);
fichero.Seek(30, SeekOrigin.Begin);
compresion = fichero.ReadByte();
fichero.Close();
if (compresion == 0)
Console.WriteLine("Sin compresión");
else
Console.WriteLine("BMP Comprimido ");
}
}
}
Ya que estamos, podemos mejorarlo un poco para que además nos muestre el ancho y el alto
de la imagen, y que compruebe antes si realmente se trata de un fichero BMP:
/*---------------------------*/
/* Ejemplo en C# nº 83: */
/* ejemplo83.cs */
/* */
/* Ficheros binarios (7): */
/* Información de un */
/* fichero BMP */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
if (! File.Exists( nombre) )
{
Console.WriteLine("No encontrado!");
}
else
{
fichero = File.OpenRead(nombre);
// Leo los dos primeros bytes
marca1 = Convert.ToChar( fichero.ReadByte() );
marca2 = Convert.ToChar( fichero.ReadByte() );
fichero.Read(datosTemp, 0, 4);
ancho = datosTemp[0] + // Convierto 4 bytes a Int32
datosTemp[1] * 256 + datosTemp[2] * 256 * 256
+ datosTemp[3] * 256 * 256 * 256;
Console.WriteLine("Ancho: {0}", ancho);
fichero.Read(datosTemp, 0, 4);
alto = datosTemp[0] + // Convierto 4 bytes a Int32
datosTemp[1] * 256 + datosTemp[2] * 256 * 256
+ datosTemp[3] * 256 * 256 * 256;
Console.WriteLine("Alto: {0}", alto);
switch (compresion) {
case 0: Console.WriteLine("Sin compresión"); break;
case 1: Console.WriteLine("Compresión RLE 8 bits"); break;
case 2: Console.WriteLine("Compresión RLE 4 bits"); break;
}
} else
Console.WriteLine("No parece un fichero BMP\n"); // Si la marca no es BM
}
}
}
/*---------------------------*/
/* Ejemplo en C# nº 84: */
/* ejemplo84.cs */
/* */
/* Ficheros binarios (8): */
/* Información de un BMP */
/* con BinaryReader */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
if (! File.Exists( nombre) )
{
Console.WriteLine("No encontrado!");
}
else
{
fichero = new BinaryReader(
File.Open(nombre, FileMode.Open));
// Leo los dos primeros bytes
marca1 = Convert.ToChar( fichero.ReadByte() );
marca2 = Convert.ToChar( fichero.ReadByte() );
alto = fichero.ReadInt32();
Console.WriteLine("Alto: {0}", alto);
switch (compresion) {
case 0: Console.WriteLine("Sin compresión"); break;
case 1: Console.WriteLine("Compresión RLE 8 bits"); break;
case 2: Console.WriteLine("Compresión RLE 4 bits"); break;
}
} else
Console.WriteLine("No parece un fichero BMP\n"); // Si la marca no es BM
}
}
}
Ejercicios propuestos:
• Localiza en Internet información sobre el formato de imágenes PCX. Crea un programa
que diga el ancho, alto y número de colores de una imagen PCX.
• Localiza en Internet información sobre el formato de imágenes GIF. Crea un programa
que diga el subformato, ancho, alto y número de colores de una imagen GIF.
también tenemos la alternativa de usar un "FileStream", que también tiene un método llamado
simplemente "Open", al que se le puede indicar el modo de apertura (FileMode, como se vieron
en el apartado 7.12) y el modo de acceso (FileAccess.Read si queremos leer, FileAccess.Write si
queremos escribir, o FileAccess.ReadWrite si queremos leer y escribir).
Una vez que hayamos indicado que queremos leer y escribir del fichero, podremos movernos
dentro de él con "Seek", leer datos con "Read" o "ReadByte", y grabar datos con "Write" o
"WriteByte":
/*---------------------------*/
/* Ejemplo en C# nº 85: */
/* ejemplo85.cs */
/* */
/* Ficheros binarios (9): */
/* Lectura y Escritura */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.IO;
nombre = "datos.dat";
datos = new byte[TAMANYO_BUFFER];
try
{
int posicion = 0;
fichero.Seek(2, SeekOrigin.Begin);
int nuevoDato = fichero.ReadByte();
Console.WriteLine("El tercer byte es un {0}", nuevoDato);
fichero.Seek(2, SeekOrigin.Begin);
fichero.WriteByte( 4 );
fichero.Seek(2, SeekOrigin.Begin);
nuevoDato = fichero.ReadByte();
Console.WriteLine("Ahora el tercer byte es un {0}", nuevoDato);
fichero.Close();
}
catch (Exception exp)
{
Console.WriteLine(exp.Message);
return;
}
}
Ejercicios propuestos:
• Un programa que vuelque todo el contenido de un fichero de texto a otro, convirtiendo
cada frase a mayúsculas. Los nombres de ambos ficheros se deben indican como
parámetros en la línea de comandos.
• Un programa que pida al usuario el nombre de un fichero de texto y una frase a
buscar, y que muestre en pantalla todas las frases de ese fichero que contengan esa
frase. Cada frase se debe preceder del número de línea (la primera línea del fichero
será la 1, la segunda será la 2, y así sucesivamente).
• Un programa que pida al usuario el nombre de un fichero y una secuencia de 4 bytes, y
diga si el fichero contiene esa secuencia de bytes.
• Un programa que duplique un fichero, copiando todo su contenido a un nuevo fichero.
El nombre de ambos ficheros se debe leer de la línea de comandos.
• Un programa que muestre el nombre del autor de un fichero de música en formato MP3
(tendrás que localizar en Internet la información sobre dicho formato).
Este tipo de variables son sencillas de usar y rápidas... si sólo vamos a manejar estructuras de
datos que no cambien, pero resultan poco eficientes si tenemos estructuras cuyo tamaño no
sea siempre el mismo.
Es el caso de una agenda: tenemos una serie de fichas, e iremos añadiendo más. Si reservamos
espacio para 10, no podremos llegar a añadir la número 11, estamos limitando el máximo. Una
solución sería la de trabajar siempre en el disco: no tenemos límite en cuanto a número de
fichas, pero es muchísimo más lento.
Lo ideal sería aprovechar mejor la memoria que tenemos en el ordenador, para guardar en ella
todas las fichas o al menos todas aquellas que quepan en memoria.
Una solución "típica" (pero mala) es sobredimensionar: preparar una agenda contando con
1000 fichas, aunque supongamos que no vamos a pasar de 200. Esto tiene varios
inconvenientes: se desperdicia memoria, obliga a conocer bien los datos con los que vamos a
trabajar, sigue pudiendo verse sobrepasado, etc.
La solución suele ser crear estructuras DINÁMICAS, que puedan ir creciendo o disminuyendo
según nos interese. En los lenguajes de programación "clásicos", como C y Pascal, este tipo de
estructuras se tienen que crear de forma básicamente artesanal, mientras que en lenguajes
modernos como C#, Java o las últimas versiones de C++, existen esqueletos ya creados que
podemos utilizar con facilidad.
Las pilas. Como una pila de libros: vamos apilando cosas en la cima, o cogiendo de la
cima. Se supone que no se puede tomar elementos de otro sitio que no sea la cima, ni
dejarlos en otro sitio distinto. De igual modo, se supone que la pila no tiene un tamaño
máximo definido, sino que puede crecer arbitrariamente.
Las colas. Como las del cine (en teoría): la gente llega por un sitio (la cola) y sale por
el opuesto (la cabeza). Al igual que antes, supondremos que un elemento no puede
entrar a la cola ni salir de ella en posiciones intermedias y que la cola puede crecer
hasta un tamaño indefinido.
Las listas, en las que se puede añadir elementos en cualquier posición, y borrarlos de
cualquier posición.
Y la cosa se va complicando: en los árboles cada elemento puede tener varios sucesores (se
parte de un elemento "raíz", y la estructura se va ramificando), etc.
Veremos ejemplos de cómo crear estructuras dinámicas de estos tipos en C#, y después
comentaremos los pasos para crear una estructura dinámica de forma "artesanal".
Así, un ejemplo básico que creara una pila, introdujera tres palabras y luego las volviera a
mostrar sería:
/*---------------------------*/
/* Ejemplo en C# */
/* pila1.cs */
/* */
/* Ejemplo de clase "Stack" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
string palabra;
yo
soy
Hola,
La implementación de una pila en C# es algo más avanzada: permite también métodos como:
• "Peek", que mira el valor que hay en la cima, pero sin extraerlo.
• "Clear", que borra todo el contenido de la pila.
• "Contains", que indica si un cierto elemento está en la pila.
• "GetType", para saber de qué tipo son los elementos almacenados en la pila.
• "ToString", que devuelve el elemento actual convertido a un string.
• "ToArray", que devuelve toda la pila convertida a un array.
• "GetEnumerator", que permite usar "enumeradores" para recorrer la pila, una
funcionalidad que veremos con algún detalle más adelante.
• También tenemos una propiedad "Count", que nos indica cuántos elementos contiene.
/*---------------------------*/
/* Ejemplo en C# */
/* cola1.cs */
/* */
/* Ejemplo de clase "Queue" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
string palabra;
Console.WriteLine( palabra );
}
que mostraría:
Hola,
soy
yo
Al igual que ocurría con la pila, la implementación de una cola que incluye C# es más avanzada
que eso, con métodos similares a los de antes:
• "Peek", que mira el valor que hay en la cabeza de la cola, pero sin extraerlo.
• "Clear", que borra todo el contenido de la cola.
• "Contains", que indica si un cierto elemento está en la cola.
• "GetType", para saber de qué tipo son los elementos almacenados en la cola.
• "ToString", que devuelve el elemento actual convertido a un string.
• "ToArray", que devuelve toda la pila convertida a un array.
• "GetEnumerator", que permite usar "enumeradores" para recorrer la cola, una
funcionalidad que veremos con algún detalle más adelante.
• Al igual que en la pila, también tenemos una propiedad "Count", que nos indica cuántos
elementos contiene.
En el caso de C#, no tenemos ninguna clase "List" que represente una lista genérica, pero sí
dos variantes especialmente útiles: una lista ordenada ("SortedList") y una lista a cuyos
elementos se puede acceder como a los de un array ("ArrayList").
8.4.1. ArrayList
En un ArrayList, podemos añadir datos en la última posición con "Add", insertar en cualquier
otra con "Insert", recuperar cualquier elemento usando corchetes, o bien ordenar toda la lista
con "Sort". Vamos a ver un ejemplo de la mayoría de sus posibilidades:
/*---------------------------*/
/* Ejemplo en C# */
/* arrayList1.cs */
/* */
/* Ejemplo de ArrayList */
Revisión 0.95 – Página 168
Introducción a la programación con C#, por Nacho Cabanes
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
// Buscamos un elemento
Console.WriteLine( "La palabra \"yo\" está en la posición {0}",
miLista.IndexOf("yo") );
// Ordenamos
miLista.Sort();
// Invertimos la lista
miLista.Reverse();
Contenido actual:
Hola,
soy
yo
La segunda palabra es: soy
Contenido tras insertar:
Hola,
Como estas?
soy
yo
La palabra "yo" está en la posición 3
Contenido tras ordenar
Como estas?
Hola,
soy
yo
Ahora "yo" está en la posición 3
Contenido dar la vuelta y tras eliminar dos:
Hola,
Como estas?
La frase "Hasta Luego"...
No está. El dato inmediatamente mayor es el 1: Hola,
Casi todo debería resultar fácil de entender, salvo quizá el símbolo ~. Esto se debe a que
BinarySearch devuelve un número negativo cuando el texto que buscamos no aparece, pero
ese número negativo tiene un significado: es el "valor complementario" de la posición del dato
inmediatamente mayor (es decir, el dato cambiando los bits 0 por 1 y viceversa). En el ejemplo
anterior, "posición" vale -2, lo que quiere decir que el dato no existe, y que el dato
inmediatamente mayor está en la posición 1 (que es el "complemento a 2" del número -2, que
es lo que indica la expresión "~posición"). En el apéndice 3 de este texto hablaremos de cómo
se representan internamente los números enteros, tanto positivos como negativos, y entonces
se verá con detalle en qué consiste el "complemento a 2".
A efectos prácticos, lo que nos interesa es que si quisiéramos insertar la frase "Hasta Luego",
su posición correcta para que todo el ArrayList permaneciera ordenado sería la 1, que viene
indicada por "~posicion".
Veremos los operadores a nivel de bits, como ~, en el tema 10, que estará dedicado a otras
características avanzadas de C#.
8.4.2. SortedList
En un SortedList, los elementos están formados por una pareja: una clave y un valor (como en
un diccionario: la palabra y su definición). Se puede añadir elementos con "Add", o acceder a
los elementos mediante su índice numérico (con "GetKey") o mediante su clave (sabiendo en
qué posición se encuentra una clave con "IndexOfKey"), como en este ejemplo:
/*---------------------------*/
/* Ejemplo en C# */
/* sortedList1.cs */
/* */
/* Ejemplo de SortedList: */
/* Diccionario esp-ing */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
public class ejemploSortedList {
Su resultado sería
Un ejemplo de diccionario, parecido al anterior (que es más rápido de consultar para un dato
concreto, pero que no se puede recorrer en orden), podría ser:
/*---------------------------*/
/* Ejemplo en C# */
/* HashTable1.cs */
/* */
/* Ejemplo de HashTable: */
/* Diccionario de inform. */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
public class ejemploHashTable {
Si un elemento que se busca no existe, se lanzaría una excepción, por lo que deberíamos
controlarlo con un bloque try..catch. Lo mismo ocurre si intentamos introducir un dato que ya
existe. Una alternativa a usar try..catch es comprobar si el dato ya existe, con el método
"Contains" (o su sinónimo "ContainsKey"), como en este ejemplo:
/*---------------------------*/
/* Ejemplo en C# */
/* HashTable2.cs */
/* */
/* Ejemplo de HashTable 2: */
/* Diccionario de inform. */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
public class ejemploHashTable2 {
Otras posibilidades son: borrar un elemento ("Remove"), vaciar toda la tabla ("Clear"), o ver si
contiene un cierto valor ("ContainsValue", mucho más lento que buscar entre las claves con
"Contains").
Una tabla hash tiene una cierta capacidad inicial, que se amplía automáticamente cuando es
necesario. Como la tabla hash es mucho más rápida cuando está bastante vacía que cuando
está casi llena, podemos usar un constructor alternativo, en el que se le indica la capacidad
inicial que queremos, si tenemos una idea aproximada de cuántos datos vamos a guardar:
/*---------------------------*/
/* Ejemplo en C# */
/* HashTable3.cs */
/* */
/* Ejemplo de HashTable */
/* y enumerador */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
public class ejemploHashTable3 {
cuyo resultado es
Contenido:
pc = personal computer
byte = 8 bits
kilobyte = 1024 bytes
Como se puede ver, los enumeradores tendrán un método "MoveNext", que intenta moverse al
siguiente elemento y devuelve "false" si no lo consigue. En el caso de las tablas hash, que tiene
dos campos (clave y valor), el enumerador a usar será un "enumerador de diccionario"
(IDictionaryEnumerator), que contiene los campos Key y Value.
Para las colecciones "normales", como las pilas y las colas, el tipo de Enumerador a usar será
un IEnumerator, con un campo Current para saber el valor actual:
/*---------------------------*/
/* Ejemplo en C# */
/* pila2.cs */
/* */
/* Ejemplo de clase "Stack" */
/* y enumerador */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
que escribiría
Revisión 0.95 – Página 175
Introducción a la programación con C#, por Nacho Cabanes
Contenido:
yo
soy
Hola,
Nota: los "enumeradores" existen también en otras plataformas, como Java, aunque allí reciben
el nombre de "iteradores".
Se puede saber más sobre las estructuras dinámicas que hay disponibles en la plataforma .Net
consultando la referencia en línea de MSDN (mucha de la cual está sin traducir al español):
http://msdn.microsoft.com/es-es/library/system.collections(en-us,VS.71).aspx#
• Utilizamos internamente un array más grande que la cantidad de datos que esperemos
que vaya a almacenar la pila.
• Creamos una función "Apilar", que añade en la primera posición libre del array
(inicialmente la 0) y después incrementa esa posición, para que el siguiente dato se
introduzca a continuación.
• Creamos también una función "Desapilar", que devuelve el dato que hay en la última
posición, y que disminuye el contador que indica la posición, de modo que el siguiente
dato que se obtuviera sería el que se introdujo con anterioridad a éste.
/*---------------------------*/
/* Ejemplo en C# */
/* pilaEstatica.cs */
/* */
/* Ejemplo de clase "Pila" */
/* basada en un array */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
using System.Collections;
string[] datosPila;
int posicionPila;
const int MAXPILA = 200;
string palabra;
// Constructor
public PilaString() {
posicionPila = 0;
datosPila = new string[MAXPILA];
}
} // Fin de la clase
Ejercicios propuestos:
• Usando esta misma estructura de programa, crear una clase "Cola", que permita
introducir datos y obtenerlos en modo FIFO (el primer dato que se introduzca debe ser
el primero que se obtenga). Debe tener un método "Encolar" y otro "Desencolar".
• Crear una clase "ListaOrdenada", que almacene un único dato (no un par clave-valor
como los SortedList). Debe contener un método "Insertar", que añadirá un nuevo dato
en orden en el array, y un "Extraer(n)", que obtenga un elemento de la lista (el número
"n").
Es decir, pondremos un asterisco entre el tipo de datos y el nombre de la variable. Ese asterisco
puede ir junto a cualquiera de ambos, también es correcto escribir
int *posicion;
El valor que guarda "posicion" es una dirección de memoria. Generalmente no podremos hacer
cosas como posicion=5; porque nada nos garantiza que la posición 5 de la memoria esté
disponible para que nosotros la usemos. Será más habitual que tomemos una dirección de
memoria que ya contiene otro dato, o bien que le pidamos al compilador que nos reserve un
espacio de memoria (más adelante veremos cómo).
posicion = №
Es más, si intentamos compilar obtendremos un mensaje de error, que nos dice que si nuestro
código no es seguro, deberemos compilarlo con la opción "unsafe":
Revisión 0.95 – Página 178
Introducción a la programación con C#, por Nacho Cabanes
mcs unsafe1.cs
unsafe1.cs(15,31): error CS0227: Unsafe code requires the `unsafe' command line
option to be specified
Compilation failed: 1 error(s), 0 warnings
Por tanto, deberemos compilar con la opción /unsafe como forma de decir al compilador "sí, sé
que este programa tiene zonas no seguras, pero aun así quiero compilarlo":
/*---------------------------*/
/* Ejemplo en C# */
/* unsafe1.cs */
/* */
/* Ejemplo de punteros (1) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// Damos un valor a x
x = 2;
// punteroAEntero será la dirección de memoria de x
punteroAEntero = &x;
// Los dos están en la misma dirección:
Console.WriteLine("x vale {0}", x);
Console.WriteLine("En punteroAEntero hay un {0}", *punteroAEntero);
x vale 2
En punteroAEntero hay un 2
x vale 5
En punteroAEntero hay un 5
/*---------------------------*/
/* Ejemplo en C# */
/* unsafe2.cs */
/* */
/* Ejemplo de punteros (2) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// Y mostramos el resultado
Console.WriteLine (i);
}
}
/*---------------------------*/
/* Ejemplo en C# */
/* unsafe3.cs */
/* */
/* Ejemplo de punteros (3) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// Rellenamos el array
for (int i = 0; i < tamanyoArray; i++)
{
datos[i] = i*10;
}
// Mostramos el array
for (int i = 0; i < tamanyoArray; i++)
{
Console.WriteLine(datos[i]);
}
}
}
Existen ciertas diferencias entre esta forma de trabajar y la que ya conocíamos: la memoria se
reserva en la pila (stack), en vez de hacerlo en la zona de memoria "habitual", conocida como
"heap" o montón, pero es un detalle sobre el que no vamos a profundizar.
/*---------------------------*/
/* Ejemplo en C# */
/* unsafe4.cs */
/* */
/* Ejemplo de punteros (4) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
El resultado sería algo parecido (porque las direcciones de memoria que obtengamos no tienen
por qué ser las mismas) a:
10
20
30
40
Por eso, en ciertas ocasiones el compilador puede avisarnos de que algunas asignaciones son
peligrosas y obligar a que usemos la palabra "fixed" para indicar al compilador que esa zona
"no debe limpiarse automáticamente".
/*---------------------------*/
/* Ejemplo en C# */
/* unsafe5.cs */
/* */
/* Ejemplo de punteros (5) */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
Como se ve en el programa anterior, en una zona "fixed" no se puede modificar el valor de esa
variables; si lo intentamos recibiremos un mensaje de error que nos avisa de que esa variable
es de "sólo lectura" (read-only). Por eso, para cambiarla, tendremos que empezar otra nueva
zona "fixed".
La idea detrás de ese "using" es que puede ocurrir que distintos programadores en distintos
puntos del mundo creen funciones o clases que se llamen igual, y, si se mezclan fuentes de
distintas procedencias, esto podría dar lugar a programas que no compilaran correctamente, o,
peor aún, que compilaran pero no funcionaran de la forma esperada.
Por eso, se recomienda usar "espacios de nombres", que permitan distinguir unos de otros. Por
ejemplo, si yo quisiera crear mi propia clase "Console" para el acceso a la consola, o mi propia
clase "Random" para manejo de números aleatorios, lo razonable es crear un nuevo espacio de
nombres.
De hecho, con entornos como SharpDevelop o Visual Studio, cuando creamos un nuevo
proyecto, el fuente "casi vacío" que se nos propone contendrá un espacio de nombres que se
llamará igual que el proyecto. Esta es apariencia del fuente si usamos VisualStudio 2008:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
}
}
}
using System;
namespace PruebaDeNamespaces
{
class Program
{
public static void Main(string[] args)
{
Console.WriteLine("Hello World!");
Vamos a un ejemplo algo más avanzado, que contenga un espacio de nombres, que cree una
nueva clase Console y que utilice las dos (la nuestra y la original, de System):
/*---------------------------*/
/* Ejemplo en C# */
/* namespaces.cs */
/* */
/* Ejemplo de espacios de */
/* nombres */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
namespace ConsolaDeNacho {
public class Console
{
public static void WriteLine(string texto)
{
System.Console.ForegroundColor = ConsoleColor.Blue;
System.Console.WriteLine("Mensaje: "+texto);
}
}
}
Como se puede ver, este ejemplo tiene dos clases Console, y ambas tienen un método
WriteLine. Una es la original de C#, que invocaríamos con "System.Console". Otra es la que
hemos creado para el ejemplo, que escribe un texto modifica y en color (ayudándose de
System.Console), y que llamaríamos mediante "ConsolaDeNacho.Console". El resultado es que
podemos tener dos clases Console accesibles desde el mismo programa, sin que existan
conflictos entre ellas. El resultado del programa sería:
Hola
Mensaje: Hola otra vez
/*---------------------------*/
/* Ejemplo en C# */
/* bits.cs */
/* */
/* Operaciones entre bits */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
}
}
El resultado es:
La variable a vale 67
y b vale 33
El complemento de a es: -68
El producto lógico de a y b es: 1
Su suma lógica es: 99
Su suma lógica exclusiva es: 98
Para comprobar que es correcto, podemos convertir al sistema binario esos dos números y
seguir las operaciones paso a paso:
67 = 0100 0011
33 = 0010 0001
Después hacemos el producto lógico de A y B, multiplicando cada bit, de modo que 1*1 = 1,
1*0 = 0, 0*0 = 0
0000 0001 = 1
Después hacemos su suma lógica, sumando cada bit, de modo que 1+1 = 1, 1+0 = 1, 0+0 =
0
0110 0011 = 99
La suma lógica exclusiva devuelve un 1 cuando los dos bits son distintos: 1^1 = 0, 1^0 = 1,
0^0 = 0
0110 0010 = 98
Desplazar los bits una posición a la izquierda es como multiplicar por dos:
Desplazar los bits una posición a la derecha es como dividir entre dos:
0010 0001 = 33
¿Qué utilidades puede tener todo esto? Posiblemente, más de las que parece a primera vista.
Por ejemplo: desplazar a la izquierda es una forma muy rápida de multiplicar por potencias de
dos; desplazar a la derecha es dividir por potencias de dos; la suma lógica exclusiva (xor) es un
método rápido y sencillo de cifrar mensajes; el producto lógico nos permite obligar a que
ciertos bits sean 0 (algo que se puede usar para comprobar máscaras de red); la suma lógica,
por el contrario, puede servir para obligar a que ciertos bits sean 1...
x += 2;
x <<= 2;
x &= 2;
x |= 2;
...
9.3. Enumeraciones
Cuando tenemos varias constantes, cuyos valores son números enteros, hasta ahora estamos
dando los valores uno por uno, así:
Hay una forma alternativa de hacerlo, especialmente útil si son números enteros consecutivos.
Se trata de enumerarlos:
(Al igual que las constantes de cualquier otro tipo, se puede escribir en mayúsculas para
recordar "de un vistazo" que son constantes, no variables)
La primera constante valdrá 0, y las demás irán aumentando de una en una, de modo que en
nuestro caso valen:
Si queremos que los valores no sean exactamente estos, podemos dar valor a cualquiera de las
contantes, y las siguientes irán aumentando de uno en uno. Por ejemplo, si escribimos
/*---------------------------*/
/* Ejemplo en C# */
/* enum.cs */
/* */
/* Ejemplo de enumeraciones */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
diasSemana.MIERCOLES);
Console.WriteLine("que equivale a: {0}",
(int) diasSemana.MIERCOLES);
}
}
y su resultado será:
Nosotros hemos usado enumeraciones muchas veces hasta ahora, sin saber realmente que lo
estábamos haciendo. Por ejemplo, el modo de apertura de un fichero (FileMode) es una
enumeración, por lo que escribimos FileMode.Open. También son enumeraciones los códigos de
color de la consola (como ConsoleColor.Red) y las teclas de la consola (como
ConsoleKey.Escape).
Nota: las enumeraciones existen también en otros lenguajes como C y C++, pero la sintaxis es
ligeramente distinta: en C# es necesario indicar el nombre de la enumeración cada vez que se
usen sus valores (como en diasSemana.MIERCOLES), mientras que en C se usa sólo el valor
(MIERCOLES).
9.4. Propiedades
Hasta ahora estábamos siguiendo la política de que los atributos de una clase sean privados, y
se acceda a ellos a través de métodos "get" (para leer su valor) y "set" (para cambiarlo). En el
caso de C#, existe una forma alternativa de conseguir el mismo efecto, empleando las llamadas
"propiedades", que tienen una forma abreviada de escribir sus métodos "get" y "set":
/*---------------------------*/
/* Ejemplo en C# */
/* propiedades.cs */
/* */
/* Ejemplo de propiedades */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
// -------------------------
// -------------------------
set
{
anchura = value;
}
}
// -------------------------
// El "Main" de prueba
public static void Main()
{
EjemploPropiedades ejemplo
= new EjemploPropiedades();
ejemplo.SetAltura(5);
Console.WriteLine("La altura es {0}",
ejemplo.GetAltura() );
ejemplo.Anchura = 6;
Console.WriteLine("La anchura es {0}",
ejemplo.Anchura );
}
}
Al igual que ocurría con las enumeraciones, ya hemos usado "propiedades" anteriormente, sin
saberlo: la longitud ("Length") de una cadena, el tamaño ("Length") y la posición actual
("Position") en un fichero, el título ("Title") de una ventana en consola, etc.
Una curiosidad: si una propiedad tiene un "get", pero no un "set", será una propiedad de sólo
lectura, no podremos hacer cosas como "Anchura = 4", porque el programa no compilaría. De
igual modo, se podría crear una propiedad de sólo escritura, definiendo su "set" pero no su
"get".
Vamos a ver solamente un ejemplo con un caso habitual: comprobar si una cadena es
numérica, alfabética o alfanumérica. Las ideas básicas son:
se usa para indicar caracteres de control, la forma correcta de escribir esa expresión
regular sería: @"\A[0-9]*\z"
• Alternativamente, se suele poder plantear muchas expresiones de otra forma más
sencilla: una cifra numérica es aquella que no contiene nada que no esté entre el cero y
el nueve, y esto se escribiría así: "[^0-9]"
using System;
using System.Text.RegularExpressions;
class PruebaExprRegulares
{
public static bool EsNumeroEntero(String cadena)
{
Regex patronNumerico = new Regex(@"\A[0-9]+\z");
return patronNumerico.IsMatch(cadena);
}
if (EsNumeroEntero("1942"))
Console.WriteLine("1942 es un número entero");
else
Console.WriteLine("1942 NO es un número entero");
Revisión 0.95 – Página 193
Introducción a la programación con C#, por Nacho Cabanes
if (EsNumeroEntero2("1942"))
Console.WriteLine("1942 es entero (forma 2)");
else
Console.WriteLine("1942 NO es entero (forma 2)");
if (EsNumeroEntero("23,45"))
Console.WriteLine("23,45 es un número entero");
else
Console.WriteLine("23,45 NO es un número entero");
if (EsNumeroEntero2("23,45"))
Console.WriteLine("23,45 es entero (forma 2)");
else
Console.WriteLine("23,45 NO es entero (forma 2)");
if (EsNumeroConDecimales("23,45"))
Console.WriteLine("23,45 es un número con decimales");
else
Console.WriteLine("23,45 NO es un número con decimales");
if (EsNumeroConDecimales("23,45,67"))
Console.WriteLine("23,45,67 es un número con decimales");
else
Console.WriteLine("23,45,67 NO es un número con decimales");
if (EsAlfabetico("hola"))
Console.WriteLine("hola es alfabetico");
else
Console.WriteLine("hola NO es alfabetico");
if (EsAlfanumerico("hola1"))
Console.WriteLine("hola1 es alfanumerico");
else
Console.WriteLine("hola1 NO es alfanumerico");
}
}
Como curiosidad, nuestra rutina de validar números con decimales comprueba la coma (,) y no
el punto (.), no sólo porque es el símbolo de puntuación utilizado en España, sino además
porque es más sencillo de comprobar, ya que el punto es un símbolo reservado en las
expresiones decimales, y quiere decir "cualquier carácter".
Las expresiones regulares son algo complejo. Por una parte, su sintaxis puede llegar a ser difícil
de seguir. Por otra parte, el manejo en C# no se limita a buscar, sino que también permite
otras operaciones, como reemplazar unas expresiones por otras. Como ver muchos más
detalles podría hacer el texto demasiado extenso, puede ser recomendable ampliar información
usando la página web de MSDN (Microsoft Developer Network):
http://msdn.microsoft.com/es-es/library/system.text.regularexpressions.regex(VS.80).aspx
/*---------------------------*/
/* Ejemplo en C# */
/* coma.cs */
/* */
/* Operador coma */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
i vale 0 y j vale 0.
i vale 1 y j vale 1.
i vale 2 y j vale 2.
i vale 3 y j vale 3.
i vale 4 y j vale 4.
i vale 5 y j vale 5.
Nota: En el lenguaje C se puede "rizar el rizo" todavía un poco más: la condición de terminación
también podría tener una coma, y entonces no se sale del bucle "for" hasta que se cumplen las
Revisión 0.95 – Página 195
Introducción a la programación con C#, por Nacho Cabanes
dos condiciones (algo que no es válido en C#, ya que la condición debe ser un "Bolean", y la
coma no es un operador válido para operaciones booleanas):
• Delegados (delegate).
• Indexadores.
• Colecciones genéricas.
• …
/*---------------------------*/
/* Ejemplo en C# */
/* consola.cs */
/* */
/* Más posibilidades de */
/* "System.Console" */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
Console.ForegroundColor = ConsoleColor.Blue;
Console.SetCursorPosition(10, 15);
Console.Write("Pulsa 1 o 2: ");
ConsoleKeyInfo tecla;
do
{
tecla = Console.ReadKey(false);
}
while ((tecla.KeyChar != '1') && (tecla.KeyChar != '2'));
}
}
(Nota: si se prueba este fuente desde Mono, habrá que compilar con "gmcs" en vez de con
"mcs", para compilar usando la versión 2.x de la plataforma .Net, no la 1.x, que no tenía estas
posibilidades).
Para comprobar el valor de una tecla, como se ve en el ejemplo anterior, tenemos que usar
una variable de tipo "ConsoleKeyInfo" (información de tecla de consola). Un ConsoleKeyInfo
tiene campos como:
• KeyChar, que representa el carácter que se escribiría al pulsar esa tecla. Por ejemplo,
podríamos hacer if (tecla.KeyChar == '1') ...
• Key, que se refiere a la tecla (porque hay teclas que no tienen un carácter visualizable,
como F1 o las teclas de cursor). Por ejemplo, para comprobar la tecla ESC podríamos
hacer if (tecla.Key == ConsoleKey.Escape) ... . Algunos de los códigos de tecla
disponibles son:
o Teclas de edición y control como, como: Backspace (Tecla RETROCESO), Tab
(Tecla TAB), Clear (Tecla BORRAR), Enter (Tecla ENTRAR), Pause (Tecla
PAUSA), Escape (Tecla ESC (ESCAPE)), Spacebar (Tecla BARRA
ESPACIADORA), PrintScreen (Tecla IMPR PANT), Insert (Tecla INS (INSERT)),
Delete (Tecla SUPR (SUPRIMIR))
o Teclas de movimiento del cursor, como: PageUp (Tecla RE PÁG), PageDown
(Tecla AV PÁG), End (Tecla FIN), Home (Tecla INICIO), LeftArrow (Tecla
FLECHA IZQUIERDA), UpArrow (Tecla FLECHA ARRIBA), RightArrow (Tecla
FLECHA DERECHA), DownArrow (Tecla FLECHA ABAJO)
o Teclas alfabéticas, como: A (Tecla A), B, C, D, E, F, G, H, I, J, K, L, M, N, O, P,
Q, R, S, T, U, V, W, X, Y, Z
o Teclas numéricas, como: D0 (Tecla 0), D1, D2, D3, D4, D5, D6, D7, D8, D9
Los colores que tenemos disponibles (y que se deben escribir precedidos con "ConsoleColor")
son: Black (negro), DarkBlue (azul marino), DarkGreen (verde oscuro) DarkCyan (verde
azulado oscuro), DarkRed (rojo oscuro), DarkMagenta (fucsia oscuro o púrpura), DarkYellow
(amarillo oscuro u ocre), Gray (gris), DarkGray (gris oscuro), Blue (azul), Green (verde), Cyan
(aguamarina o verde azulado claro), Red (rojo), Magenta (fucsia), Yellow (amarillo), White
(blanco).
La forma más cómoda de conseguirlo es usando herramientas que incluyan un editor visual,
como Visual Studio o SharpDevelop.
A pesar de que existen versiones gratuitas de Visual Studio, vamos a ver el caso de
SharpDevelop , que necesita un ordenador menos potente y tiene un manejo muy similar (en el
Apéndice 4 tienes los cambios para Visual Studio).
Cuando entramos a SharpDevelop, diríamos que queremos crear una "nueva solución", y en el
menú escogeríamos "Aplicación Windows":
Debajo de la ventana de código hay una pestaña llamada "Diseño", que nos permite acceder al
diseñador visual:
Para poder incluir botones y otros elementos visuales, debemos escoger la ventana de
"Herramientas" en la parte inferior de la pantalla, y luego el panel "Windows Forms" en esta
ventana:
Para incluir un botón, podemos hacer clic en el elemento "Button" del panel izquierdo, y luego
hacer un clic en la parte de la ventana en la que queremos que aparezca. De igual modo,
podríamos añadir otros elementos.
Por ejemplo, vamos a añadir un botón (Button) y una etiqueta de texto (Label), para hacer un
primer programa que cambie el texto de la etiqueta cuando pulsemos el botón.
Las propiedades de cada uno de estos elementos aparecen en la parte derecha, en una nueva
ventana. Estas propiedades las podremos cambiar directamente en ese panel, o bien desde
código. Algunas de esas propiedades son:
Si queremos que al pulsar el botón cambie el texto, tendremos que modificar el código que
corresponde al "evento" de pulsación del botón. Lo conseguimos simplemente haciendo doble
clic en el botón, y aparece este texto:
Dentro de ese método escribiremos lo que queremos que ocurra al hacer clic en el botón. Por
ejemplo, para que el texto de la etiqueta "label1" pase a ser "Hola", haríamos:
Ejercicios propuestos:
• Un programa que muestre una ventana con 3 recuadros de texto (TextBox) y un botón
"Calcular suma". Cuando se pulse el botón, en el tercer recuadro deberá aparecer la
suma de los números introducidos en los dos primeros recuadros.
• Un programa que muestre una ventana con un recuadro de texto y 3 etiquetas. En el
recuadro de texto se escribirá un número (en sistema decimal). Cada vez que en el
recuadro de texto se pulse una tecla, se mostrará en las 3 etiquetas de texto el
equivalente de ese número en binario, octal y hexadecimal. Pista: en la ventana de
propiedades, se deberá escoger "Eventos", para ver las posibles acciones relacionadas
con el TextBox.
La segunda variante es indicar además qué botones queremos mostrar, y qué iconos de aviso:
Y la tercera variante permite indicar además cual será el botón por defecto:
Como se ve en estos ejemplos, tenemos algunos valores predefinidos para indicar qué botones
o iconos queremos mostrar:
Si queremos que el usuario responda tecleando, no tenemos ninguna ventana predefinida que
nos lo permita (sí existe un "InputBox" en otros entornos de programación, como Visual Basic),
así que deberíamos crear nosotros esa ventana de introducción de datos desde el editor visual
o mediante código, elemento por elemento.
Vamos a crear una ventana principal que tenga una casilla de texto (TextBox) en la que no se
podrá escribir, sino que sólo se mostrarán resultados, y un botón que hará que aparezca la
ventana secundaria, en la que sí podremos introducir datos.
Para crear el segundo formulario (la ventana auxiliar), usamos la opción "Agregar / Nuevo
Elemento", del menú "Proyecto":
En esta nueva ventana, entramos a la vista de diseño y añadimos dos casillas de texto
(TextBox), con sus etiquetas aclaratorias (Label), y el botón de Aceptar:
Llega el momento de añadir el código a nuestro programa. Por una parte, haremos que el
botón "Aceptar" cierre la ventana. Lo podemos conseguir con doble clic, para que nos aparezca
la función que se lanzará con el suceso Click del botón (cuando se pulse el ratón sobre él), y
añadimos la orden "Close()" en el cuerpo de esa función, así:
Además, para que desde la ventana principal se puedan leer los datos de ésta, podemos hacer
que sus componentes sean públicos, o, mejor, crear un método "Get" que devuelva el
contenido de estos componentes. Por ejemplo, podemos devolver el nombre y el apellido como
parte de una única cadena de texto, así:
Ya sólo falta que desde la ventana principal se muestre la ventana secundaria y se lean los
valores al terminar. Para eso, añadimos un atributo en la ventana principal, que represente la
ventana auxiliar:
public MainForm()
{
InitializeComponent();
ventanaIntro = new Introduccion();
}
Finalmente, en el suceso Click del botón hacemos que se muestre la ventana secundaria usando
ShowDialog, que espera a que se cierre ésta antes de permitirnos seguir trabajando en la
ventana principal, y después leemos el valor que se había tecleado en dicha ventana:
// Dibujamos
ventanaGrafica.DrawLine(contornoRojo, 200, 100, 300, 400);
ventanaGrafica.FillEllipse(rellenoAzul, new Rectangle(0, 0, 200, 300));
Los métodos para dibujar líneas, rectángulos, elipses, curvas, etc. son parte de la clase
Graphics. Algunos de los métodos que ésta contiene y que pueden ser útiles para realizar
dibujos sencillos son:
Esta imagen debería estar en la carpeta del programa ejecutable (que quizá no sea la misma
que el fuente), y puede estar en formato BMP, GIF, PNG, JPG o TIFF.
Se puede encontrar más detalles en la referencia en línea (MSDN), por ejemplo en la página
http://msdn.microsoft.com/es-es/library/system.drawing.graphics_methods.aspx
Dentro de ese tipo de datos DateTime, tenemos las herramientas para saber el día (Day), el
mes (Month) o el año (Year) de una fecha, entre otros. También podemos calcular otras fechas
sumando a la actual una cierta cantidad de segundos (AddSeconds), días (AddDays), etc. Un
ejemplo básico de su uso sería:
/*---------------------------*/
/* Ejemplo en C# */
/* fechas.cs */
/* */
/* Ejemplo básico de */
/* manejo de fechas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
class ejemploFecha
{
Algunas de las propiedades más útiles son: Now (fecha y hora actual de este equipo), Today
(fecha actual); Day (día del mes), Month (número de mes), Year (año); Hour (hora), Minute
(minutos), Second (segundos), Millisecond (milisegundos); DayOfWeek (día de la semana: su
nombre en inglés); DayOfYear (día del año).
Y para calcular nuevas fechas, podemos usar métodos que generan un nuevo objeto DateTime,
como: AddDays , AddHours, AddMilliseconds, AddMinutes, AddMonths, AddSeconds, AddHours,
o bien un Add más genérico (para sumar una fecha a otra) y un Subtract también genérico
(para restar una fecha de otra).
Cuando restamos dos fechas, obtenemos un dato de tipo "intervalo de tiempo" (TimeSpan), del
que podemos saber detalles como la cantidad de días (sin decimales, Days, o con decimales,
TotalDays), como se ve en este ejemplo:
/*---------------------------*/
/* Ejemplo en C# */
/* restarfechas.cs */
/* */
/* Ejemplo ampliado de */
/* manejo de fechas */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using System;
class ejemploFecha
{
public static void Main()
{
DateTime fechaActual = DateTime.Now;
DateTime fechaNacimiento = new DateTime(1990, 9, 18);
TimeSpan diferencia =
fechaActual.Subtract(fechaNacimiento) ;
}
}
También podemos hacer una pausa en la ejecución: Si necesitamos que nuestro programa se
detenga una cierta cantidad de tiempo, no hace falta que usemos un "while" que compruebe la
hora continuamente, sino que podemos "bloquear" (Sleep) el subproceso (o hilo, "Thread") que
representa nuestro programa una cierta cantidad de milésimas de segundo con:
Thread.Sleep(5000);
También tenemos un método "GetFiles" que nos permite obtener la lista de ficheros que
contiene un directorio. Así, podríamos listar todo el contenido de un directorio con:
listaFicheros = Directory.GetFiles(miDirectorio);
foreach(string fich in listaFicheros)
Console.WriteLine(fich);
(la clase Directory está declarada en el espacio de nombres System.IO, por lo que deberemos
añadirlo entre los "using" de nuestro programa).
En los actuales sistemas operativos multitarea se da por sentado que no es necesario esperar a
que termine otra la tarea, sino que nuestro programa puede proseguir. Si aun así, queremos
esperar a que se complete la otra tarea, lo conseguiríamos con "WaitForExit", añadiendo esta
segunda línea:
proc.WaitForExit();
Para acceder a SQLite desde C#, tenemos disponible alguna adaptación de la biblioteca
original. Una de ellas de es System.Data.SQLite, disponible en http://sqlite.phxsoftware.com/
Con ella, los pasos a seguir para leer información desde una base de datos serían:
using System;
using System.Data.SQLite; //Utilizamos la DLL
// Creamos la conexion a la BD
// El Data Source contiene la ruta del archivo de la BD
SQLiteConnection conexion =
new SQLiteConnection
("Data Source=prueba.sqlite;Version=3;New=False;Compress=True;");
conexion.Open();
Deberemos compilar con la versión 2 (o superior) de la plataforma .Net. En Mono, esto supone
emplear el compilador "gmcs" en vez de "mcs":
mono pruebaSQLite.exe
La base de datos de prueba se podría haber creado previamente desde el propio entorno de
SQLite, o con algún gestor auxiliar, como la extensión de Firefox llamada SQLite Manager o
como la utilidad SQLite Database Browser.
Otra alternativa es utilizar un fuente similar, que creara una base de datos y que introdujera en
ella algún dato, lo que se podría hacer así:
using System;
using System.Data.SQLite; //Utilizamos la DLL
{
public static void Main()
{
Console.WriteLine("Creando la base de datos...");
// Creamos la tabla
string creacion = "CREATE TABLE gente "
+"(codigo INT PRIMARY KEY, nombre VARCHAR(40), edad INT);";
SQLiteCommand cmd = new SQLiteCommand(creacion, conexion);
cmd.ExecuteNonQuery();
Console.WriteLine("Creada.");
}
}
Nota: el primer ejemplo daba por sentando que en la tabla "gente" existían dos campos (o
más), de los cuales el primero era el "nombre" y el segundo era la "edad"; en este segundo
ejemplo, estos son el segundo y el tercer dato, respectivamente, porque el primero es un
"código", incluido a modo de ejemplo de cómo se puede crear una clave primaria.
Tao.SDL es una adaptación de esta librería, que permite emplearla desde C#. Se puede
descargar desde http://www.mono-project.com/Tao
SDL no es una librería especialmente sencilla, y tampoco lo acaba de ser Tao.SDL, así que los
fuentes siguientes pueden resultar difíciles de entender, a pesar de realizar tareas muy básicas.
Por eso, muchas veces es preferible "ocultar" los detalles de SDL creando nuestras propias
clases "Hardware", "Imagen", etc. Puedes ver un ejemplo de esa forma de trabajar aquí:
http://www.nachocabanes.com/videojuegos/deathpit/
Vamos a ver un primer ejemplo, básico pero completo, que muestre cómo entrar a modo
gráfico, cargar una imagen, dibujarla en pantalla, esperar cinco segundos y volver al sistema
operativo. Tras el fuente comentaremos las principales funciones.
/*---------------------------*/
/* Ejemplo en C# */
/* sdl01.cs */
/* */
/* Primer acercamiento */
/* a SDL */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using Tao.Sdl;
using System; // Para IntPtr (puntero: imágenes, etc)
// Inicializamos SDL
Sdl.SDL_Init(Sdl.SDL_INIT_EVERYTHING);
pantallaOculta = Sdl.SDL_SetVideoMode(
anchoPantalla,
altoPantalla,
bitsColor,
flags);
Environment.Exit(4);
}
// Dibujamos la imagen
short x = 400;
short y = 300;
short anchoImagen = 50;
short altoImagen = 50;
Sdl.SDL_Rect origen = new Sdl.SDL_Rect(0,0,anchoImagen,altoImagen);
Sdl.SDL_Rect dest = new Sdl.SDL_Rect(x,y,anchoImagen,altoImagen);
Sdl.SDL_BlitSurface(imagen, ref origen, pantallaOculta, ref dest);
// Y esperamos 5 segundos
System.Threading.Thread.Sleep( 5000 );
• SDL_Init es la rutina de inicialización, que entra a modo gráfico, con cierto ancho y alto
de pantalla, cierta cantidad de colores y ciertas opciones adicionales.
• El tipo SDL_Rect define un "rectángulo" a partir de su origen (x e y), su ancho y su
alto, y se usa en muchas operaciones.
• SDL_SetClipRect indica la zona de recorte (clipping) del tamaño de la pantalla, para que
no tengamos que preocuparnos por si dibujamos una imagen parcialmente (o
completamente) fuera de la pantalla visible.
• SDL_LoadBMP carga una imagen en formato BMP (si sólo usamos SDL "puro", no
podremos emplear otros tipos que permitan menores tamaños, como el JPG, o
transparencia, como el PNG). El tipo de dato que se obtiene es un "IntPtr" (del que no
daemos más detalles), y la forma de comprobar si realmente se ha podido cargar la
imagen es mirando si el valor obtenido es IntPtr.Zero (y en ese caso, no se habría
podido cargar) u otro distinto (y entonces la imagen se habría leido sin problemas).
• SDL_BlitSurface vuelca un rectángulo (SDL_Rect) sobre otro, y lo usamos para ir
dibujando todas las imágenes en una pantalla oculta, y finalmente volcar toda esa
pantalla oculta a la pantalla visible, con lo que se evitan parpadeos (es una técnica que
se conoce como "doble buffer").
• SDL_Flip vuelca esa pantalla oculta a la pantalla visible (el paso final de ese "doble
buffer").
• Para la pausa no hemos usado ninguna función de SDL, sino Thread.Sleep, que ya
habíamos comentado en el apartado sobre temporización.
• SDL_Quit libera los recursos (algo que generalmente haríamos desde un destructor).
Si no vamos a usar imágenes comprimidas (PNG o JPG), ni tipos de letra TTF, ni sonidos en
formato MP3, ni funciones adicionales de dibujo (líneas, recuadros, círculos, etc). No
necesitaremos ninguna DLL adicional.
Un segundo ejemplo algo más detallado podría permitirnos mover el personaje con las flechas
del teclado, a una velocidad de 50 fotogramas por segundo, así:
/*---------------------------*/
/* Ejemplo en C# */
/* sdl02.cs */
/* */
/* Segundo acercamiento */
/* a SDL */
/* */
/* Introduccion a C#, */
/* Nacho Cabanes */
/*---------------------------*/
using Tao.Sdl;
using System; // Para IntPtr (puntero: imágenes, etc)
// Inicializamos SDL
Sdl.SDL_Init(Sdl.SDL_INIT_EVERYTHING);
pantallaOculta = Sdl.SDL_SetVideoMode(
anchoPantalla,
altoPantalla,
bitsColor,
flags);
// Parte repetitiva
bool terminado = false;
short x = 400;
short y = 300;
short anchoImagen = 50;
short altoImagen = 50;
Sdl.SDL_Event suceso;
int numkeys;
byte[] teclas;
do
{
// Comprobamos sucesos
Sdl.SDL_PollEvent(out suceso);
teclas = Sdl.SDL_GetKeyState(out numkeys);
// Borramos pantalla
Sdl.SDL_Rect origen = new Sdl.SDL_Rect(0,0,
anchoPantalla,altoPantalla);
Sdl.SDL_FillRect(pantallaOculta, ref origen, 0);
// Dibujamos en sus nuevas coordenadas
origen = new Sdl.SDL_Rect(0,0,anchoImagen,altoImagen);
Sdl.SDL_Rect dest = new Sdl.SDL_Rect(x,y,
anchoImagen,altoImagen);
Revisión 0.95 – Página 219
Introducción a la programación con C#, por Nacho Cabanes
// Y esperamos 20 ms
System.Threading.Thread.Sleep( 20 );
} while (!terminado);
En un programa real, esta estructura puede ser engorrosa y resultar difícil de leer. Por eso,
suele ser preferible crear nuestras propias funciones que la oculten un poco: funciones como
"Inicializar", "CargarImagen", "ComprobarTeclas", "DibujarImagenes", etc.
(Nota: en este apartado se asumirá que el lector entiende algunos conceptos básicos de redes,
como qué es una dirección IP, qué significa "localhost" o qué diferencias hay entre el protocolo
TCP y el UDP).
Como primer ejemplo, vamos a ver cómo podríamos recibir una página web (por ejemplo, la
página principal de "www.nachocabanes.com"), línea a línea como si se tratara de un fichero de
texto (StreamReader), y mostrar sólo las líneas que contengan un cierto texto (por ejemplo, la
palabra "Pascal"):
using System;
using System.IO; // Para Stream
using System.Net; // Para System.Net.WebClient
Otra posibilidad que tampoco es complicada (aunque sí algo más que ésta última) es la de
comunicar dos ordenadores, para enviar información desde uno y recibirla desde el otro. Se
puede hacer de varias formas. Una de ellas es usando directamente el protocolo TCP:
emplearemos un TcpClient para enviar y un TcpListener para recibir, y en ambos casos
trataremos los datos como un tipo especial de fichero binario, un NetworkStream:
using System;
using System.IO; // Para Stream
using System.Text; // Para Encoding
using System.Net; // Para Dns, IPAddress
using System.Net.Sockets; // Para NetworkStream
conexion.Write(secuenciaLetras, 0, secuenciaLetras.Length);
conexion.Close();
cliente.Close();
}
cliente.Close();
listener.Stop();
return frase;
}
1
Esperando...
Entonces lanzaríamos otra sesión del mismo programa en el mismo ordenador (porque estamos
conectando a la dirección "localhost"), y en esta nueva sesión escogeríamos la opción de Enviar
(2):
Prueba de texto
Recibido
Enviando... Enviado
Esta misma idea se podría usar como base para programas más elaborados, que comunicaran
diferentes equipos (en este caso, la dirección no sería "localhost", sino la IP del otro equipo),
como podría ser un chat o cualquier juego multijugador en el que hubiera que avisar a otros
jugadores de los cambios realizados por cada uno de ellos.
Esto se puede conseguir a un nivel algo más alto, usando los llamados "Sockets" en vez de los
TcpClient, o de un modo "no fiable", usando el protocolo UDP en vez de TCP, pero nosotros no
veremos más detalles de ninguno de ambos métodos.
Para eliminar esos fallos que hacen que un programa no se comporte como debería, se usan
unas herramientas llamadas "depuradores". Estos nos permiten avanzar paso a paso para ver
cómo avanza realmente nuestro programa, y también nos dejan ver los valores de las variables.
using System;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
int a, b, c;
a = 5;
b = a + 2;
b -= 3;
c = -3;
c *= 2;
++c;
a *= b;
}
}
}
Para avanzar paso a paso y ver los valores de las variables, entramos al menú "Depurar". En él
aparece la opción "Paso a paso por instrucciones" (al que corresponde la tecla F11):
Si escogemos esa opción del menú o pulsamos F11, aparece una ventana inferior con la lista de
variables, y un nuevo cursor amarillo, que se queda al principio del programa:
Cada vez que pulsemos nuevamente F11 (o vayamos al menú, o al botón correspondiente de la
barra de herramientas), el depurador analiza una nueva línea de programa, muestra los valores
de las variables correspondientes, y se vuelve a quedar parado, realzando con fondo amarillo la
siguiente línea que se analizará:
Aquí hemos avanzado desde el principio del programa, pero eso no es algo totalmente habitual.
Es más frecuente que supongamos en qué zona del programa se encuentra el error, y sólo
queramos depurar una zona de programa. La forma de conseguirlo es escoger otra de las
opciones del menú de depuración: "Alternar puntos de ruptura" (tecla F9). Aparecerá una
marca granate en la línea actual:
Si ahora iniciamos la depuración del programa, saltará sin detenerse hasta ese punto, y será
entonces cuando se interrumpa. A partir de ahí, podemos seguir depurando paso a paso como
antes.
Una forma sencilla de crear una batería de pruebas es comprobando los resultados de
operaciones conocidas. Vamos a ver un ejemplo, para un programa que calcule las soluciones
de una ecuación de segundo grado.
float discriminante;
if (discriminante < 0)
{
x1 = -9999;
x2 = -9999;
}
else if (discriminante == 0)
{
x1 = - b / (2*a);
x2 = -9999;
}
else
{
x1 = (float) ((- b + Math.Sqrt(discriminante))/ 2*a);
x2 = (float) ((- b - Math.Sqrt(discriminante))/ 2*a);
}
}
Revisión 0.95 – Página 227
Introducción a la programación con C#, por Nacho Cabanes
Es decir, si alguna solución no existe, se devuelve un valor falso, que no sea fácil que se
obtenga en un caso habitual, como -9999.
x2 -1 = 0 x1 = 1, x2 = -1
x2 = 0 x1 = 0, x2 = No existe (solución única)
x2 -3x = 0 x1 = 3, x2 = 0
2x2 -2 = 0 x1 = 1, x2 = -1
using System;
Console.Write("Probando x2 - 1 = 0 ...");
SegundoGrado.Resolver((float)1, (float) 0, (float) -1,
out soluc1, out soluc2);
if ((soluc1 == 1) && (soluc2 == -1))
Console.WriteLine("OK");
else
Console.WriteLine("Falla");
Console.Write("Probando x2 = 0 ...");
SegundoGrado.Resolver((float)1, (float)0, (float)0,
out soluc1, out soluc2);
if ((soluc1 == 0) && (soluc2 == -9999))
Console.WriteLine("OK");
else
Console.WriteLine("Falla");
Vemos que en uno de los casos, la solución no es correcta. Revisaríamos los pasos que da
nuestro programa, corregiríamos el fallo y volveríamos a lanzar las pruebas automatizadas. En
este caso, el problema es que falta un paréntesis, para dividir entre (2*a), de modo que
estamos diviendo entre 2 y luego multiplicando por a.
La ventaja de crear baterías de pruebas es que es una forma muy rápida de probar un
programa, por lo que se puede aplicar tras cada pocos cambios para comprobar que todo es
correcto. El inconveniente es que NO GARANTIZA que el programa sea correcto, sino sólo que
no falla en ciertos casos. Por ejemplo, si la batería de pruebas anterior solo contuviera las tres
primeras pruebas, no habría descubierto el fallo del programa.
Por tanto, las pruebas sólo permiten asegurar que el programa falla en caso de encontrar
problemas, pero no permiten asegurar nada en caso de que no se encuentren problemas:
puede que aun así exista un fallo que no hayamos detectado.
Para crear las baterías de pruebas, lo que más ayuda es la experiencia y el conocimiento del
problema. Algunos expertos recomiendan que, si es posible, las pruebas las realice un equipo
de desarrollo distinto al que ha realizado el proyecto principal, para evitar que se omitan cosas
que se "den por supuestas".
Casi todos esos diagramas caen fuera del alcance de este texto: en una introducción a la
programación se realizan programas de pequeño tamaño, para los que no es necesaria una
gran planificación.
Aun así, hay un tipo de documentación que sí debe estar presente en cualquier problema: los
comentarios que aclaren todo aquello que no sea obvio.
Por eso, este apartado se va a centrar en algunas de las pautas que los expertos suelen
recomendar para los comentarios en los programas, y también veremos como a partir de estos
comentarios se puede generar documentación adicional de forma casi automática.
http://www.variablenotfound.com/2007/12/13-consejos-para-comentar-tu-cdigo.html
Ojo a las tabulaciones. Hay editores de texto que usan el carácter ASCII (9) y otros, lo
sustituyen por un número determinado de espacios, que suelen variar según las preferencias
personales del desarrollador. Lo mejor es usar espacios simples o asegurarse de que esto es lo
que hace el IDE correspondiente.
5. Sé correcto
Evita comentarios del tipo "ahora compruebo que el estúpido usuario no haya introducido un
número negativo", o "este parche corrige el efecto colateral producido por la patética
implementación del inepto desarrollador inicial".
Relacionado e igualmente importante: cuida la ortografía.
6. No pierdas el tiempo
No comentes si no es necesario, no escribas nada más que lo que necesites para transmitir la
idea. Nada de diseños realizados a base de caracteres ASCII, ni florituras, ni chistes, ni poesías,
ni chascarrillos.
Hay incluso quien opina que los comentarios que describen un bloque deberían escribirse antes
de codificarlo, de forma que, en primer lugar, sirvan como referencia para saber qué es lo que
hay que hacer y, segundo, que una vez codificado éstos queden como comentarios para la
posteridad. Un ejemplo:
Console.WriteLine("Resultado: " +
new Calculator()
.Set(0)
.Add(10)
.Multiply(2)
.Substract(4)
.Get()
);
Pero en algunos lenguajes modernos, como Java y C#, existe una posibilidad adicional que
puede resultar muy útil: usar comentarios que nos ayuden a crear de forma automática cierta
documentación del programa.
Esta documentación típicamente será una serie páginas HTML enlazadas, o bien varios ficheros
XML.
Por ejemplo, la herramienta (gratuita) Doxygen genera páginas como ésta a partir de un fuente
en C#:
Lo habitual es que estos tipos de comentarios se utilicen al principio de cada clase y de cada
método, así:
/**
* Personaje: uno de los tipos de elementos graficos del juego
*
* @see ElemGrafico Juego
* @author 1-DAI 2008/09
*/
Revisión 0.95 – Página 233
Introducción a la programación con C#, por Nacho Cabanes
(...)
Además, es habitual que tengamos a nuestra disposición ciertas "palabras reservadas" para
poder afinar la documentación, como @returns, que da información sobre el valor que devuelve
una función, o como @see, para detallar qué otras clases de nuestro programa están
relacionadas con la actual.
Así, comparando el fuente anterior y la documentación que se genera a partir de él, podemos
ver que:
En el lenguaje Java, documentación como esta se puede crear con la herramienta JavaDoc, que
es parte de la distribución "oficial" del lenguaje. En cambio, en C#, la herramienta estándar
genera documentación en formato XML, que puede resultar menos legible, pero se pueden
emplear alternativas gratuitas como Doxygen.
Eso sí, suele ocurrir que realmente un texto de 2000 letras que se guarde en el ordenador
ocupe más de 2000 bytes, porque se suele incluir información adicional sobre los tipos de letra
que se han utilizado, cursivas, negritas, márgenes y formato de página, etc.
Un byte se queda corto a la hora de manejar textos o datos algo más largos, con lo que se
recurre a un múltiplo suyo, el kilobyte, que se suele abreviar Kb o K.
En teoría, el prefijo kilo querría decir "mil", luego un kilobyte debería ser 1000 bytes, pero en
los ordenadores conviene buscar por comodidad una potencia de 2 (pronto veremos por qué),
por lo que se usa 210 =1024. Así, la equivalencia exacta es 1 K = 1024 bytes.
Los K eran unidades típicas para medir la memoria de ordenadores: 640 K ha sido mucho
tiempo la memoria habitual en los IBM PC y similares. Por otra parte, una página
mecanografiada suele ocupar entre 2 K (cerca de 2000 letras) y 4 K.
Cuando se manejan datos realmente extensos, se pasa a otro múltiplo, el megabyte o Mb, que
es 1000 K (en realidad 1024 K) o algo más de un millón de bytes. Por ejemplo, en un diskette
"normal" caben 1.44 Mb, y en un Compact Disc para ordenador (Cd-Rom) se pueden almacenar
hasta 700 Mb. La memoria principal (RAM) de un ordenador actual suele andar por encima de
los 512 Mb, y un disco duro actual puede tener una capacidad superior a los 80.000 Mb.
Para estas unidades de gran capacidad, su tamaño no se suele medir en megabytes, sino en el
múltiplo siguiente: en gigabytes, con la correspondencia 1 Gb = 1024 Mb. Así, son cada vez
más frecuentes los discos duros con una capacidad de 120, 200 o más gigabytes.
Y todavía hay unidades mayores, pero que aún se utilizan muy poco. Por ejemplo, un terabyte
son 1024 gigabytes.
Ejercicios propuestos:
Revisión 0.95 – Página 235
Introducción a la programación con C#, por Nacho Cabanes
Un bit es demasiado pequeño para un uso normal (recordemos: sólo puede tener dos valores: 0
ó 1), por lo que se usa un conjunto de ellos, 8 bits, que forman un byte. Las matemáticas
elementales (combinatoria) nos dicen que si agrupamos los bits de 8 en 8, tenemos 256
posibilidades distintas (variaciones con repetición de 2 elementos tomados de 8 en 8: VR2,8):
00000000
00000001
00000010
00000011
00000100
...
11111110
11111111
Por tanto, si en vez de tomar los bits de 1 en 1 (que resulta cómodo para el ordenador, pero no
para nosotros) los utilizamos en grupos de 8 (lo que se conoce como un byte), nos
encontramos con 256 posibilidades distintas, que ya son más que suficientes para almacenar
una letra, o un signo de puntuación, o una cifra numérica o algún otro símbolo.
Por ejemplo, se podría decir que cada vez que encontremos la secuencia 00000010 la
interpretaremos como una letra A, y la combinación 00000011 como una letra B, y así
sucesivamente.
También existe una correspondencia entre cada grupo de bits y un número del 0 al 255: si
usamos el sistema binario de numeración (que aprenderemos dentro de muy poco), en vez del
sistema decimal, tenemos que:
...
1111 1110 (binario) = 254 (decimal)
1111 1111 (binario) = 255 (decimal)
En la práctica, existe un código estándar, el código ASCII (American Standard Code for
Information Interchange, código estándar americano para intercambio de información), que
relaciona cada letra, número o símbolo con una cifra del 0 al 255 (realmente, con una
secuencia de 8 bits): la "a" es el número 97, la "b" el 98, la "A" el 65, la "B", el 32, el "0" el 48,
el "1" el 49, etc. Así se tiene una forma muy cómoda de almacenar la información en
ordenadores, ya que cada letra ocupará exactamente un byte (8 bits: 8 posiciones elementales
de memoria).
Aun así, hay un inconveniente con el código ASCII: sólo los primeros 127 números son
estándar. Eso quiere decir que si escribimos un texto en un ordenador y lo llevamos a otro, las
letras básicas (A a la Z, 0 al 9 y algunos símbolos) no cambiarán, pero las letras internacionales
(como la Ñ o las vocales con acentos) puede que no aparezcan correctamente, porque se les
asignan números que no son estándar para todos los ordenadores.
Nota: Eso de que realmente el ordenador trabaja con ceros y unos, por lo que le resulta más
fácil manejar los números que son potencia de 2 que los números que no lo son, es lo que
explica que el prefijo kilo no quiera decir "exactamente mil", sino que se usa la potencia de 2
más cercana: 210 =1024. Por eso, la equivalencia exacta es 1 K = 1024 bytes.
La idea de este código es que se pueda compartir información entre distintos ordenadores o
sistemas informáticos. Para ello, se hace corresponder una letra o carácter a cada secuencia de
varios bits.
El código ASCII estándar es de 7 bits, lo que hace que cada grupo de bits desde el 0000000
hasta el 1111111 (0 a 127 en decimal) corresponda siempre a la misma letra.
Por ejemplo, en cualquier ordenador que use código ASCII, la secuencia de bits 1000001 (65 en
decimal) corresponderá a la letra "A" (a, en mayúsculas).
Los códigos estándar "visibles" van del 32 al 127. Los códigos del 0 al 31 son códigos de
control: por ejemplo, el caracter 7 (BEL) hace sonar un pitido, el caracter 13 (CR) avanza de
línea, el carácter 12 (FF) expulsa una página en la impresora (y borra la pantalla en algunos
ordenadores), etc.
| 0 1 2 3 4 5 6 7 8 9 |
| 000 NUL SOH STX ETX EOT ENQ ACK BEL BS HT |
| 010 LF VT FF CR SO SI DLE DC1 DC2 DC3 |
| 020 DC4 NAK SYN ETB CAN EM SUB ESC FS GS |
| 030 RS US SP ! " # $ % & ' |
| 040 ( ) * + , - . / 0 1 |
| 050 2 3 4 5 6 7 8 9 : ; |
| 060 < = > ? @ A B C D E |
| 070 F G H I J K L M N O |
| 080 P Q R S T U V W X Y |
| 090 Z [ \ ] ^ _ ` a b c |
| 100 d e f g h i j k l m |
| 110 n o p q r s t u v w |
| 120 x y z { | } ~ • |
Hoy en día, casi todos los ordenadores incluyen un código ASCII extendido de 8 bits, que
permite 256 símbolos (del 0 al 255), lo que permite que se puedan usar también vocales
acentuadas, eñes, y otros símbolos para dibujar recuadros, por ejemplo, aunque estos símbolos
que van del número 128 al 255 no son estándar, de modo que puede que otro ordenador no los
interprete correctamente si el sistema operativo que utiliza es distinto.
Una alternativa más moderna es el estándar Unicode, que permite usar caracteres de más de 8
bits (típicamente 16 bits, que daría lugar a 65.536 símbolos distintos), lo que permite que la
transferencia de información entre distintos sistemas no tenga problemas de distintas
interpretaciones.
(aunque realmente nosotros lo hacemos automáticamente: no nos paramos a pensar este tipo
de cosas cuando sumamos o multiplicamos dos números).
Para los ordenadores no es cómodo contar hasta 10. Como partimos de "casillas de memoria"
que están completamente vacías (0) o completamente llenas (1), sólo les es realmente cómodo
contar con 2 cifras: 0 y 1.
Por eso, dentro del ordenador cualquier número se deberá almacenar como ceros y unos, y
entonces los números se deberán desglosar en potencias de 2 (el llamado "sistema binario"):
13 = 1 · 8 + 1 · 4 + 0 · 2 + 1 · 1
13 = 1 · 2 3 + 1 · 2 2 + 0 · 2 1 + 1 · 2 0
En general, convertir un número binario al sistema decimal es fácil: lo expresamos como suma
de potencias de 2 y sumamos:
Convertir un número de decimal a binario resulta algo menos intuitivo. Una forma sencilla es ir
dividiendo entre las potencias de 2, y coger todos los cocientes de las divisiones:
1 / 2 = 0 (resto: 1)
1 / 1 = 1 (se terminó).
Si "juntamos" los cocientes que hemos obtenido, aparece el número binario que buscábamos:
109 decimal = 0110 1101 binario
(Nota: es frecuente separar los números binarios en grupos de 4 cifras -medio byte- para
mayor legibilidad, como yo he hecho en el ejemplo anterior; a un grupo de 4 bits se le llama
nibble).
Si leemos esos restos de abajo a arriba, obtenemos el número binario: 1101101 (7 cifras, si
queremos completarlo a 8 cifras rellenamos con ceros por la izquierda: 01101101).
¿Y se pueden hacer operaciones con números binarios? Sí, casi igual que en decimal:
Ejercicios propuestos:
1. Expresar en sistema binario los números decimales 17, 101, 83, 45.
2. Expresar en sistema decimal los números binarios de 8 bits: 01100110, 10110010,
11111111, 00101101
3. Sumar los números 01100110+10110010, 11111111+00101101. Comprobar el
resultado sumando los números decimales obtenidos en el ejercicio anterior.
4. Multiplicar los números binarios de 4 bits 0100·1011, 1001·0011. Comprobar el
resultado convirtiéndolos a decimal.
Por eso, se han buscado otros sistemas de numeración que resulten más "compactos" que el
sistema binario cuando haya que expresar cifras medianamente grandes, pero que a la vez
mantengan con éste una correspondencia algo más sencilla que el sistema decimal. Los más
usados son el sistema octal y, sobre todo, el hexadecimal.
de modo que
254 = 3 · 8 2 + 7 · 8 1 + 6 · 8 0
o bien
Hemos conseguido otra correspondencia que, si bien nos resulta a nosotros más incómoda que
usar el sistema decimal, al menos es más compacta: el número 254 ocupa 3 cifras en decimal,
y también 3 cifras en octal, frente a las 8 cifras que necesitaba en sistema binario.
Pero además existe una correspondencia muy sencilla entre el sistema octal y el sistema
binario: si agrupamos los bits de 3 en 3, el paso de binario a octal es rapidísimo
de modo que
El paso desde el octal al binario y al decimal también es sencillo. Por ejemplo, el número 423
(octal) sería 423 (octal) = 100 010 011 (binario)
o bien
Ejercicios propuestos:
1. Expresar en sistema octal los números decimales 17, 101, 83, 45.
2. Expresar en sistema octal los números binarios de 8 bits: 01100110, 10110010,
11111111, 00101101
3. Expresar en el sistema binario los números octales 171, 243, 105, 45.
4. Expresar en el sistema decimal los números octales 162, 76, 241, 102.
El sistema octal tiene un inconveniente: se agrupan los bits de 3 en 3, por lo que convertir de
binario a octal y viceversa es muy sencillo, pero un byte está formado por 8 bits, que no es
múltiplo de 3.
Sería más cómodo poder agrupar de 4 en 4 bits, de modo que cada byte se representaría por 2
cifras. Este sistema de numeración trabajará en base 16 (2 4 =16), y es lo que se conoce como
sistema hexadecimal.
Pero hay una dificultad: estamos acostumbrados al sistema decimal, con números del 0 al 9, de
modo que no tenemos cifras de un solo dígito para los números 10, 11, 12, 13, 14 y 15, que
utilizaremos en el sistema hexadecimal. Para representar estas cifras usaremos las letras de la
A a la F, así:
0 (decimal) = 0 (hexadecimal)
1 (decimal) = 1 (hexadecimal)
2 (decimal) = 2 (hexadecimal)
3 (decimal) = 3 (hexadecimal)
4 (decimal) = 4 (hexadecimal)
5 (decimal) = 5 (hexadecimal)
6 (decimal) = 6 (hexadecimal)
7 (decimal) = 7 (hexadecimal)
8 (decimal) = 8 (hexadecimal)
9 (decimal) = 9 (hexadecimal)
10 (decimal) = A (hexadecimal)
11 (decimal) = B (hexadecimal)
12 (decimal) = C (hexadecimal)
13 (decimal) = D (hexadecimal)
14 (decimal) = E (hexadecimal)
15 (decimal) = F (hexadecimal)
de modo que
254 = 15 · 16 1 + 14 · 16 0
o bien
de modo que
o bien
254 = 13 · 16 3 + 4 · 16 2 + 3 · 16 1 + 11 · 16 0
es decir
Ahora vamos a dar el paso inverso: convertir de hexadecimal a decimal, por ejemplo el
número A2B5
110010100100100101010100111 =>
0110 0101 0010 0100 1010 1010 0111 = 6524AA7
Ejercicios propuestos:
1. Expresar en sistema hexadecimal los números decimales 18, 131, 83, 245.
2. Expresar en sistema hexadecimal los números binarios de 8 bits: 01100110, 10110010,
11111111, 00101101
3. Expresar en el sistema binario los números hexadecimales 2F, 37, A0, 1A2.
4. Expresar en el sistema decimal los números hexadecimales 1B2, 76, E1, 2A.
Es un método muy sencillo, pero que tiene el inconveniente de que las operaciones en
las que aparecen números negativos no se comportan correctamente. Vamos a ver un
ejemplo, con números de 8 bits:
Complemento a 1: se cambian los ceros por unos para expresar los números
negativos.
También es un método sencillo, en el que las operaciones con números negativos salen
bien, y que sólo tiene como inconveniente que hay dos formas de expresar el número 0
(0000 0000 o 1111 1111), lo que complica algunos trabajos internos del ordenador.
Ejercicio propuesto: convertir los números decimales 13, 34, -13, -34 a sistema
binario, usando complemento a uno para expresar los números negativos. Calcular (en
binario) el resultado de las operaciones 13+34, (-13)+(-34), 13+(-34) y comprobar que
los resultados que se obtienen son los correctos.
Complemento a 2: para los negativos, se cambian los ceros por unos y se suma uno
al resultado.
Es un método que parece algo más complicado, pero que no es difícil de seguir, con el
que las operaciones con números negativos salen bien, y no tiene problemas para
expresar el número 0 (00000000).
Ejercicio propuesto: convertir los números decimales 13, 34, -13, -34 a sistema
binario, usando complemento a dos para expresar los números negativos. Calcular (en
En general, todos los formatos que permiten guardar números negativos usan el primer bit para
el signo. Por eso, si declaramos una variable como "unsigned", ese primer bit se puede utilizar
como parte de los datos, y podemos almacenar números más grandes. Por ejemplo, un
"unsigned int" en MsDos podría tomar valores entre 0 y 65.535, mientras que un "int" (con
signo) podría tomar valores entre +32.767 y –32.768.
Nosotros usaremos Visual Studio 2008 en su versión Express, que es la que se puede usar
libremente ("gratis"), y, por tanto, es la más razonablemente para principiantes, además de que
necesita un ordenador menos potente qu la versión profesional.
http://www.microsoft.com/express/downloads/#Visual_Studio_2008_Express_Downloads
En concreto, en vez de descargar el paquete que llaman "Visual C#, y que es apenas un
"descargador" para instalar con conexión a Internet, puede ser preferible descargar
directamente la imagen ISO en español -Spanish- del DVD -900 Mb-, para poder instalar en
cualquier ordenador y en cualquier momento, sin necesidad de estar conectado a Internet:
http://www.microsoft.com/express/downloads/#2008-All
Esa imagen ISO se deberá grabar en DVD usando cualquer programa de grabación de CD y
DVD. Al introducir y ejecutar el DVD deberá aparecer un menú como éste:
E indicar si queremos algún componente adicional, como Silverlight (una alternativa al Flash de
Adobe) o el gestor de bases de datos SQL Server, en su versión Express (para un uso de
principiante, yo recomendaría no instalar ninguno de los dos):
Y para empezar un proyecto nuevo, deberíamos ir al menú Archivo y escoger la opción "Nuevo
Proyecto":
En la parte derecha de la pantalla aparecerán las clases que forman nuestro proyecto (por
ahora, sólo "Program.cs", que es el esqueleto básico, "Form1.cs", que representa nuestra
primera ventana), y en la parte izquierda aparecerá nuestra ventana vacía.
El "cuadro de herramientas", que nos permite colocar componentes visuales, como botones y
casillas de texto, aparece "escondido" a la izquierda. Si hacemos un clic se despliega, y
podemos fijarlo con el "pincho" que aparece en la esquina superior dercha, si no queremos que
se vuelva a "esconder" automáticamente.
Podemos desplegar la categoría "componentes comunes" para acceder a los más habituales:
A partir de ahí, el manejo sería muy similar a lo que vimos de SharpDevelop en el apartado 8.2,
con la diferencia de que no es necesario compilar para ver los errores, sino que se nos van
indicando "al vuelo", a medida que tecleamos:
http://www.microsoft.com/express/downloads/#2010-All
La instalación y el manejo son muy similares a los de la versión 2008, salvo por el detalle de
que está en inglés y de que la nueva estética juega con tonos blancos y azules. Esta es la
pantalla de instalación:
Y esta sería la ventana de trabajo, con nuestro formulario, la vista de clases a la derecha
(solution explorer), el cuadro de herramientas a la izquierda (toolbox), y las propiedades de
cada componente (properties) en la parte inferior derecha:
Índice alfabético
r @
-, 20 @, 35, 234
[
--, 28 [,] (arrays), 69
[] (arrays), 64
!
\
!, 41
!=, 39 \, 34
# ^
#, 31 ^, 187
% {
%, 20 { y }, 12, 38
%=, 29
|
& |, 187
&, 187 ||, 41
& (dirección de una variable, 178
&&, 41 ~
~, 123, 187
*
*, 20 +
* (punteros), 178
*=, 29 +, 20
++, 28
+=, 29
,
,, 195 <
<, 39
. <<, 187
.Net, 9
=
/ =, 21
/, 20 -=, 29
/*, 24 ==, 39
/**, 233
//, 24 >
///, 233
/=, 29 >, 39
>>, 187
:
A
:, 46
: (goto), 60 Abs, 103
Acceso aleatorio, 150
Acceso secuencial, 150
? Acos, 103
?, 46 Add (ArrayList), 168
al azar, números, 102
E G
E, 104 get, 190
ejecutable, 9 GetEnumerator, 174
elevado a, 103 GetFiles, 212
else, 40 GetKey, 171
Encapsulación, 109 gigabyte, 235
End, 153 gmcs, 198
Enqueue, 167 goto, 60
ensamblador, 8 goto case, 48
ensambladores, 9
entero, 27 H
enteros negativos, 243
Entorno, 212 Herencia, 109
enum, 188 Hexadecimal, 32
Enumeraciones, 188 Hora, 209
enumeradores, 174
Environment, 213
Environment.Exit, 107 I
ERRORLEVEL, 107 Identificadores, 23
escritura indentada, 39 if, 37
Espacios de nombres, 185 Igual a, 39
Estructuras, 71 Incremento, 28
Estructuras alternativas, 37 IndexOf, 76
Estructuras anidadas, 73 IndexOfKey, 171
Estructuras de control, 37 inseguro (bloque "unsafe"), 178
Estructuras dinámicas, 165 Inserción directa, 87
Estructuras repetitivas, 50 Insert, 77, 168
Euclides, 106 instante actual:, 102
Excepciones, 146 int, 21, 27
EXE, 9 intérprete, 9
Exists (ficheros), 145 interrumpir un programa, 107
Exp, 103 Introducción de datos, 25
Exponencial, 103 IOException, 158
Expresiones regulares, 192 iterativa, 106
F J
factorial, 104 JavaDoc, 234
false, 36
falso, 36
Fecha y hora, 209
Fibonacci, 106
Revisión 0.95 – Página 3
Introducción a la programación con C#, por Nacho Cabanes
K not, 187
Notepad++, 19
Key, 198 Now, 209
KeyAvailable, 197 null (fin de fichero), 143
KeyChar, 198 Números aleatorios, 102
kilobyte, 235 números enteros, 19
Números reales, 29
L
O
LastIndexOf, 76
Lectura y escritura en un mismo fichero, 162 O, 41
Length (cadenas), 75 Objetos, 109
Length (fichero), 153 octal, 240
lenguaje C, 7 ocultación de datos, 112
lenguaje máquina, 7 OpenOrCreate, 158
LIFO, 166 OpenRead, 151
línea de comandos, 106 OpenText, 142
Líneas, dibujar, 208 OpenWrite, 154
lista, 168 Operaciones abreviadas, 29
Operaciones aritméticas, 20
Operaciones con bits, 187
Ll operador coma, 195
llaves, 12 Operador condicional, 46
Operadores lógicos, 41
Operadores relacionales, 39
L or, 187
Log, 103 Ordenaciones simples, 86
Log10, 103 OSVersion, 213
Logaritmo, 103 out, 192
long, 27 override, 129
Longitud de una cadena, 75
P
M Parámetros de Main, 106
Main, 12 Parámetros de salida, 192
máquina virtual, 9 Parámetros de una función, 93
matemáticas, funciones, 103 parámetros por referencia, 100
Mayor que, 39 parámetros por valor, 100
mayúsculas, 76 Pascal, 7
mcs, 18 Paso a paso (depuración), 224
pausa, 211
megabyte, 235
Peek, 167
Memoria dinámica, 165
Pi, 104
Menor que, 39
pila, 166
MessageBox, 203
Polimorfismo, 109, 123
métodos, 120
POO, 109
Microsoft, 246
pop, 166
Mientras (condición repetitiva), 50
Posición del cursor, 197
Millisecond, 102
Posición en el fichero, 152
Modificar parámetros, 99
postdecremento, 28
Módulo (resto de división), 20
postincremento, 28
Mono, 11, 13 Potencia, 103
Month, 209 Pow, 103
Mostrar el valor de una variable, 22 Precedencia de los operadores, 20
MoveNext, 175 predecremento, 28
Multiplicación, 20 preincremento, 28
Prioridad de los operadores, 20
N private, 116
Process.Start, 212
n, 33 Producto lógico, 187
namespace, 185 programa, 7
Negación, 20 Programación orientada a objetos, 109
Net, 9 Propiedades, 190
new (arrays), 64 protected, 116
new (redefinir métodos), 115 Prueba de programas, 227
nibble, 240 Pseudocódigo, 9
No, 41 public, 12, 115
Revisión 0.95 – Página 4
Introducción a la programación con C#, por Nacho Cabanes