C# Ya
C# Ya
C# Ya
http://www.tutorialesprogramacionya.com/csharpya/index.php?inicio=0
12345
El objetivo fundamental de este tutorial es permitir que el estudiante pueda resolver problemas de distinta
índole (matemáticos, administrativos, gráficos, contables etc.) empleando como herramienta la
computadora.
Hay que tener en cuenta que para llegar a ser programador se debe recorrer un largo camino donde cada
tema es fundamental para conceptos futuros. Es importante no dejar temas sin entender y relacionar.
La programación a diferencia de otras materias como podría ser la historia requiere un estudio metódico y
ordenado (en historia se puede estudiar la edad media sin tener grandes conocimientos de la edad antigua)
La programación es una actividad nueva para el estudiante, no hay en los estudios primarios y secundarios
una materia parecida.
Es bueno tenerse paciencia cuando los problemas no se resuelven por completo, pero es de fundamental
importancia dedicar tiempo al análisis individual de los problemas.
Qué es un programa?
Programa: Conjunto de instrucciones que entiende un ordenador para realizar una actividad.
Todo programa tiene un objetivo bien definido: un procesador de texto es un programa que permite cargar,
modificar e imprimir textos, un programa de ajedrez permite jugar al ajedrez contra el ordenador u otro
contrincante humano.
La actividad fundamental del programador es resolver problemas empleando el ordenador como herramienta
fundamental.
Estos son los elementos esenciales que intervienen en el desarrollo de un diagrama de flujo.
Podemos identificar:
Datos conocidos:
Horas trabajadas en el mes.
Pago por hora.
Proceso:
Cálculo del sueldo multiplicando la cantidad de horas por el pago por hora.
Información resultante:
Sueldo mensual.
Esta forma de expresar un problema identificando sus datos conocidos, procesos e información resultante
puede llegar a ser engorrosa para problemas complejos donde hay muchos datos conocidos y procesos. Es
por eso que resulta mucho más efectivo representar los pasos para la resolución del problema mediante un
diagrama de flujo.
El diagrama de flujo nos da una idea del orden de ejecución de las actividades en el tiempo. Primero
cargamos los datos de entrada, luego hacemos las operaciones necesarias y por último mostramos los
resultados.
2 - Creación de un proyecto en C#
Descarga del entorno para programar con C#.
Podemos utilizar el Visual Studio Community que tiene entre otros lenguajes el C# y lo podemos descargar
de aquí.. También si ya tiene instalado el Visual Studio 2013, 2012 o 2010 puede hacer este curso.
Codificación del problema con el lenguaje C#.
No debemos perder de vista que el fin último es realizar un programa de computación que permita
automatizar una actividad para que muchos procesos sean desarrollados por la computadora.
El diagrama de flujo es un paso intermedio para poder ser interpretado por la computadora.
El paso siguiente es la codificación del diagrama de flujo en un lenguaje de computación, en nuestro caso
emplearemos el lenguaje C#.
Lenguaje de computación: Conjunto de instrucciones que son interpretadas por una computadora para
realizar operaciones, mostrar datos por pantalla, sacar listados por impresora, entrar datos por teclado, etc.
Para el ejemplo planteado la variable HorasTrabajadas almacena la cantidad de horas trabajadas por el
operario. La variable ValorHora almacena el precio de una hora de trabajo. La variable Sueldo almacena el
sueldo a abonar al operario.
En el ejemplo tenemos tres variables.
Tipos de variable:
Una variable puede almacenar:
- Valores Enteros (100, 260, etc.)
- Valores Reales (1.24, 2.90, 5.00, etc.)
- Cadenas de caracteres ("Juan", "Compras", "Listado", etc.)
Hay que tener en cuenta que el entorno de programación "Microsoft Visual Studio" no ha sido desarrollado
pensando en un principiante de la programación. Lo mismo ocurre con el propio lenguaje C#, es decir su
origen no tiene como objetivo el aprendizaje de la programación. Debido a estos dos puntos veremos que a
medida que avanzamos con el tutorial muchos conceptos que iremos dejando pendientes se irán aclarando.
Pasos.
2 - Creación del proyecto. Para esto seleccionamos desde el menú la opción "Archivo" -> "Nuevo" ->
"proyecto..."
Aparece un diálogo donde debemos indicar del lado izquierdo que utilizaremos el lenguaje Visual C#-
>Windows->Escritorio clásica, del lado de la derecha seleccionamos "Aplicación de consola" y en la parte
inferior definimos el "nombre", "ubicación" y "nombre de la solución" (podemos usar el mismo texto para el
"nombre de la solución" y "nombre"):
Podemos ver que el entorno nos generó automáticamente el esqueleto de nuestro programa:
3 - Si vamos al Explorador de archivos de windows podemos ver que tenemos creada la carpeta con nuestro
programa:
4 - Una vez que finalizamos de trabajar con el proyecto lo cerramos desde el menú de opciones Archivo ->
Cerrar solución:
3 - Codificación del diagrama de flujo en C#
Ahora debemos codificar el diagrama de flujo utilizando las instrucciones del lenguaje C#. Como hemos
visto el entorno de programación del Visual C# nos creó un esqueleto básico sobre el cual continuaremos el
programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace CalculoSueldo
{
class Program
{
static void Main(string[] args)
{
}
}
}
A medida que avancemos en el curso veremos que significa una clase y namespace, cual es el objetivo del
using etc. por el momento nos centraremos donde codificaremos nuestros diagramas de flujo.
La codificación del diagrama de flujo la haremos dentro de la función Main (la función Main es la primera
que se ejecuta al iniciarse un programa)
El programa completo para el calculo del sueldo de un operario conociendo la cantidad de horas trabajadas y
el costo por hora es:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace CalculoSueldo
{
class Program
{
static void Main(string[] args)
{
int horasTrabajadas;
float costoHora;
float sueldo;
string linea;
Console.Write("Ingrese Horas trabajadas por el operario:");
linea = Console.ReadLine();
horasTrabajadas = int.Parse(linea);
Console.Write("Ingrese el pago por hora:");
linea = Console.ReadLine();
costoHora = float.Parse(linea);
sueldo = horasTrabajadas * costoHora;
Console.Write("El sueldo total del operario es:");
Console.Write(sueldo);
Console.ReadKey();
}
}
}
Para probar el funcionamiento del programa debemos presionar el ícono con un triángulo verde (o la tecla
especial F5 o desde el menú elegir la opción "Depurar" -> "Iniciar depuración"):
La ejecución del programa permite ingresar la cantidad de horas trabajadas por un operario y su pago por
hora, mostrando seguidamente el sueldo que se debe pagar, un ejemplo de la ejecución de este programa es :
Si el texto de la ventana donde se ejecuta el programa es muy pequeña podemos modificar su tamaño de
fuente desde el menú que tiene a la izquierda:
1. Por el momento haremos todo el algoritmo dentro de la función Main. Es decir el resto siempre lo
crea el entorno del Visual C#.
2. Si observamos el diagrama de flujos vemos que debemos definir tres variables: (horasTrabajadas,
costoHora,sueldo), aquí es donde debemos definir que tipos de datos se almacenarán en las mismas.
La cantidad de horas normalmente será un valor entero (ej. 100 - 150 - 230 etc.), pero el costo de la
hora es muy común que sea un valor real (ej. 5.35 - 7.50 etc.) y como el sueldo resulta de multiplicar
las horas trabajadas por el costo por hora el mismo deberá ser real.
int horasTrabajadas;
float costoHora;
float sueldo;
Utilizamos la palabra clave int para definir variables enteras (en C# las palabras claves deben ir
obligatoriamente en minúsculas, sino se produce un error sintáctico) Luego de la palabra clave
debemos indicar el nombre de la variable, por ejemplo: horasTrabajadas (se propone que el nombre
de la variable comience con minúscula y en caso de estar constituida por dos palabras o más palabras
deben ir en mayúsculas el primer caracter (un nombre de variable no puede tener espacios en blanco,
empezar con un número, ni tampoco utilizar caracteres especiales)
Debemos buscar siempre nombres de variables que nos indiquen que almacenan (no es conveniente
llamar a nombres de variables con letras individuales)
Con esta sintaxis todo lo que se encuentra contenido entre comillas aparecerá exactamente en la
ventana de la "Console".
Si disponemos una variable:
Console.Write(sueldo);
Luego cada vez que necesitemos ingresar por teclado un conjunto de caracteres utilizaremos la
función ReadLine del objeto Console con la siguiente sintaxis:
linea = Console.ReadLine();
Un segundo paso es copiar el contenido de la variable linea en una variable de tipo int:
horasTrabajadas = int.Parse(linea);
costoHora = float.Parse(linea);
La variable linea almacena temporalmente los datos que ingresa el operador del programa, para
luego copiarse a la variable respectiva (como vemos si queremos convertir un string a tipo de dato
entero utilizamos la función Parse del objeto int (int.Parse))
Las operaciones que indicamos en el diagrama de flujo mediante la figura rectángulo la codificamos
tal cual:
Podemos ver una relación entre las instrucciones que debemos utilizar para cada símbolo del diagrama de
flujo:
En el diagrama de flujo no indicamos la definición de variables:
int horasTrabajadas;
float costoHora;
float sueldo;
string linea;
No representamos con símbolos los mensajes a mostrar previo a la carga de datos por teclado:
Console.Write("Ingrese Horas trabajadas por el operario:");
Como hemos visto hasta ahora hay muchas partes de nuestro código que no entendemos pero son
indispensables para la implementación de nuestros programas, a medida que avancemos con el curso
muchos de estos conceptos se irán aclarando.
Recordar que cuando terminamos con un programa debemos cerrarlo antes de crear uno nuevo mediante
Archivo -> Cerrar solución:
4 - Errores sintácticos y lógicos
Confeccionaremos un problema y agregaremos adrede una serie de errores tipográficos. Este tipo de errores
siempre son detectados por el COMPILADOR, antes de ejecutar el programa.
A los errores tipográficos, como por ejemplo la falta de puntos y comas, nombres de variables incorrectas,
falta de paréntesis, palabras claves mal escritas, etc. los llamamos errores SINTACTICOS.
Un programa no se puede ejecutar sin corregir absolutamente todos los errores sintácticos.
Existe otro tipo de errores llamados ERRORES LOGICOS. Este tipo de errores en programas grandes (miles
de líneas) son más difíciles de localizar. Por ejemplo un programa que permite hacer la facturación pero la
salida de datos por impresora es incorrecta.
Problema:
Diagrama de flujo:
Proyecto:
Recordemos que si tenemos un proyecto abierto actualmente debemos cerrarlo desde Archivo -> Cerrar
solución:
Codificamos el algoritmo en C# e introducimos dos errores sintáctico:
1 - Disponemos el nombre del objeto Console con minúsculas.
2 - Tratamos de imprimir el nombre de la variable superficie con el primer caracter en mayúsculas.
Como podemos observar aparece subrayado la línea donde disponemos console con minúsculas como en la
línea que imprimimos la variable superficie con mayúsculas. Si modificamos y corregimos los dos errores
sintácticos podremos ejecutar nuestro programa.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace SuperficieCuadrado
{
class Program
{
static void Main(string[] args)
{
int lado;
int superficie;
String linea;
Console.Write("Ingrese el valor del lado del cuadrado:");
linea = Console.ReadLine();
lado = int.Parse(linea);
superficie = lado * lado;
Console.Write("La superficie del cuadrado es:");
Console.Write(superficie);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace SuperficieCuadrado
{
class Program
{
static void Main(string[] args)
{
int lado;
int superficie;
String linea;
Console.Write("Ingrese el valor del lado del cuadrado:");
linea = Console.ReadLine();
lado = int.Parse(linea);
superficie = lado * lado * lado;
Console.Write("La superficie del cuadrado es:");
Console.Write(superficie);
Console.ReadKey();
}
}
}
Como podemos observar si ejecutamos el programa no presenta ningún error de compilación. Pero luego de
ingresar el valor del lado del cuadrado (por ejemplo el valor 10) obtenemos como resultado un valor
incorrecto (imprime el 1000), esto debido que definimos incorrectamente la fórmula para calcular la
superficie del cuadrado:
superficie = lado * lado * lado;
Problema:
Realizar la carga de dos números enteros por teclado e imprimir su suma y su producto.
Diagrama de flujo:
Tenemos dos entradas num1 y num2, dos operaciones: realización de la suma y del producto de los valores
ingresados y dos salidas, que son los resultados de la suma y el producto de los valores ingresados. En el
símbolo de impresión podemos indicar una o más salidas, eso queda a criterio del programador, lo mismo
para indicar las entradas por teclado.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace SumaProductoNumeros
{
class Program
{
static void Main(string[] args)
{
int num1, num2, suma, producto;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1 = int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
suma = num1 + num2;
producto = num1 * num2;
Console.Write("La suma de los dos valores es:");
Console.WriteLine(suma);
Console.Write("El producto de los dos valores es:");
Console.WriteLine(producto);
Console.ReadKey();
}
}
}
Recordemos que tenemos que seguir todos los pasos vistos para la creación de un proyecto.
Problemas propuestos
1. Realizar la carga del lado de un cuadrado, mostrar por pantalla el perímetro del mismo (El perímetro
de un cuadrado se calcula multiplicando el valor del lado por cuatro)
2. Escribir un programa en el cual se ingresen cuatro números, calcular e informar la suma de los dos
primeros y el producto del tercero y el cuarto.
3. Realizar un programa que lea cuatro valores numéricos e informar su suma y promedio.
4. Se debe desarrollar un programa que pida el ingreso del precio de un artículo y la cantidad que lleva
el cliente. Mostrar lo que debe abonar el comprador.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace PerimetroCuadrado
{
class Program
{
static void Main(string[] args)
{
int lado,perimetro;
string linea;
Console.Write("Ingrese el lado del cuadrado:");
linea=Console.ReadLine();
lado=int.Parse(linea);
perimetro=lado * 4;
Console.Write("El perímetro del cuadrado es:");
Console.Write(perimetro);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace SumaProductos4Numeros
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3,num4,suma,producto;
string linea;
Console.Write("Ingrese primer valor:");
linea=Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea=Console.ReadLine();
num2=int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea=Console.ReadLine();
num3=int.Parse(linea);
Console.Write("Ingrese cuarto valor:");
linea=Console.ReadLine();
num4=int.Parse(linea);
suma=num1 + num2;
producto=num3 * num4;
Console.Write("La suma de los dos primero valores es:");
Console.WriteLine(suma);
Console.Write("El producto del tercer y cuarto valor es:");
Console.Write(producto);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace SumaPromedio
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3,num4,suma,promedio;
string linea;
Console.Write("Ingrese primer valor:");
linea=Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea=Console.ReadLine();
num2=int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea=Console.ReadLine();
num3=int.Parse(linea);
Console.Write("Ingrese cuarto valor:");
linea=Console.ReadLine();
num4=int.Parse(linea);
suma=num1 + num2 + num3 + num4;
promedio=suma/4;
Console.Write("La suma de los cuatro valores es:");
Console.WriteLine(suma);
Console.Write("El promedio es:");
Console.Write(promedio);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace CostoCompra
{
class Program
{
static void Main(string[] args)
{
int cantidad;
float precio,importe;
string linea;
Console.Write("Ingrese la cantidad de artículos a llevar:");
linea=Console.ReadLine();
cantidad=int.Parse(linea);
Console.Write("Ingrese el valor unitario del producto:");
linea=Console.ReadLine();
precio=float.Parse(linea);
importe=precio * cantidad;
Console.Write("El importe total a pagar es:");
Console.Write(importe);
Console.ReadKey();
}
}
}
Podemos observar: El rombo representa la condición. Hay dos opciones que se pueden tomar. Si la
condición da verdadera se sigue el camino del verdadero, o sea el de la derecha, si la condición da falsa se
sigue el camino de la izquierda.
Se trata de una estructura CONDICIONAL SIMPLE porque por el camino del verdadero hay actividades y
por el camino del falso no hay actividades.
Por el camino del verdadero pueden existir varias operaciones, entradas y salidas, inclusive ya veremos que
puede haber otras estructuras condicionales.
Problema:
Ingresar el sueldo de una persona, si supera los 3000 pesos mostrar un mensaje en pantalla indicando que
debe abonar impuestos.
Diagrama de flujo:
Podemos observar lo siguiente: Siempre se hace la carga del sueldo, pero si el sueldo que ingresamos supera
3000 pesos se mostrará por pantalla el mensaje "Esta persona debe abonar impuestos", en caso que la
persona cobre 3000 o menos no aparece nada por pantalla.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalSimple1
{
class Program
{
static void Main(string[] args)
{
float sueldo;
string linea;
Console.Write("Ingrese el sueldo:");
linea=Console.ReadLine();
sueldo=float.Parse(linea);
if (sueldo>3000)
{
Console.Write("Esta persona debe abonar impuestos");
}
Console.ReadKey();
}
}
}
La palabra clave "if" indica que estamos en presencia de una estructura condicional; seguidamente
disponemos la condición entre paréntesis. Por último encerrada entre llaves las instrucciones de la rama del
verdadero.
Es necesario que las instrucciones a ejecutar en caso que la condición sea verdadera estén encerradas entre
llaves { }, con ellas marcamos el comienzo y el fin del bloque del verdadero.
Ejecutando el programa e ingresamos un sueldo superior a 3000 pesos. Podemos observar como aparece en
pantalla el mensaje "Esta persona debe abonar impuestos", ya que la condición del if es verdadera.
Volvamos a ejecutar el programa y carguemos un sueldo menor o igual a 3000 pesos. No debe aparecer
mensaje en pantalla.
Cuando se presenta la elección tenemos la opción de realizar una actividad u otra. Es decir tenemos
actividades por el verdadero y por el falso de la condición. Lo más importante que hay que tener en cuenta
que se realizan las actividades de la rama del verdadero o las del falso, NUNCA se realizan las actividades
de las dos ramas.
Representación gráfica:
En una estructura condicional compuesta tenemos entradas, salidas, operaciones, tanto por la rama del
verdadero como por la rama del falso.
Problema:
Realizar un programa que solicite ingresar dos números distintos y muestre por pantalla el mayor de ellos.
Diagrama de flujo:
Se hace la entrada de num1 y num2 por teclado. Para saber cual variable tiene un valor mayor preguntamos
si el contenido de num1 es mayor (>) que el contenido de num2, si la respuesta es verdadera vamos por la
rama de la derecha e imprimimos num1, en caso que la condición sea falsa vamos por la rama de la
izquierda (Falsa) e imprimimos num2.
Como podemos observar nunca se imprimen num1 y num2 simultáneamente.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalCompuesta1
{
class Program
{
static void Main(string[] args)
{
int num1, num2;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1 = int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
if (num1 > num2)
{
Console.Write(num1);
}
else
{
Console.Write(num2);
}
Console.ReadKey();
}
}
}
Cotejemos el diagrama de flujo y la codificación y observemos que el primer bloque de llaves después del if
representa la rama del verdadero y el segundo bloque de llaves representa la rama del falso.
Compilemos el programa, si hubo errores sintácticos corrijamos y carguemos dos valores, como por
ejemplo:
Si ingresamos los valores 10 y 4 la condición del if retorna verdadero y ejecuta el primer bloque.
Un programa se controla y corrige probando todos sus posibles resultados.
Ejecutemos nuevamente el programa e ingresemos:
Cuando a un programa le corregimos todos los errores sintácticos y lógicos ha terminado nuestra tarea y
podemos entregar el mismo al USUARIO que nos lo solicitó.
Operadores
En una condición deben disponerse únicamente variables, valores constantes y operadores relacionales.
>Operadores Relacionales:
> (mayor)
< (menor)
>= (mayor o igual)
<= (menor o igual)
== (igual)
!= (distinto)
Operadores Matemáticos
+ (más)
- (menos)
* (producto)
/ (división)
% (resto de una división) Ej.: x=13%5; {se guarda 3}
Hay que tener en cuenta que al disponer una condición debemos seleccionar que operador relacional se
adapta a la pregunta.
Ejemplos:
Los problemas que se pueden presentar son infinitos y la correcta elección del operador sólo se alcanza con
la práctica intensiva en la resolución de problemas.
Problemas propuestos
1. Realizar un programa que lea por teclado dos números, si el primero es mayor al segundo informar
su suma y diferencia, en caso contrario informar el producto y la división del primero respecto al
segundo.
2. Se ingresan tres notas de un alumno, si el promedio es mayor o igual a siete mostrar un mensaje
"Promocionado".
3. Se ingresa por teclado un número positivo de uno o dos dígitos (1..99) mostrar un mensaje indicando
si el número tiene uno o dos dígitos.
(Tener en cuenta que condición debe cumplirse para tener dos dígitos, un número entero)
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalCompuesta2
{
class Program
{
static void Main(string[] args)
{
int num1,num2;
string linea;
Console.Write("Ingrese primer valor:");
linea=Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea=Console.ReadLine();
num2=int.Parse(linea);
if (num1>num2)
{
int suma,diferencia;
suma=num1 + num2;
diferencia=num1 - num2;
Console.Write("La suma de los dos valores es:");
Console.WriteLine(suma);
Console.Write("La diferencia de los dos valores es:");
Console.WriteLine(diferencia);
}
else
{
int producto,division;
producto=num1 * num2;
division=num1 / num2;
Console.Write("El producto de los dos valores es:");
Console.WriteLine(producto);
Console.Write("La división de los dos valores es:");
Console.WriteLine(division);
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalSimple2
{
class Program
{
static void Main(string[] args)
{
int nota1,nota2,nota3;
string linea;
Console.Write("Ingrese primer nota:");
linea=Console.ReadLine();
nota1=int.Parse(linea);
Console.Write("Ingrese segunda nota:");
linea=Console.ReadLine();
nota2=int.Parse(linea);
Console.Write("Ingrese tercer nota:");
linea=Console.ReadLine();
nota3=int.Parse(linea);
int promedio;
promedio=(nota1 + nota2 + nota3) / 3;
if (promedio>=7)
{
Console.Write("Promocionado");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalCompuesta3
{
class Program
{
static void Main(string[] args)
{
int num;
string linea;
Console.Write("Ingrese un valor entero de 1 o 2 dígitos:");
linea=Console.ReadLine();
num=int.Parse(linea);
if (num<10)
{
Console.Write("Tiene un dígito");
}
else
{
Console.Write("Tiene dos dígitos");
}
Console.ReadKey();
}
}
}
El diagrama de flujo que se presenta contiene dos estructuras condicionales. La principal se trata de una
estructura condicional compuesta y la segunda es una estructura condicional simple y está contenida por la
rama del falso de la primer estructura.
Es común que se presenten estructuras condicionales anidadas aún más complejas.
Problema:
Confeccionar un programa que pida por teclado tres notas de un alumno, calcule el promedio e imprima
alguno de estos mensajes:
Si el promedio es >=7 mostrar "Promocionado".
Si el promedio es >=4 y <7 mostrar "Regular".
Si el promedio es <4 mostrar "Reprobado".
Diagrama de flujo:
Analicemos el siguiente diagrama. Se ingresan tres valores por teclado que representan las notas de un
alumno, se obtiene el promedio sumando los tres valores y dividiendo por 3 dicho resultado (Tener en
cuenta que si el resultado es un valor real solo se almacena la parte entera).
Primeramente preguntamos si el promedio es superior o igual a 7, en caso afirmativo va por la rama del
verdadero de la estructura condicional mostramos un mensaje que indica "Promocionado" (con comillas
indicamos un texto que debe imprimirse en pantalla).
En caso que la condición nos de falso, por la rama del falso aparece otra estructura condicional, porque
todavía debemos averiguar si el promedio del alumno es superior o igual a cuatro o inferior a cuatro.
Estamos en presencia de dos estructuras condicionales compuestas.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalAnidada1
{
class Program
{
static void Main(string[] args)
{
int nota1,nota2,nota3;
string linea;
Console.Write("Ingrese primer nota:");
linea = Console.ReadLine();
nota1=int.Parse(linea);
Console.Write("Ingrese segunda nota:");
linea = Console.ReadLine();
nota2 = int.Parse(linea);
Console.Write("Ingrese tercer nota:");
linea = Console.ReadLine();
nota3 = int.Parse(linea);
int promedio=(nota1 + nota2 + nota3) / 3;
if (promedio>=7)
{
Console.Write("Promocionado");
}
else
{
if (promedio>=4)
{
Console.Write("Regular");
}
else
{
Console.Write("Reprobado");
}
}
Console.ReadKey();
}
}
}
Codifiquemos y ejecutemos este programa. Al correr el programa deberá solicitar por teclado la carga de
tres notas y mostrarnos un mensaje según el promedio de las mismas.
Podemos definir un conjunto de variables del mismo tipo en una misma línea:
int nota1,nota2,nota3;
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalAnidada2
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segunda valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea = Console.ReadLine();
num3 = int.Parse(linea);
if (num1>num2)
{
if (num1>num3)
{
Console.Write(num1);
}
else
{
Console.Write(num3);
}
}
else
{
if (num2>num3)
{
Console.Write(num2);
}
else
{
Console.Write(num3);
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalAnidada3
{
class Program
{
static void Main(string[] args)
{
int num;
string linea;
Console.Write("Ingrese un valor:");
linea = Console.ReadLine();
num=int.Parse(linea);
if (num==0)
{
Console.Write("Se ingresó el cero");
}
else
{
if (num>0)
{
Console.Write("Se ingresó un valor positivo");
}
else
{
Console.Write("Se ingresó un valor negativo");
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalAnidada4
{
class Program
{
static void Main(string[] args)
{
int num;
string linea;
Console.Write("Ingrese un valor de hasta tres dígitos positivo:");
linea = Console.ReadLine();
num=int.Parse(linea);
if (num<10)
{
Console.Write("Tiene un dígito");
}
else
{
if (num<100)
{
Console.Write("Tiene dos dígitos");
}
else
{
if (num<1000)
{
Console.Write("Tiene tres dígitos");
}
else
{
Console.Write("Error en la entrada de datos.");
}
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraCondicionalAnidada5
{
class Program
{
static void Main(string[] args)
{
int totalPreguntas,totalCorrectas;
string linea;
Console.Write("Ingrese la cantidad total de preguntas del examen:");
linea = Console.ReadLine();
totalPreguntas=int.Parse(linea);
Console.Write("Ingrese la cantidad total de preguntas contestadas
correctamente:");
linea = Console.ReadLine();
totalCorrectas=int.Parse(linea);
int porcentaje=totalCorrectas * 100 / totalPreguntas;
if (porcentaje>=90)
{
Console.Write("Nivel máximo");
}
else
{
if (porcentaje>=75)
{
Console.Write("Nivel medio");
}
else
{
if (porcentaje>=50)
{
Console.Write("Nivel regular");
}
else
{
Console.Write("Fuera de nivel");
}
}
}
Console.ReadKey();
}
}
}
Estos dos operadores se emplean fundamentalmente en las estructuras condicionales para agrupar varias
condiciones simples.
Operador &&
Traducido se lo lee como ?Y?. Si la Condición 1 es verdadera Y la condición 2 es verdadera luego ejecutar
la rama del verdadero.
Cuando vinculamos dos o más condiciones con el operador ?&&?, las dos condiciones deben ser verdaderas
para que el resultado de la condición compuesta de Verdadero y continúe por la rama del verdadero de la
estructura condicional.
La utilización de operadores lógicos permiten en muchos casos plantear algoritmos más cortos y
comprensibles.
Problema:
Confeccionar un programa que lea por teclado tres números distintos y nos muestre el mayor.
Diagrama de flujo:
Este ejercicio está resuelto sin emplear operadores lógicos en un concepto anterior del tutorial. La primera
estructura condicional es una ESTRUCTURA CONDICIONAL COMPUESTA con una CONDICION
COMPUESTA.
Podemos leerla de la siguiente forma:
Si el contenido de la variable num1 es mayor al contenido de la variable num2 Y si el contenido de la
variable num1 es mayor al contenido de la variable num3 entonces la CONDICION COMPUESTA resulta
Verdadera.
Si una de las condiciones simples da falso la CONDICION COMPUESTA da Falso y continua por la rama
del falso.
Es decir que se mostrará el contenido de num1 si y sólo si num1>num2 y num1>num3.
En caso de ser Falsa la condición, analizamos el contenido de num2 y num3 para ver cual tiene un valor
mayor.
En esta segunda estructura condicional no se requieren operadores lógicos al haber una condición simple.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionCompuesta1
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea = Console.ReadLine();
num3 = int.Parse(linea);
if (num1>num2 && num1>num3)
{
Console.Write(num1);
}
else
{
if (num2>num3)
{
Console.Write(num2);
}
else
{
Console.Write(num3);
}
}
Console.ReadKey();
}
}
}
Operador ||
Traducido se lo lee como ?O?. Si la condición 1 es Verdadera O la condición 2 es Verdadera, luego ejecutar
la rama del Verdadero.
Cuando vinculamos dos o más condiciones con el operador ?Or", con que una de las dos condiciones sea
Verdadera alcanza para que el resultado de la condición compuesta sea Verdadero.
Problema:
Se carga una fecha (día, mes y año) por teclado. Mostrar un mensaje si corresponde al primer trimestre del
año (enero, febrero o marzo) Cargar por teclado el valor numérico del día, mes y año.
Ejemplo: dia:10 mes:1 año:2010.
Diagrama de flujo:
La carga de una fecha se hace por partes, ingresamos las variables dia, mes y año.
Mostramos el mensaje "Corresponde al primer trimestre" en caso que el mes ingresado por teclado sea igual
a 1, 2 ó 3.
En la condición no participan las variables dia y año.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionCompuesta2
{
class Program
{
static void Main(string[] args)
{
int dia,mes,año;
string linea;
Console.Write("Ingrese nro de día:");
linea = Console.ReadLine();
dia = int.Parse(linea); ;
Console.Write("Ingrese nro de mes:");
linea = Console.ReadLine();
mes=int.Parse(linea);
Console.Write("Ingrese nro de año:");
linea = Console.ReadLine();
año=int.Parse(linea);
if (mes==1 || mes==2 || mes==3)
{
Console.Write("Corresponde al primer trimestre");
}
Console.ReadLine();
}
}
}
Problemas propuestos
1. Realizar un programa que pida cargar una fecha cualquiera, luego verificar si dicha fecha
corresponde a Navidad.
2. Se ingresan tres valores por teclado, si todos son iguales se imprime la suma del primero con el
segundo y a este resultado se lo multiplica por el tercero.
3. Se ingresan por teclado tres números, si todos los valores ingresados son menores a 10, imprimir en
pantalla la leyenda "Todos los números son menores a diez".
4. Se ingresan por teclado tres números, si al menos uno de los valores ingresados es menor a 10,
imprimir en pantalla la leyenda "Alguno de los números es menor a diez".
5. Escribir un programa que pida ingresar la coordenada de un punto en el plano, es decir dos valores
enteros x e y (distintos a cero).
Posteriormente imprimir en pantalla en que cuadrante se ubica dicho punto. (1º Cuadrante si x > 0 Y
y > 0 , 2º Cuadrante: x < 0 Y y > 0, etc.)
6. De un operario se conoce su sueldo y los años de antigüedad. Se pide confeccionar un programa que
lea los datos de entrada e informe:
a) Si el sueldo es inferior a 500 y su antigüedad es igual o superior a 10 años, otorgarle un aumento
del 20 %, mostrar el sueldo a pagar.
b)Si el sueldo es inferior a 500 pero su antigüedad es menor a 10 años, otorgarle un aumento de 5 %.
c) Si el sueldo es mayor o igual a 500 mostrar el sueldo en pantalla sin cambios.
7. Escribir un programa en el cual: dada una lista de tres valores numéricos distintos se calcule e
informe su rango de variación (debe mostrar el mayor y el menor de ellos)
Solución
R using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas3
{
class Program
{
static void Main(string[] args)
{
int dia,mes,año;
string linea;
Console.Write("Ingrese nro de día:");
linea = Console.ReadLine();
dia=int.Parse(linea);
Console.Write("Ingrese nro de mes:");
linea = Console.ReadLine();
mes=int.Parse(linea);
Console.Write("Ingrese nro de año:");
linea = Console.ReadLine();
año = int.Parse(linea);
if (mes==12 && dia==25)
{
Console.Write("La fecha ingresada corresponde a navidad.");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas4
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea = Console.ReadLine();
num3 = int.Parse(linea);
if (num1==num2 && num1==num3)
{
int suma=num1 + num2;
Console.Write("La suma del primero y segundo:");
Console.WriteLine(suma);
int producto=suma * num3;
Console.Write("La suma del primero y segundo multiplicado por el
tercero:");
Console.Write(producto);
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas5
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea=Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea=Console.ReadLine();
num2=int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea=Console.ReadLine();
num3=int.Parse(linea);
if (num1<10 && num2<10 && num3<10)
{
Console.Write("Todos los números son menores a diez");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas6
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea = Console.ReadLine();
num3 = int.Parse(linea);
if (num1<10 || num2<10 || num3<10)
{
Console.Write("Alguno de los números es menor a diez");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas7
{
class Program
{
static void Main(string[] args)
{
int x, y;
string linea;
Console.Write("Ingrese coordenada x:");
linea = Console.ReadLine();
x = int.Parse(linea);
Console.Write("Ingrese coordenada y:");
linea = Console.ReadLine();
y = int.Parse(linea);
if (x > 0 && y > 0)
{
Console.Write("Se encuentra en el primer cuadrante");
}
else
{
if (x < 0 && y > 0)
{
Console.Write("Se encuentra en el segundo cuadrante");
}
else
{
if (x < 0 && y < 0)
{
Console.Write("Se encuentra en el tercer cuadrante");
}
else
{
Console.Write("Se encuentra en el cuarto cuadrante");
}
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas8
{
class Program
{
static void Main(string[] args)
{
float sueldo;
int antiguedad;
string linea;
Console.Write("Ingrese sueldo del empleado:");
linea = Console.ReadLine();
sueldo=float.Parse(linea);
Console.Write("Ingrese su antiguedad en años:");
linea = Console.ReadLine();
antiguedad=int.Parse(linea);
if (sueldo<500 && antiguedad>10)
{
float aumento=sueldo * 0.20f;
float sueldoTotal=sueldo+aumento;
Console.Write("Sueldo a pagar:");
Console.Write(sueldoTotal);
}
else
{
if (sueldo<500)
{
float aumento=sueldo * 0.05f;
float sueldoTotal=sueldo+aumento;
Console.Write("Sueldo a pagar:");
Console.Write(sueldoTotal);
}
else
{
Console.Write("Sueldo a pagar:");
Console.Write(sueldo);
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CondicionesCompuestas9
{
class Program
{
static void Main(string[] args)
{
int num1,num2,num3;
string linea;
Console.Write("Ingrese primer valor:");
linea = Console.ReadLine();
num1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
num2 = int.Parse(linea);
Console.Write("Ingrese tercer valor:");
linea = Console.ReadLine();
num3 = int.Parse(linea);
Console.Write("Rango de valores:");
if (num1<num2 && num1<num3)
{
Console.Write(num1);
}
else
{
if (num2<num3)
{
Console.Write(num2);
}
else
{
Console.Write(num3);
}
}
Console.Write("-");
if (num1>num2 && num1>num3)
{
Console.Write(num1);
}
else
{
if (num2>num3)
{
Console.Write(num2);
}
else
{
Console.Write(num3);
}
}
Console.ReadKey();
}
}
}
Retornar
Una estructura repetitiva permite ejecutar una instrucción o un conjunto de instrucciones varias veces.
No debemos confundir la representación gráfica de la estructura repetitiva while (Mientras) con la estructura
condicional if (Si)
Funcionamiento: En primer lugar se verifica la condición, si la misma resulta verdadera se ejecutan las
operaciones que indicamos por la rama del Verdadero.
A la rama del verdadero la graficamos en la parte inferior de la condición. Una línea al final del bloque de
repetición la conecta con la parte superior de la estructura repetitiva.
En caso que la condición sea Falsa continúa por la rama del Falso y sale de la estructura repetitiva para
continuar con la ejecución del algoritmo.
El bloque se repite MIENTRAS la condición sea Verdadera.
Importante: Si la condición siempre retorna verdadero estamos en presencia de un ciclo repetitivo infinito.
Dicha situación es un error de programación, nunca finalizará el programa.
Problema 1:
Sin conocer las estructuras repetitivas podemos resolver el problema empleando una estructura secuencial.
Inicializamos una variable con el valor 1, luego imprimimos la variable, incrementamos nuevamente la
variable y así sucesivamente.
Diagrama de flujo:
Si continuamos con el diagrama no nos alcanzarían las próximas 5 páginas para finalizarlo. Emplear una
estructura secuencial para resolver este problema produce un diagrama de flujo y un programa en C# muy
largo.
Al ejecutarse la condición retorna VERDADERO porque el contenido de x (1) es menor o igual a 100. Al
ser la condición verdadera se ejecuta el bloque de instrucciones que contiene la estructura while. El bloque
de instrucciones contiene una salida y una operación.
Se imprime el contenido de x, y seguidamente se incrementa la variable x en uno.
La operación x=x + 1 se lee como "en la variable x se guarda el contenido de x más 1". Es decir, si x
contiene 1 luego de ejecutarse esta operación se almacenará en x un 2.
Lo más difícil es la definición de la condición de la estructura while y qué bloque de instrucciones se van a
repetir. Observar que si, por ejemplo, disponemos la condición x >=100 ( si x es mayor o igual a 100) no
provoca ningún error sintáctico pero estamos en presencia de un error lógico porque al evaluarse por primera
vez la condición retorna falso y no se ejecuta el bloque de instrucciones que queríamos repetir 100 veces.
No existe una RECETA para definir una condición de una estructura repetitiva, sino que se logra con una
práctica continua solucionando problemas.
Una vez planteado el diagrama debemos verificar si el mismo es una solución válida al problema (en este
caso se debe imprimir los números del 1 al 100 en pantalla), para ello podemos hacer un seguimiento del
flujo del diagrama y los valores que toman las variables a lo largo de la ejecución:
x
1
2
3
4
.
.
100
101 Cuando x vale 101 la condición de la estructura repetitiva retorna falso,
en este caso finaliza el diagrama.
Importante: Podemos observar que el bloque repetitivo puede no ejecutarse ninguna vez si la condición
retorna falso la primera vez.
La variable x debe estar inicializada con algún valor antes que se ejecute la operación x=x + 1 en caso de no
estar inicializada aparece un error de compilación.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile1
{
class Program
{
static void Main(string[] args)
{
int x;
x = 1;
while (x <= 100)
{
Console.Write(x);
Console.Write(" - ");
x = x + 1;
}
Console.ReadKey();
}
}
}
Recordemos que un problema no estará 100% solucionado si no hacemos el programa en C# que muestre los
resultados buscados.
Probemos algunas modificaciones de este programa y veamos que cambios se deberían hacer para:
Respuestas:
Escribir un programa que solicite la carga de un valor positivo y nos muestre desde 1 hasta el valor
ingresado de uno en uno.
Ejemplo: Si ingresamos 30 se debe mostrar en pantalla los números del 1 al 30.
Diagrama de flujo:
Podemos observar que se ingresa por teclado la variable n. El operador puede cargar cualquier valor.
Si el operador carga 10 el bloque repetitivo se ejecutará 10 veces, ya que la condición es ?Mientras x<=n ?,
es decir ?mientras x sea menor o igual a 10?; pues x comienza en uno y se incrementa en uno cada vez que
se ejecuta el bloque repetitivo.
A la prueba del diagrama la podemos realizar dándole valores a las variables; por ejemplo, si ingresamos 5
el seguimiento es el siguiente:
n x
5 1 (Se imprime el contenido de x)
2 " "
3 " "
4 " "
5 " "
6 (Sale del while porque 6 no es menor o igual a 5)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile2
{
class Program
{
static void Main(string[] args)
{
int n,x;
string linea;
Console.Write("Ingrese el valor final:");
linea=Console.ReadLine();
n=int.Parse(linea);
x=1;
while (x<=n)
{
Console.Write(x);
Console.Write(" - ");
x = x + 1;
}
Console.ReadKey();
}
}
}
Los nombres de las variables n y x pueden ser palabras o letras (como en este caso)
Problema 3:
Desarrollar un programa que permita la carga de 10 valores por teclado y nos muestre posteriormente la
suma de los valores ingresados y su promedio.
Diagrama de flujo:
En este problema, a semejanza de los anteriores, llevamos un CONTADOR llamado x que nos sirve para
contar las vueltas que debe repetir el while.
También aparece el concepto de ACUMULADOR (un acumulador es un tipo especial de variable que se
incrementa o decrementa con valores variables durante la ejecución del programa)
Hemos dado el nombre de suma a nuestro acumulador. Cada ciclo que se repita la estructura repetitiva, la
variable suma se incrementa con el contenido ingresado en la variable valor.
Este es un seguimiento del diagrama planteado. Los números que toma la variable valor dependerá de qué
cifras cargue el operador durante la ejecución del programa.
El promedio se calcula al salir de la estructura repetitiva (es decir primero sumamos los 10 valores
ingresados y luego los dividimos por 10)
Hay que tener en cuenta que cuando en la variable valor se carga el primer valor (en este ejemplo 5) al
cargarse el segundo valor (16) el valor anterior 5 se pierde, por ello la necesidad de ir almacenando en la
variable suma los valores ingresados.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile3
{
class Program
{
static void Main(string[] args)
{
int x,suma,valor,promedio;
string linea;
x=1;
suma=0;
while (x<=10)
{
Console.Write("Ingrese un valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
suma=suma+valor;
x=x+1;
}
promedio=suma/10;
Console.Write("La suma de los 10 valores es:");
Console.WriteLine(suma);
Console.Write("El promedio es:");
Console.Write(promedio);
Console.ReadKey();
}
}
}
Problema 4:
Diagrama de flujo:
Podemos observar que dentro de una estructura repetitiva puede haber estructuras condicionales (inclusive
puede haber otras estructuras repetitivas que veremos más adelante)
En este problema hay que cargar inicialmente la cantidad de piezas a ingresar ( n ), seguidamente se cargan
n valores de largos de piezas.
Cada vez que ingresamos un largo de pieza (largo) verificamos si es una medida correcta (debe estar entre
1.20 y 1.30 el largo para que sea correcta), en caso de ser correcta la CONTAMOS (incrementamos la
variable cantidad en 1)
Al contador cantidad lo inicializamos en cero porque inicialmente no se ha cargado ningún largo de medida.
Cuando salimos de la estructura repetitiva porque se han cargado n largos de piezas mostramos por pantalla
el contador cantidad (que representa la cantidad de piezas aptas)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile4
{
class Program
{
static void Main(string[] args)
{
int x,cantidad,n;
float largo;
string linea;
x=1;
cantidad=0;
Console.Write("Cuantas piezar procesará:");
linea = Console.ReadLine();
n=int.Parse(linea);
while (x<=n)
{
Console.Write("Ingrese la medida de la pieza:");
linea = Console.ReadLine();
largo=float.Parse(linea);
if (largo>=1.20 && largo<=1.30)
{
cantidad = cantidad +1;
}
x=x + 1;
}
Console.Write("La cantidad de piezas aptas son:");
Console.Write(cantidad);
Console.ReadKey();
}
}
}
Problemas propuestos
Ha llegado la parte fundamental, que es el momento donde uno desarrolla individualmente un algoritmo para
la resolución de problemas.
El tiempo a dedicar a esta sección EJERCICIOS PROPUESTOS debe ser mucho mayor que el empleado a
la sección de EJERCICIOS RESUELTOS.
La experiencia dice que debemos dedicar el 80% del tiempo a la resolución individual de problemas y el
otro 20% al análisis y codificación de problemas ya resueltos por otras personas.
Es de vital importancia para llegar a ser un buen PROGRAMADOR poder resolver problemas en forma
individual.
1. Escribir un programa que solicite ingresar 10 notas de alumnos y nos informe cuántos tienen notas
mayores o iguales a 7 y cuántos menores.
2. Se ingresan un conjunto de n alturas de personas por teclado. Mostrar la altura promedio de las
personas.
3. En una empresa trabajan n empleados cuyos sueldos oscilan entre $100 y $500, realizar un programa
que lea los sueldos que cobra cada empleado e informe cuántos empleados cobran entre $100 y $300
y cuántos cobran más de $300. Además el programa deberá informar el importe que gasta la empresa
en sueldos al personal.
4. Realizar un programa que imprima 25 términos de la serie 11 - 22 - 33 - 44, etc. (No se ingresan
valores por teclado)
5. Mostrar los múltiplos de 8 hasta el valor 500. Debe aparecer en pantalla 8 - 16 - 24, etc.
6. Realizar un programa que permita cargar dos listas de 15 valores cada una. Informar con un mensaje
cual de las dos listas tiene un valor acumulado mayor (mensajes "Lista 1 mayor", "Lista 2 mayor",
"Listas iguales")
Tener en cuenta que puede haber dos o más estructuras repetitivas en un algoritmo.
7. Desarrollar un programa que permita cargar n números enteros y luego nos informe cuántos valores
fueron pares y cuántos impares.
Emplear el operador ?%? en la condición de la estructura condicional:
8. if (valor%2==0) //Si el if da verdadero luego es par.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile5
{
class Program
{
static void Main(string[] args)
{
int x,nota,conta1,conta2;
string linea;
x=1;
conta1=0;
conta2=0;
while (x<=10)
{
Console.Write("Ingrese nota:");
linea = Console.ReadLine();
nota=int.Parse(linea);
if (nota>=7)
{
conta1=conta1 + 1;
}
else
{
conta2=conta2 + 1;
}
x=x + 1;
}
Console.Write("Cantidad de alumnos con notas mayores o iguales a 7:");
Console.WriteLine(conta1);
Console.Write("Cantidad de alumons con notas menores a 7:");
Console.Write(conta2);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile6
{
class Program
{
static void Main(string[] args)
{
int n,x;
float altura,suma,promedio;
string linea;
Console.Write("Cuantas personas hay:");
linea = Console.ReadLine();
n=int.Parse(linea);
x=1;
suma=0;
while (x<=n)
{
Console.Write("Ingrese la altura:");
linea = Console.ReadLine();
altura=float.Parse(linea);
suma=suma + altura;
x=x + 1;
}
promedio=suma/n;
Console.Write("Altura promedio:");
Console.Write(promedio);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile7
{
class Program
{
static void Main(string[] args)
{
int n,x,conta1,conta2;
float sueldo,gastos;
string linea;
Console.Write("Cuantos empleados tiene la empresa:");
linea = Console.ReadLine();
n=int.Parse(linea);
x=1;
conta1=0;
conta2=0;
gastos=0;
while (x<=n) {
Console.Write("Ingrese el sueldo del empleado:");
linea = Console.ReadLine();
sueldo=float.Parse(linea);
if (sueldo<=300) {
conta1=conta1 + 1;
} else {
conta2=conta2 + 1;
}
gastos=gastos+sueldo;
x=x + 1;
}
Console.Write("Cantidad de empleados con sueldos entre 100 y 300:");
Console.WriteLine(conta1);
Console.Write("Cantidad de empleados con sueldos mayor a 300:");
Console.WriteLine(conta2);
Console.Write("Gastos total de la empresa en sueldos:");
Console.WriteLine(gastos);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile8
{
class Program
{
static void Main(string[] args)
{
int x,termino;
x=1;
termino=11;
while (x<=25)
{
Console.Write(termino);
Console.Write(" - ");
x=x + 1;
termino=termino + 11;
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile9
{
class Program
{
static void Main(string[] args)
{
int mult8;
mult8=8;
while (mult8<=500)
{
Console.Write(mult8);
Console.Write(" - ");
mult8=mult8 + 8;
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile10
{
class Program
{
static void Main(string[] args)
{
int valor,x,suma1,suma2;
string linea;
x=1;
suma1=0;
suma2=0;
Console.Write("Primer lista");
while (x<=15)
{
Console.Write("Ingrese valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
suma1=suma1 + valor;
x=x + 1;
}
Console.Write("Segunda lista");
x=1;
while (x<=15)
{
Console.Write("Ingrese valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
suma2=suma2 + valor;
x=x + 1;
}
if (suma1>suma2)
{
Console.Write("Lista 1 mayor.");
}
else
{
if (suma2>suma1)
{
Console.Write("Lista2 mayor.");
}
else
{
Console.Write("Listas iguales.");
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaWhile11
{
class Program
{
static void Main(string[] args)
{
int n,x,valor,pares,impares;
string linea;
x=1;
pares=0;
impares=0;
Console.Write("Cuantos números ingresará:");
linea = Console.ReadLine();
n=int.Parse(linea);
while (x<=n) {
Console.Write("Ingrese el valor:");
linea = Console.ReadLine();
valor = int.Parse(linea); ;
if (valor%2==0)
{
pares=pares + 1;
}
else
{
impares=impares + 1;
}
x=x + 1;
}
Console.Write("Cantadad de pares:");
Console.WriteLine(pares);
Console.Write("Cantidad de impares:");
Console.Write(impares);
Console.ReadKey();
}
}
}
Representación gráfica:
En su forma más típica y básica, esta estructura requiere una variable entera que cumple la función de un
CONTADOR de vueltas. En la sección indicada como "inicialización contador", se suele colocar el nombre
de la variable que hará de contador, asignándole a dicha variable un valor inicial. En la sección de
"condición" se coloca la condición que deberá ser verdadera para que el ciclo continúe (en caso de un falso,
el ciclo se detendrá). Y finalmente, en la sección de "incremento contador" se coloca una instrucción que
permite modificar el valor de la variable que hace de contador (para permitir que alguna vez la condición sea
falsa)
Cuando el ciclo comienza, antes de dar la primera vuelta, la variable del for toma el valor indicado en la
sección de de "inicialización contador". Inmediatamente se verifica, en forma automática, si la condición es
verdadera. En caso de serlo se ejecuta el bloque de operaciones del ciclo, y al finalizar el mismo se ejecuta la
instrucción que se haya colocado en la tercer sección.
Seguidamente, se vuelve a controlar el valor de la condición, y así prosigue hasta que dicha condición
entregue un falso.
Si conocemos la cantidad de veces que se repite el bloque es muy sencillo emplear un for, por ejemplo si
queremo que se repita 50 veces el bloque de instrucciones puede hacerse así:
La variable del for puede tener cualquier nombre. En este ejemplo se la ha definido con el nombre f.
Analicemos el ejemplo:
La variable f PUEDE ser modificada dentro del bloque de operaciones del for, aunque esto podría causar
problemas de lógica si el programador es inexperto.
La variable f puede ser inicializada en cualquier valor y finalizar en cualquier valor. Además, no es
obligatorio que la instrucción de modificación sea un incremento del tipo contador (f++).
Cualquier instrucción que modifique el valor de la variable es válida. Si por ejemplo se escribe f=f+2 en
lugar de f++, el valor de f será incrementado de a 2 en cada vuelta, y no de a 1. En este caso, esto significará
que el ciclo no efectuará las 50 vueltas sino sólo 25.
Problema 1:
Diagrama de flujo:
Podemos observar y comparar con el problema realizado con el while. Con la estructura while el
CONTADOR x sirve para contar las vueltas. Con el for el CONTADOR f cumple dicha función.
Inicialmente f vale 1 y como no es superior a 100 se ejecuta el bloque, imprimimos el contenido de f, al
finalizar el bloque repetitivo se incrementa la variable f en 1, como 2 no es superior a 100 se repite el bloque
de instrucciones.
Cuando la variable del for llega a 101 sale de la estructura repetitiva y continúa la ejecución del algoritmo
que se indica después del círculo.
La variable f (o como sea que se decida llamarla) debe estar definida como una variable más.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor1
{
class Program
{
static void Main(string[] args)
{
int f;
for(f=1;f<=100;f++)
{
Console.Write(f);
Console.Write("-");
}
Console.ReadKey();
}
}
}
Problema 2:
: Desarrollar un programa que permita la carga de 10 valores por teclado y nos muestre posteriormente la
suma de los valores ingresados y su promedio. Este problema ya lo desarrollamos, lo resolveremos
empleando la estructura for.
Diagrama de flujo:
En este caso, a la variable del for (f) sólo se la requiere para que se repita el bloque de instrucciones 10
veces.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor2
{
class Program
{
static void Main(string[] args)
{
int suma,f,valor,promedio;
string linea;
suma=0;
for(f=1;f<=10;f++)
{
Console.Write("Ingrese valor:");
linea=Console.ReadLine();
valor=int.Parse(linea);
suma=suma+valor;
}
Console.Write("La suma es:");
Console.WriteLine(suma);
promedio=suma/10;
Console.Write("El promedio es:");
Console.Write(promedio);
Console.ReadKey();
}
}
}
Problema 3:
Escribir un programa que lea 10 notas de alumnos y nos informe cuántos tienen notas mayores o iguales a 7
y cuántos menores.
Diagrama de flujo:
Los contadores aprobados y reprobados deben imprimirse FUERA de la estructura repetitiva.
Es fundamental inicializar los contadores aprobados y reprobados en cero antes de entrar a la estructura for.
Importante: Un error común es inicializar los contadores dentro de la estructura repetitiva. En caso de
hacer esto los contadores se fijan en cero en cada ciclo del for, por lo que al finalizar el for como máximo el
contador puede tener el valor 1.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor3
{
class Program
{
static void Main(string[] args)
{
int aprobados,reprobados,f,nota;
string linea;
aprobados=0;
reprobados=0;
for(f=1;f<=10;f++)
{
Console.Write("Ingrese la nota:");
linea = Console.ReadLine();
nota=int.Parse(linea);
if (nota>=7)
{
aprobados=aprobados+1;
}
else
{
reprobados=reprobados+1;
}
}
Console.Write("Cantidad de aprobados:");
Console.WriteLine(aprobados);
Console.Write("Cantidad de reprobados:");
Console.Write(reprobados);
Console.ReadKey();
}
}
}
Problema 4:
Escribir un programa que lea 10 números enteros y luego muestre cuántos valores ingresados fueron
múltiplos de 3 y cuántos de 5. Debemos tener en cuenta que hay números que son múltiplos de 3 y de 5 a la
vez.
Diagrama de flujo:
Tengamos en cuenta que el operador matemático % retorna el resto de dividir un valor por otro, en este caso:
valor%3 retorna el resto de dividir el valor que ingresamos por teclado, por tres.
Veamos: si ingresamos 6 el resto de dividirlo por 3 es 0, si ingresamos 12 el resto de dividirlo por 3 es 0.
Generalizando: cuando el resto de dividir por 3 al valor que ingresamos por teclado es cero, se trata de un
múltiplo de dicho valor.
Ahora bien ¿por qué no hemos dispuesto una estructura if anidada? Porque hay valores que son múltiplos de
3 y de 5 a la vez. Por lo tanto con if anidados no podríamos analizar los dos casos.
Es importante darse cuenta cuando conviene emplear if anidados y cuando no debe emplearse.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor4
{
class Program
{
static void Main(string[] args)
{
int mul3,mul5,valor,f;
string linea;
mul3=0;
mul5=0;
for(f=1;f<=10;f++)
{
Console.Write("Ingrese un valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (valor%3==0)
{
mul3=mul3+1;
}
if (valor%5==0)
{
mul5=mul5+1;
}
}
Console.Write("Cantidad de valores ingresados múltiplos de 3:");
Console.WriteLine(mul3);
Console.Write("Cantidad de valores ingresados múltiplos de 5:");
Console.Write(mul5);
Console.ReadKey();
}
}
}
Problema 5:
Escribir un programa que lea n números enteros y calcule la cantidad de valores mayores o iguales a 1000.
Este tipo de problemas también se puede resolver empleando la estructura repetitiva for. Lo primero que se
hace es cargar una variable que indique la cantidad de valores a ingresar. Dicha variable se carga antes de
entrar a la estructura repetitiva for.
La estructura for permite que el valor inicial o final dependa de una variable cargada previamente por
teclado.
Diagrama de flujo:
Tenemos un contador llamado cantidad y f que es el contador del for.
La variable entera n se carga previo al inicio del for, por lo que podemos fijar el valor final del for con la
variable n.
Por ejemplo si el operador carga 5 en n la estructura repetitiva for se ejecutará 5 veces.
La variable valor se ingresa dentro de la estructura repetitiva, y se verifica si el valor de la misma es mayor o
igual a 1000, en dicho caso se incrementa en uno el contador cantidad.
Fuera de la estructura repetitiva imprimimos el contador cantidad que tiene almacenado la cantidad de
valores ingresados mayores o iguales a 1000.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor5
{
class Program
{
static void Main(string[] args)
{
int cantidad,n,f,valor;
string linea;
cantidad=0;
Console.Write("Cuantos valores ingresará:");
linea = Console.ReadLine();
n=int.Parse(linea);
for(f=1;f<=n;f++)
{
Console.Write("Ingrese el valor:");
linea = Console.ReadLine();
valor = int.Parse(linea);
if (valor>=1000)
{
cantidad=cantidad+1;
}
}
Console.Write("La cantidad de valores ingresados mayores o iguales a 1000
son:");
Console.Write(cantidad);
Console.ReadKey();
}
}
}
Problemas propuestos
Ha llegado nuevamente la parte fundamental, que es el momento donde uno desarrolla individualmente un
algoritmo para la resolución de un problema.
1. Confeccionar un programa que lea n pares de datos, cada par de datos corresponde a la medida de la
base y la altura de un triángulo. El programa deberá informar:
a) De cada triángulo la medida de su base, su altura y su superficie.
b) La cantidad de triángulos cuya superficie es mayor a 12.
2. Desarrollar un programa que solicite la carga de 10 números e imprima la suma de los últimos 5
valores ingresados.
3. Desarrollar un programa que muestre la tabla de multiplicar del 5 (del 5 al 50)
4. Confeccionar un programa que permita ingresar un valor del 1 al 10 y nos muestre la tabla de
multiplicar del mismo (los primeros 12 términos)
Ejemplo: Si ingreso 3 deberá aparecer en pantalla los valores 3, 6, 9, hasta el 36.
5. Realizar un programa que lea los lados de n triángulos, e informar:
a) De cada uno de ellos, qué tipo de triángulo es: equilátero (tres lados iguales), isósceles (dos lados
iguales), o escaleno (ningún lado igual)
b) Cantidad de triángulos de cada tipo.
c) Tipo de triángulo que posee menor cantidad.
6. Escribir un programa que pida ingresar coordenadas (x,y) que representan puntos en el plano.
Informar cuántos puntos se han ingresado en el primer, segundo, tercer y cuarto cuadrante. Al
comenzar el programa se pide que se ingrese la cantidad de puntos a procesar.
7. Se realiza la carga de 10 valores enteros por teclado. Se desea conocer:
a) La cantidad de valores ingresados negativos.
b) La cantidad de valores ingresados positivos.
c) La cantidad de múltiplos de 15.
d) El valor acumulado de los números ingresados que son pares.
8. Se cuenta con la siguiente información:
Las edades de 50 estudiantes del turno mañana.
Las edades de 60 estudiantes del turno tarde.
Las edades de 110 estudiantes del turno noche.
Las edades de cada estudiante deben ingresarse por teclado.
a) Obtener el promedio de las edades de cada turno (tres promedios)
b) Imprimir dichos promedios (promedio de cada turno)
c) Mostrar por pantalla un mensaje que indique cual de los tres turnos tiene un promedio de edades
mayor.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor6
{
class Program
{
static void Main(string[] args)
{
int basetri,altura,superficie,cantidad,f,n;
string linea;
cantidad=0;
Console.Write("Cuantos triángulos procesará:");
linea = Console.ReadLine();
n=int.Parse(linea);
for(f=1;f<=n;f++)
{
Console.Write("Ingrese el valor de la base:");
linea = Console.ReadLine();
basetri=int.Parse(linea);
Console.Write("Ingrese el valor de la altura:");
linea = Console.ReadLine();
altura=int.Parse(linea);
superficie=basetri*altura/2;
Console.Write("La superficie es:");
Console.WriteLine(superficie);
if (superficie>12)
{
cantidad=cantidad+1;
}
}
Console.Write("La cantidad de triángulos con superficie superior a 12
son:");
Console.Write(cantidad);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor7
{
class Program
{
static void Main(string[] args)
{
int f,valor,suma;
string linea;
suma=0;
for(f=1;f<=10;f++)
{
Console.Write("Ingrese un valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (f>5)
{
suma=suma+valor;
}
}
Console.Write("La suma de los últimos 5 valores es:");
Console.Write(suma);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor8
{
class Program
{
static void Main(string[] args)
{
int f;
for(f=5;f<=50;f=f+5)
{
Console.Write(f);
Console.Write("-");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor9
{
class Program
{
static void Main(string[] args)
{
int f,valor;
string linea;
Console.Write("Ingrese un valor entre 1 y 10:");
linea = Console.ReadLine();
valor=int.Parse(linea);
for(f=valor;f<=valor*12;f=f+valor)
{
Console.Write(f);
Console.Write("-");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor10
{
class Program
{
static void Main(string[] args)
{
int f,lado1,lado2,lado3,cant1,cant2,cant3,n;
string linea;
cant1=0;
cant2=0;
cant3=0;
Console.Write("Ingrese la cantidad de triángulos:");
linea = Console.ReadLine();
n=int.Parse(linea);
for(f=1;f<=n;f++)
{
Console.Write("Ingrese lado 1:");
linea = Console.ReadLine();
lado1=int.Parse(linea);
Console.Write("Ingrese lado 2:");
linea = Console.ReadLine();
lado2 = int.Parse(linea);
Console.Write("Ingrese lado 3:");
linea = Console.ReadLine();
lado3 = int.Parse(linea);
if (lado1==lado2 && lado1==lado3)
{
Console.WriteLine("Es un triángulo equilatero.");
cant1++;
}
else
{
if (lado1==lado2 || lado1==lado3 || lado2==lado3)
{
Console.WriteLine("Es un triángulo isósceles.");
cant2++;
}
else
{
cant3++;
Console.WriteLine("Es un triángulo escaleno.");
}
}
}
Console.Write("Cantidad de triángulos equilateros:");
Console.WriteLine(cant1);
Console.Write("Cantidad de triángulos isósceles:");
Console.WriteLine(cant2);
Console.Write("Cantidad de triángulos escalenos:");
Console.WriteLine(cant3);
if (cant1<cant2 && cant1<cant3)
{
Console.Write("Hay menor cantidad de triángulos equilateros.");
}
else
{
if (cant2<cant3)
{
Console.Write("Han menor cantidad de triángulos isósceles");
}
else
{
Console.Write("Han menor cantidad de triángulos escalenos");
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor11
{
class Program
{
static void Main(string[] args)
{
int n,f,x,y,cant1,cant2,cant3,cant4;
string linea;
cant1=0;
cant2=0;
cant3=0;
cant4=0;
Console.Write("Cantidad de puntos:");
linea = Console.ReadLine();
n=int.Parse(linea);
for(f=1;f<=n;f++)
{
Console.Write("Ingrese coordenada x:");
linea = Console.ReadLine();
x=int.Parse(linea);
Console.Write("Ingrese coordenada y:");
linea = Console.ReadLine();
y=int.Parse(linea);
if (x>0 && y>0)
{
cant1++;
}
else
{
if (x<0 && y>0)
{
cant2++;
}
else
{
if (x<0 && y<0)
{
cant3++;
}
else
{
if (x>0 && y<0)
{
cant4++;
}
}
}
}
}
Console.Write("Cantidad de puntos en el primer cuadrante:");
Console.WriteLine(cant1);
Console.Write("Cantidad de puntos en el segundo cuadrante:");
Console.WriteLine(cant2);
Console.Write("Cantidad de puntos en el tercer cuadrante:");
Console.WriteLine(cant3);
Console.Write("Cantidad de puntos en el cuarto cuadrante:");
Console.WriteLine(cant4);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor12
{
class Program
{
static void Main(string[] args)
{
int f,valor,negativos,positivos,mult15,sumapares;
string linea;
negativos=0;
positivos=0;
mult15=0;
sumapares=0;
for(f=1;f<=10;f++)
{
Console.Write("Ingrese valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (valor<0)
{
negativos++;
}
else
{
if (valor>0)
{
positivos++;
}
}
if (valor%15==0)
{
mult15++;
}
if (valor%2==0)
{
sumapares=sumapares+valor;
}
}
Console.Write("Cantidad de valores negativos:");
Console.WriteLine(negativos);
Console.Write("Cantidad de valores positivos:");
Console.WriteLine(positivos);
Console.Write("Cantidad de valores múltiplos de 15:");
Console.WriteLine(mult15);
Console.Write("Suma de los valores pares:");
Console.WriteLine(sumapares);
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaFor13
{
class Program
{
static void Main(string[] args)
{
int f,edad,suma1,suma2,suma3,pro1,pro2,pro3;
string linea;
suma1=0;
suma2=0;
suma3=0;
for(f=1;f<=50;f++)
{
Console.Write("Ingrese edad:");
linea = Console.ReadLine();
edad=int.Parse(linea);
suma1=suma1+edad;
}
pro1=suma1/50;
Console.Write("Promedio de edades del turno mañana:");
Console.WriteLine(pro1);
for(f=1;f<=60;f++)
{
Console.Write("Ingrese edad:");
linea = Console.ReadLine();
edad = int.Parse(linea) ;
suma2=suma2+edad;
}
pro2=suma2/60;
Console.Write("Promedio de edades del turno tarde:");
Console.WriteLine(pro2);
for(f=1;f<=110;f++)
{
Console.Write("Ingrese edad:");
linea = Console.ReadLine();
edad=int.Parse(linea);
suma3=suma3+edad;
}
pro3=suma3/110;
Console.Write("Promedio de edades del turno noche:");
Console.WriteLine(pro3);
if (pro1<pro2 && pro1<pro3)
{
Console.Write("El turno mañana tiene un promedio menor de edades.");
}
else
{
if (pro2<pro3)
{
Console.Write("El turno tarde tiene un promedio menor de edades.");
}
else
{
Console.Write("El turno noche tiene un promedio menor de edades.");
}
}
Console.ReadKey();
}
}
}
Retornar
Representación gráfica:
Problema 1:
Escribir un programa que solicite la carga de un número entre 0 y 999, y nos muestre un mensaje de cuántos
dígitos tiene el mismo. Finalizar el programa cuando se cargue el valor 0.
Diagrama de flujo:
No hay que confundir los rombos de las estructuras condicionales con los de las estructuras repetitivas do
while.
En este problema por lo menos se carga un valor. Si se carga un valor mayor o igual a 100 se trata de un
número de tres cifras, si es mayor o igual a 10 se trata de un valor de dos dígitos, en caso contrario se trata
de un valor de un dígito. Este bloque se repite hasta que se ingresa en la variable valor el número 0 con lo
que la condición de la estructura do while retorna falso y sale del bloque repetitivo finalizando el programa.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaDoWhile1
{
class Program
{
static void Main(string[] args)
{
int valor;
string linea;
do {
Console.Write("Ingrese un valor entre 0 y 999 (0 finaliza):");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (valor>=100)
{
Console.WriteLine("Tiene 3 dígitos.");
}
else
{
if (valor>=10)
{
Console.WriteLine("Tiene 2 dígitos.");
}
else
{
Console.WriteLine("Tiene 1 dígito.");
}
}
} while (valor!=0);
}
}
}
Problema 2:
Escribir un programa que solicite la carga de números por teclado, obtener su promedio. Finalizar la carga
de valores cuando se cargue el valor 0.
Cuando la finalización depende de algún valor ingresado por el operador conviene el empleo de la estructura
do while, por lo menos se cargará un valor (en el caso más extremo se carga 0, que indica la finalización de
la carga de valores)
Diagrama de flujo:
Es importante analizar este diagrama de flujo.
Definimos un contador cant que cuenta la cantidad de valores ingresados por el operador (no lo incrementa
si ingresamos 0)
El valor 0 no es parte de la serie de valores que se deben sumar.
Definimos el acumulador suma que almacena todos los valores ingresados por teclado.
La estructura repetitiva do while se repite hasta que ingresamos el valor 0. Con dicho valor la condición del
ciclo retorna falso y continúa con el flujo del diagrama.
Disponemos por último una estructura condicional para el caso que el operador cargue únicamente un 0 y
por lo tanto no podemos calcular el promedio ya que no existe la división por 0.
En caso que el contador cant tenga un valor distinto a 0 el promedio se obtiene dividiendo el acumulador
suma por el contador cant que tiene la cantidad de valores ingresados antes de introducir el 0.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaDoWhile2
{
class Program
{
static void Main(string[] args)
{
int suma,cant,valor,promedio;
string linea;
suma=0;
cant=0;
do {
Console.Write("Ingrese un valor (0 para finalizar):");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (valor!=0) {
suma=suma+valor;
cant++;
}
} while (valor!=0);
if (cant!=0) {
promedio=suma/cant;
Console.Write("El promedio de los valores ingresados es:");
Console.Write(promedio);
} else {
Console.Write("No se ingresaron valores.");
}
Console.ReadLine();
}
}
}
El contador cant DEBE inicializarse antes del ciclo, lo mismo que el acumulador suma. El promedio se
calcula siempre y cuando el contador cant sea distinto a 0.
Problema 3:
Realizar un programa que permita ingresar el peso (en kilogramos) de piezas. El proceso termina cuando
ingresamos el valor 0. Se debe informar:
a) Cuántas piezas tienen un peso entre 9.8 Kg. y 10.2 Kg.?, cuántas con más de 10.2 Kg.? y cuántas con
menos de 9.8 Kg.?
b) La cantidad total de piezas procesadas.
Diagrama de flujo:
Los tres contadores cont1, cont2, y cont3 se inicializan en 0 antes de entrar a la estructura repetitiva.
A la variable suma no se la inicializa en 0 porque no es un acumulador, sino que guarda la suma del
contenido de las variables cont1, cont2 y cont3.
La estructura se repite hasta que se ingresa el valor 0 en la variable peso. Este valor no se lo considera un
peso menor a 9.8 Kg., sino que indica que ha finalizado la carga de valores por teclado.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaDoWhile3
{
class Program
{
static void Main(string[] args)
{
int cant1,cant2,cant3,suma;
float peso;
string linea;
cant1=0;
cant2=0;
cant3=0;
do {
Console.Write("Ingrese el peso de la pieza (0 pera finalizar):");
linea = Console.ReadLine();
peso=float.Parse(linea);
if (peso>10.2)
{
cant1++;
}
else
{
if (peso>=9.8)
{
cant2++;
}
else
{
if (peso>0)
{
cant3++;
}
}
}
} while(peso!=0);
suma=cant1+cant2+cant3;
Console.Write("Piezas aptas:");
Console.WriteLine(cant2);
Console.Write("Piezas con un peso superior a 10.2:");
Console.WriteLine(cant1);
Console.Write("Piezas con un peso inferior a 9.8:");
Console.WriteLine(cant3);
Console.ReadLine();
}
}
}
Problemas propuestos
1. Realizar un programa que acumule (sume) valores ingresados por teclado hasta ingresar el 9999 (no
sumar dicho valor, indica que ha finalizado la carga). Imprimir el valor acumulado e informar si
dicho valor es cero, mayor a cero o menor a cero.
2. En un banco se procesan datos de las cuentas corrientes de sus clientes. De cada cuenta corriente se
conoce: número de cuenta y saldo actual. El ingreso de datos debe finalizar al ingresar un valor
negativo en el número de cuenta.
Se pide confeccionar un programa que lea los datos de las cuentas corrientes e informe:
a)De cada cuenta: número de cuenta y estado de la cuenta según su saldo, sabiendo que:
3. Estado de la cuenta 'Acreedor' si el saldo es >0.
4. 'Deudor' si el saldo es <0.
5. 'Nulo' si el saldo es =0.
namespace EstructuraRepetitivaDoWhile4
{
class Program
{
static void Main(string[] args)
{
int suma,valor;
string linea;
suma=0;
do {
Console.Write("Ingrese un valor:");
linea = Console.ReadLine();
valor=int.Parse(linea);
if (valor!=9999)
{
suma=suma+valor;
}
}while (valor!=9999);
Console.Write("El valor acumulado es ");
Console.WriteLine(suma);
if (suma==0)
{
Console.WriteLine("El valor acumulado es cero.");
}
else
{
if (suma>0)
{
Console.WriteLine("El valor acumulado es positivo.");
}
else
{
Console.WriteLine("El valor acumulado es negativo");
}
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EstructuraRepetitivaDoWhile5
{
class Program
{
static void Main(string[] args)
{
int cuenta;
float saldo,suma;
string linea;
suma=0;
do {
Console.Write("Ingrese número de cuenta:");
linea = Console.ReadLine();
cuenta=int.Parse(linea);
if (cuenta>=0)
{
Console.Write("Ingrese saldo:");
linea = Console.ReadLine();
saldo=float.Parse(linea);
if (saldo>0)
{
Console.WriteLine("Saldo Acreedor.");
suma=suma+saldo;
}
else
{
if (saldo<0)
{
Console.WriteLine("Saldo Deudor.");
}
else
{
Console.WriteLine("Saldo Nulo.");
}
}
}
} while(cuenta>=0);
Console.Write("Total de saldos Acreedores:");
Console.Write(suma);
Console.ReadKey();
}
}
}
12 - Cadenas de caracteres
En C# hemos visto que cuando queremos almacenar un valor entero definimos una variable de tipo int, si
queremos almacenar un valor con decimales definimos una variable de tipo float. Ahora si queremos
almacenar una cadena de caracteres (por ejemplo un nombre de una persona) debemos definir una variable
de tipo string.
En realidad hemos estado utilizando en todos los problemas planteados desde el principio la definición de
una variable de tipo string donde almacenamos cualquier dato que carga el operador por teclado, esto
debido a que la clase Console tiene el método ReadLine que carga un string.
Más adelante veremos en profundidad y detenimiento los conceptos de del manejo de string, por ahora solo
nos interesa la mecánica para trabajar con cadenas de caracteres.
Problema 1:
Solicitar el ingreso del nombre y edad de dos personas. Mostrar el nombre de la persona con mayor edad.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CadenaDeCaracteres1
{
class Program
{
static void Main(string[] args)
{
String nombre1,nombre2;
int edad1,edad2;
String linea;
Console.Write("Ingrese el nombre:");
nombre1=Console.ReadLine();
Console.Write("Ingrese edad:");
linea=Console.ReadLine();
edad1=int.Parse(linea);
Console.Write("Ingrese el nombre:");
nombre2=Console.ReadLine();
Console.Write("Ingrese edad:");
linea=Console.ReadLine();
edad2=int.Parse(linea);
Console.Write("La persona de mayor edad es:");
if (edad1>edad2)
{
Console.Write(nombre1);
}
else
{
Console.Write(nombre2);
}
Console.ReadKey();
}
}
}
Para almacenar un nombre debemos definir una variable de tipo string y su ingreso por teclado se hace
llamando al método ReadLine del objeto Console:
nombre1=Console.ReadLine();
No tenemos que hacer ninguna conversión como sucede cuando cargamos un valor de tipo int o float.
Problema 2:
Solicitar el ingreso del apellido, nombre y edad de dos personas. Mostrar el nombre de la persona con
mayor edad. Realizar la carga del apellido y nombre en una variable de tipo string.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CadenaDeCaracteres2
{
class Program
{
static void Main(string[] args)
{
string apenom1,apenom2;
int edad1,edad2;
string linea;
Console.Write("Ingrese el apellido y el nombre:");
apenom1=Console.ReadLine();
Console.Write("Ingrese edad:");
linea = Console.ReadLine();
edad1=int.Parse(linea);
Console.Write("Ingrese el apellido y el nombre:");
apenom2=Console.ReadLine();
Console.Write("Ingrese edad:");
linea = Console.ReadLine();
edad2=int.Parse(linea);
Console.Write("La persona de mayor edad es:");
if (edad1>edad2) {
Console.Write(apenom1);
} else {
Console.Write(apenom2);
}
Console.ReadKey();
}
}
}
Problema 3:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace CadenaDeCaracteres3
{
class Program
{
static void Main(string[] args)
{
string apellido1,apellido2;
Console.Write("Ingrese primer apellido:");
apellido1=Console.ReadLine();
Console.Write("Ingrese segundo apellido:");
apellido2=Console.ReadLine();
if (apellido1==apellido2)
{
Console.Write("Los apellidos son iguales");
}
else
{
Console.Write("Los apellidos son distintos");
}
Console.ReadKey();
}
}
}
Para comparar si el contenido de dos string son iguales se utiliza el operador == como si se estuvieran
comparando dos enteros.
La condición se verifica verdadero si los contenidos de los dos string son exactamente iguales, es decir si
cargamos "Martinez" en apellido1 y "martinez" en apellido2 luego retorna falso ya que no es lo mismo la
"M" mayúscula y la "m" minúscula.
En el caso que necesitemos considerar igual caracteres mayúsculas y minúsculas veremos más adelante
como resolverlo
13 - Declaración de una clase y definición de
objetos.
La programación orientada a objetos se basa en la programación de clases; a diferencia de la programación
estructurada, que está centrada en las funciones.
Una clase es un molde del que luego se pueden crear múltiples objetos, con similares características.
Una clase es una plantilla (molde), que define atributos (variables) y métodos (funciones)
La clase define los atributos y métodos comunes a los objetos de ese tipo, pero luego, cada objeto tendrá
sus propios valores y compartirán las mismas funciones.
Debemos crear una clase antes de poder crear objetos (instancias) de esa clase. Al crear un objeto de una
clase, se dice que se crea una instancia de la clase o un objeto propiamente dicho.
Problema 1:
Confeccionar una clase que permita carga el nombre y la edad de una persona. Mostrar los datos cargados.
Imprimir un mensaje si es mayor de edad (edad>=18)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase1
{
class Persona
{
private string nombre;
private int edad;
El nombre de la clase debe hacer referencia al concepto (en este caso la hemos llamado Persona):
class Persona
Veremos más adelante que un atributo es normalmente definido con la cláusula private (con esto no
permitimos el acceso al atributo desde otras clases)
A los atributos se tiene acceso desde cualquier función o método de la clase (salvo la main)
Luego de definir los atributos de la clase debemos declarar los métodos o funciones de la clase. La sintaxis
es parecida a la main (sin la cláusula static):
En el método inicializar (que será el primero que deberemos llamar desde la main) cargamos por teclado los
atributos nombre y edad. Como podemos ver el método inicializar puede hacer acceso a dos atributos de la
clase Persona.
El segundo método tiene por objetivo imprimir el contenido de los atributos nombre y edad (los datos de
los atributos se cargaron al ejecutarse previamente el método inicializar:
Console.Write("Nombre:");
Console.WriteLine(nombre);
Console.Write("Edad:");
Console.WriteLine(edad);
El tercer método tiene por objetivo mostrar un mensaje si la persona es mayor o no de edad:
Por último en la main declaramos un objeto de la clase Persona y llamamos a los métodos en un orden
adecuado:
Problema 2:
Desarrollar un programa que cargue los lados de un triángulo e implemente los siguientes métodos:
inicializar los atributos, imprimir el valor del lado mayor y otro método que muestre si es equilátero o no.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase2
{
class Triangulo
{
private int lado1, lado2, lado3;
Este problema requiere definir tres atributos de tipo entero donde almacenamos los valores de los lados del
triángulo:
El primer método que deberá llamarse desde la main es el Inicializar donde cargamos los tres atributos por
teclado:
Como podemos observar cuando un problema se vuelve más complejo es más fácil y ordenado separar los
distintos algoritmos en varios métodos y no codificar todo en la main.
El último método de esta clase verifica si los tres enteros ingresados son iguales:
Problema 3:
Desarrollar una clase que represente un punto en el plano y tenga los siguientes métodos: cargar los valores
de x e y, imprimir en que cuadrante se encuentra dicho punto (concepto matemático, primer cuadrante si x e
y son positivas, si x<0 e y>0 segundo cuadrante, etc.)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase3
{
class Punto
{
private int x, y;
void ImprimirCuadrante()
{
if (x>0 && y>0)
{
Console.Write("Se encuentra en el primer cuadrante.");
}
else
{
if (x<0 && y>0)
{
Console.Write("Se encuentra en el segundo cuadrante.");
}
else
{
if (x<0 && y<0)
{
Console.Write("Se encuentra en el tercer cuadrante.");
}
else
{
if (x>0 && y<0)
{
Console.Write("Se encuentra en el cuarto cuadrante.");
}
else
{
Console.Write("El punto no está en un cuadrante.");
}
}
}
}
Console.ReadKey();
}
private int x, y;
void ImprimirCuadrante()
{
if (x>0 && y>0)
{
Console.Write("Se encuentra en el primer cuadrante.");
}
else
{
if (x<0 && y>0)
{
Console.Write("Se encuentra en el segundo cuadrante.");
}
else
{
if (x<0 && y<0)
{
Console.Write("Se encuentra en el tercer cuadrante.");
}
else
{
if (x>0 && y<0)
{
Console.Write("Se encuentra en el cuarto cuadrante.");
}
else
{
Console.Write("El punto no está en un cuadrante.");
}
}
}
}
Console.ReadKey();
}
La Main no tiene grandes diferencias con los problemas realizados anteriormente, declaramos un objeto de
la clase Punto, creamos el objeto mediante el operador new y seguidamente llamamos a los métodos
Inicializar e ImprimirCuadrante en ese orden:
Problema 4:
Desarrollar una clase que represente un Cuadrado y tenga los siguientes métodos: cargar el valor de su lado,
imprimir su perímetro y su superficie.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase4
{
class Cuadrado
{
private int lado;
En este problema es interesante ver como no definimos dos atributos donde se almacenan la superficie y el
perímetro del cuadrado, esto debido a que solo estos datos se los requiere en el método donde se imprimen:
Esto significa que la variable perimetro es una variable local al método ImprimirPerimetro. Esta variable es
local a dicho método y solo se la puede acceder dentro del método. La diferencia fundamental entre una
variable local y un atributo de la clase es que al atributo se lo puede acceder desde cualquier método de la
clase y la variable local solo existe mientras se está ejecutando el método.
Problemas propuestos
1. Confeccionar una clase que represente un empleado. Definir como atributos su nombre y su sueldo.
Confeccionar los métodos para la carga, otro para imprimir sus datos y por último uno que imprima
un mensaje si debe pagar impuestos (si el sueldo supera a 3000)
2. Implementar la clase operaciones. Se deben cargar dos valores enteros, calcular su suma, resta,
multiplicación y división, cada una en un método, imprimir dichos resultados.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase5
{
class Empleado
{
string nombre;
float sueldo;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaClase5
{
class Operaciones
{
private int valor1, valor2;
Los parámetros los podemos imaginar como variables locales al método, pero su valor se inicializa con
datos que llegan cuando lo llamamos.
Problema 1:
Confeccionar una clase que permita ingresar valores enteros por teclado y nos muestre la tabla de multiplicar
de dicho valor. Finalizar el programa al ingresar el -1.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Tabla
{
class TablaMultiplicar
{
public void CargarValor()
{
int valor;
string linea;
do
{
Console.Write("Ingrese un valor (-1 para finalizar):");
linea = Console.ReadLine();
valor = int.Parse(linea);
if (valor != -1)
{
Calcular(valor);
}
} while (valor != -1);
}
public void Calcular(int v)
{
for(int f=v;f<=v*10;f=f+v)
{
Console.Write(f+"-");
}
Console.WriteLine();
}
El método Calcular recibe un parámetro de tipo entero, luego lo utilizamos dentro del método para mostrar
la tabla de multiplicar de dicho valor, para esto inicializamos la variable f con el valor que llega en el
parámetro. Luego de cada ejecución del for incrementamos el contador f con el valor de v.
Un método puede no tener parámetros como hemos visto en problemas anteriores o puede tener uno o más
parámetros (en caso de tener más de un parámetro los mismos se separan por coma)
El método CargarValores no tiene parámetros y tiene por objetivo cargar un valor entero por teclado y
llamar al método Calcular para que muestre la tabla de multiplicar del valor que le pasamos por teclado:
Como vemos al método Calcular lo llamamos por su nombre y entre paréntesis le pasamos el dato a enviar
(debe ser un valor o variable entera)
En este problema en la Main solo llamamos al método CargarValor, ya que el método Calcular luego es
llamado por el método CargarValor:
Cuando un método retorna un dato en vez de indicar la palabra clave void previo al nombre del método
indicamos el tipo de dato que retorna. Luego dentro del algoritmo en el momento que queremos que finalice
el mismo y retorne el dato empleamos la palabra clave return con el valor respectivo.
Problema 2:
Confeccionar una clase que permita ingresar tres valores por teclado. Luego mostrar el mayor y el menor.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace EspacioMayorMenor
{
class MayorMenor
{
Si vemos la sintaxis que calcula el mayor de tres valores enteros es similar al algoritmo visto en conceptos
anteriores:
Lo primero que podemos observar que el método retorna un entero y recibe tres parámetros:
Dentro del método verificamos cual de los tres parámetros almacena un valor mayor, a este valor lo
almacenamos en una variable local llamada "m", al valor almacenado en esta variable lo retornamos al final
con un return.
mayor=CalcularMayor(valor1,valor2,valor3);
Debemos asignar a una variable el valor devuelto por el método CalcularMayor. Luego el contenido de la
variable mayor lo mostramos:
Un vector es una estructura de datos que permite almacenar un CONJUNTO de datos del MISMO tipo.
Con un único nombre se define un vector y por medio de un subíndice hacemos referencia a cada elemento
del mismo (componente)
Problema 1:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector1
{
class PruebaVector1
{
private int[] sueldos;
En el método de Cargar lo primero que hacemos es crear el vector (en C# los vectores son objetos por lo que
es necesario proceder a su creación mediante el operador new):
Cuando creamos el vector indicamos entre corchetes la cantidad de elementos que se pueden almacenar
posteriormente en el mismo.
Para cargar cada componente debemos indicar entre corchetes que elemento del vector estamos accediendo:
La estructura de programación que más se adapta para cargar en forma completa las componentes de un
vector es un for, ya que sabemos de antemano la cantidad de valores a cargar.
Cuando f vale cero estamos accediendo a la primer componente del vector (en nuestro caso sería):
sueldos[f] = int.Parse(linea);
Lo mas común es utilizar una estructura repetitiva for para recorrer cada componente del vector.
Utilizar el for nos reduce la cantidad de código, si no utilizo un for debería en forma secuencial implementar
el siguiente código:
string linea;
Console.Write("Ingrese valor de la componente:");
linea=Console.ReadLine();
sueldos[0]=int.Parse(linea);
Console.Write("Ingrese valor de la componente:");
linea=Console.ReadLine();
sueldos[1]=int.Parse(linea);
Console.Write("Ingrese valor de la componente:");
linea=Console.ReadLine();
sueldos[2]=int.Parse(linea);
Console.Write("Ingrese valor de la componente:");
linea=Console.ReadLine();
sueldos[3]=int.Parse(linea);
Console.Write("Ingrese valor de la componente:");
linea=Console.ReadLine();
sueldos[4]=int.Parse(linea);
Siempre que queremos acceder a una componente del vector debemos indicar entre corchetes la
componente, dicho valor comienza a numerarse en cero y continua hasta un número menos del tamaño del
vector, en nuestro caso creamos el vector con 5 elementos:
Por último en este programa creamos un objeto en la Main y llamamos a lo métodos de Cargar e Imprimir el
vector:
Problema 2:
Definir un vector de 5 componentes de tipo float que representen las alturas de 5 personas.
Obtener el promedio de las mismas. Contar cuántas personas son más altas que el promedio y cuántas más
bajas.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector2
{
class PruebaVector2
{
private float[] alturas;
private float promedio;
alturas=new float[5];
En otro método procedemos a sumar todas sus componentes y obtener el promedio. El promedio lo
almacenamos en un atributo de la clase ya que lo necesitamos en otro método:
public void CalcularPromedio()
{
float suma;
suma=0;
for(int f=0; f < 5; f++)
{
suma=suma+alturas[f];
}
promedio=suma/5;
Console.WriteLine("Promedio de alturas:"+promedio);
}
Por último en un tercer método comparamos cada componente del vector con el atributo promedio, si el
valor almacenado supera al promedio incrementamos un contador en caso que sea menor al promedio
incrementamos otro contador:
Importante:
En este problema podemos observar una ventaja de tener almacenadas todas las alturas de las personas. Si
no conociéramos los vectores tenemos que cargar otra vez las alturas por teclado para compararlas con el
promedio.
Mientras el programa está en ejecución tenemos el vector alturas a nuestra disposición. Es importante tener
en cuenta que cuando finaliza la ejecución del programa se pierde el contenido de todas las variables
(simples y vectores)
Problema 3:
Una empresa tiene dos turnos (mañana y tarde) en los que trabajan 8 empleados (4 por la mañana y 4 por la
tarde)
Confeccionar un programa que permita almacenar los sueldos de los empleados agrupados por turno.
Imprimir los gastos en sueldos de cada turno.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector3
{
class PruebaVector3
{
private float[] turnoMan;
private float[] turnoTar;
Definimos dos atributos de tipo vector donde almacenaremos los sueldos de los empleados de cada turno:
turnoMan=new float[4];
turnoTar=new float[4];
En otro método procedemos a sumar las componentes de cada vector y mostrar dichos acumuladores:
float man=0;
float tar=0;
for(int f = 0; f < 4; f++)
{
man=man+turnoMan[f];
tar=tar+turnoTar[f];
}
Console.WriteLine("Total de gastos del turno de la mañana:"+man);
Console.WriteLine("Total de gastos del turno de la tarde:"+tar);
Problemas propuestos
1. Desarrollar un programa que permita ingresar un vector de 8 elementos, e informe:
El valor acumulado de todos los elementos del vector.
El valor acumulado de los elementos del vector que sean mayores a 36.
Cantidad de valores mayores a 50.
2. Realizar un programa que pida la carga de dos vectores numéricos enteros de 4 elementos. Obtener
la suma de los dos vectores, dicho resultado guardarlo en un tercer vector del mismo tamaño. Sumar
componente a componente.
3. Se tienen las notas del primer parcial de los alumnos de dos cursos, el curso A y el curso B, cada
curso cuenta con 5 alumnos.
Realizar un programa que muestre el curso que obtuvo el mayor promedio general.
4. Cargar un vector de 10 elementos y verificar posteriormente si el mismo está ordenado de menor a
mayor.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector4
{
class PruebaVector4
{
private int[] vec;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector5
{
class PruebaVector5
{
private int[] vec1;
private int[] vec2;
private int[] vecSuma;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector6
{
class PruebaVector6
{
private int[] cursoa;
private int[] cursob;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector7
{
class PruebaVector7
{
private int[] vec;
Retornar
16 - Vector (Tamaño de un vector)
Como hemos visto cuando se crea un vector indicamos entre corchetes su tamaño:
sueldos=new int[5];
Luego cuando tenemos que recorrer dicho vector disponemos una estructura repetitiva for:
for(int f=0;f<5;f++)
{
Console.Write("Ingrese valor de la componente:");
string linea;
linea=Console.ReadLine();
sueldos[f]=int.Parse(linea);
}
Como vemos el for se repite mientras el contador f vale menos de 5. Este estructura repetitiva es idéntica
cada vez que recorremos el vector.
Que pasa ahora si cambiamos el tamaño del vector cuando lo creamos:
sueldos=new int[7];
Con esto tenemos que cambiar todos los for que recorren dicho vector. Ahora veremos que un vector al ser
un objeto tiene una propiedad llamada Length que almacena su tamaño. Luego podemos modificar todos los
for con la siguiente sintaxis:
for(int f=0;f<sueldos.Length;f++)
{
Console.Write("Ingrese valor de la componente:");
string linea;
linea=Console.ReadLine();
sueldos[f]=int.Parse(linea);
}
También podemos pedir al usuario que indique el tamaño del vector en tiempo de ejecución, en estos casos
se hace imprescindible el empleo de la propiedad Length.
Problema 1:
Se desea almacenar los sueldos de operarios. Cuando se ejecuta el programa se debe pedir la cantidad de
sueldos a ingresar. Luego crear un vector con dicho tamaño.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector8
{
class PruebaVector8
{
private int[] sueldos;
Luego para la creación del mismo ingresamos una variable entera y la utilizamos como subíndice en el
momento de la creación del vector:
Luego las estructuras repetitivas las acotamos accediendo a la propiedad Length del vector:
Problemas propuestos
1. Desarrollar un programa que permita ingresar un vector de n elementos, ingresar n por teclado.
Luego imprimir la suma de todos sus elementos
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector9
{
class PruebaVector9
{
private int[] vec;
17 - Vectores paralelos
Este concepto se da cuando hay una relación entre las componentes de igual subíndice (misma posición) de
un vector y otro.
Si tenemos dos vectores de 5 elementos cada uno. En uno se almacenan los nombres de personas en el otro
las edades de dichas personas.
Decimos que el vector nombres es paralelo al vector edades si en la componente 0 de cada vector se
almacena información relacionada a una persona (Juan - 12 años)
Es decir hay una relación entre cada componente de los dos vectores.
Esta relación la conoce únicamente el programador y se hace para facilitar el desarrollo de algoritmos que
procesen los datos almacenados en las estructuras de datos.
Problema 1:
Desarrollar un programa que permita cargar 5 nombres de personas y sus edades respectivas. Luego de
realizar la carga por teclado de todos los datos imprimir los nombres de las personas mayores de edad
(mayores o iguales a 18 años)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector10
{
class PruebaVector10
{
private string[] nombres;
private int[] edades;
nombres=new string[5];
edades=new int[5];
Podemos utilizar la propiedad Length de cualquiera de los dos vectores, ya que tienen el mismo tamaño.
Para imprimir los nombres de las personas mayores de edad verificamos cada componente del vector de
edades, en caso que sea igual o mayor o 18 procedemos a mostrar el elemento de la misma posición del otro
vector:
Problema 1:
Confeccionar un programa que permita cargar los nombres de 5 operarios y sus sueldos respectivos. Mostrar
el sueldo mayor y el nombre del operario.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector11
{
class PruebaVector11
{
private string[] nombres;
private float[] sueldos;
Definimos los dos vectores paralelos donde almacenaremos los nombres y los sueldos de los operarios:
nombres=new string[5];
sueldos=new float[5];
for(int f=0;f < nombres.Length;f++)
{
Console.Write("Ingrese el nombre del empleado:");
nombres[f] = Console.ReadLine();
Console.Write("Ingrese el sueldo:");
string linea;
linea = Console.ReadLine();
sueldos[f]=float.Parse(linea);
}
Para obtener el mayor sueldo y el nombre del operario realizar los siguientes pasos:
Inicializamos una variable mayor con la primer componente del vector sueldos:
mayor=sueldos[0];
Inicializamos una variable pos con el valor 0, ya que decimos primeramente que el mayor es la primer
componente del vector:
pos=0;
Accedemos a cada componente para controlar si supera lo que tiene la variable mayor:
En caso de ser verdadera la condición asignamos a la variable mayor este nuevo valor sueldos[f]
mayor=sueldos[f];
y a la variable pos le cargamos la variable f que indica la componente que estamos analizando:
pos=f
Cuando salimos de la estructura repetitiva imprimimos la variable mayor que contiene el mayor sueldo y
para imprimir el nombre del operario conociendo la posición del mayor sueldo imprimimos el elemento que
ocupa la posición que indica la variable pos en el vector paralelo:
Problemas propuestos
1. Cargar un vector de n elementos. imprimir el menor y un mensaje si se repite dentro del vector.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector12
{
class PruebaVector12
{
private int[] vec;
private int menor;
public void Cargar()
{
Console.Write("Cuantos elementos desea cargar:");
string linea;
linea = Console.ReadLine();
int n=int.Parse(linea);
vec=new int[n];
for(int f=0;f < vec.Length;f++)
{
Console.Write("Ingrese componente:");
linea = Console.ReadLine();
vec[f]=int.Parse(linea);
}
}
El contenido de la componente vec[0] sea menor o igual al contenido de la componente vec[1] y así
sucesivamente.
Si se cumple lo dicho anteriormente decimos que el vector está ordenado de menor a mayor. Igualmente
podemos ordenar un vector de mayor a menor.
Se puede ordenar tanto vectores con componentes de tipo int, float como string. En este último caso el
ordenamiento es alfabético.
Problema 1:
Se debe crear un vector donde almacenar 5 sueldos. Ordenar el vector sueldos de menor a mayor.
Esta primera aproximación tiene por objetivo analizar los intercambios de elementos dentro del vector.
El algoritmo consiste en comparar si la primera componente es mayor a la segunda, en caso que la condición
sea verdadera, intercambiamos los contenidos de las componentes.
1200
750
820
550
490
En este ejemplo: ¿es 1200 mayor a 750? La respuesta es verdadera, por lo tanto intercambiamos el contenido
de la componente 0 con el de la componente 1.
Luego comparamos el contenido de la componente 1 con el de la componente 2: ¿Es 1200 mayor a 820?
La respuesta es verdadera entonces intercambiamos.
Si hay 5 componentes hay que hacer 4 comparaciones, por eso el for se repite 4 veces.
Generalizando: si el vector tiene N componentes hay que hacer N-1 comparaciones.
Cuando f = 0 f = 1 f = 2 f = 3
Podemos ver cómo el valor más grande del vector desciende a la última componente. Empleamos una
variable auxiliar (aux) para el proceso de intercambio:
aux=sueldos[f];
sueldos[f]=sueldos[f+1];
sueldos[f+1]=aux;
750
820
550
490
1200
Analizando el algoritmo podemos comprobar que el elemento mayor del vector se ubica ahora en el último
lugar.
Podemos definir otros vectores con distintos valores y comprobar que siempre el elemento mayor queda al
final.
Pero todavía con este algoritmo no se ordena un vector. Solamente está ordenado el último elemento del
vector.
Ahora bien, con los 4 elementos que nos quedan podemos hacer el mismo proceso visto anteriormente, con
lo cual quedará ordenado otro elemento del vector. Este proceso lo repetiremos hasta que quede ordenado
por completo el vector.
Como debemos repetir el mismo algoritmo podemos englobar todo el bloque en otra estructura repetitiva.
Realicemos una prueba del siguiente algoritmo:
Cuando k = 0
f = 0 f = 1 f = 2 f = 3
750 750 750 750
1200 820 820 820
820 1200 550 550
550 550 1200 490
490 490 490 1200
Cuando k = 1
f = 0 f = 1 f = 2 f = 3
750 750 750 750
820 550 550 550
550 820 490 490
490 490 820 820
1200 1200 1200 1200
Cuando k = 2
f = 0 f = 1 f = 2 f = 3
550 550 550 550
750 490 490 490
490 750 750 750
820 820 820 820
1200 1200 1200 1200
Cuando k = 3
f = 0 f = 1 f = 2 f = 3
490 490 490 490
550 550 550 550
750 750 750 750
820 820 820 820
1200 1200 1200 1200
¿Porque repetimos 4 veces el for externo?
Como sabemos cada vez que se repite en forma completa el for interno queda ordenada una componente del
vector. A primera vista diríamos que deberíamos repetir el for externo la cantidad de componentes del
vector, en este ejemplo el vector sueldos tiene 5 componentes.
Si observamos, cuando quedan dos elementos por ordenar, al ordenar uno de ellos queda el otro
automáticamente ordenado (podemos imaginar que si tenemos un vector con 2 elementos no se requiere el
for externo, porque este debería repetirse una única vez)
Una última consideración a este ALGORITMO de ordenamiento es que los elementos que se van ordenando
continuamos comparándolos.
Ejemplo: En la primera ejecución del for interno el valor 1200 queda ubicado en la posición 4 del vector. En
la segunda ejecución comparamos si el 820 es mayor a 1200, lo cual seguramente será falso.
Podemos concluir que la primera vez debemos hacer para este ejemplo 4 comparaciones, en la segunda
ejecución del for interno debemos hacer 3 comparaciones y en general debemos ir reduciendo en uno la
cantidad de comparaciones.
Si bien el algoritmo planteado funciona, un algoritmo más eficiente, que se deriva del anterior es el plantear
un for interno con la siguiente estructura: (f=0 ; f<4-k; f++)
Es decir restarle el valor del contador del for externo.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector13
{
class PruebaVector13
{
private int[] sueldos;
También podemos ordenar vectores cuyas componentes sean de tipo String. Para esto no podemos utilizar el
operador > sino debemos utilizar un método de la clase String:
string cad1="juan";
string cad2="analia";
if (cad1.CompareTo(cad2)>0)
{
Console.Write(cad1 + " es mayor alfabéticamente que " + cad2);
}
El método CompareTo retorna un valor mayor a cero si cad1 es mayor alfabéticamente. En este ejemplo
cad1 tiene un valor alfabéticamente mayor a cad2, luego el CompareTo retorna un valor mayor a cero.
Si los dos string son exactamente iguales el método CompareTo retorna un cero, y finalmente si cad1 es
menor alfabéticamente retorna un valor menor a cero.
Problema 2:
Definir un vector donde almacenar los nombres de 5 paises. Confeccionar el algoritmo de ordenamiento
alfabético.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector14
{
class PruebaVector14
{
private string[] paises;
paises=new string[5];
Para el ordenamiento utilizamos el método CompareTo para verificar si tenemos que intercambiar las
componentes:
if (paises[f].CompareTo(paises[f + 1])>0)
string aux;
aux = paises[f];
paises[f] = paises[f + 1];
paises[f + 1] = aux;
Problemas propuestos
1. Cargar un vector de n elementos de tipo entero. Ordenar posteriormente el vector.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector15
{
class PruebaVector15
{
private int[] vec;
Problema 1:
Confeccionar un programa que permita cargar los nombres de 5 alumnos y sus notas respectivas. Luego
ordenar las notas de mayor a menor. Imprimir las notas y los nombres de los alumnos.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector16
{
class PruebaVector16
{
private string[] nombres;
private int[] notas;
Creamos los dos vectores paralelos con cinco elementos cada uno:
nombres=new string[5];
notas=new int[5];
En el proceso de ordenamiento dentro de los dos for verificamos si debemos intercambiar los elementos del
vector notas:
En el caso que la nota de la posición 'f' sea menor a de la posición siguiente 'f+1' procedemos a intercambiar
las notas:
int auxnota;
auxnota = notas[f];
notas[f] = notas[f + 1];
notas[f + 1] = auxnota;
y simultánemamente procedemos a intercambiar los elementos del vector paralelo (con esto logramos que
los dos vectores continuen siendo vectores paralelos):
string auxnombre;
auxnombre = nombres[f];
nombres[f] = nombres[f + 1];
nombres[f + 1] = auxnombre;
Como vemos utilizamos dos auxiliares distintos porque los elementos de los dos vectores son de distinto tipo
(int y string)
Problemas propuestos
1. Cargar en un vector los nombres de 5 paises y en otro vector paralelo la cantidad de habitantes del
mismo. Ordenar alfabéticamente e imprimir los resultados. Por último ordenar con respecto a la
cantidad de habitantes (de mayor a menor) e imprimir nuevamente.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaVector17
{
class PruebaVector17
{
private string[] paises;
private int[] habitantes;
Problema 1:
Crear una matriz de 3 filas por 5 columnas con elementos de tipo int, cargar sus componentes y luego
imprimirlas.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz1
{
class Matriz1
{
private int[,] mat;
Para definir una matriz debemos disponer una coma dentro de los corchetes:
Para crear la matriz, es decir hacer la reserva de espacio de todas sus componentes debemos utilizar el
operador new y mediante dos subíndices indicamos la cantidad de filas y columnas que tendrá la matriz:
mat=new int[3,5];
Luego debemos pasar a cargar sus 15 componentes (cada fila almacena 5 componentes y tenemos 3 filas)
Lo más cómodo es utilizar un for anidado, el primer for que incrementa el contador f lo utilizamos para
recorrer las filas y el contador interno llamado c lo utilizamos para recorrer las columnas.
Cada vez que se repite en forma completa el for interno se carga una fila completa, primero se carga la fila
cero en forma completa, luego la fila uno y finalmente la fila 2.
Siempre que accedemos a una posición de la matriz debemos disponer dos subíndices que hagan referencia a
la fila y columna mat[f,c]):
Para imprimir la matriz de forma similar utilizamos dos for para acceder a cada elemento de la matriz:
Cada vez que se ejecuta todas las vueltas del for interno tenemos en pantalla una fila completa de la matriz,
por eso pasamos a ejecutar un salto de línea (con esto logramos que en pantalla los datos aparezcan en forma
matricial):
Console.WriteLine();
Problema 2:
Crear y cargar una matriz de 4 filas por 4 columnas. Imprimir la diagonal principal.
x - - -
- x - -
- - x -
- - - x
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz2
{
class Matriz2
{
private int[,] mat;
Problema 3:
Crear y cargar una matriz de 3 filas por 4 columnas. Imprimir la primer fila. Imprimir la última fila e
imprimir la primer columna.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz3
{
class Matriz3
{
private int[,] mat;
mat=new int[3,4];
Luego de cargarla el primer método que codificamos es el que imprime la primer fila. Disponemos un for
para recorrer las columnas, ya que la fila siempre será la cero. Como son cuatro los elementos de la primer
fila el for se repite esta cantidad de veces:
Para imprimir la última fila el algoritmo es similar, disponemos un for que se repita 4 veces y en el
subíndice de la fila disponemos el valor 2 (ya que la matriz tiene 3 filas):
Para imprimir la primer columna el for debe repetirse 3 veces ya que la matriz tiene 3 filas. Dejamos
constante el subíndice de la columna con el valor cero:
Console.WriteLine("Primer columna:");
for(int f = 0; f < 3; f++)
{
Console.WriteLine(mat[f,0]);
}
Problemas propuestos
1. Crear una matriz de 2 filas y 5 columnas. Realizar la carga de componentes por columna (es decir
primero ingresar toda la primer columna, luego la segunda columna y así sucesivamente)
Imprimir luego la matriz.
Solución
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz4
{
class Matriz4
{
private int[,] mat;
int[,] mat;
Creación:
mat=new int[3,4];
Como las matrices son objetos en C# disponemos de un método llamado GetLength que le pasamos como
parámetro la dimension y nos retorna el valor de dicha dimensión.
Si queremos saber la cantidad de filas que tiene la matriz debemos llamar al método GetLength con el valor
cero:
La primer dimensión son la cantidad de filas y la segunda dimensión son la cantidad de columnas de la
matriz.
Problema 1:
Crear una matriz de n * m filas (cargar n y m por teclado) Imprimir la matriz completa y la última fila.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz5
{
class Matriz5
{
private int[,] mat;
Ahora las estructuras repetitivas las acotamos preguntando a la misma matriz la cantidad de filas y la
cantidad de columnas:
El algoritmo de impresión es idéntico al visto anteriormente con la modificación de las condiciones de los
for:
Para imprimir la última fila debemos disponer un valor fijo en el subíndice de la fila (en este caso no
podemos disponer un número fijo sino preguntarle a la misma matriz la cantidad de filas y restarle uno ya
que las filas comienzan a numerarse a partir de cero: mat[mat.GetLength(0)-1,c]
También la condición del for debemos preguntar a la matriz la cantidad de columnas mat.GetLength(1):
Console.WriteLine("Ultima fila");
for(int c = 0; c < mat.GetLength(1); c++)
{
Console.Write(mat[mat.GetLength(0)-1,c]+" ");
}
Problema 2:
Crear una matriz de n * m filas (cargar n y m por teclado) Imprimir el mayor elemento y la fila y columna
donde se almacena.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz6
{
class Matriz6
{
private int[,] mat;
Para obtener el mayor elemento de la matriz y la fila y columna donde se ubica debemos inicializar una
variable mayor con el elemento de la fila cero y columna cero (esto lo hacemos suponiendo que en dicha
posición se almacena el mayor):
int mayor=mat[0,0];
int filamay=0;
int columnamay=0;
Luego mediante dos for recorremos todos los elementos de la matriz y cada vez que encontramos un
elemento mayor al actual procedemos a actualizar la variable mayor y la posición donde se almacena:
Problemas propuestos
1. Crear una matriz de n * m filas (cargar n y m por teclado) Intercambiar la primer fila con la segundo.
Imprimir luego la matriz.
2. Crear una matriz de n * m filas (cargar n y m por teclado) Imprimir los cuatro valores que se
encuentran en los vértices de la misma (mat[0][0] etc.)
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz7
{
class Matriz7
{
private int[,] mat;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz8
{
class Matriz8
{
private int[,] mat;
Retornar
Problema 1:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz9
{
class Matriz9
{
private string[] empleados;
private int[,] sueldos;
private int[] sueldostot;
Para resolver este problema lo primero que hacemos es definir una matriz donde se almacenarán los sueldos
mensuales de cada empleado, un vector de tipo string donde almacenaremos los nombre de cada empleado y
finalmente definimos un vector paralelo a la matriz donde almacenaremos la suma de cada fila de la matriz:
En el método de cargar inicializamos el vector con los nombres de los empleados y la matriz paralela donde
se almacenan los últimos tres sueldos (previo a cargar procedemos a crear el vector y la matriz):
empleados=new String[4];
sueldos=new int[4,3];
for(int f = 0; f < empleados.Length; f++)
{
Console.Write("Ingrese el nombre del empleado:");
empleados[f]=Console.ReadLine();
for(int c = 0; c < sueldos.GetLength(1); c++)
{
Console.Write("Ingrese sueldo:");
string linea;
linea = Console.ReadLine();
sueldos[f,c]=int.Parse(linea);
}
}
El método sumar sueldos crea el vector donde se almacenará la suma de cada fila de la matriz. Mediante dos
for recorremos toda la matriz y sumamos cada fila:
El método ImprimirTotalPagado tiene por objetivo mostrar los dos vectores (el de nombre de los empleados
y el que almacena la suma de cada fila de la matriz):
Por último para obtener el nombre del empleado con mayor sueldo acumulado debemos inicializar dos
variables auxiliares con el primer elemento del vector de empleados y en otra auxiliar guardamos la primer
componente del vector sueldostot:
int may=sueldostot[0];
string nom=empleados[0];
for(int f = 0; f < sueldostot.Length; f++)
{
if (sueldostot[f] > may)
{
may=sueldostot[f];
nom=empleados[f];
}
}
Console.WriteLine("El empleado con mayor sueldo es "+ nom + " que tiene un
sueldo de "+may);
Problemas propuestos
1. Se desea saber la temperatura media trimestral de cuatro paises. Para ello se tiene como dato las
temperaturas medias mensuales de dichos paises.
Se debe ingresar el nombre del país y seguidamente las tres temperaturas medias mensuales.
Seleccionar las estructuras de datos adecuadas para el almacenamiento de los datos en memoria.
a - Cargar por teclado los nombres de los paises y las temperaturas medias mensuales.
b - Imprimir los nombres de las paises y las temperaturas medias mensuales de las mismas.
c - Calcular la temperatura media trimestral de cada país.
c - Imprimr los nombres de las provincias y las temperaturas medias trimestrales.
b - Imprimir el nombre de la provincia con la temperatura media trimestral mayor.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Matriz10
{
class Matriz10
{
private string[] paises;
private int[,] tempmen;
private int[] temptri;
Como podemos ver la fila cero tiene reservado dos espacios, la fila uno reserva cuatro espacios y la última
fila reserva espacio para tres componentes.
Primero creamos la cantidad de filas dejando vacío el espacio que indica la cantidad de columnas:
mat=new int[3][];
Luego debemos ir creando cada fila de la matriz indicando la cantidad de elementos de la respectiva fila:
mat[0]=new int[2];
mat[1]=new int[4];
mat[2]=new int[3];
Luego la forma para acceder a sus componentes debe ser utilizando corchetes abiertos y cerrados para cada
índice:
mat[0][0]=120;
Dará un error si queremos cargar la tercer componente de la fila cero (esto debido a que no existe):
mat[0][2]=230;
Console.Write(mat.Length);
Problema 1:
Confeccionaremos un programa que permita crear una matriz irregular y luego imprimir la matriz en forma
completa.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace MatrizIrregular1
{
class MatrizIrregular1
{
private int[][] mat;
Primero creamos la cantidad de filas que tendrá la matriz (en los corchetes para las columnas no disponemos
valor):
Dentro del primer for pedimos que ingrese la cantidad de elementos que tendrá cada fila y utilizamos el
operador new nuevamente, pero en este caso se están creando cada fila de la matriz (C# trata a cada fila
como un vector):
Dentro del for interno hacemos la carga de las componentes propiamente dicho de la matriz (podemos ir
cargando cada fila a medida que las vamos creando):
Luego imprimimos la matriz en forma completa teniendo cuidado las condiciones que disponemos en cada
for.
El primer for se repite tantas veces como filas tiene la matriz: f<mat.Length y
el for interno se repite tantas veces como elementos tiene la fila que estamos procesando c<mat [f].Length:
Problemas propuestos
1. Confeccionar una clase para administrar una matriz irregular de 5 filas y 1 columna la primer fila, 2
columnas la segunda fila y así sucesivamente hasta 5 columnas la última fila (crearla sin la
intervención del operador)
Realizar la carga por teclado e imprimir posteriormente.
2. Confeccionar una clase para administrar los días que han faltado los 3 empleados de una empresa.
Definir un vector de 3 elementos de tipo string para cargar los nombres y una matriz irregular para
cargar los días que han faltado cada empleado (cargar el número de día que faltó)
Cada fila de la matriz representan los días de cada empleado.
Mostrar los empleados con la cantidad de inasistencias.
Cuál empleado faltó menos días.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace MatrizIrregular2
{
class MatrizIrregular2
{
private int[][] mat;
public void Cargar()
{
mat=new int[5][];
for(int f = 0; f < mat.Length; f++)
{
mat[f]=new int[f+1];
for(int c = 0; c < mat[f].Length; c++)
{
Console.Write("Ingrese componente:");
string linea = Console.ReadLine();
mat[f][c]=int.Parse(linea);
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace MatrizIrregular3
{
class MatrizIrregular3
{
private string[] nombres;
private int[][] dias;
Retornar
25 - Constructor de la clase
En C# podemos definir un método que se ejecute inicialmente y en forma automática. Este método se lo
llama constructor.
Se desea guardar los sueldos de 5 operarios en un vector. Realizar la creación y carga del vector en el
constructor.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaConstructor1
{
class Operarios
{
private int[] sueldos;
public Operarios()
{
sueldos=new int[5];
for(int f = 0; f < sueldos.Length; f++)
{
Console.Write("Ingrese el sueldo:");
string linea = Console.ReadLine();
sueldos[f]=int.Parse(linea);
}
}
Como podemos ver es el mismo problema que resolvimos cuando vimos vectores. La diferencia es que
hemos sustituido el método Cargar con el constructor:
public Operarios()
{
sueldos=new int[5];
for(int f = 0; f < sueldos.Length; f++)
{
Console.Write("Ingrese el sueldo:");
string linea = Console.ReadLine();
sueldos[f]=int.Parse(linea);
}
}
Como la clase se llama Operarios el constructor tiene el mismo nombre, no disponemos la palabra clave
void ya que el constructor no puede retornar datos.
La ventaja de plantear un constructor en lugar de definir un método con cualquier nombre es que se llamará
en forma automática cuando se crea un objeto de esta clase:
op.Imprimir();
Problema 2:
Plantear una clase llamada Alumno y definir como atributos su nombre y su edad. En el constructor realizar
la carga de datos. Definir otros dos métodos para imprimir los datos ingresados y un mensaje si es mayor o
no de edad (edad >=18)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaConstructor2
{
class Alumno
{
private string nombre;
private int edad;
public Alumno()
{
Console.Write("Ingrese nombre:");
nombre = Console.ReadLine();
Console.Write("Ingrese edad:");
string linea = Console.ReadLine();
edad=int.Parse(linea);
}
Declaramos la clase Alumno, sus dos atributos y definimos el constructor con el mismo nombre de la clase:
class Alumno
{
private string nombre;
private int edad;
public Alumno()
{
Console.Write("Ingrese nombre:");
nombre = Console.ReadLine();
Console.Write("Ingrese edad:");
string linea = Console.ReadLine();
edad=int.Parse(linea);
}
En la main el constructor se llama en forma automática cuando creamos un objeto de la clase Alumno:
Los otros dos métodos deben llamarse por su nombre y en el orden que necesitemos:
alumno1.Imprimir();
alumno1.EsMayorEdad();
Problemas propuestos
1. Confeccionar una clase que represente un empleado. Definir como atributos su nombre y su sueldo.
En el constructor cargar los atributos y luego en otro método imprimir sus datos y por último uno que
imprima un mensaje si debe pagar impuestos (si el sueldo supera a 3000)
2. Implementar la clase operaciones. Se deben cargar dos valores enteros en el constructor, calcular su
suma, resta, multiplicación y división, cada una en un método, imprimir dichos resultados.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaConstructor3
{
class EmpleadoFabrica
{
string nombre;
float sueldo;
public EmpleadoFabrica()
{
Console.Write("Ingrese el nombre del empleado:");
nombre = Console.ReadLine();
Console.Write("Ingrese su sueldo:");
string linea = Console.ReadLine();
sueldo = float.Parse(linea);
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PruebaConstructor4
{
class OperacionesCalculo
{
int valor1, valor2;
public OperacionesCalculo()
{
Console.Write("Ingrese primer valor:");
string linea = Console.ReadLine();
valor1=int.Parse(linea);
Console.Write("Ingrese segundo valor:");
linea = Console.ReadLine();
valor2=int.Parse(linea);
}
26 - Colaboración de clases
Normalmente un problema resuelto con la metodología de programación orientada a objetos no interviene
una sola clase, sino que hay muchas clases que interactúan y se comunican.
Problema 1:
Un banco tiene 3 clientes que pueden hacer depósitos y extracciones. También el banco requiere que al final
del día calcule la cantidad de dinero que hay depositada.
Cliente
atributos
nombre
monto
métodos
constructor
Depositar
Extraer
RetornarMonto
Banco
atributos
3 Cliente (3 objetos de la clase Cliente)
métodos
constructor
Operar
DepositosTotales
Creamos un proyecto llamado: Colaboracion1 y dentro del proyecto creamos dos clases llamadas: Cliente y
Banco.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Colaboracion1
{
class Cliente
{
private string nombre;
private int monto;
class Banco
{
private Cliente cliente1, cliente2, cliente3;
public Banco()
{
cliente1=new Cliente("Juan");
cliente2=new Cliente("Ana");
cliente3=new Cliente("Pedro");
}
Los atributos de una clase normalmente son privados para que no se tenga acceso directamente desde otra
clase, los atributos son modificados por los métodos de la misma clase:
El constructor recibe como parámetro el nombre del cliente y lo almacena en el atributo respectivo e
inicializa el atributo monto en cero:
Los métodos Depositar y Extraer actualizan el atributo monto con el dinero que llega como parámetro (para
simplificar el problema no hemos validado que cuando se extrae dinero el atributo monto quede con un valor
negativo):
El método RetornarMonto tiene por objetivo comunicar al Banco la cantidad de dinero que tiene el cliente
(recordemos que como el atributo monto es privado de la clase, debemos tener un método que lo retorne):
Por último el método imprimir muestra nombre y el monto de dinero del cliente:
Como podemos observar la clase Cliente no tiene función Main. Entonces donde definimos objetos de la
clase Cliente?
La respuesta a esta pregunta es que en la clase Banco definimos tres objetos de la clase Cliente.
Veamos ahora la clase Banco que requiere la colaboración de la clase Cliente.
Primero definimos tres atributos de tipo Cliente:
class Banco
{
private Cliente cliente1, cliente2, cliente3;
En le constructor creamos los tres objetos (cada vez que creamos un objeto de la clase Cliente debemos
pasar a su constructor el nombre del cliente, recordemos que su monto de depósito se inicializa con cero):
public Banco()
{
cliente1=new Cliente("Juan");
cliente2=new Cliente("Ana");
cliente3=new Cliente("Pedro");
}
El método operar del banco (llamamos a los métodos Depositar y Extraer de los clientes):
El método DepositosTotales obtiene el monto depositado de cada uno de los tres clientes, procede a
mostrarlos y llama al método imprimir de cada cliente para poder mostrar el nombre y depósito:
Por último en la Main definimos un objeto de la clase Banco (la clase Banco es la clase principal en nuestro
problema):
Problema 2:
Plantear un programa que permita jugar a los dados. Las reglas de juego son: se tiran tres dados si los tres
salen con el mismo valor mostrar un mensaje que "gano", sino "perdió".
Dado
atributos
valor
métodos
constructor
Tirar
Imprimir
RetornarValor
JuegoDeDados
atributos
3 Dado (3 objetos de la clase Dado)
métodos
constructor
Jugar
Creamos un proyecto llamado: Colaboracion2 y dentro del proyecto creamos dos clases llamadas: Dado y
JuegoDeDados.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Colaboracion2
{
class Dado
{
private int valor;
private static Random aleatorio;
public Dado()
{
aleatorio = new Random();
}
class JuegoDeDados
{
private Dado dado1,dado2,dado3;
public JuegoDeDados()
{
dado1=new Dado();
dado2=new Dado();
dado3=new Dado();
}
public void Jugar()
{
dado1.Tirar();
dado1.Imprimir();
dado2.Tirar();
dado2.Imprimir();
dado3.Tirar();
dado3.Imprimir();
if (dado1.RetornarValor()==dado2.RetornarValor() &&
dado1.RetornarValor()==dado3.RetornarValor())
{
Console.WriteLine("Ganó");
}
else
{
Console.WriteLine("Perdió");
}
Console.ReadKey();
}
La clase Dado define el atributo "valor" donde almacenamos un valor aleatorio que representa el número
que sale al tirarlo.
Definimos otro atributo de la clase Random. Esta clase nos facilita la generación de un número aleatorio que
nos indicará el valor del dato. Como luego se crearán tres objetos de la clase dado y nosotros solo
requerimos un objeto de la clase Random luego definimos el atributo de tipo static, con esto todos los
objetos de la clase Dado acceden al mismo objeto de la clase Random:
public Dado()
{
aleatorio = new Random();
}
El método Tirar almacena el valor aleatorio (para generar un valor aleatorio utilizamos el método Next de la
clase Random, el mismo genera un valor entero comprendido entre los dos parámetros que le pasamos
(nunca genera la cota superior):
El método Imprimir de la clase Dado muestra por pantalla el valor del dado:
La clase JuegoDeDatos define tres atributos de la clase Dado (con esto decimos que la clase Dado colabora
con la clase JuegoDeDados):
class JuegoDeDados
{
private Dado dado1,dado2,dado3;
public JuegoDeDados()
{
dado1=new Dado();
dado2=new Dado();
dado3=new Dado();
}
En el método Jugar llamamos al método Tirar de cada dado, pedimos que se imprima el valor generado y
finalmente procedemos a verificar si se ganó o no:
En la Main creamos solo un objeto de la clase principal (en este caso la clase principal es el JuegoDeDados):
Problemas propuestos
1. Plantear una clase Club y otra clase Socio.
La clase Socio debe tener los siguientes atributos privados: nombre y la antigüedad en el club (en
años). En el constructor pedir la carga del nombre y su antigüedad. La clase Club debe tener como
atributos 3 objetos de la clase Socio. Definir una responsabilidad para imprimir el nombre del socio
con mayor antigüedad en el club.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Colaboracion3
{
class Socio
{
private string nombre;
private int antiguedad;
public Socio()
{
Console.Write("Ingrese el nombre del socio:");
nombre = Console.ReadLine(); ;
Console.Write("Ingrese la antiguedad:");
string linea = Console.ReadLine();
antiguedad=int.Parse(linea);
}
class Club
{
private Socio socio1, socio2, socio3;
public Club()
{
socio1=new Socio();
socio2=new Socio();
socio3=new Socio();
}
Retornar
27 - Concepto de propiedad
La mayoría de los lenguajes de programación orientado a objetos acceden a sus atributos a través de
métodos. Esto lo vimos en el concepto anterior cuando accedíamos al atributo monto de un cliente:
cliente3.Depositar(200);
cliente3.Extraer(150);
En C# normalmente este tipo de problemas se lo resuelve implementado una propiedad. Veamos el mismo
problemas resolviéndolo utilizando propiedades.
Problema 1:
El problema era : Un banco tiene 3 clientes que pueden hacer depósitos y extracciones. También el banco
requiere que al final del día calcule la cantidad de dinero que hay depositada.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades1
{
class Cliente
{
private string nombre;
private int monto;
public string Nombre
{
set
{
nombre = value;
}
get
{
return nombre;
}
}
class Banco
{
private Cliente cliente1, cliente2, cliente3;
public Banco()
{
cliente1 = new Cliente();
cliente1.Nombre = "Juan";
cliente1.Monto = 0;
cliente2 = new Cliente();
cliente2.Nombre = "Ana";
cliente2.Monto = 0;
cliente3 = new Cliente();
cliente3.Nombre = "Pedro";
cliente3.Monto = 0;
}
Lo más importante es entender que una propiedad es una forma de acceder al contenido de un atributo, tanto
para consultar su valor como modificarlo.
La propiedad Nombre mediante el modificador set inicializa el atributo nombre con el valor que llega del
objeto:
cliente1.Nombre = "Juan";
Como vemos donde definimos el objeto cliente1 accedemos a la propiedad mediante el operador punto y le
asignamos un valor (en este caso un string porque la propiedad es de tipo string)
Si queremos consultar el atributo nombre lo podemos hacer mediante la propiedad Nombre. Es común
definir el nombre que le damos a la propiedad con el mismo nombre que tiene el atributo pero con el primer
caracter en mayúsculas:
//atributo en minúsculas
private int monto;
//nombre de la propiedad con el mismo nombre pero en mayúsculas.
public int Monto
{
set
{
monto = value;
}
get
{
return monto;
}
}
Podemos observar que la sintaxis para acceder a las propiedades donde definimos objetos es mucho mas
intuitiva y sencillas, por ejemplo para saber cuanto dinero hay en el banco la sintaxis con propiedades es:
Y como la vimos anteriormente por medio de un método que retorna el monto tenemos la siguiente sintaxis:
int t = cliente1.RetornarMonto () +
cliente2.RetornarMonto () +
cliente3.RetornarMonto ();
Lo primero que nos viene a la mente es porque no definir los atributos con el modificador public :
Ahora veamos que cuando consultamos o inicializamos una propiedad en realidad lo que está sucediendo es
la ejecución de un método (set o get) donde podemos disponer código donde validar el valor asignado. Por
ejemplo si disponemos la restricción que el Monto siempre debe ser positivo para que se almacene, luego
debemos codificar la propiedad con la siguiente sintaxis:
Es decir si el valor que le asignamos a la propiedad Monto es negativo luego no se inicializa el atributo
monto con dicho valor.
Si ejecutamos este código luego debe mostrar un mensaje indicando que "No se puede tener monto
negativo":
cliente1.Monto = -100;
Problema 2:
Plantear un programa que permita jugar a los dados. Las reglas de juego son: se tiran tres dados si los tres
salen con el mismo valor mostrar un mensaje que "gano", sino "perdió".
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades2
{
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Colaboracion2
{
class Dado
{
private int valor;
public Dado()
{
aleatorio = new Random();
}
class JuegoDeDados
{
private Dado dado1, dado2, dado3;
public JuegoDeDados()
{
dado1 = new Dado();
dado2 = new Dado();
dado3 = new Dado();
}
Luego cuando queremos consultar el valor del dado desde el jugo de dados por medio de la sintaxis
siguiente podemos comparar si los tres dados tienen el mismo número:
Algo importante es poder restringir la ejecución del set o get desde fuera de la clase, por ejemplo en este
caso queremos evitar que desde la clase JuegoDeDados se puede cambiar el valor del dado con la siguiente
sintaxis:
dado1.Valor=7;
La línea anterior provocará un error ya que sección del set de la propiedad la hemos definido de tipo private
(con esto hacemos que solo los métodos de la clase puedan ejecuta el set. La sintaxis para acceder a la
propiedad Valor desde la clase es:
Problemas propuestos
1. Plantear una clase Club y otra clase Socio.
La clase Socio debe tener los siguientes atributos privados: nombre y la antigüedad en el club (en
años) Definir dos propiedades para poder acceder al nombre y la antigüedad del socio(no permitir
cargar un valor negativo en la antigüedad). La clase Club debe tener como atributos 3 objetos de la
clase Socio. Definir una responsabilidad para imprimir el nombre del socio con mayor antigüedad en
el club.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades3
{
class Socio
{
private string nombre;
private int antiguedad;
class Club
{
private Socio socio1, socio2, socio3;
public Club()
{
socio1 = new Socio();
socio1.Nombre = "Juan";
socio1.Antiguedad = 7;
socio2 = new Socio();
socio2.Nombre = "Ana";
socio2.Antiguedad = 3;
socio3 = new Socio();
socio3.Nombre = "Martin";
socio3.Antiguedad = 25;
}
Retornar
28 - Herencia
Vimos en el concepto anterior que dos clases pueden estar relacionadas por la colaboración. Ahora veremos
otro tipo de relaciones entre clases que es la Herencia.
La herencia significa que se pueden crear nuevas clases partiendo de clases existentes, que tendrá todas los
atributos, propiedades y los métodos de su 'superclase' o 'clase padre' y además se le podrán añadir otros
atributos, propiedades y métodos propios.
clase padre
Clase de la que desciende o deriva una clase. Las clases hijas (descendientes) heredan (incorporan)
automáticamente los atributos, propiedades y métodos de la la clase padre.
Subclase
Clase desciendiente de otra. Hereda automáticamente los atributos, propiedades y métodos de su superclase.
Es una especialización de otra clase. Admiten la definición de nuevos atributos y métodos para aumentar la
especialización de la clase.
Vehiculo
FordK Renault 9
Siempre hacia abajo en la jerarquía hay una especialización (las subclases añaden nuevos atributos,
propiedades y métodos.
Software
DeAplicacion DeBase
El primer tipo de relación que habíamos visto entre dos clases, es la de colaboración. Recordemos que es
cuando una clase contiene un objeto de otra clase como atributo.
Cuando la relación entre dos clases es del tipo "...tiene un..." o "...es parte de...", no debemos implementar
herencia. Estamos frente a una relación de colaboración de clases no de herencia.
Si tenemos una ClaseA y otra ClaseB y notamos que entre ellas existe una relacion de tipo "... tiene un...",
no debe implementarse herencia sino declarar en la clase ClaseA un atributo de la clase ClaseB.
Por ejemplo: tenemos una clase Auto, una clase Rueda y una clase Volante. Vemos que la relación entre
ellas es: Auto "...tiene 4..." Rueda, Volante "...es parte de..." Auto; pero la clase Auto no debe derivar de
Rueda ni Volante de Auto porque la relación no es de tipo-subtipo sino de colaboración. Debemos declarar
en la clase Auto 4 atributos de tipo Rueda y 1 de tipo Volante.
Luego si vemos que dos clase responden a la pregunta ClaseA "..es un.." ClaseB es posible que haya una
relación de herencia.
Por ejemplo:
Problema 1:
Ahora plantearemos el primer problema utilizando herencia. Supongamos que necesitamos implementar dos
clases que llamaremos Suma y Resta. Cada clase tiene como atributo valor1, valor2 y resultado. Las
propiedades a definir son Valor1, Valor2 y Resultado, el método Operar (que en el caso de la clase "Suma"
suma los dos Valores y en el caso de la clase "Resta" hace la diferencia entre Valor1 y Valor2.
Si analizamos ambas clases encontramos que muchas propiedades son idénticos. En estos casos es bueno
definir una clase padre que agrupe dichas propiedades, atributos y responsabilidades comunes.
Operacion
Suma Resta
Solamente el método operar es distinto para las clases Suma y Resta (esto hace que no lo podamos disponer
en la clase Operacion), luego las propiedades Valor1, Valor2 son idénticos a las dos clases, esto hace que
podamos disponerlos en la clase Operacion. Lo mismo las propiedades Valor1, Valor2 y Resultado se
definirán en la clase padre Operacion.
Crear un proyecto y luego crear cuatro clases llamadas: Operacion, Suma, Resta y Prueba
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Herencia1
{
class Prueba
{
static void Main(string[] args)
{
Suma suma1 = new Suma();
suma1.Valor1 = 10;
suma1.Valor2 = 7;
suma1.Operar();
Console.WriteLine("La suma de " + suma1.Valor1 + " y " +
suma1.Valor2 + " es " + suma1.Resultado);
Console.ReadKey();
}
}
}
La clase Operación define tres atributos y sus tres propiedades que las acceden:
Ya veremos que definimos los atributos con este nuevo modificador de acceso (protected) para que la
subclase tenga acceso a dichos atributos. Si los definimos private las subclases no pueden acceder a dichos
atributos.
Ahora veamos como es la sintaxis para indicar que una clase hereda de otra:
Disponemos dos puntos y seguidamente el nombre de la clase padre (con esto estamos indicando que todas
las propiedades de la clase Operación son también propiedades de la clase Suma.
El método Operar puede acceder a las propiedades heredadas (siempre y cuando los mismos se declaren
protected, en caso que sean private si bien lo hereda de la clase padre solo los pueden modificar métodos de
dicha clase padre)
Ahora podemos decir que la clase Suma tiene tres propiedades y un método.
Luego en otra clase creamos un objeto de la clase Suma:
class Prueba
{
static void Main(string[] args)
{
Suma suma1 = new Suma();
suma1.Valor1 = 10;
suma1.Valor2 = 7;
suma1.Operar();
Console.WriteLine("La suma de " + suma1.Valor1 + " y " +
suma1.Valor2 + " es " + suma1.Resultado);
Console.ReadKey();
}
}
Podemos llamar tanto al método propio de la clase Suma "Operar()" como acceder a las propiedades
heredadas de la clase Operacion. Quien utilice la clase Suma solo debe conocer que métodos y propiedades
públicas tiene (independientemente que pertenezcan a la clase Suma o a una clase superior)
La clase Operación agrupa en este caso un conjunto de atributos y propiedades comunes a un conjunto de
subclases (Suma, Resta). No tiene sentido definir objetos de la clase Operacion.
El planteo de jerarquías de clases es una tarea compleja que requiere un perfecto entendimiento de todas las
clases que intervienen en un problema, cuales son sus atributos, propiedades y responsabilidades.
Problema 2:
Confeccionar una clase Persona que tenga como atributos el nombre y la edad (definir las propiedades para
poder acceder a dichos atributos). Definir como responsabilidad un método para imprimir.
Plantear una segunda clase Empleado que herede de la clase Persona. Añadir un atributo sueldo ( y su
propiedad) y el método para imprimir su sueldo.
Definir un objeto de la clase Persona y llamar a sus métodos y propiedades. También crear un objeto de la
clase Empleado y llamar a sus métodos y propiedades.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Herencia2
{
public class Persona
{
protected string nombre;
protected int edad;
class Prueba
{
static void Main(string[] args)
{
Persona persona1 = new Persona();
persona1.Nombre = "Juan";
persona1.Edad = 25;
Console.WriteLine("Los datos de la persona son:");
persona1.Imprimir();
Console.ReadKey();
}
}
}
La clase Persona define los atributos protegidos (protected) nombre y edad. Luego las propiedades públicas
Nombre y Edad (que acceden a los atributos para modificarlos o consultar sus valores.
El método imprimir es público para que se lo pueda llamar desde donde definimos un objeto de esta clase.
La clase Empleado hereda de la clase Persona y agrega un atributo llamado sueldo y la respectiva propiedad
Sueldo para acceder al atributo. Como la clase Empleado define otro método Imprimir debemos anteceder la
palabla clave new (con esto indicamos que sobreescribimos el método existente en la clase padre.
Para llamar desde el método imprimir de la clase Empleado al método imprimir de la clase Persona es con la
sintaxis base.Imprimir()
Por último en la clase Prueba creamos un objeto de la clase Persona y un objeto de la clase Empleado:
class Prueba
{
static void Main(string[] args)
{
Persona persona1 = new Persona();
persona1.Nombre = "Juan";
persona1.Edad = 25;
Console.WriteLine("Los datos de la persona son:");
persona1.Imprimir();
Console.ReadKey();
}
}
Retornar
Problema 1:
Plantear tres clases A, B, C que B herede de A y C herede de B. Definir un constructor a cada clase que
muestre un mensaje. Luego definir un objeto de la clase C.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Herencia3
{
public class A
{
public A()
{
Console.WriteLine("Constructor de la clase A");
}
}
public class B : A
{
public B()
{
Console.WriteLine("Constructor de la clase B");
}
}
public class C : B
{
public C()
{
Console.WriteLine("Constructor de la clase C");
}
}
class Prueba
{
static void Main(string[] args)
{
C obj1 = new C();
Console.ReadKey();
}
}
}
Como se puede ver con la ejecución del programa la salida por pantalla es:
Constructor de la clase A
Constructor de la clase B
Constructor de la clase C
Es decir cuando creamos un objeto de la clase C lo primero que se ejecuta es el constructor de la clase de
nivel superior (en este caso la clase A), luego el constructor de la clase B y finalmente se ejecuta el
constructor de la clase C.
Problema 2:
Plantear tres clases A, B, C que B herede de A y C herede de B. Definir un constructor a cada clase que
reciba como parámetro un entero. Luego definir un objeto de la clase C.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Herencia4
{
public class A
{
public A(int a)
{
Console.WriteLine(a);
}
}
public class B : A
{
public B(int b):base(b/2)
{
Console.WriteLine(b);
}
}
public class C : B
{
public C(int c):base(c/2)
{
Console.WriteLine(c);
}
}
class Prueba
{
static void Main(string[] args)
{
C obj1 = new C(20);
Console.ReadKey();
}
}
}
Como podemos ver la clase el constructor de la clase C debe llamar en forma explícita al constructor de la
clase padre mediante la palabra clave base con el valor a pasar (en este ejemplo le pasamos el parámetro c
dividido por dos):
Si ejecutamos el programa podemos ver que aparece por pantalla los números:
5
10
20
Donde definimos el objeto obj1 pasamos el valor 20, el constructor de la clase C llama al constructor de la
clase B pasando el valor 10, luego el constructor de la clase B llama al constructor de la clase C pasando el
valor 5. Como vimos anteriormente primero se ejecuta el constructor de la clase A mostrando el valor 5,
seguidamente el constructor de la clase B mostrando el valor 10 y finalmente se ejecuta el constructor de la
clase A mostrando el 20.
Retornar
30 - Clase parcial (partial class)
Hasta ahora hemos visto que una clase se la implementa en forma completa dentro de un archivo. El
lenguaje C# permite la implementación de una clase en dos o más archivos. Para esto hay que agregarle el
modificador partial cuando declaramos la clase.
Este concepto es ámpliamente utilizado por el entorno del Visual Studio .Net en la generación de interfaces
visuales.
Como veremos en conceptos futuros es necesario presentar "partial class" para su entendimiento.
Una clase parcial no es más ni menos que crear una clase completa y luego agrupar métodos y propiedades
en dos o más archivos.
Problema 1:
Plantear una clase Rectangulo, definir dos propiedades: Lado1 y Lado2. Definir dos métodos
RetornarSuperficie y RetornarPerimetro. Dividir la clase en dos archivos utilizando el concepto de "partial
class".
Programa:
Archivo1.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ClaseParcial1
{
partial class Rectangulo
{
private int lado1;
public int Lado1
{
set
{
lado1 = value;
}
get
{
return lado1;
}
}
private int lado2;
public int Lado2
{
set
{
lado2 = value;
}
get
{
return lado2;
}
}
}
}
Archivo2.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ClaseParcial1
{
partial class Rectangulo
{
public int RetornarSuperficie()
{
int sup = Lado1 * Lado2;
return sup;
}
Program.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ClaseParcial1
{
class Program
{
static void Main(string[] args)
{
Rectangulo rectangulo1 = new Rectangulo();
rectangulo1.Lado1 = 5;
rectangulo1.Lado2 = 10;
Console.WriteLine("La superficie del rectángulo es:" +
rectangulo1.RetornarSuperficie());
Console.WriteLine("El perímetro del rectángulo es:" +
rectangulo1.RetornarPerimetro());
Console.ReadKey();
}
}
}
8.
Retornar
En C# existen varias librerías de clase para implementar interfaces visuales. Utilizaremos las Windows
Forms.
Para crear una aplicación que utilice esta librería debemos crear un proyecto. Los pasos son los siguientes:
1. Desde el menú de opciones del Visual Studio .Net seleccionamos la opción: Archivo -> Nuevo ->
Proyecto...
2. Seleccionamos la plantilla "Aplicación de Windows Forms".
3. Ahora ya tenemos un esqueleto para desarrollar nuestra aplicación. Si vemos la ventana del
"Explorador de soluciones tenemos tres archivos generados en forma automática: Program.cs,
Form1.cs y Form1.Designer.cs:
En la parte central tenemos el Form listo para disponer controles con el mouse.
4. Ahora podemos seleccionar un control visual de la ventana "Cuadro de herramientas" que se
encuentra a la izquierda (seleccionemos el control Button) y seguidamente presionemos el botón
izquierdo del mouse dentro del formulario que se encuentra en la parte central del Visual Studio .net:
5. Ahora podemos analizar la ventana "Propiedades" que nos muestra las propiedades del objeto
seleccionado del formulario. Podemos por ejemplo si tenemos seleccionado el botón cambiar la
propiedad text (la misma cambia la etiqueta que muestra el botón):
6. Cuando ejecutamos la aplicación el resultado podemos ver que es muy distinto a la interfaz en modo
texto vista hasta el momento:
7. Por último vamos a ver los contenidos de los archivos generados automáticamente por el Visual
Studio .Net.
Program.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace WindowsFormsApplication1
{
static class Program
{
///
/// Punto de entrada principal para la aplicación.
///
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
}
}
Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
}
}
Form1.Designer.cs
namespace WindowsFormsApplication1
{
partial class Form1
{
///
/// Variable del diseñador necesaria.
///
private System.ComponentModel.IContainer components = null;
///
/// Limpiar los recursos que se estén usando.
///
/// true si los recursos administrados se deben desechar; false en caso
contrario.
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
///
/// Método necesario para admitir el Diseñador. No se puede modificar
/// el contenido de este método con el editor de código.
///
private void InitializeComponent()
{
this.button1 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// button1
//
this.button1.Location = new System.Drawing.Point(188, 214);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(75, 23);
this.button1.TabIndex = 0;
this.button1.Text = "Hola Mundo";
this.button1.UseVisualStyleBackColor = true;
//
// Form1
//
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(284, 261);
this.Controls.Add(this.button1);
this.Name = "Form1";
this.Text = "Form1";
this.ResumeLayout(false);
#endregion
Retornar
: Desarrollar un programa que muestre un objeto de cada una de las siguientes clases: MonthCalendar,
TextBox y Button
Por ejemplo si disponemos dos objetos de la clase Button y seleccionamos uno de ellos podremos editar las
propiedades del mismo en la "ventana de propiedades":
A medida que seleccionamos un objeto en la ventana de "Diseño" podemos ver como se actualiza la
"ventana de propiedades", por ejemplo la propiedad Text de la clase Button permite fijar la etiqueta que
muestra el botón.
El formulario también es un objeto, esto quiere decir que si lo seleccionamos luego la "ventana de
propiedades" nos muestra las propiedades de la clase Form:
Problema propuesto
1. Elaborar una interfaz gráfica que muestre una calculadora (utilizar objetos de la clase Button y un
objeto de la clase TextBox donde se mostrarían los resultados y se cargarían los datos), tener en
cuenta que solo se debe implementar la interfaz y no la funcionalidad de una calculadora.
Retornar
Para disponer el código para dicho evento debemos hacer doble clic sobre dicho evento (esto hace que se
active la ventana del editor y genere automáticamente el método asociado a dicho evento):
Problema:
Confeccionar un programa que al presionar un botón se muestre en un objeto de la clase Label el string
"Hola Mundo".
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplication5
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Hay que tener en cuenta que la clase anterior es parcial (el archivo Form1.Designer.cs contiene la definición
de los dos objetos y la inicialización de sus propiedades y evento):
namespace WindowsFormsApplication5
{
partial class Form1
{
///
/// Variable del diseñador requerida.
///
private System.ComponentModel.IContainer components = null;
///
/// Limpiar los recursos que se estén utilizando.
///
/// true si los recursos administrados se deben eliminar;
/// false en caso contrario, false.
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
///
/// Método necesario para admitir el Diseñador. No se puede modificar
/// el contenido del método con el editor de código.
///
private void InitializeComponent()
{
this.label1 = new System.Windows.Forms.Label();
this.button1 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// label1
//
this.label1.AutoSize = true;
this.label1.Location = new System.Drawing.Point(48, 102);
this.label1.Name = "label1";
this.label1.Size = new System.Drawing.Size(35, 13);
this.label1.TabIndex = 0;
this.label1.Text = "label1";
//
// button1
//
this.button1.Location = new System.Drawing.Point(51, 148);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(75, 23);
this.button1.TabIndex = 1;
this.button1.Text = "Presionar";
this.button1.UseVisualStyleBackColor = true;
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// Form1
//
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(292, 273);
this.Controls.Add(this.button1);
this.Controls.Add(this.label1);
this.Name = "Form1";
this.Text = "Form1";
this.ResumeLayout(false);
this.PerformLayout();
#endregion
Al ejecutar el programa si presionamos el botón vemos como cambia el contenido de la Label (esto debido a
que en el evento Click del botón cambiamos el valor de la propiedad Text del objeto de la clase Label):
Problema propuesto
1. Disponer 7 objetos de la clase Button con los días de la semana. Fijar en los atributos Text de cada
botón los días de la semana. Al presionar un botón mostrar en un objeto de la clase Label el día
seleccionado.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplication6
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Retornar
El primer control que vamos a analizar es la clase Label. Un control de tipo Label nos permite mostrar
básicamente un texto inicializando la propiedad Text.
Las propiedades más comunes de este control son:
Text: Es el string que muestra el control.
BackColor: Define el color de fondo del control.
ForeColor: Define el color del texto.
Font: Define la fuente del texto.
BorderStyle: Define si la label tiene un borde visible.
AutoSize: Permite o no redimensionar la label en forma automática.
Cursor: Definimos el ícono del cursor a mostrar cuando disponemos el mouse dentro del control.
Visible: Determina si el control está visible u oculto cuando ejecutamos el programa.
Problema propuesto
1. Crear una aplicación que muestre en 6 objetos de la clase Label con algunos nombres de controles
visuales contenidos en la pestaña de "controles comunes" del cuadro de herramientas
Problema 1:
Confeccionar un formulario que muestre tres objetos de la clase Button, disponer como etiqueta en cada
botón los valores 1,2 y 3. Cuando se presiona el botón mostrar en el título del formulario el valor de la
etiqueta del botón presionado.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationButton1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Para el evento click de cada botón inicializamos la propiedad Text del formulario con la propiedad Text del
botón presionado (como la clase Form1 hereda de la clase Form luego accedemos a la propiedad Text sin
anteceder nombre alguno: Text = button1.Text; ):
Problema 2:
Modificar el problema anterior para que se acumulen en el título del formulario los valores de los botones
presionados.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationButton2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Concatenamos el valor actual de la propiedad Text del formulario con el valor de la propiedad Text del
botón respectivo:
Problema 3:
Similar al problema anterior solo permitir mostrar hasta 10 caracteres en el título del formulario.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationButton3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Como la propiedad Text es de tipo string luego podemos acceder a la propiedad Length para conocer la
cantidad de caracteres almacenados:
Problema propuesto
1. Elaborar una interfaz gráfica que muestre una calculadora (utilizar objetos de la clase Button y un
objeto de la clase Label donde se muestra el valor ingresado), tener en cuenta que solo se debe
implementar la interfaz y la carga de un valor de hasta 12 dígitos.
2.
3. SOLUCION:
4.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationButton4
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Retornar
Problema 1:
Confeccionar un programa que permita ingresar dos valores enteros por teclado y al presionar un botón
mostrar en una Label la suma de dichos valores.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationTextBox1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Para saber el valor almacenado en un TextBox disponemos de la propiedad Text. Como la propiedad Text es
de tipo string debemos convertir dicho valor a tipo entero mediante el método estático Parse de la clase int.
Luego para recuperar como enteros los dos valores almacenados en los TextBox:
Y finalmente cargamos en un objeto de la clase Label el resultado de la suma. Como la variable suma es un
entero debemos llamar al método ToString() para retornar dicho valor como string:
label4.Text = suma.ToString();
Problema 2:
Solicitar que se ingrese una clave. Si se ingresa la cadena "abc123" mostrar un mensaje de clave correcta en
caso contrario mostrar clave incorrecta.
Utilizar un control de tipo TextBox para el ingreso de la clave y una Label para mostrar el resultado al
presionar un botón.
Inicializar la propiedad UseSystemPasswordChar con el valor true (esto hace que cuando el operador tipee
caracteres dentro del TextBox se visualicen como asteriscos)
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationTextBox2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Para verificar si la clave es correcta comparamos la cadena cargada en el textBox1 con la cadena "abc123".
Problema 3:
Disponer un control de tipo TextBox e inicializar la propiedad Multiline con el valor true (esto permite
ingresar múltiples líneas dentro de un TextBox.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationTextBox3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
MessageBox.Show(textBox2.Text);
Problema propuesto
1. Solicitar el ingreso de una clave de hasta 10 caracteres en un control de tipo TextBox (inicializar la
propiedad MaxLength con el valor 10)
Mostrar en un cuadro de mensajes la clave ingresada al presionar un botón.
SOLUCION:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationTextBox4
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Retornar
38 - Controles comunes - CheckBox
El control CheckBox permite implementar un cuadro de selección (básicamente un botón de dos estados:
seleccionado o no seleccionado)
Problema 1:
Confeccionar un programa que muestre 3 objetos de la clase CheckBox con etiquetas de tres idiomas.
Cuando se presiona un botón mostrar en la barra de títulos del Form todos los CheckBox seleccionados
hasta el momento.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationCheckBox1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
La clase CheckBox tiene una propiedad llamada Checked (si tiene el valor true significa que el CheckBox
esta seleccionado, en caso contrario no esta seleccionado.
En el evento Click del botón primero borramos el contenido del título del Form:
Text = "";
if (checkBox1.Checked == true)
{
Text = Text + "(Inglés)";
}
Problema 2:
Disponer un control Label que muestre el siguiente mensaje: "Esta de acuerdo con las normas del servicio?",
luego un CheckBox y finalmente un objeto de tipo Button desactivo (propiedad Enabled con false). Cuando
se tilde el CheckBox debemos activar el botón (para esto debemos responder al evento)
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationCheckBox2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Problema propuesto
1. Disponer tres objetos de la clase CheckBox con nombres de navegadores web. Cuando se presione
un botón mostrar en el título del Form los programas seleccionados.
SOLUCION:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationCheckBox3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Retornar
Problema 1:
Confeccionar un programa que muestre 3 objetos de la clase RadioButton que permitan configurar el ancho
y alto del Form. Cuando se presione un botón actualizar el ancho y alto.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationRadioButton1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
El control RadioButton tiene una propiedad llamada Checked que almacena true o false, por eso que por
medio de un conjunto de if verificamos cual de los radio esta seleccionado:
if (radioButton1.Checked == true)
{
Width = 640;
Height = 480;
}
else
{
if (radioButton2.Checked == true)
{
Width = 800;
Height = 600;
}
else
{
if (radioButton3.Checked == true)
{
Width = 1024;
Height = 768;
}
}
}
Para cambiar el ancho y alto del Form accedemos a las propiedades Width y Height.
Problemas propuestos
1. Permitir el ingreso de dos números en controles de tipo TextBox y mediante dos controles de tipo
RadioButton permitir seleccionar si queremos sumarlos o restarlos. Al presionar un botón mostrar en
el título del Form el resultado de la operación.
SOLUCION:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationRadioButton2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Para inicializar los string que contendrá el ComboBox debemos acceder a la propiedad Items
Un evento muy útil con este control es cuando el operador selecciona un Item de la lista. Para capturar la
selección de un item debemos codificar el evento SelectedIndexChanged.
Problema 1:
Cargar en un ComboBox los nombres de varios colores. Al seleccionar alguno mostrar en la barra de título
del Form el string seleccionado.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationComboBox1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Problema 2:
Disponer tres controles de tipo ComboBox con valores entre 0 y 255 (cada uno representa la cantidad de
rojo, verde y azul). Luego al presionar un botón pintar el fondo del Form con el color que se genera
combinando los valores de los ComboBox.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationComboBox2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
La carga manual de cada ComboBox nos haría perder mucho tiempo en tiempo de diseño por lo que lo
hacemos mediante un algoritmo. Cuando se carga el Form se ejecuta el evento Load donde mediante un for
procedemos a añadir los 256 valores:
La propiedad Items del ComboBox tiene un método llamado Add que añade un elemento a la lista (como
debemos pasar un string como parámetro convertimos a la variable entera f a string)
Luego para dejar seleccionado por defecto el primer item añadido inicializamos la propiedad SelectedIndex:
comboBox1.SelectedIndex = 0;
comboBox2.SelectedIndex = 0;
comboBox3.SelectedIndex = 0;
En el evento Click del botón procedemos a extraer el valor seleccionado de cada ComboBox y lo
convertimos a entero:
Para cambiar el color de fondo del Form actualizamos la propiedad BackColor. El color lo generamos
llamando al método estático FromArgb de la clase Color:
Problemas propuestos
1. Solicitar el ingreso del nombre de una persona y seleccionar de un control ComboBox un país. Al
presionar un botón mostrar en la barra del título del Form el nombre ingresado y el país
seleccionado.
SOLUCION:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace WindowsFormsApplicationComboBox3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Retornar
41 - Estructuras dinámicas
Conocemos algunas estructuras de datos como son los vectores y matrices. No son las únicas. Hay muchas
situaciones donde utilizar alguna de estas estructuras nos proporcionará una solución muy ineficiente
(cantidad de espacio que ocupa en memoria, velocidad de acceso a la información, etc.)
Ejemplo 1. Imaginemos que debemos realizar un procesador de texto, debemos elegir la estructura de datos
para almacenar en memoria las distintas líneas que el operador irá tipeando. Una solución factible es utilizar
una matriz de caracteres. Pero como sabemos debemos especificar la cantidad de filas y columnas que
ocupará de antemano. Podría ser por ejemplo 2000 filas y 200 columnas. Con esta definición estamos
reservando de antemano 800000 bytes de la memoria, no importa si el operador después carga una línea con
20 caracteres, igualmente ya se ha reservado una cantidad de espacio que permanecerá ociosa.
Tiene que existir alguna estructura de datos que pueda hacer más eficiente la solución del problema anterior.
Ejemplo 2. ¿Cómo estarán codificadas las planillas de cálculo? ¿Reservarán espacio para cada casilla de la
planilla al principio? Si no la lleno, ¿lo mismo se habrá reservado espacio?
Utilizar una matriz para almacenar todas las casillas de una planilla de cálculo seguro será ineficiente.
Bien, todos estos problemas y muchos más podrán ser resueltos en forma eficiente cuando conozcamos estas
nuevas estructuras de datos (Listas, árboles)
42 - Estructuras dinámicas: Listas
Una lista es un conjunto de nodos, cada uno de los cuales tiene dos campos: uno de información y un
apuntador al siguiente nodo de la lista. Además un apuntador externo señala el primer nodo de la lista.
La información puede ser cualquier tipo de dato simple, estructura de datos o inclusive uno o más objetos.
La dirección al siguiente nodo es un puntero.
Como decíamos, una lista es una secuencia de nodos (en este caso cuatro nodos). La información de los
nodos en este caso es un entero y siempre contiene un puntero que guarda la dirección del siguiente nodo.
raiz es otro puntero externo a la lista que contiene la dirección del primer nodo.
Tipos de listas.
Según el mecanismo de inserción y extracción de nodos en la lista tenemos los siguientes tipos:
Una lista se comporta como una pila si las inserciones y extracciones las hacemos por un mismo lado de la
lista. También se las llama listas LIFO (Last In First Out - último en entrar primero en salir)
Una lista se comporta como una cola si las inserciones las hacemos al final y las extracciones las hacemos
por el frente de la lista. También se las llama listas FIFO (First In First Out - primero en entrar primero en
salir)
Una lista se comporta como genérica cuando las inserciones y extracciones se realizan en cualquier parte de
la lista.
Podemos en algún momento insertar un nodo en medio de la lista, en otro momento al final, borrar uno del
frente, borrar uno del fondo o uno interior, etc.
Retornar
Importante: Una pila al ser una lista puede almacenar en el campo de información cualquier tipo de valor
(int, char, float, vector de caracteres, un objeto, etc)
Para estudiar el mecanismo de utilización de una pila supondremos que en el campo de información
almacena un entero (para una fácil interpretación y codificación)
Inicialmente la PILA está vacía y decimos que el puntero raiz apunta a null (Si apunta a null decimos que no
tiene una dirección de memoria):
Insertamos un valor entero en la pila: insertar(10)
Luego de realizar la inserción la lista tipo pila queda de esta manera: un nodo con el valor 10 y raiz apunta a
dicho nodo. El puntero del nodo apunta a null ya que no hay otro nodo después de este.
Ahora el primer nodo de la pila es el que almacena el valor cuatro. raiz apunta a dicho nodo. Recordemos
que raiz es el puntero externo a la lista que almacena la dirección del primer nodo. El nodo que acabamos de
insertar en el campo puntero guarda la dirección del nodo que almacena el valor 10.
Ahora qué sucede si extraemos un nodo de la pila. ¿Cuál se extrae? Como sabemos en una pila se extrae el
último en entrar.
Problema 1:
Confeccionar una clase que administre una lista tipo pila (se debe poder insertar, extraer e imprimir los datos
de la pila)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListasTipoPila1
{
class Pila
{
class Nodo
{
public int info;
public Nodo sig;
}
public Pila()
{
raiz = null;
}
class Nodo
{
public int info;
public Nodo sig;
}
Para declarar un nodo debemos utilizar una clase. En este caso la información del nodo (info) es un entero y
siempre el nodo tendrá una referencia de tipo Nodo, que le llamamos sig.
El puntero sig apunta al siguiente nodo o a null en caso que no exista otro nodo. Este puntero es interno a la
lista. Para poder acceder a los atributos los definimos de tipo public.
También definimos un puntero de tipo Nodo llamado raiz. Este puntero tiene la dirección del primer nodo de
la lista. En caso de estar vacía la lista, raiz apunta a null (es decir no tiene dirección)
El puntero raiz es fundamental porque al tener la dirección del primer nodo de la lista nos permite acceder a
los demás nodos.
public Pila()
{
raiz = null;
}
En el constructor de la clase hacemos que raiz guarde el valor null. Tengamos en cuenta que si raiz tiene
almacenado null la lista está vacía, en caso contrario tiene la dirección del primer nodo de la lista.
Uno de los métodos más importantes que debemos entender en una pila es el de Insertar un elemento en la
pila.
Al método llega la información a insertar, en este caso en particular es un valor entero.
Nodo nuevo;
- Creación del nodo (creación de un objeto):
nuevo = new Nodo();
Cuando se ejecuta el operador new se reserva espacio para el nodo. Realmente se crea el nodo cuando se
ejecuta el new.
nuevo.info = x;
En el campo info almacenamos lo que llega en el parámetro x. Por ejemplo si llega un 5 el nodo queda:
Por último queda enlazar el nodo que acabamos de crear al principio de la lista.
Si la lista está vacía debemos guardar en el campo sig del nodo el valor null para indicar que no hay otro
nodo después de este, y hacer que raiz apunte al nodo creado (sabemos si una lista esta vacía si raiz
almacena un null)
if (raiz == null)
{
nuevo.sig = null;
raiz = nuevo;
}
Gráficamente podemos observar que cuando indicamos raiz=nuevo, el puntero raiz guarda la dirección del
nodo apuntado por nuevo.
Tener en cuenta que cuando finaliza la ejecución del método el puntero nuevo desaparece, pero no el nodo
creado con el operador new.
En caso que la lista no esté vacía, el puntero sig del nodo que acabamos de crear debe apuntar al que es hasta
este momento el primer nodo, es decir al nodo que apunta raiz actualmente.
else
{
nuevo.sig = raiz;
raiz = nuevo;
}
Como primera actividad cargamos en el puntero sig del nodo apuntado por nuevo la dirección de raiz, y
posteriormente raiz apunta al nodo que acabamos de crear, que será ahora el primero de la lista.
Antes de los enlaces tenemos:
nuevo.sig = raiz;
Ahora tenemos:
raiz = nuevo;
La lista queda:
El método Extraer:
El objetivo del método extraer es retornar la información del primer nodo y además borrarlo de la lista.
Si la lista no está vacía guardamos en una variable local la información del primer nodo:
el nodo que previamente estaba apuntado por raiz es eliminado automáticamente, al no tener ninguna
referencia.
Retornamos la información:
return informacion;
En caso de estar vacía la pila retornamos el número entero máximo y lo tomamos como código de error (es
decir nunca debemos guardar el entero mayor en la pila)
return int.MaxValue;
Es muy importante entender gráficamente el manejo de las listas. La interpretación gráfica nos permitirá
plantear inicialmente las soluciones para el manejo de listas.
Por último expliquemos el método para recorrer una lista en forma completa e imprimir la información de
cada nodo:
Definimos un puntero auxiliar reco y hacemos que apunte al primer nodo de la lista:
Nodo reco=raiz;
Disponemos una estructura repetitiva que se repetirá mientras reco sea distinto a null. Dentro de la estructura
repetitiva hacemos que reco avance al siguiente nodo:
while (reco!=null)
{
Console.Write(reco.info+"-");
reco=reco.sig;
}
reco=reco.sig;
Estamos diciendo que reco almacena la dirección que tiene el puntero sig del nodo apuntado actualmente por
reco.
Gráficamente:
Al analizarse la condición:
while (reco!=null)
reco=reco.sig;
La condición del while nuevamente se valúa en verdadera y avanza el puntero reco al siguiente nodo:
reco=reco.sig;
Ahora sí reco apunta a null y ha llegado el final de la lista (Recordar que el último nodo de la lista tiene
almacenado en el puntero sig el valor null, con el objetivo de saber que es el último nodo)
Para poder probar esta clase recordemos que debemos definir un objeto de la misma y llamar a sus métodos:
Insertamos 3 enteros, luego imprimimos la pila, extraemos uno de la pila y finalmente imprimimos
nuevamente la pila.
Problema 2:
Agregar a la clase Pila un método que retorne la cantidad de nodos y otro que indique si esta vacía.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListasTipoPila1
{
class Pila
{
class Nodo
{
public int info;
public Nodo sig;
}
public Pila()
{
raiz = null;
}
Para verificar si la pila esta vacía verificamos el contenido de la variable raiz, si tiene null luego la lista esta
vacía y por lo tanto retornamos un true:
El algoritmo para saber la cantidad de nodos es similar al imprimir, pero en lugar de mostrar la información
del nodo procedemos a incrementar un contador:
Para probar esta clase en la main creamos un objeto de la clase Pila insertamos tres enteros:
pila1.Imprimir();
Luego mientras el método Vacia nos retorne un false (lista no vacía) procedemos a llamar al método extraer:
while (pila1.Vacia()==false)
{
Console.WriteLine(pila1.Extraer());
}
Problemas propuestos
1. Agregar un método a la clase Pila que retorne la información del primer nodo de la Pila sin borrarlo.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListasTipoPila1
{
class Pila
{
class Nodo
{
public int info;
public Nodo sig;
}
public Pila()
{
raiz = null;
}
Retornar
44 - Estructuras dinámicas: Listas tipo Pila -
Problema de aplicación
Hasta ahora hemos visto como desarrollar los algoritmos para administrar una lista tipo Pila, hemos visto
que hay bastante complejidad en el manejo de punteros pero todo esto acarrea ventajas en la solución de
problemas que requieren una estructura de tipo Pila.
Este práctico tiene por objetivo mostrar la importancia de las pilas en las Ciencias de la Computación y más
precisamente en la programación de software de bajo nivel.
Todo compilador o intérprete de un lenguaje tiene un módulo dedicado a analizar si una expresión está
correctamente codificada, es decir que los paréntesis estén abiertos y cerrados en un orden lógico y bien
balanceados.
Se debe desarrollar una clase que tenga las siguientes responsabilidades (clase Formula):
Para la solución de este problema la clase formula tendrá un atributo de la clase Pila.
Veamos como nos puede ayudar el empleo de una pila para solucionar este problema.
Primero cargaremos la fórmula en un TextBox.
}(2+[3-12]*{8/3})
Incorrecta: llega una } de cerrado y la pila está vacía.
{[2+4}]
Incorrecta: llega una llave } y en el tope de la pila hay un corchete [.
{[2+4]
Incorrecta: al finalizar el análisis del último caracter en la pila queda pendiente una
llave {.
Programa:
archivo: Pila.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Formula
{
class Pila
{
class Nodo
{
public char simbolo;
public Nodo sig;
}
public Pila()
{
raiz = null;
}
archivo: Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace Formula
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Primero declaramos y definimos la clase Pila. Almacenamos en cada nodo un caracter y llamamos al campo
de información símbolo.
No es necesario implementar los métodos Imprimir, Cantidad, etc. Porque no se requieren para este
problema.
En el método Click verifica si la fórmula están correctos los parentesis, corchetes y llaves.
Pila pila1;
pila1 = new Pila();
string cadena = textBox1.Text;
En caso de ser un ) cerrado debemos extraer un carácter de la pila y verificar si no coincide con el paréntesis
de apertura ( la fórmula está incorrecta:
if (cadena.ElementAt(f) == ')')
{
if (pila1.Extraer() != '(')
{
Text = "Incorrecta";
return;
}
}
Al finalizar el análisis de toda la cadena si la pila está vacía podemos afirmar que la fórmula está
correctamente balanceada, en caso contrario quiere decir que faltan símbolos de cerrado y es incorrecta:
if (pila1.Vacia())
{
Text = "Correcta";
}
else
{
Text = "Incorrecta";
}
Es importante entender que la clase Formula utiliza un objeto de la clase Pila para resolver el algoritmo de
verificar el balanceo de la fórmula, pero no accede directamente a los nodos de la lista.
Retornar
Confeccionaremos un programa que permita administrar una lista tipo cola. Desarrollaremos los métodos de
Insertar, Extraer, Vacia e Imprimir.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListasTipoCola1
{
class Cola
{
class Nodo
{
public int info;
public Nodo sig;
}
public Cola()
{
raiz=null;
fondo=null;
}
La declaración del nodo es igual a la clase Pila. Luego definimos dos punteros externos:
class Nodo
{
public int info;
public Nodo sig;
}
raíz apunta al principio de la lista y fondo al final de la lista. Utilizar dos punteros tiene como ventaja que
cada vez que tengamos que insertar un nodo al final de la lista no tengamos que recorrerla. Por supuesto que
es perfectamente válido implementar una cola con un único puntero externo a la lista.
En el constructor inicializamos a los dos punteros en null (Realmente esto es opcional ya que los atributos de
una clase en C# se inicializan automáticamente con null):
public Cola()
{
raiz=null;
fondo=null;
}
El método vacía retorna true si la lista no tiene nodos y false en caso contrario:
En la inserción luego de crear el nodo tenemos dos posibilidades: que la cola esté vacía, en cuyo caso los
dos punteros externos a la lista deben apuntar al nodo creado, o que haya nodos en la lista.
Nodo nuevo;
nuevo = new Nodo ();
nuevo.info = info;
nuevo.sig = null;
if (Vacia ())
{
raiz = nuevo;
fondo = nuevo;
}
else
{
fondo.sig = nuevo;
fondo = nuevo;
}
Recordemos que definimos un puntero llamado nuevo, luego creamos el nodo con el operador new y
cargamos los dos campos, el de información con lo que llega en el parámetro y el puntero con null ya que se
insertará al final de la lista, es decir no hay otro después de este.
Debemos enlazar el puntero sig del último nodo con el nodo recién creado:
fondo.sig = nuevo;
Y por último el puntero externo fondo debe apuntar al nodo apuntado por nuevo:
fondo = nuevo;
Con esto ya tenemos correctamente enlazados los nodos en la lista tipo cola. Recordar que el puntero nuevo
desaparece cuando se sale del método insertar, pero el nodo creado no se pierde porque queda enlazado en la
lista.
Si la lista no está vacía guardamos en una variable local la información del primer nodo:
Para saber si hay un solo nodo verificamos si los dos punteros raiz y fondo apuntan a la misma dirección de
memoria:
if (raiz == fondo)
Luego hacemos:
raiz = null;
fondo = null;
En caso de haber 2 o más nodos debemos avanzar el puntero raiz al siguiente nodo:
raiz = raiz.sig;
Ya tenemos la lista correctamente enlazada (raiz apunta al primer nodo y fondo continúa apuntando al
último nodo)
Retornar
Este práctico tiene por objetivo mostrar la importancia de las colas en las Ciencias de la Computación y más
precisamente en las simulaciones.
Las simulaciones permiten analizar situaciones de la realidad sin la necesidad de ejecutarlas realmente.
Tiene el beneficio que su costo es muy inferior a hacer pruebas en la realidad.
Programa:
archivo: Cola.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaTipoCola2
{
class Cola
{
class Nodo
{
public int info;
public Nodo sig;
}
private Nodo raiz, fondo;
public Cola()
{
raiz = null;
fondo = null;
}
archivo: Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace ListaTipoCola2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
La clase Form1 define tres objetos de la clase Label para mostrar los resultados de la simulación.
El método más importante es el click del botón, veamos las distintas partes de dicho método:
La variable estado almacena un cero si el cajero está libre y un uno cuando está ocupado.
La variable llegada almacena en que minuto llegará el próximo cliente (debemos generar un valor entre 2 y
3)
La variable salida almacenará en que minuto terminará el cliente de ser atendido (como al principio el cajero
está vacío inicializamos esta variable con -1)
Luego definimos un objeto de la clase Cola para poder almacenar las personas que llegan al cajero y se lo
encuentran ocupado.
Creamos un objeto de la clase Random para poder utilizar el método Next que nos retorna un valor aleatorio
en el rango que le pasamos como parámetros (si pasamos un 0 y 2 luego nos puede retornar un 0 o un 1)
Disponemos un for que se repita 600 veces (600 minutos o lo que es lo mismo 10 horas)
Dentro del for hay dos if fundamentales que verifican que sucede cuando llega una persona o cuando una
persona se retira:
if (llegada == minuto)
{
............
}
if (salida == minuto)
{
............
}
Cuando llega una persona al cajero primero verificamos si el cajero está desocupado:
if (llegada == minuto)
{
if (estado==0)
{
Si está desocupado lo ocupamos cambiando el valor de la variable estado y generando en que minuto esta
persona dejará el cajero (un valor aleatorio entre 2 y 4 minutos):
estado = 1;
salida = minuto + 2 + ale.Next(0, 3);
Si el cajero está ocupado procedemos a cargar dicha persona en la cola (insertamos el minuto que llega):
else
{
cola.Insertar(minuto);
}
El otro if importante es ver que sucede cuando sale la persona del cajero:
if (salida == minuto) {
Si sale una persona del cajero cambiamos el valor de la variable estado, incrementamos en uno el contador
cantAtendidos y si la cola no está vacía extraemos una persona, cambiamos a uno la variable estado y
generamos en que minuto dejará esta persona el cajero:
estado = 0;
cantAtendidas++;
if (!cola.Vacia())
{
cola.Extraer();
estado = 1;
salida = minuto + 2 + ale.Next(0, 3);
}
label1.Text="Atendidos:" + cantAtendidas.ToString();
label2.Text="En cola" + cola.Cantidad().ToString();
label3.Text="Minuto llegada:" + cola.Extraer().ToString();
Problemas propuestos
1. Un supermercado tiene tres cajas para la atención de los clientes.
Las cajeras tardan entre 7 y 11 minutos para la atención de cada cliente.
Los clientes llegan a la zona de cajas cada 2 ó 3 minutos. (Cuando el cliente llega, si todas las cajas
tienen 6 personas, el cliente se marcha del supermercado)
Cuando el cliente llega a la zona de cajas elige la caja con una cola menor.
SOLUCION:
Archivo: Cola.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaTipoCola3
{
class Cola
{
class Nodo
{
public int info;
public Nodo sig;
}
public Cola()
{
raiz = null;
fondo = null;
}
Archivo: Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace ListaTipoCola3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
}
llegada = minuto + 2 + ale.Next(0, 2);
}
if (salida1 == minuto)
{
cantAte1++;
estado1 = 0;
if (!cola1.Vacia())
{
estado1 = 1;
int m = cola1.Extraer();
salida1 = minuto + 7 + ale.Next(0, 5);
tiempoEnCola = tiempoEnCola + (minuto - m);
cantidadEnCola++;
}
}
if (salida2 == minuto)
{
cantAte2++;
estado2 = 0;
if (!cola2.Vacia())
{
estado2 = 1;
int m = cola2.Extraer();
salida2 = minuto + 7 + ale.Next(0, 5);
tiempoEnCola = tiempoEnCola + (minuto - m);
cantidadEnCola++;
}
}
if (salida3 == minuto)
{
cantAte3++;
estado3 = 0;
if (!cola3.Vacia())
{
estado3 = 1;
int m = cola3.Extraer();
salida3 = minuto + 7 + ale.Next(0, 5);
tiempoEnCola = tiempoEnCola + (minuto - m);
cantidadEnCola++;
}
}
}
label1.Text="Clientes atendidos por caja: caja1=" + cantAte1.ToString() + "
caja2=" + cantAte2.ToString() + " caja3=" + cantAte3.ToString();
label2.Text="Se marchan sin hacer compras:" + marchan.ToString();
if (cantidadEnCola > 0)
{
int tiempoPromedio = tiempoEnCola / cantidadEnCola;
label3.Text="Tiempo promedio en cola:" + tiempoPromedio.ToString();
}
}
}
}
Retornar
Métodos a desarrollar:
Extrae la información del nodo de la posición indicada (pos). Se debe eliminar el nodo.
int Mayor()
int PosMayor()
Debe retornar true si la lista está ordenada de menor a mayor, false en caso contrario.
bool Ordenada()
Debe retornar true si existe la información que llega en el parámetro, false en caso contrario.
bool Vacia()
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaGenerica1
{
class ListaGenerica
{
class Nodo
{
public int info;
public Nodo sig;
}
public ListaGenerica()
{
raiz = null;
}
Primero con un if verificamos que exista esa posición en la lista (por ejemplo si la lista tiene 4 nodos
podemos insertar hasta la posición 5, es decir uno más allá del último):
Ahora debemos analizar si la inserción es al principio de la lista, al final o en medio ya que los enlaces
varían según donde se lo inserta.
if (pos == 1)
Si llega un 1 luego enlazamos el puntero sig del nodo que creamos con la dirección del primer nodo de la
lista (raiz apunta siempre al primer nodo de la lista) y luego desplazamos raiz al nodo que acabamos de
crear:
nuevo.sig = raiz;
raiz = nuevo;
if (pos == Cantidad () + 1)
y enlazamos el puntero sig del último nodo de la lista con la dirección del nodo que acabamos de crear
(disponemos en sig del nodo creado el valor null ya que no hay otro nodo más adelante)
reco.sig = nuevo;
nuevo.sig = null;
Si no se inserta al principio o al final significa que tenemos que insertar en medio de la lista.
Disponemos un for donde avanzamos un puntero auxiliar y nos detenemos una posición antes a donde
tenemos que insertarlo:
En caso que exista verificamos si el nodo a extraer es el primero de la lista (este análisis debe hacerse ya que
si es el primero de la lista se modifica el puntero raiz):
if (pos == 1)
Si es el primero guardamos en una variable auxiliar la información del nodo y avanzamos el puntero raiz:
informacion = raiz.info;
raiz = raiz.sig;
Si el nodo a extraer no está al principio de la lista avanzamos con una estructura repetitiva hasta el nodo
anterior a extraer:
Luego definimos otro puntero auxiliar y lo disponemos en el siguiente nodo a donde está apuntando reco:
Ahora enlazamos el puntero sig del nodo apuntado por reco al nodo siguiente del nodo apuntado por prox
(es decir el nodo apuntado por prox queda fuera de la lista):
reco.sig = prox.sig;
El método borrar es muy similar al método extraer con la diferencia de que no retorna valor:
El método intercambiar recibe dos enteros que representan las posiciones de los nodos que queremos
intercambiar sus informaciones:
Definimos un puntero auxiliar llamado reco1, lo inicializamos con la dirección del primer nodo y mediante
un for avanzamos hasta la posición almacenada en pos1:
De forma similar con un segundo puntero auxiliar avanzamos hasta la posición indicada por pos2:
if (!Vacia ())
Suponemos que el mayor es el primero de la lista e inicializamos un puntero auxiliar con la dirección del
segundo nodo de la lista:
Cada vez que encontramos un nodo con información mayor que la variable may la actualizamos con este
nuevo valor y avanzamos el puntero reco para visitar el siguiente nodo:
El método que retorna la posición del mayor es similar al anterior con la salvedad que debemos almacenar
en otro auxiliar la posición donde se almacena el mayor:
El método que debe retornar si está ordenada la lista de menor a mayor es:
Lo primero que verificamos si la lista tiene más de un nodo significa que debemos controlarla:
if (Cantidad()>1)
Disponemos dos punteros auxiliares con las direcciones del primer y segundo nodo de la lista:
Nodo reco1=raiz;
Nodo reco2=raiz.sig;
while (reco2!=null)
{
controlamos si la información del segundo nodo es menor al nodo anterior significa que la lista no está
ordenada y podemos parar el análisis retornando un false
if (reco2.info<reco1.info)
{
return false;
Dentro del while avanzamos los dos punteros a sus nodos siguientes respectivamente.
reco2=reco2.sig;
reco1=reco1.sig;
Fuera del while retornamos true indicando que la lista está ordenada de menor a mayor
return true;
El método existe:
Nodo reco=raiz;
while (reco!=null)
{
y en cada nodo que visitamos controlamos si el parámetro x es igual a la información del nodo, en caso
afirmativo salimos del método retornando true:
if (reco.info==x)
return true;
reco=reco.sig;
Fuera del while retornamos false indicando que ningún nodo coincide con el parámetro x:
return false;
Problemas propuestos
1. Plantear una clase para administrar una lista genérica implementando los siguientes métodos:
a) Insertar un nodo al principio de la lista.
b) Insertar un nodo al final de la lista.
c) Insertar un nodo en la segunda posición. Si la lista está vacía no se inserta el nodo.
d) Insertar un nodo en la ante última posición.
e) Borrar el primer nodo.
f) Borrar el segundo nodo.
g) Borrar el último nodo.
h) Borrar el nodo con información mayor.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaGenerica2
{
class ListaGenerica
{
class Nodo
{
public int info;
public Nodo sig;
}
public ListaGenerica()
{
raiz = null;
}
void InsertarPrimero(int x)
{
Nodo nuevo = new Nodo();
nuevo.info = x;
nuevo.sig = raiz;
raiz = nuevo;
}
}
public void Imprimir()
{
Nodo reco = raiz;
while (reco != null)
{
Console.Write (reco.info + "-");
reco = reco.sig;
}
Console.WriteLine();
}
Retornar
Ejemplo:
listaOrdenada.Insertar(10)
listaOrdenada.Insertar(5)
listaOrdenada.Insertar(7)
listaOrdenada.Insertar(50)
Podemos observar que si recorremos la lista podemos acceder a la información de menor a mayor.
No se requiere un método para ordenar la lista, sino que siempre permanece ordenada, ya que se inserta
ordenada.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaOrdenada1
{
class ListaOrdenada
{
class Nodo
{
public int info;
public Nodo sig;
}
public ListaOrdenada()
{
raiz = null;
}
void Insertar(int x)
{
Nodo nuevo = new Nodo();
nuevo.info = x;
if (raiz == null)
{
raiz = nuevo;
}
else
{
if (x < raiz.info)
{
nuevo.sig = raiz;
raiz = nuevo;
}
else
{
Nodo reco = raiz;
Nodo atras = raiz;
while (x >= reco.info && reco.sig != null)
{
atras = reco;
reco = reco.sig;
}
if (x >= reco.info)
{
reco.sig = nuevo;
}
else
{
nuevo.sig = reco;
atras.sig = nuevo;
}
}
}
}
if (raiz==null)
{
raiz=nuevo;
}
else
{
Si no está vacía la lista, verificamos si lo debemos insertar en la primera posición de la lista (analizamos si la
información a insertar es menor a lo apuntado por raiz en el campo info):
if (x<raiz.info)
{
nuevo.sig=raiz;
raiz=nuevo;
}
else
{
Nodo reco=raiz;
Nodo atras=raiz;
while (x>=reco.info && reco.sig!=null)
{
atras=reco;
reco=reco.sig;
}
Cuando salimos del while si la condición (x>=reco.info) continua siendo verdadera significa que se inserta
al final de la lista, en caso contrario se inserta en medio de la lista:
if (x>=reco.info)
{
reco.sig=nuevo;
}
else
{
nuevo.sig=reco;
atras.sig=nuevo;
}
Retornar
Como ejemplo pensemos que debemos almacenar un menú de opciones en una lista, la opción a seleccionar
puede ser la siguiente o la anterior, podemos desplazarnos en ambas direcciones.
Observemos que una lista doblemente encadenada tiene dos punteros por cada nodo, uno apunta al nodo
siguiente y otro al nodo anterior.
Seguimos teniendo un puntero (raiz) que tiene la dirección del primer nodo.
El puntero sig del último nodo igual que las listas simplemente encadenadas apunta a null, y el puntero ant
del primer nodo apunta a null.
Se pueden plantear Listas tipo pila, cola y genéricas con enlace doble.
Hay que tener en cuenta que el requerimiento de memoria es mayor en las listas doblemente encadenadas ya
que tenemos dos punteros por nodo.
class Nodo {
public int info;
public Nodo sig, ant;
}
Resolveremos algunos métodos para administrar listas genéricas empleando listas doblemente encadenadas
para analizar la mecánica de enlace de nodos.
Muchos de los métodos, para listas simple y doblemente encadenadas no varía, como por ejemplo: el
constructor, Vacia, Cantidad, etc.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaGenericaDoble1
{
class ListaGenericaDoble
{
class Nodo {
public int info;
public Nodo ant,sig;
}
public ListaGenericaDoble ()
{
raiz=null;
}
Primero con un if verificamos que exista esa posición en la lista (por ejemplo si la lista tiene 4 nodos
podemos insertar hasta la posición 5, es decir uno más allá del último):
Ahora debemos analizar si la inserción es al principio de la lista, al final o en medio ya que los enlaces
varían según donde se lo inserta.
if (pos == 1)
{
Si llega un 1 luego enlazamos el puntero sig del nodo que creamos con la dirección del primer nodo de la
lista (raiz apunta siempre al primer nodo de la lista)
Verificamos si raiz está apuntando actualmente a un nodo, en caso afirmativo enlazamos el puntero ant con
el nodo que acabamos de crear y luego desplazamos raiz al nodo creado:
nuevo.sig = raiz;
if (raiz!=null)
raiz.ant=nuevo;
raiz = nuevo;
if (pos == Cantidad () + 1)
{
y enlazamos el puntero sig del último nodo de la lista con la dirección del nodo que acabamos de crear
(disponemos en sig del nodo creado el valor null ya que no hay otro nodo más adelante) El puntero ant del
nodo que creamos lo enlazamos con el nodo que era último hasta este momento y está siendo apuntado por
reco:
reco.sig = nuevo;
nuevo.ant=reco;
nuevo.sig = null;
Si no se inserta al principio o al final significa que tenemos que insertar en medio de la lista.
Disponemos un for donde avanzamos un puntero auxiliar y nos detenemos una posición antes a donde
tenemos que insertarlo:
Disponemos otro puntero auxiliar que apunte al nodo próximo a donde está apuntando reco. Ahora
enlazamos el puntero sig del nodo apuntado por reco con la dirección del nodo creado y el puntero sig del
nodo creado con la dirección del nodo siguiente. El puntero ant del nodo apuntado por nuevo lo enlazamos
con el nodo apuntado por raiz y el puntero ant del nodo apuntado por siguiente lo apuntamos a nuevo (con
esto tenemos actualizados los cuatro punteros internos a la lista):
En caso que exista verificamos si el nodo a extraer es el primero de la lista (este análisis debe hacerse ya que
si es el primero de la lista se modifica el puntero raiz):
if (pos == 1)
{
Si es el primero guardamos en una variable auxiliar la información del nodo y avanzamos el puntero raiz,
luego si raiz apunta a un nodo disponemos el puntero ant de dicho nodo a null:
informacion = raiz.info;
raiz = raiz.sig;
if (raiz!=null)
raiz.ant=null;
Si el nodo a extraer no está al principio de la lista avanzamos con una estructura repetitiva hasta el nodo
anterior a extraer:
Luego definimos otro puntero auxiliar y lo disponemos en el siguiente nodo a donde está apuntando reco:
Ahora enlazamos el puntero sig del nodo apuntado por reco al nodo siguiente del nodo apuntado por prox
(es decir el nodo apuntado por prox queda fuera de la lista) disponemos finalmente otro puntero llamado
siguiente que apunte al nodo que se encuentra una posición más adelante del nodo apuntado por prox, si
dicho puntero apunta a un nodo actualizamos el puntero ant de dicho nodo con la dirección del nodo
apuntado por reco:
reco.sig = prox.sig;
Nodo siguiente=prox.sig;
if (siguiente!=null)
siguiente.ant=reco;
informacion = prox.info;
El método borrar es muy similar al método extraer con la diferencia de que no retorna valor:
El método intercambiar recibe dos enteros que representan las posiciones de los nodos que queremos
intercambiar sus informaciones:
Definimos un puntero auxiliar llamado reco1, lo inicializamos con la dirección del primer nodo y mediante
un for avanzamos hasta la posición almacenada en pos1:
De forma similar con un segundo puntero auxiliar avanzamos hasta la posición indicada por pos2:
if (!Vacia ())
{
Suponemos que el mayor es el primero de la lista e inicializamos un puntero auxiliar con la dirección del
segundo nodo de la lista:
Cada vez que encontramos un nodo con información mayor que la variable may la actualizamos con este
nuevo valor y avanzamos el puntero reco para visitar el siguiente nodo:
return may;
El método que retorna la posición del mayor es similar al anterior con la salvedad que debemos almacenar
en otro auxiliar la posición donde se almacena el mayor:
El método que debe retornar si está ordenada la lista de menor a mayor es:
Lo primero que verificamos si la lista tiene más de un nodo significa que debemos controlarla:
if (Cantidad()>1)
{
Disponemos dos punteros auxiliares con las direcciones del primer y segundo nodo de la lista:
Nodo reco1=raiz;
Nodo reco2=raiz.sig;
while (reco2!=null)
{
controlamos si la información del segundo nodo es menor al nodo anterior significa que la lista no está
ordenada y podemos parar el análisis retornando un false
if (reco2.info<reco1.info)
{
return false;
Dentro del while avanzamos los dos punteros a sus nodos siguientes respectivamente.
reco2=reco2.sig;
reco1=reco1.sig;
Fuera del while retornamos true indicando que la lista está ordenada de menor a mayor
return true;
El método existe:
Nodo reco=raiz;
while (reco!=null)
{
y en cada nodo que visitamos controlamos si el parámetro x es igual a la información del nodo, en caso
afirmativo salimos del método retornando true:
if (reco.info==x)
return true;
reco=reco.sig;
Fuera del while retornamos false indicando que ningún nodo coincide con el parámetro x:
return false;
Problemas propuestos
1. Plantear una clase para administrar una lista genérica doblemente encadenada implementando los
siguientes métodos:
a) Insertar un nodo al principio de la lista.
b) Insertar un nodo al final de la lista.
c) Insertar un nodo en la segunda posición. Si la lista está vacía no se inserta el nodo.
d) Insertar un nodo en la ante última posición.
e) Borrar el primer nodo.
f) Borrar el segundo nodo.
g) Borrar el último nodo.
h) Borrar el nodo con información mayor.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaGenericaDoble2
{
class ListaGenericaDoble
{
class Nodo
{
public int info;
public Nodo ant, sig;
}
public ListaGenericaDoble()
{
raiz = null;
}
void InsertarPrimero(int x)
{
Nodo nuevo = new Nodo();
nuevo.info = x;
nuevo.sig = raiz;
if (raiz != null)
raiz.ant = nuevo;
raiz = nuevo;
}
}
public void Imprimir()
{
Nodo reco = raiz;
while (reco != null)
{
Console.Write(reco.info + "-");
reco = reco.sig;
}
Console.WriteLine();
}
Retornar
50 - Estructuras dinámicas: Listas genéricas
circulares
Una lista circular simplemente encadenada la podemos representar gráficamente:
Observemos que el puntero sig del último nodo apunta al primer nodo. En este tipo de listas si avanzamos
raiz no perdemos la referencia al nodo anterior ya que es un círculo.
El puntero ant del primer nodo apunta al último nodo de la lista y el puntero sig del último nodo de la lista
apunta al primero.
Resolveremos algunos métodos para administrar listas genéricas circulares doblemente encadenadas para
analizar la mecánica de enlace de nodos.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ListaCircular1
{
public class ListaCircular
{
class Nodo
{
public int info;
public Nodo ant, sig;
}
public ListaCircular()
{
raiz = null;
}
Si la lista está vacía luego tanto el puntero sig y ant apuntan a si mismo ya que debe ser circular (y raiz
apunta al nodo creado):
if (raiz==null)
{
nuevo.sig=nuevo;
nuevo.ant=nuevo;
raiz=nuevo;
En caso que la lista no esté vacía disponemos un puntero al final de la lista (el puntero ant del primer nodo
tiene dicha dirección):
}
else
{
Nodo ultimo=raiz.ant;
nuevo.sig=raiz;
nuevo.ant=ultimo;
raiz.ant=nuevo;
ultimo.sig=nuevo;
Finalmente hacemos que raiz apunte al nodo creado luego de haber hecho todos los enlaces:
raiz=nuevo;
El algoritmo es idéntico al método que inserta al principio con la salvedad que no desplazamos raiz con la
dirección del nodo creado (es decir al insertar en la posición anterior del primer nodo lo que estamos
haciendo realmente es insertar al final de la lista):
Para imprimir la lista ya no podemos disponer un puntero reco que apunte al primer nodo y que se detenga
cuando encuentre un nodo que el atributo sig almacene null.
Si la lista no está vacía disponemos un puntero en el primer nodo y utilizamos un do/while para recorrer la
lista. La condición del do/while es que se repita mientras el puntero reco sea distinto a raiz (es decir que no
haya dado toda la vuelta a la lista):
if (!Vacia())
{
Nodo reco=raiz;
do {
Console.Write(reco.info + "-");
reco = reco.sig;
} while (reco!=raiz);
Console.WriteLine();
}
}
Debemos primero identificar si es el primero de la lista (ya que en este caso se modifica el puntero externo
raiz):
if (Cantidad()==1)
{
raiz=null;
Si no disponemos un puntero al final de la lista, avanzamos raiz y enlazamos el último nodo con el segundo
de la lista:
}
else
{
Nodo ultimo=raiz.ant;
raiz = raiz.sig;
ultimo.sig=raiz;
raiz.ant=ultimo;
}
En caso que queremos borrar un nodo que se encuentra en medio de la lista o inclusive al final debemos
recorrer con un for hasta el nodo que queremos borrar y luego disponemos un puntero en el nodo anterior y
otro puntero en el nodo siguiente. Seguidamente procedemos a enlazar los nodos:
Retornar
Este concepto será de gran utilidad para el capítulo de la estructura de datos tipo árbol.
La recursividad es un concepto difícil de entender en principio, pero luego de analizar diferentes problemas
aparecen puntos comunes.
En C# los métodos pueden llamarse a sí mismos. Si dentro de un método existe la llamada a sí mismo
decimos que el método es recursivo.
Cuando un método se llama a sí mismo, se asigna espacio en la pila para las nuevas variables locales y
parámetros.
Al volver de una llamada recursiva, se recuperan de la pila las variables locales y los parámetros antiguos y
la ejecución se reanuda en el punto de la llamada al método.
Problema 1:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad1
{
public class Recursividad
{
void Repetir()
{
Repetir();
}
}
}
Problema 2:
Implementación de un método recursivo que reciba un parámetro de tipo entero y luego llame en forma
recursiva con el valor del parámetro menos 1.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad2
{
public class Recursividad
{
void Imprimir(int x)
{
Console.Write(x + " ");
Imprimir(x - 1);
}
Desde la Main se llama a la función Imprimir y se le envía el valor 5. El parámetro x recibe el valor 5. Se
ejecuta el algoritmo de la función, imprime el contenido del parámetro (5) y seguidamente se llama a una
función, en este caso a sí misma (por eso decimos que es una función recursiva), enviándole el valor 4.
El parámetro x recibe el valor 4 y se imprime en pantalla el cuatro, llamando nuevamente a la función
imprimir enviándole el valor 3.
Si continuamos este algoritmo podremos observar que en pantalla se imprime:
5 4 3 2 1 0 ?1 ?2 ?3 . . . . . . . . .
hasta que se bloquee el programa.
Tener en cuenta que cada llamada a una función consume 4 bytes por la llamada y en este caso 4 bytes por el
parámetro x. Como nunca finaliza la ejecución completa de las funciones se desborda la pila estática por las
sucesivas llamadas.
Problema 3:
Programa:
void imprimir(int x) {
if (x>0) {
System.out.println(x);
imprimir(x-1);
}
}
Ahora si podemos ejecutar este programa y observar los resultados en pantalla. Se imprimen los números 5 4
3 2 1 y no se bloquea el programa.
Analice qué sucede cada vez que el if (x>0) se evalúa como falso, ¿a qué línea del programa retorna?
Problema 4:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad4
{
public class Recursividad
{
void Imprimir(int x)
{
if (x > 0)
{
Imprimir(x - 1);
Console.WriteLine(x);
}
}
Con este ejemplo se presenta una situación donde debe analizarse línea a línea la ejecución del programa y
el porque de estos resultados.
Cuando llamamos desde la misma función le enviamos el valor de x menos 1 y la memoria queda de la
siguiente forma:
Debemos entender que el parámetro x en la nueva llamada está en otra parte de la memoria y que almacena
un 4, nosotros le llamaremos x prima.
Comienza a ejecutarse la función, la condición del if se valúa como verdadero por lo que entra al bloque y
llama recursivamente a la función Imprimir pasándole el valor 3 al parámetro.
Nuevamente la condición se valúa como verdadero y llama a la función enviándole un 2, lo mismo ocurre
cuando le envía un 1 y un 0.
void Imprimir(int x)
{
if (x > 0)
{
Imprimir(x - 1);
Console.WriteLine(x);
}
}
Cuando x vale 0 la condición del if se valúa como falsa y sale de la función Imprimir.
¿Qué línea ahora se ejecuta ?
Vuelve a la función Main ? NO.
Recordemos que la última llamada de la función Imprimir se había hecho desde la misma función Imprimir
por lo que vuelve a la línea:
Console.WriteLine(x);
Ahora si analicemos que valor tiene el parámetro x. Observemos la pila de llamadas del gráfico:
Console.WriteLine(x);
Es importante tener en cuenta que siempre en una función recursiva debe haber un if para finalizar la
recursividad ( en caso contrario la función recursiva será infinita y provocará que el programa se bloquee)
Problema 5:
Otro problema típico que se presenta para analizar la recursividad es el obtener el factorial de un número.
Recordar que el factorial de un número es el resultado que se obtiene de multiplicar dicho número por el
anterior y así sucesivamente hasta llegar a uno.
Ej. el factorial de 4 es 4 * 3 * 2 * 1 es decir 24.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad5
{
public class Recursividad
{
La función factorial es recursiva porque desde la misma función llamamos a la función Factorial.
Debemos hacer el seguimiento del problema para analizar como se calcula.
La memoria en la primera llamada:
fact recibe el valor 4 y valor se cargará con el valor que se obtenga con el producto de fact por el valor
devuelto por la función Factorial (llamada recursiva)
Es importantísimo entender la liberación del espacio de las variables locales y los parámetros en las
sucesivas llamadas recursivas.
Por último la función Main recibe "valor", en este caso el valor 24.
Problema 6:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad6
{
class Recursivdad
{
int[] vec = { 312, 614, 88, 22, 54 };
void Procesar()
{
Ordenar(vec, vec.Length);
}
void Imprimir()
{
for (int f = 0 ; f < vec.Length ; f++)
Console.WriteLine (vec [f] + " ");
Console.WriteLine();
}
Hasta ahora hemos visto problemas que se pueden resolver tanto con recursividad como con estructuras
repetitivas.
Es muy importante tener en cuenta que siempre que podamos emplear un algoritmo no recursivo será mejor
(ocupa menos memoria de ram y se ejecuta más rápidamente)
Pero hay casos donde el empleo de recursividad hace mucho más sencillo el algoritmo (tener en cuenta que
no es el caso de los tres problemas vistos previamente)
Retornar
Problema 1:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Recursividad7
{
public class Recursividad
{
class Nodo
{
public int info;
public Nodo sig;
}
void InsertarPrimero(int x)
{
Nodo nuevo = new Nodo ();
nuevo.info = x;
nuevo.sig=raiz;
raiz=nuevo;
}
Cuando llamamos al método recursivo le enviamos raiz y el parámetro reco recibe esta dirección. Si reco es
distinto a null llamamos recursivamente al método enviándole la dirección del puntero sig del nodo.
Por lo que el parámetro reco recibe la dirección del segundo nodo.
Podemos observar como en las distintas llamadas recursivas el parámetro reco apunta a un nodo. Cuando se
van desapilando las llamadas recursivas se imprime primeramente el 10 luego el 4 y por último el 5.
Problema 2:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
namespace Recursividad8
{
public class Recursividad
{
using System.IO;
Para recorrer y visitar todos los directorios y archivos de un directorio debemos implementar un algoritmo
recursivo que reciba como parámetro el directorio inicial donde comenzaremos a recorrer:
El método estático GetFiles de la clase Directory retorna todos los archivos contenidos en la carpeta que le
pasamos como parámetro. Seguidamente imprimimos todos los archivos:
String[] archivos=Directory.GetFiles(inicio);
Console.WriteLine(inicio + " (Directorio)");
for(int f=0;f < archivos.Length;f++)
{
Console.WriteLine(archivos[f]+" (Archivo)");
}
Ahora obtenemos todos los directorios contenidos en la carpeta actual y llamamos en forma recursiva para
cada directorio:
String[]directorios=Directory.GetDirectories(inicio);
for(int f=0;f < directorios.Length;f++)
{
Leer(directorios[f]);
}
Problema 3:
Desarrollar un programa que permita recorrer un laberinto e indique si tiene salida o no.
Para resolver este problema al laberinto lo representaremos con una matriz de 10 x 10 Label.
El valor:
Programa:
archivo: Form1.Designer.cs
namespace Laberinto
{
partial class Form1
{
///
/// Variable del diseñador requerida.
///
private System.ComponentModel.IContainer components = null;
///
/// Limpiar los recursos que se estén utilizando.
///
/// true si los recursos administrados se deben eliminar; false en caso
contrario, false.
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
///
/// Método necesario para admitir el Diseñador. No se puede modificar
/// el contenido del método con el editor de código.
///
private void InitializeComponent()
{
this.button1 = new System.Windows.Forms.Button();
this.button2 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// button1
//
this.button1.Location = new System.Drawing.Point(11, 15);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(75, 23);
this.button1.TabIndex = 0;
this.button1.Text = "Verificar";
this.button1.UseVisualStyleBackColor = true;
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// button2
//
this.button2.Location = new System.Drawing.Point(93, 15);
this.button2.Name = "button2";
this.button2.Size = new System.Drawing.Size(124, 23);
this.button2.TabIndex = 1;
this.button2.Text = "Otro laberinto";
this.button2.UseVisualStyleBackColor = true;
this.button2.Click += new System.EventHandler(this.button2_Click);
//
// Form1
//
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(345, 428);
this.Controls.Add(this.button2);
this.Controls.Add(this.button1);
this.Name = "Form1";
this.Text = "Form1";
this.Load += new System.EventHandler(this.Form1_Load);
this.ResumeLayout(false);
#endregion
archivo: Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace Laberinto
{
public partial class Form1 : Form
{
private Label[,] mat;
private bool salida;
public Form1()
{
InitializeComponent();
}
Primero verificamos si la coordenada a procesar del laberinto se encuentra dentro de los límites correctos y
además no hayamos encontrado la salida hasta el momento:
if (mat[fil,col].Text=="s")
salida = true;
if (mat[fil,col].Text=="0")
{
En caso de estar en el pasillo procedemos a fijar dicha Label con el caracter "9" e intentamos desplazarnos
en las cuatro direcciones (arriba, abajo, derecha e izquierda), este desplazamiento lo logramos llamando
recursivamente:
mat[fil,col].Text="9";
mat[fil,col].BackColor=Color.Red;
Recorrer(fil, col + 1);
Recorrer(fil + 1, col);
Recorrer(fil - 1, col);
Recorrer(fil, col - 1);
Problemas propuestos
1. Desarrollar el juego del Buscaminas. Definir una matriz de 10*10 de Button y disponer una 'b' para
las bombas (10 diez) un cero en los botones que no tienen bombas en su perímetro, un 1 si tiene una
bomba en su perímetro y así sucesivamente. Cuando se presiona un botón si hay un cero proceder en
forma recursiva a destapar los botones que se encuentran a sus lados. Disponer el mismo color de
frente y fondo de los botones para que el jugador no pueda ver si hay bombas o no.
SOLUCION:
archivo: Form1.Designer.cs
namespace Buscaminas
{
partial class Form1
{
///
/// Variable del diseñador requerida.
///
private System.ComponentModel.IContainer components = null;
///
/// Limpiar los recursos que se estén utilizando.
///
/// true si los recursos administrados se deben eliminar; false en caso
contrario, false.
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
///
/// Método necesario para admitir el Diseñador. No se puede modificar
/// el contenido del método con el editor de código.
///
private void InitializeComponent()
{
this.button1 = new System.Windows.Forms.Button();
this.SuspendLayout();
//
// button1
//
this.button1.Location = new System.Drawing.Point(13, 13);
this.button1.Name = "button1";
this.button1.Size = new System.Drawing.Size(75, 23);
this.button1.TabIndex = 0;
this.button1.Text = "Reiniciar";
this.button1.UseVisualStyleBackColor = true;
this.button1.Click += new System.EventHandler(this.button1_Click);
//
// Form1
//
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(496, 448);
this.Controls.Add(this.button1);
this.Name = "Form1";
this.Text = "Form1";
this.Load += new System.EventHandler(this.Form1_Load);
this.ResumeLayout(false);
#endregion
archivo: Form1.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace Buscaminas
{
public partial class Form1 : Form
{
private Button[,] mat;
public Form1()
{
InitializeComponent();
}
void Reiniciar()
{
Text="";
for (int f = 0; f < 10; f++)
{
for (int c = 0; c < 10; c++)
{
mat[f,c].Text="0";
mat[f,c].Enabled = true; ;
mat[f,c].ForeColor=Color.LightGray;
mat[f,c].BackColor=Color.LightGray;
}
}
DisponerBombas();
ContarBombasPerimetro();
}
void DisponerBombas()
{
int cantidad = 10;
Random ale = new Random();
do
{
int fila = ale.Next(0, 10);
int columna = ale.Next(0, 10);
if (mat[fila,columna].Text!="b")
{
mat[fila,columna].Text="b";
cantidad--;
}
}
while (cantidad != 0);
}
void ContarBombasPerimetro()
{
for (int f = 0; f < 10; f++)
{
for (int c = 0; c < 10; c++)
{
if (mat[f,c].Text=="0")
{
int cant = ContarCoordenada(f, c);
mat[f,c].Text=cant.ToString();
}
}
}
}
void DesactivarJuego()
{
for (int f = 0; f < 10; f++)
{
for (int c = 0; c < 10; c++)
{
mat[f,c].Enabled=false;
}
}
}
void VerificarTriunfo()
{
int cant = 0;
for (int f = 0; f < 10; f++)
{
for (int c = 0; c < 10; c++)
{
Color col = mat[f,c].BackColor;
if (col == Color.Yellow)
cant++;
}
}
if (cant == 90)
{
Text="Ganooooooooo";
DesactivarJuego();
}
}
Retornar
Nivel de un árbol: El nodo A está en el nivel 1 sus descendientes directos están en el nivel 2 y así
sucesivamente.
El nivel del árbol está dado por el nodo de máximo nivel.
Ej. Este árbol es de nivel 3.
Grado de un nodo: es el número de nodos hijos que tiene dicho nodo (solo se tiene en cuenta los nodos
interiores)
Ej. El nodo A tiene grado 3.
El nodo B tiene grado 2.
Los otros nodos no tienen grado porque no tienen descendientes.
Longitud de camino del nodo x: Al número de arcos que deben ser recorridos para llegar a un nodo x,
partiendo de la raiz.
La raiz tiene longitud de camino 1, sus descendientes directos tienen longitud de camino 2, etc. En forma
general un nodo en el nivel i tiene longitud de camino i.
Árbol binario: Un árbol es binario si cada nodo tiene como máximo 2 descendientes.
Para cada nodo está definido el subárbol izquierdo y el derecho.
Para el nodo A el subárbol izquierdo está constituido por los nodos B, D y E. Y el subárbol derecho está
formado por los nodos C y F.
Lo mismo para el nodo B tiene el subárbol izquierdo con un nodo (D) y un nodo en el subárbol derecho (E).
El nodo D tiene ambos subárboles vacíos.
El nodo C tiene el subárbol izquierdo vacío y el subárbol derecho con un nodo (F).
Árbol binario perfectamente equilibrado: Si para cada nodo el número de nodos en el subárbol izquierdo
y el número de nodos en el subárbol derecho, difiere como mucho en una unidad.
Hay que tener en cuenta todos los nodos del árbol.
Hay nodos hoja en los niveles 4, 3 y 2. No debería haber nodos hojas en el nivel 2.
Árbol binario ordenado: Si para cada nodo del árbol, los nodos ubicados a la izquierda son inferiores al
que consideramos raíz para ese momento y los nodos ubicados a la derecha son mayores que la raíz.
No hace falta analizar los nodos hoja. Si todas las respuestas son afirmativas podemos luego decir que se
trata de un árbol binario ordenado.
Retornar
Insertamos el 400
Insertamos el valor 100. Debemos analizar si raíz es distinto a null verificamos si 100 es mayor o menor a la
información del nodo apuntado por raíz, en este caso es menor y como el subárbol izquierdo es null
debemos insertarlo allí.
Insertamos el 200. Hay que tener en cuenta que siempre comenzamos las comparaciones a partir de raíz. El
200 es menor que 400, descendemos por el subárbol izquierdo. Luego analizamos y vemos que el 200 es
mayor a 100, debemos avanzar por derecha. Como el subárbol derecho es null lo insertamos en dicha
posición.
Ej. Si en el árbol anterior necesitamos verificar si está almacenado el 700, primero verificamos si la
información del nodo apuntado por raíz es 700, en caso negativo verificamos si la información a buscar
(700) es mayor a la información de dicho nodo (400) en caso afirmativo descendemos por el subárbol
derecho en caso contrario descendemos por el subárbol izquierdo.
Este proceso lo repetimos hasta encontrar la información buscada o encontrar un subárbol vacío.
Recorrer: Pasar a través del árbol enumerando cada uno de sus nodos una vez.
Visitar: Realizar algún procesamiento del nodo.
Pre-orden:
- Visitar la raíz.
- Recorrer el subárbol izquierdo en pre-orden.
- Recorrer el subárbol derecho en pre-orden.
Entre-orden
- Recorrer el subárbol izquierdo en entre-orden.
- Visitar la raíz.
- Recorrer el subárbol derecho en entre-orden.
Post-orden
- Recorrer el subárbol izquierdo en post-orden.
- Recorrer el subárbol derecho en post-orden.
- Visitar la raíz.
Ejemplo:
Veamos como se imprimen las informaciones de los nodos según su recorrido:
Recorrido preorden:
Recorrido entreorden:
Recorrido postorden:
Es decir que el orden de impresión de la información es:
Retornar
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ArbolBinarioOrdenado1
{
public class ArbolBinarioOrdenado {
class Nodo
{
public int info;
public Nodo izq, der;
}
Nodo raiz;
public ArbolBinarioOrdenado()
{
raiz=null;
}
Creamos un nodo y disponemos los punteros izq y der a null, guardamos la información que llega al método
en el nodo.
Si el árbol está vacío, apuntamos raíz al nodo creado; en caso de no estar vacío, dentro de una estructura
repetitiva vamos comparando info con la información del nodo, si info es mayor a la del nodo descendemos
por el subárbol derecho en caso contrario descendemos por el subárbol izquierdo.
Cuando se encuentra un subárbol vacío insertar el nodo en dicho subárbol. Para esto llevamos un puntero
anterior dentro del while.
El método recursivo void ImprimirPre (Nodo reco) lo primero que verifica con un if si reco está apuntando a
un nodo (esto es verdad si reco es distinto a null), en caso afirmativo ingresa al bloque del if y realiza:
- Visitar la raiz.
- Recorrer el subárbol izquierdo en pre-orden.
- Recorrer el subárbol derecho en pre-orden.
La visita en este caso es la impresión de la información del nodo y los recorridos son las llamadas recursivas
pasando las direcciones de los subárboles izquierdo y derecho.
Los algoritmos de los recorridos en entreorden y postorden son similares. La diferencia es que la visita la
realizamos entre las llamadas recursivas en el recorrido en entre orden:
y por último en el recorrido en postorden la visita la realizamos luego de las dos llamadas recursivas:
Problema 2:
Confeccionar una clase que permita insertar un entero en un árbol binario ordenado verificando que no se
encuentre previamente dicho número.
Desarrollar los siguientes métodos:
1 - Retornar la cantidad de nodos del árbol.
2 - Retornar la cantidad de nodos hoja del árbol.
3 - Imprimir en entre orden.
4 - Imprimir en entre orden junto al nivel donde se encuentra dicho nodo.
5 - Retornar la altura del árbol.
6 - Imprimir el mayor valor del árbol.
7 - Borrar el nodo menor del árbol.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ArbolBinarioOrdenado2
{
public class ArbolBinarioOrdenado {
class Nodo
{
public int info;
public Nodo izq, der;
}
private Nodo raiz;
private int cant;
private int altura;
public ArbolBinarioOrdenado()
{
raiz=null;
}
Para verificar si existe un elemento de información en el árbol disponemos un puntero reco en el nodo
apuntado por raiz. Dentro de un while verificamos si la información del parámetro coincide con la
información del nodo apuntado por reco, en caso afirmativo salimos del método retornando true, en caso
contrario si la información a buscar es mayor a la del nodo procedemos a avanzar reco con la dirección del
subárbol derecho:
Para retornar la cantidad de nodos del árbol procedemos a inicializar un atributo de la clase llamado cant con
cero. Llamamos al método recursivo y en cada visita al nodo incrementamos el atributo cant en uno:
Para imprimir todos los nodos en entre orden junto al nivel donde se encuentra planteamos un método
recursivo que llegue la referencia del nodo a imprimir junto al nivel de dicho nodo. Desde el método no
recursivo pasamos la referencia a raiz y un uno (ya que raiz se encuentra en el primer nivel)
Cada vez que descendemos un nivel le pasamos la referencia del subárbol respectivo junto al nivel que se
encuentra dicho nodo:
Para obtener la altura del árbol procedemos en el método no recursivo a inicializar el atributo altura con el
valor cero. Luego llamamos al método recursivo con la referencia a raiz que se encuentra en el nivel uno.
Cada vez que visitamos un nodo procedemos a verificar si el parámetro nivel supera al atributo altura, en
dicho caso actualizamos el atributo altura con dicho nivel.
Para imprimir el mayor valor del árbol debemos recorrer siempre por derecha hasta encontrar un nodo que
almacene null en der:
Para borrar el menor valor del árbol lo primero que comprobamos es si el subárbol izquierdo es nulo luego
el menor del árbol es el nodo apuntado por raiz. Luego si el subárbol izquierdo no está vacío procedemos a
descender siempre por la izquierda llevando un puntero en el nodo anterior. Cuando llegamos al nodo que
debemos borrar procedemos a enlazar el puntero izq del nodo que se encuentra en el nivel anterior con la
referencia del subárbol derecho del nodo a borrar:
Retornar
Tipos enteros
Según el valor entero máximo a almacenar podemos elegir entre: int (?2147483,648, 2147483647) (es el que
normalmente hemos elegido hasta este momento cada vez que necesitamos almacenar un valor entero), byte
(podemos almacenar un valor entre 0 y 255), sbyte (-128 y 127), short (-32768, 32767), ushort (0, 65535),
uint (0, 4294967295), long (?9223372036854775808, 9223372036854775807) y ulong (0,
18446744073709551615)
Problema 1:
Confeccionar un programa que defina variables enteras una por cada tipo. Mostrar por pantalla el valor
almacenado en cada variable
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace TiposDatosPrimitivos1
{
class Program
{
static void Main(string[] args)
{
byte v1 = 200;
Console.WriteLine("Valor almacenado en la variable de tipo byte:" + v1);
sbyte v2=-100;
Console.WriteLine("Valor almacenado en la variable de tipo sbyte:" + v2);
short v3 = 30000;
Console.WriteLine("Valor almacenado en la variable de tipo short:" + v3);
ushort v4 = 60000;
Console.WriteLine("Valor almacenado en la variable de tipo ushort:" + v4);
int v5 = -2000000000;
Console.WriteLine("Valor almacenado en la variable de tipo int:" + v5);
uint v6 = 2000000000;
Console.WriteLine("Valor almacenado en la variable de tipo uint:" + v6);
long v7 = -1000000000000000000;
Console.WriteLine("Valor almacenado en la variable de tipo long:" + v7);
ulong v8 = 1000000000000000000;
Console.WriteLine("Valor almacenado en la variable de tipo ulong:" + v8);
Console.ReadKey();
}
}
}
Problema 2:
Mostrar el valor máximo y mínimo que puede almacenar cada tipo de dato entero en C#.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace TiposDatosPrimitivos2
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Valor máximo y mínimo para tipo de dato byte.");
Console.WriteLine("Mínimo:" + byte.MinValue);
Console.WriteLine("Máximo:" + byte.MaxValue);
Console.WriteLine("Valor máximo y mínimo para tipo de dato sbyte.");
Console.WriteLine("Mínimo:" + sbyte.MinValue);
Console.WriteLine("Máximo:" + sbyte.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato short.");
Console.WriteLine("Mínimo:" + short.MinValue);
Console.WriteLine("Máximo:" + short.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato ushort.");
Console.WriteLine("Mínimo:" + ushort.MinValue);
Console.WriteLine("Máximo:" + ushort.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato int.");
Console.WriteLine("Mínimo:" + int.MinValue);
Console.WriteLine("Máximo:" + int.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato uint.");
Console.WriteLine("Mínimo:" + uint.MinValue);
Console.WriteLine("Máximo:" + uint.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato long.");
Console.WriteLine("Mínimo:" + long.MinValue);
Console.WriteLine("Máximo:" + long.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato ulong.");
Console.WriteLine("Mínimo:" + ulong.MinValue);
Console.WriteLine("Máximo:" + ulong.MaxValue);
Console.ReadKey();
}
}
}
Los tipos de datos primitivos tienen asociados clases con una serie de propiedades y métodos. Las
propiedades MinValue y MaxValue nos permiten acceder a los valores mínimos y máximos que pueden
almacenar cada tipo de dato.
Tipos reales
Podemos almacenar la parte entera y la parte fraccionaria. Disponemos tres tipos de datos reales: float,
double y decimal.
Problema 3:
Mostrar el valor máximo y mínimo que puede almacenar cada tipo de dato real en C#.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace TiposDatosPrimitivos3
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Valor mínimo y máximo para tipo de dato float.");
Console.WriteLine("Mínimo:" + float.MinValue);
Console.WriteLine("Máximo:" + float.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato double.");
Console.WriteLine("Mínimo:" + double.MinValue);
Console.WriteLine("Máximo:" + double.MaxValue);
Console.WriteLine("Valor mínimo y máximo para tipo de dato decimal.");
Console.WriteLine("Mínimo:" + decimal.MinValue);
Console.WriteLine("Máximo:" + decimal.MaxValue);
Console.ReadKey();
}
}
}
El tipo de dato decimal es el que más se adecua para almacenar datos monetarios (tiene una precisión de 28
dígitos decimales)
Tipo char
El tipo de dato primitivo char puede almacenar un caracter Unicode.
char letra='A';
Tipo lógico
Puede almacenar el valor true o false.
bool encontrado=false;
Problema 4:
Mostrar cuantos bytes de memoria requieren cada tipo de dato primitivo en C# (utilizar el operador sizeof)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace TiposDatosPrimitivos4
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Bytes reservados para el tipo byte:" + sizeof(byte));
Console.WriteLine("Bytes reservados para el tipo sbyte:" + sizeof(sbyte));
Console.WriteLine("Bytes reservados para el tipo short:" + sizeof(short));
Console.WriteLine("Bytes reservados para el tipo ushort:" +
sizeof(ushort));
Console.WriteLine("Bytes reservados para el tipo int:" + sizeof(int));
Console.WriteLine("Bytes reservados para el tipo uint:" + sizeof(uint));
Console.WriteLine("Bytes reservados para el tipo long:" + sizeof(long));
Console.WriteLine("Bytes reservados para el tipo ulong:" + sizeof(ulong));
Console.WriteLine("Bytes reservados para el tipo float:" + sizeof(float));
Console.WriteLine("Bytes reservados para el tipo double:" +
sizeof(double));
Console.WriteLine("Bytes reservados para el tipo decimal:" +
sizeof(decimal));
Console.WriteLine("Bytes reservados para el tipo char:" + sizeof(char));
Console.WriteLine("Bytes reservados para el tipo bool:" + sizeof(bool));
Console.ReadKey();
}
}
}
Retornar
57 - Definición de variables con tipos implícitos
El lenguaje C# es fuertemente tipado (es decir que toda variable en tiempo de compilación tiene que
identificar que tipo de variable se trata)
Por ejemplo siempre que definimos una variable le antecedemos su tipo:
int x;
double altura;
bool existe.
C# nos permite mediante la palabra clave var definir e inicializar una variable y que el compilador se
encargue de definir su tipo teniendo en cuenta el valor que le asignamos:
var x=10;
var altura=1.92;
El compilador se fija del lado derecho del operador de asignación el dato que estamos asignando y a partir
de dicho valor define la variable.
Solo las variables definidas en un método podemos utilizar la palabra clave var.
Problema 1:
Confeccionar un programa que defina variables con tipo de dato implícitos para almacenar un entero, un
real, un caracter y un valor lógico. Imprimir sus valores.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace DatosImplicitos1
{
class Program
{
static void Main(string[] args)
{
var edad = 44;
var distancia = 7.55;
var continua = true;
var tecla = 'A';
Console.WriteLine(edad);
Console.WriteLine(distancia);
Console.WriteLine(continua);
Console.WriteLine(tecla);
Console.ReadKey();
}
}
}
Problema propuesto
1. Imprimir los valores de 1 al 1000 utilizando un for y definiendo una variable implícita
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace DatosImplicitos2
{
class Program
{
static void Main(string[] args)
{
for (var x = 1; x <= 1000; x++)
{
Console.Write(x + "-");
}
Console.ReadKey();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PalabraClaveThis1
{
class Persona
{
private string nombre;
private int edad;
Esto significa que this almacena una referencia al objeto actual. Es común no disponer esta palabra clave
antecediendola a cada atributo o llamada de método:
this.nombre = nom;
this.edad = ed;
this.Imprimir();
nombre = nom;
edad = ed;
Imprimir();
Pero hay caso donde debemos anteceder esta palabra clave como por ejemplo en los caso que los parámetros
de un método coincidan con el mismo nombre que tienen atributos de la clase:
Retornar
Problema 1:
Almacenar los sueldos de 5 operarios en un vector, imprimir los elementos recorriendo el vector con la
estructura repetitiva foreach.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Estructuraforeach1
{
class SueldoEmpleados
{
private int[] sueldos;
La variable s almacena la primera vez el primer elemento del vector sueldos, seguidamente se ejecuta el
bloque del foreach (en este caso imprimimos el contenido de la variable s)
Es decir que s almacena en cada vuelta del foreach un elemento del vector.
Con la estructura foreach recorremos en forma completa el vector y en cada iteración tenemos acceso a un
elemento del vector que se copia en una variable auxiliar.
Podemos utilizar la palabra clave var para definir en forma implícita la variable que almacena sucesivamente
los elementos del vector:
Problema propuesto
1. Crear un vector de n elementos de tipo entero (n se ingresa por teclado) Mostrar cuantos elementos
son superiores a 100 (emplear el foreach para recorrer el vector)
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Estructuraforeach2
{
class Mayores100
{
private int[] vec;
Retornar
60 - Estructura condicional switch
La estructura condicional switch remplaza en algunos casos un conjunto de if.
switch(variable) {
case valor1:
Instrucciones
break;
case valor2:
Instrucciones
break;
case valor3:
Instrucciones
break;
.
.
.
default:
Instrucciones
break;
}
Luego de la palabra clave switch entre paréntesis indicamos una variable, luego con una serie de case
verificamos si dicha variable almacena un valor igual a [valor1, valor2, valor3 etc.] en el caso de ser igual se
ejecutan las instrucciones contenidas en dicho case.
Si todos los case son falsos, luego se ejecutan las instrucciones contenidas después de la plabra default.
Problema 1:
Ingresar un valor entero entre 1 y 5. Luego mostrar en castellano el valor ingresado. Si se ingresa un valor
fuera de dicho rango mostrar un mensaje indicando tal situación
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Estructuraswitch1
{
class Program
{
static void Main(string[] args)
{
Console.Write("Ingrese un valor entre 1 y 5:");
int valor = int.Parse(Console.ReadLine());
switch (valor)
{
case 1: Console.Write("uno");
break;
case 2: Console.Write("dos");
break;
case 3: Console.Write("tres");
break;
case 4: Console.Write("cuatro");
break;
case 5: Console.Write("cinco");
break;
default:
Console.Write("Se ingreso un valor fuera de rango");
break;
}
Console.ReadKey();
}
}
}
Es obligatorio que esté entre paréntesis la variable luego de la palabra clave switch.
Luego de cada case debemos indicar el valor con el que se comparará la variable (siempre debe ser un valor
constante y no podemos disponer una variable luego de la palabra case.
Es necesario la palabra break luego de cada bloque de instrucciones por cada case.
Problema 2:
Ingresar un número entre uno y cinco en castellano. Luego mostrar en formato numérico. Si se ingresa un
valor fuera de dicho rango mostrar un mensaje indicando tal situación
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Estructuraswitch2
{
class Program
{
static void Main(string[] args)
{
Console.Write("Ingrese un número en castellano entre uno y cinco:");
string nro = Console.ReadLine();
switch (nro)
{
case "uno": Console.Write(1);
break;
case "dos": Console.Write(2);
break;
case "tres": Console.Write(3);
break;
case "cuatro": Console.Write(4);
break;
case "cinco": Console.Write(5);
break;
default: Console.Write("Debe ingresar un valor entre uno y cinco");
break;
}
Console.ReadKey();
}
}
}
Esto quiere decir que podemos utilizar en el switch variables string para ser comparadas.
Retornar
Problema 1:
Implementar un método que reciba dos enteros y luego imprima de uno en uno desde el valor menor hasta el
valor mayor.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosValor1
{
class Program
{
public void MostrarRango(int menor, int mayor)
{
for (var x = menor; x <= mayor; x++)
{
Console.Write(x + " ");
}
}
Como podemos ver los parámetros se separan por coma y son utilizados en el método (en este método los
dos parámetros por valor se llaman menor y mayor):
Dentro del método se utilizan los parámetros cual si fueran variables locales:
Problema 2:
Confeccionar un método que reciba como parámetros tres valores enteros y retorne el mayor de los mismos.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosValor2
{
class Program
{
public int Mayor(int v1, int v2, int v3)
{
if (v1 >= v2 && v1 >= v3)
{
return v1;
}
else
{
if (v2 >= v3)
{
return v2;
}
else
{
return v3;
}
}
}
El método Mayor recibe tres parámetros y retorna un entero. Se inicializa con public ya que lo llamamos
desde donde definimos un objeto desde la Main:
Desde la Main llamamos al método pasando tres variables enteras e imprimimos el entero que retorna:
Problema propuesto
1. Confeccionar un método que reciba un entero entre 1 y 10 y retorne el valor en castellano.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosValor3
{
class Program
{
public string RetornarCastellano(int x)
{
string valor;
switch (x)
{
case 1: valor = "uno";
break;
case 2: valor = "dos";
break;
case 3: valor = "tres";
break;
case 4: valor = "cuatro";
break;
case 5: valor = "cinco";
break;
case 6: valor = "seis";
break;
case 7: valor = "siete";
break;
case 8: valor = "ocho";
break;
case 9: valor = "nueve";
break;
case 10: valor = "diez";
break;
default: valor = "valor fuera de rango";
break;
}
return valor;
}
Retornar
62 - Parámetros por referencia (ref)
Hasta ahora habíamos visto que podemos pasar datos a un método para que sean utilizados en el mismo. Si
queremos que un método retorne más de un dato es común utilizar los parámetros por referencia.
La idea central es pasar a un método las direcciones de las variables para que el método pueda modificar los
datos de las variables pasadas.
Problema 1:
Implementar un método que se le envíen dos variables y nos devuelvan sus contenidos intercambiados.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosReferencia1
{
class Program
{
public void Intercambiar(ref int valor1, ref int valor2)
{
int aux = valor1;
valor1 = valor2;
valor2 = aux;
}
Para indicar que un parámetro es por referencia debemos anteceder la palabra clave "ref":
El objetivo fundamental de un parámetro por referencia es modificar su contenido dentro del método
(asignamos valor2 a valor1 y en valor2 almacenamos el contenido de valor1):
Siempre que llamamos a un método que recibe un parámetro por referencia debe estar inicializada la
variable que le pasamos:
int x1 = 10;
int x2 = 20;
Console.WriteLine("Los valores antes de intercambiar son " + x1 + " " +
x2);
p.Intercambiar(ref x1, ref x2);
Problema propuesto
1. Confeccionar un método que reciba por referencia tres enteros y nos los retorne en forma ordenada
de menor a mayor.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosReferencia2
{
class Program
{
public void Ordenados(ref int v1, ref int v2, ref int v3)
{
int menor;
int intermedio;
int mayor;
if (v1 <= v2 & v1 <= v3)
{
menor = v1;
if (v2 < v3)
{
intermedio = v2;
mayor = v3;
}
else
{
intermedio = v3;
mayor = v2;
}
}
else
{
if (v2 <= v3)
{
menor = v2;
if (v1 < v3)
{
intermedio = v1;
mayor = v3;
}
else
{
intermedio = v3;
mayor = v1;
}
}
else
{
menor = v3;
if (v1 < v2)
{
intermedio = v1;
mayor = v2;
}
else
{
intermedio = v2;
mayor = v1;
}
}
}
v1 = menor;
v2 = intermedio;
v3 = mayor;
}
Retornar
Problema 1:
Implementar un programa que permita crear, cargar y obtener el menor y mayor valor de un vector. La
obtención del mayor y menor hacerlo en un único método que retorne dichos dos valores.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosReferencia3
{
class Program
{
private int[] vec;
public Program()
{
Console.Write("Tamaño del vector:");
int tam = int.Parse(Console.ReadLine());
vec = new int[tam];
}
Para definir un parámetro por referencia que no requiere que la variable que le enviemos esté inicializada le
antecedemos la palabra clave out:
Lo mismo cuando llamamos al método debemos anteceder las variables que le pasamos con la palabra clave
out:
Como podemos ver las variables ma y me no tienen un valor previo a la llamada al método MayorMenor.
Problema propuesto
1. Confeccionar un método que me retorne dos valores aleatorios comprendidos entre 1 y 100 mediante
parámetros por referencia.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosReferencia4
{
class Program
{
public void DosValoresAleatorios(out int v1, out int v2)
{
Random r = new Random();
v1 = r.Next(1, 101);
v2 = r.Next(1, 101);
}
Retornar
64 - Número variable de parámetros
Hasta ahora hemos planteado métodos que reciben una cantidad fija de parámetros, pero C# permite
implementar métodos con una cantidad variable de parámetros (es decir que llamemos al método pasando en
algunas circunstancias pasándole dos parámetros y en otras pasándole 5)
Problema 1:
Implementar un método que le envíe una cantidad n de enteros y me retorne la suma de los mismos.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosVariables1
{
class Program
{
public int Sumar(params int[] p)
{
int su = 0;
for (var f = 0; f < p.Length; f++)
{
su = su + p[f];
}
return su;
}
Como podemos ver le antecedemos al nombre del vector la palabra clave params:
public int Sumar(params int[] p) { int su = 0; for (var f = 0; f < p.Length; f++) { su = su + p[f]; } return su; }
La diferencia fundamental es que desde donde llamamos al método no le pasamos un vector de enteros, sino
una lista de parámetros enteros:
Console.WriteLine(p.Sumar(3,4,5));
Un método solo puede tener un solo parámetro de este tipo y debe ser siempre el último, por ejemplo:
Problema propuesto
1. Confeccionar un método que reciba un string con la cadena "suma" o "producto" y seguidamente una
lista de enteros. El método debe retornar la suma o producto de todos los valores enviados.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosVariables2
{
class Program
{
public int Operar(string operacion, params int[] v)
{
if (operacion == "suma")
{
int suma = 0;
for (var f = 0; f < v.Length; f++)
{
suma = suma + v[f];
}
return suma;
}
if (operacion == "producto")
{
int producto = 1;
for (var f = 0; f < v.Length; f++)
{
producto=producto * v[f];
}
return producto;
}
return int.MaxValue;
}
Retornar
65 - Parámetros opcionales
Para definir un parámetro opcional debemos asignarle un dato en la declaración del método:
Como vemos debemos asignarle un valor en la declaración del método, luego cuando llamamos a este
método podemos hacerlo pasando 1,2 o 3 parámetros:
ob1.Imprimir("Hola");
ob1.Imprimir("Hola",40);
ob1.Imprimir("Hola",40,12);
Los parámetros opcionales deben ser siempre los últimos que indiquemos. El parámetro mensaje no es
opcional por lo que si o si debe especificarse.
Problema 1:
Implementar un método que muestre un mensaje en la pantalla con dos parámetros opciones que indiquen la
columna y la fila donde imprimir.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosOpcionales1
{
class Program
{
public void Imprimir(string mensaje, int col = 1, int fil = 1)
{
Console.SetCursorPosition(col, fil);
Console.Write(mensaje);
}
static void Main(string[] args)
{
Program ob1 = new Program();
ob1.Imprimir("Hola");
ob1.Imprimir("Hola", 40);
ob1.Imprimir("Hola", 40, 12);
Console.ReadKey();
}
}
}
Problema propuesto
1. Confeccionar un método que muestre los primeros 10 números pares. En caso que le pasemos un
parámetro opcional mostrar tantos pares como indica el parámetro.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosOpcionales2
{
class Program
{
public void Pares(int cant = 10)
{
int par = 2;
for (var x = 1; x <= cant; x++)
{
Console.Write(par + "-");
par = par + 2;
}
}
static void Main(string[] args)
{
Program ob1 = new Program();
ob1.Pares();
Console.WriteLine();
Console.WriteLine();
ob1.Pares(20);
Console.ReadKey();
}
}
}
Retornar
C# presenta la posibilidad de llamar a un método pasando los parámetros en cualquier orden, siempre que
cuando lo llamemos indiquemos previo al valor del parámetro el nombre del parámetro.
Problema 1:
Implementar un método que muestre un mensaje en la pantalla indicando la fila y columna donde debe
imprimirse. Luego llamar al método pasando los nombres y valores de los parámetros.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ParametrosNombre1
{
class Program
{
public void Imprimir(string mensaje, int columna, int fila)
{
Console.SetCursorPosition(columna, fila);
Console.WriteLine(mensaje);
}
Como podemos observar previo al dato a enviar le antecedemos el nombre del parámetro y luego de dos
puntos el dato a enviar:
Retornar
67 - Sobrecarga de métodos
La sobrecarga de métodos permite definir dos o más métodos con el mismo nombre, pero que difieren en
cantidad o tipo de parámetros.
Esta característica del lenguaje nos facilita la implementación de algoritmos que cumplen la misma función
pero que difieren en los parámetros.
Problema 1:
Implementar dos métodos que sumen dos enteros en el primer caso y que concatenen dos string en el
segundo.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaParametros1
{
class Program
{
public int Sumar(int x1, int x2)
{
int s = x1 + x2;
return s;
}
Como podemos ver definimos dos métodos llamados Sumar con dos parámetros cada uno (el primero con
parámetros de tipo int y el segundo con parámetros de tipo string):
Cuando llamamos a los métodos el compilador sabe con cual enlazarlos según el tipo de datos enviados.
Problema 2:
Plantear una clase llamada Ventana que defina cuatro métodos sobrecargados que muestren un mensaje en la
consola.
El primero lo muestrar donde se encuentra actualmente el cursor.
El segundo lo muestra en una determinada columna y fila.
El tercero lo muestra en una determinada columna,fila y con un color de letra.
Y por último similar al anterior más un color de fondo.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaParametros2
{
class Ventana
{
public void Mostrar(string mensaje)
{
Console.Write(mensaje);
}
Como podemos observar hemos definido cuatro métodos llamados Mostrar que difieren en la cantidad de
parámetros.
El primero recibe un string y lo muestra en la consola:
El segundo recibe tres parámetros con el mensaje, la columna y fila donde mostrarlo:
Como podemos observar la clase Console tiene un método llamado SetCursorPosition que le pasamos la
columna y fila donde queremos que se posicione el cursor previo a la salida de datos llamando al método
Write.
El tercer método recibe cuatro parámetros y como podemos ver desde dentro de este método llamamos al
método Mostrar que recibe tres parámetros:
Problema propuesto
1. Plantear una clase que sobrecargue un método que permita cargar por referencia (out) distintos tipos
de datos primitivos por teclado.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaParametros3
{
class EntradaTeclado
{
public void Leer(out int valor)
{
valor = int.Parse(Console.ReadLine());
}
Retornar
Problema 1:
Implementar una clase que represente un titulo en pantalla. Definir un constructor que reciba el string del
título y otro constructor que reciba el string del título y la columna y fila donde mostrarlo.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaConstructor1
{
class Titulo
{
private string tit;
private int columna;
private int fila;
public Titulo(string t)
{
tit = t;
columna = 1;
fila = 1;
}
Hemos planteado dos constructores, uno que recibe un string y otro que recibe un string y dos enteros (los
mismos inicializan atributos de la clase):
public Titulo(string t)
{
tit = t;
columna = 1;
fila = 1;
}
en la Main definimos dos objetos de la clase Titulo, el primero llama al constructor con un parámetro, por lo
que el mensaje aparece en la columna 1 y fila 1:
El segundo objeto que creamos se llama al constructor que tiene tres parámetros:
Si desde un constructor queremos llamar a otro constructor de la clase lo debemos hacer con la palabra this
seguida por los parámetros del constructor a llamar:
public Titulo(string t)
: this(t, 1, 1)
{
}
En el ejemplo anterior cuando llamamos al constructor que tiene un parámetro el mismo primero llama al
constructor que tiene tres parámetros y luego se ejecuta el código del constructor inicial (en este ejemplo no
tenemos código dentro del constructor)
Retornar
69 - Métodos estáticos
En C# podemos definir métodos que se crean independientemente a la definición de objetos. Un método
estático puede llamarse sin tener que crear un objeto de dicha clase. Un método estático tiene ciertas
restricciones:
Si recordamos cada vez que creamos un programa en C# debemos especificar el método Main:
El método Main es estático para que el sistema operativo pueda llamarlo directamente sin tener que crear un
objeto de la clase que lo contiene.
Problema 1:
Implementar una clase llamada Operacion. Definir dos métodos estáticos que permitan sumar y restar dos
valores enteros.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace MetodosEstaticos1
{
class Operacion
{
public static int Sumar(int x1, int x2)
{
int s = x1+x2;
return s;
}
class Program
{
static void Main(string[] args)
{
Console.Write("La suma de 2+4 es ");
Console.WriteLine(Operacion.Sumar(2, 4));
Console.Write("La resta de 6-2 es ");
Console.WriteLine(Operacion.Restar(6, 2));
Console.ReadKey();
}
}
}
Agregamos la palabra clave static antes de indicar el valor que retorna el método:
Es importante notar que no se crean objetos de la clase Operacion para poder llamar a los métodos Sumar y
Restar.
Problema propuesto
1. Plantear una clase llamada VectorEnteros que defina tres métodos estáticos. El primero retorna el
mayor elemento del vector, el segundo el menor elemento y el tercero la suma de sus componentes.
SOLUCION:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace MetodosEstaticos2
{
class VectorEnteros
{
public static int Mayor(int[] vec)
{
int may = vec[0];
for (int x = 0; x < vec.Length; x++)
{
if (vec[x] > may)
{
may = vec[x];
}
}
return may;
}
class Program
{
static void Main(string[] args)
{
int[] v = { 5, 6, 20 };
VectorEnteros.Imprimir(v);
Console.WriteLine("El mayor elemento del vector es :" +
VectorEnteros.Mayor(v));
Console.WriteLine("El menor elemento del vector es :" +
VectorEnteros.Menor(v));
Console.Write("La suma de los elementos del vector es :" +
VectorEnteros.SumaElementos(v));
Console.ReadKey();
}
}
}
Retornar
Problema 1:
Definir un atributo estático que almacene la cantidad de objetos creados de dicha clase.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace AtributosEstaticos1
{
class Persona
{
private string nombre;
private int edad;
public static int cantidad;
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.cantidad);
Persona per1 = new Persona("juan", 30);
per1.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.cantidad);
Persona per2 = new Persona("ana", 20);
per2.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.cantidad);
Persona per3 = new Persona("luis", 10);
per3.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.cantidad);
Console.ReadKey();
}
}
}
Un atributo estático de tipo entero se inicializa en cero cuando lo definimos (lo definimos de tipo public para
poder acceder a su valor desde afuera de la clase):
Luego de crear un objeto de la clase Persona se ejecuta el constructor de la clase donde se incrementa en uno
el atributo estático cantidad:
No importa cuantos objeto de la clase Persona se creen luego existe un solo atributo cantidad:
Si queremos encapsular el atributo cantidad en la clase Persona luego podemos definir una propiedad
estática.
Problema 2:
Definir una propiedad estática para acceder a un atributo estático que guarda la cantidad de objetos creados
de dicha clase.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PropiedadesEstaticas1
{
class Persona
{
private string nombre;
private int edad;
private static int cantidad;
class Program
{
static void Main(string[] args)
{
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.Cantidad);
Persona per1 = new Persona("juan", 30);
per1.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.Cantidad);
Persona per2 = new Persona("ana", 20);
per2.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.Cantidad);
Persona per3 = new Persona("luis", 10);
per3.Imprimir();
Console.WriteLine("Valor del atributo estático cantidad:" +
Persona.Cantidad);
Console.ReadKey();
}
}
}
Esto hace necesario implementar una propiedad estática para acceder a su contenido:
Luego para acceder a su valor desde fuera debemos preguntar el valor de la propiedad Cantidad ya que el
atributo cantidad es privado:
Si queremos pulir un poco más este problema podemos eliminar la parte del set de la propiedad Cantidad:
public static int Cantidad
{
get
{
return cantidad;
}
}
Esto hace que no podamos por error asignar a la propiedad Cantidad un valor (genera un error en tiempo de
compilación, ya que se trata de una propiedad de solo lectura):
Persona.Cantidad = 71;
El incremento del atributo cantidad solo se hace en el constructor de la clase Persona cada vez que creamos
un objeto.
Retornar
71 - Clases estáticas
Una clase se la puede hacer estática siempre y cuando todos sus atributos sean estáticos, lo mismo que sus
métodos.
Problema 1:
Definir una clase estática llamada Operaciones. Implementar cuatro métodos que permitan
sumar,restar,multiplicar y dividir dos enteros.:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ClaseEstatica1
{
static class Operaciones
{
public static int Sumar(int valor1, int valor2)
{
return valor1 + valor2;
}
public static int Restar(int valor1, int valor2)
{
return valor1 - valor2;
}
public static int Multiplicar(int valor1, int valor2)
{
return valor1 * valor2;
}
public static int Dividir(int valor1, int valor2)
{
return valor1 / valor2;
}
}
class Program
{
static void Main(string[] args)
{
Console.WriteLine("10+5 es " + Operaciones.Sumar(10, 5));
Console.WriteLine("10-5 es " + Operaciones.Restar(10, 5));
Console.WriteLine("10*5 es " + Operaciones.Multiplicar(10, 5));
Console.WriteLine("10/5 es " + Operaciones.Dividir(10, 5));
Console.ReadKey();
}
}
}
Como definimos la clase de tipo static estamos obligados a definir los cuatro métodos de tipo static:
Luego si queremos definir alguno de los métodos no estático se produce un error en tiempo de compilación:
Retornar
72 - Enumeraciones
Una enumeración es un conjunto de constantes enteras que tienen asociado un nombre para cada valor.
Luego mediante if podemos analizar el valor de esa variable y proceder de acuerdo al valor existente.
Problema 1:
Definir un tipo de dato enumerado para cada tipo de carta de una baraja española.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Enumeracion1
{
class JuegoDeCartas
{
public enum TipoCarta { oro, basto, copa, espada };
Para definir un tipo de dato enumerado utilizamos la palabra clave enum y entre paréntesis escribimos todos
los valores que puede almacenar una variable de dicho tipo:
Internamente el primer valor esta asociado con el número cero (oro), el segundo se asocia con el 1 y así
sucesivamente.
Para definir una variable lo hacemos como cuando definimos cualquier tipo de variable, le antecedemos el
nuevo tipo de dato creado (en nuestro caso hemos creado el TipoCarta):
private TipoCarta cartaActual;
El atributo cartaActual puede almacenar cualquiera de los cuatro valores y para inicializar utilizamos la
sintaxis:
cartaActual = TipoCarta.oro;
cartaActual = 2; // error
Tengamos en cuenta que si no le asignamos un valor a la constante comienzan a asignarse un valor a partir
de cero en forma correlativa:
Con esto bits16 tiene asociado el entero 0, bits32 tiene asociado el valor 1 y bits54 tiene asociado el valor 2.
Con esto bits16 tiene asociado el entero 16, bits32 tiene asociado el valor 17 y bits54 tiene asociado el valor
18.
Si necesitamos el valor entero de una variable de tipo enumeración debemos utilizar el operador cast (es
decir anteceder entre paréntesis el tipo de dato que necesitamos que la convierta):
Console.Write((int)cartaActual);
Por defecto cuando definimos una enumeración se trabaja con tipos int, pero podemos especificar otro tipo
de enteros con otros tamaños:
Problema propuesto
1. Elaborar una calculadora para trabajar con valores enteros(utilizar objetos de la clase Button y un
objeto de la clase Label donde se muestra el valor ingresado) definir un tipo de dato enumerado para
las cuatro operaciones básicas. Cuando se presione algunos de los botones de operaciones almacenar
en una memoria el valor ingresado y el tipo de operación pendiente en una variable del tipo de dato
enumerado creado. Cuando se presione el botón igual verificar el tipo de operación pendiente y
proceder a obtener el resultado y mostrarlo en la Label.
SOLUCION:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace CalculadoraEnumeracion
{
public partial class Form1 : Form
{
public enum TipoOperacion { ninguna ,suma, resta, multiplicacion, division };
public Form1()
{
InitializeComponent();
}
Retornar
73 - Sobrecarga de operadores
La sobrecarga de operadores en C# permite redefinir la acción de un operador en relación a una clase.
Por ejemplo podemos plantear una clase Vector y luego redefinir el operador + para dicha clase. Luego
cuando sumamos dos objetos de esa clase vector podemos generar otro objeto de dicha clase que resulte de
la suma de sus componentes.
El empleo de la sobrecarga de operadores debe hacerse con mucho cuidado de no desvirtuar el concepto que
representa dicho operador (por ejemplo sobrecargar el operador "-" para la clase Vector y que genere la
suma de sus componentes)
Problema 1:
Plantear una clase VectorEnteros que permita crear un vector de 5 elementos y sobrecargue el operador +
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaOperadores1
{
class VectorEnteros
{
private int []vec;
public VectorEnteros()
{
vec = new int[5];
}
}
class Program
{
static void Main(string[] args)
{
VectorEnteros v1 = new VectorEnteros();
Console.WriteLine("Carga del primer vector");
v1.Cargar();
VectorEnteros v2 = new VectorEnteros();
Console.WriteLine("Carga del segundo vector");
v2.Cargar();
Console.WriteLine("Primer Vector");
v1.Imprimir();
Console.WriteLine("Segundo Vector");
v2.Imprimir();
VectorEnteros vt;
vt = v1 + v2;
Console.WriteLine("Vector Resultante");
vt.Imprimir();
Console.ReadKey();
}
}
}
Luego mediante un for cargamos cada elemento del vector de enteros con los datos de las componentes
homólogas de los otros dos vectores:
Como estamos en la clase VectorEnteros podemos acceder a los atributos privados vec.
return su;
Luego cuando utilizamos el operador + con dos objetos de la clase VectorEnteros el resultado el otro objeto
de la clase VectorEnteros:
VectorEnteros vt;
vt = v1 + v2;
Console.WriteLine("Vector Resultante");
vt.Imprimir();
Como podemos ver no creamos el objeto vt sino la llamada al operador + con dos objetos de la clase
VectorEnteros retorna un objeto de la clase VectorEnteros.
Problema 2:
Plantear una clase VectorEnteros que permita crear un vector de 5 elementos y sobrecargue el operador * de
un objeto de la clase VectorEnteros con un valor de tipo int (el resultado debe ser otro objeto de la clase
VectorEnteros donde cada componente se obtiene de multiplicar su valor por el valor entero)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaOperadores2
{
class VectorEnteros
{
private int[] vec;
public VectorEnteros()
{
vec = new int[5];
}
}
class Program
{
static void Main(string[] args)
{
VectorEnteros v1 = new VectorEnteros();
Console.WriteLine("Carga del vector");
v1.Cargar();
VectorEnteros vr;
Console.WriteLine("Primer Vector");
v1.Imprimir();
vr = v1 * 10;
Console.WriteLine("Vector resultante");
vr.Imprimir();
Console.ReadKey();
}
}
}
Como vemos ahora estamos sobrecargando el operador "*". El método tiene dos parámetros uno de tipo
VectorEnteros y otro de tipo int:
No lo hemos hecho pero podríamos también sobrecargar el operador "*" y recibir como parámetro dos
objetos de la clase VectorEnteros:
Operación unaria
Los ejemplos anteriores mostraban la sobrecarga de operadores binarios (un operador y dos operandos), un
operador unario afecta solo un operado.
Problema 3:
Sobrecargar el operador ++ en la clase VectorEnteros (se debe incrementar en uno cada elemento)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaOperadores3
{
class VectorEnteros
{
private int[] vec;
public VectorEnteros()
{
vec = new int[5];
}
class Program
{
static void Main(string[] args)
{
VectorEnteros v1 = new VectorEnteros();
Console.WriteLine("Carga del vector");
v1.Cargar();
Console.WriteLine("Impresión del vector");
v1.Imprimir();
v1++;
Console.WriteLine("Impresión del vector luego del operador ++");
v1.Imprimir();
Console.ReadKey();
}
}
}
Cuando se sobrecargan los operadores relacionales estamos obligados a implementar en pares, es decir si
emplementamos el == debemos implementar el != en forma obligatoria (sino se genera un error sintáctico.
Problema 4:
Sobrecargar el operador == en la clase VectorEnteros (retornar true si los cinco enteros son iguales)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SobrecargaOperadores4
{
class VectorEnteros
{
private int[] vec;
public VectorEnteros()
{
vec = new int[5];
}
class Program
{
static void Main(string[] args)
{
VectorEnteros v1 = new VectorEnteros();
Console.WriteLine("Carga del primer vector");
v1.Cargar();
VectorEnteros v2 = new VectorEnteros();
Console.WriteLine("Carga del segundo vector");
v2.Cargar();
if (v1 == v2)
Console.Write("Todos los elementos son iguales");
else
Console.Write("No todos los elementos son iguales");
Console.ReadKey();
}
}
}
Retornar
74 - Indizadores
Los indizadores permiten acceder mediante subíndices a un objeto de una clase que lo implementa. El
objetivo es facilitar la implementación de un algoritmo.
El indizador evita que implementemos una serie de métodos para administrar el objeto.
using System.Linq;
using System.Text;
namespace Indizador1
{
class Program
{
class A
{
private int []vec;
public A()
{
vec=new int[3];
}
Definimos un indizador con parámetro entero llamado indice(veremos más adelante que el parámetro puede
ser de otro tipo o inclusive tener más de uno) Utilizamos la palabra this para indicar que es un indizador.
En la seccion del set se ejecuta cuando le asignamos un valor donde definimos un objeto de dicha clase
(obj1[0] = 7;) es cero lo recibe el parámetro indice y el 7 lo almacena la palabra clave value. Por último la
sección del get se ejecuta cuando lo accedemos (Console.WriteLine(obj1[0]);
obj1[0] = 7;
obj1[1] = 34;
obj1[2] = 45;
Console.WriteLine(obj1[0]);
Console.WriteLine(obj1[1]);
Console.WriteLine(obj1[2]);
Console.ReadKey();
Problema 1:
Confeccionar una clase llamada Cliente definir las propiedades Nombre y Dinero (en el constructor
inicializar dichas propiedades)
Desarrollar otra clase llamada Banco donde debemos definir y crear un vector de tres elementos de tipo
Cliente.
Implementar un indizador en la clase Banco que permita acceder a cada cliente por un subíndice entero.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Indizador2
{
class Cliente
{
private string nombre;
public string Nombre
{
set
{
nombre = value;
}
get
{
return nombre;
}
}
private int dinero;
public int Dinero
{
set
{
dinero = value;
}
get
{
return dinero;
}
}
public Cliente(string nom, int din)
{
Nombre = nom;
Dinero = din;
}
}
class Banco
{
private Cliente[] clientes;
public Banco()
{
clientes = new Cliente[3];
}
class Program
{
static void Main(string[] args)
{
Banco banco1 = new Banco();
Cliente cli1 = new Cliente("juan", 1000);
Cliente cli2 = new Cliente("ana", 2000);
Cliente cli3 = new Cliente("luis", 1500);
banco1[0] = cli1;
banco1[1] = cli2;
banco1[2] = cli3;
Console.WriteLine("Datos de los clientes.");
Console.WriteLine(banco1[0].Nombre);
Console.WriteLine(banco1[0].Dinero);
Console.WriteLine();
Console.WriteLine(banco1[1].Nombre);
Console.WriteLine(banco1[1].Dinero);
Console.WriteLine();
Console.WriteLine(banco1[2].Nombre);
Console.WriteLine(banco1[2].Dinero);
Console.WriteLine();
Console.ReadKey();
}
}
}
Para ver las ventajas que presenta implementar un indizador en la clase Banco podemos ver como se
inicializan los tres clientes que tiene el banco:
Banco banco1 = new Banco();
Cliente cli1 = new Cliente("juan", 1000);
Cliente cli2 = new Cliente("ana", 2000);
Cliente cli3 = new Cliente("luis", 1500);
banco1[0] = cli1;
banco1[1] = cli2;
banco1[2] = cli3;
Como podemos ver le asignamos al objeto indicando un subíndice la referencia del objeto de la clase
Cliente.
Para poder imprimir el nombre y dinero depositado que tiene cada cliente lo hacemos accediendo al
subíndice y seguidamente la propiedad del cliente que necesitamos mostrar:
Problema 2:
Implementar una clase llamada Tablero del juego de la Batalla Naval. Permitir mediante un indizador de dos
dimensiones acceder a las casillas del tablero.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Indizador3
{
public enum DatoCasilla { agua, barco };
class Tablero
{
private DatoCasilla [,] mat;
public Tablero()
{
mat = new DatoCasilla[10, 10];
}
class Program
{
static void Main(string[] args)
{
Tablero tablero1 = new Tablero();
tablero1[0, 0] = DatoCasilla.barco;
tablero1[0, 1] = DatoCasilla.barco;
tablero1[0, 2] = DatoCasilla.barco;
tablero1[0, 9] = DatoCasilla.barco;
tablero1[1, 9] = DatoCasilla.barco;
tablero1[2, 9] = DatoCasilla.barco;
tablero1.Graficar();
Console.WriteLine();
if (tablero1[0, 0] == DatoCasilla.barco)
{
Console.WriteLine("Hay un barco en esta casilla");
}
else
{
if (tablero1[0, 0] == DatoCasilla.agua)
{
Console.WriteLine("Agua");
}
}
Console.ReadKey();
}
}
}
public Tablero()
{
mat = new DatoCasilla[10, 10];
}
Imprimimos el tablero, si en la componente hay "agua" mostramos un cero y si hay "barco" mostramos un
guión (cuando se creó la matriz en el constructor se inicializan todas las componentes con el valor 0 que es
"agua"):
Ahora veamos lo más importante que es la definición del indizador con dos subíndices:
class Program
{
static void Main(string[] args)
{
Tablero tablero1 = new Tablero();
Seguidamente disponemos un barco en la primer fila de la matriz utilizando un indizador (estas tres líneas
acceden al set del indizador pasando los dos subíndices y el valor a almacenar):
tablero1[0, 0] = DatoCasilla.barco;
tablero1[0, 1] = DatoCasilla.barco;
tablero1[0, 2] = DatoCasilla.barco;
tablero1[0, 9] = DatoCasilla.barco;
tablero1[1, 9] = DatoCasilla.barco;
tablero1[2, 9] = DatoCasilla.barco;
Para saber que hay en una casilla cualquiera del tablero podemos utilizar el indizador (se ejecuta el get):
if (tablero1[0, 0] == DatoCasilla.barco)
{
Console.WriteLine("Hay un barco en esta casilla");
}
else
{
if (tablero1[0, 0] == DatoCasilla.agua)
{
Console.WriteLine("Agua");
}
}
Retornar
Vamos a utilizar la versión SQL Server Express que la podemos descargar en forma gratuita del sitio de
Microsoft.
Descargaremos el SQL Server de 64bit (si tiene un sistema operativo con dichas características):
Y también descargaremos el Microsoft® SQL Server Management Studio, que nos permitirá administrar las
bases de datos (crearlas por ejemplo):
Ejecutamos el primer archivo que descargamos (Motor de base de datos SQL Server)
SQLEXPR_x64_ESN.exe y seguimos las instrucciones que nos propone el diálogo (Elegimos nueva
instalación independiente):
Procedemos a ejecutar el archivo que descargamos y aparece el "Centro de instalación de SQL Server",
elegimos "Nueva instalación independiente de SQL Server o agregar características a una instalación
existente" (tengamos en cuenta que estamos agregando características al SQL Server que ya instalamos):
En tipo de instalación dejamos por defecto "Realización de una nueva instalación de SQL Server":
Aceptamos los términos de licencia:
Por defecto dejamos la selección de características:
Finalmente ya tenemos las herramientas de administración instaladas:
Retornar
Para ejecutar comando SQL sobre una determinada base de datos procedemos a posicionar la flecha del
mouse sobre una base de datos, por ejemplo "base1" y presionamos el botón derecho del mouse
seleccionando "Nueva consulta":
En la ventana de consultas podemos escribir y ejecutar comandos SQL:
Si escribimos luego del comando insert un select podemos ver el registro añadido:
Retornar
El primer paso es conectarnos con nuestro servidor de base de datos que instalamos en los conceptos
anteriores y vimos como podemos comunicarnos utilizando el Microsoft SQL Server Management Studio.
Problema
Confeccionar una aplicación que permita conectarnos con nuestro servidor de base de datos y seleccionar
una determinada base de datos. Luego de la conexión cerrar la misma.
Para conectarnos con el servidor y seleccionar una base de datos debemos utilizar la clase SqlConnection:
El string que le pasamos al constructor del SqlConnection le indicamos como primer dato el nombre de
nuestro servidor, en su máquina lo puede ubicar cuando arranca el SQL Server Management Studio:
Algo importante si su nombre incluye la barra "\" luego disponga dos seguidas ya que C# esta barra es un
caracter de control y por lo tanto para disponer una debemos disponer "\\".
El segundo dato que se le pasa es el nombre de la base de datos con la que nos comunicaremos, en nuestro
caso creamos previamente la base de datos "base1" y finalmente el último dato obligatorio es indicar
"integrated security = true", esto hace que se utilice el usuario propio de Windows y no necesitemos por el
momento crear otros usuarios propios de SQL Server.
Creemos un nuevo proyecto en Visual Studio y demos como nombre: BaseDatos1. La interfaz visual es un
simple botón:
using System.Data.SqlClient;
namespace PruebaBaseDatos1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
using System.Data.SqlClient;
Para el evento Click del button1 implementamos primero la creación de un objeto de tipo SqlConnection:
Abrimos la conexión:
conexion.Open();
Cerramos la conexión:
conexion.Close();
Si todo está correcto podemos seguir con los próximos conceptos para pedir que SQL Server ejecute
comando y recuperar los resultados.
Retornar
79 - SqlCommand (insert): Insertar registro
Vimos en el concepto anterior como conectarnos al servidor de SQL Server y seleccionar una base de datos.
Veremos ahora como pedir que el SQL Server ejecute un comando "INSERT".
Trabajaremos con la base de datos "base1" que creamos en conceptos anteriores y con la tabla "articulos"
que también creamos previamente con la estructura:
Problema
Implementar una interfaz visual para el alta o carga de registros en la tabla artículos.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos2
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Lo primero que hacemos para el evento Click del botón es crear un objeto de la clase SqlConnection y
proceder a abrir la conexión:
Tenemos que guardar en la variable cadena un comando INSERT válido tomando los datos ingresados en
los dos TextBox:
Como el campo descripción es de tipo varchar debemos incluir las comillas simples:
Si cargamos por teclado por ejemplo el producto: manzanas y con un precio : 20, luego la variable cadena
tiene el valor:
Como podemos ver es una forma sencilla de elaborar un comando SQL siempre y cuando tengamos pocos
campos que inicializar.
Creamos un objeto de la clase SqlCommand y le pasamos al constructor un string con el comando SQL y la
referencia a la conexión.
Seguidamente llamamos al método ExecuteNonQuery que procede a comunicarse con el servidor para que
se ejecute el comando SQL configurado en la línea anterior:
comando.ExecuteNonQuery();
Y de esta forma poder ver los registros que almacena la tabla "articulos":
Retornar
Veremos ahora como recuperar los datos almacenados en una tabla para mostrarlos en pantalla.
Problema
Para recuperar datos de una tabla tenemos que utilizar el comando SQL select indicando el nombre de la
tabla de donde se recuperan los datos y los campos propiamente dichos.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos3
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
using System.Data.SqlClient;
Seguidamente seguimos con la creación del objeto de la clase SqlCommand pasando el string con el
comando SQL y la referencia a la conexión:
Lo nuevo aparece cuando queremos recuperar los datos que genera el SQL Server. Para esto llamamos al
método ExecuteReader del objeto SqlCommand:
Este método retorna un objeto de la clase SqlDataReader que almacena el resultado del comando SQL
select.
Para acceder a cada fila retornada por el comando SQL select debemos disponer una estructura repetitiva
while y llamar en cada vuelta del ciclo al método Read:
Imaginemos que el comando SQL select retorna 5 filas. Cada llamada a Read avanza un puntero al siguiente
registro y dentro del while lo recuperamos mediante el objeto registros y mediante un subíndice indicamos el
nombre de campo de la tabla a recuperar.
Por ejemplo si la tabla articulos estuviera vacía la primera vez que se ejecuta el while el método Read
retorna false.
Problema
En el evento Click procedemos a buscar el código del artículo ingresado en el TextBox. El código fuente es:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos4
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
using System.Data.SqlClient;
En el evento Click del botón buscar procedemos a conectarnos con el motor de base de datos y abrir la
conexión:
Seguidamente confeccionamos un string con el comando SQL select para recuperar la descripción y el
precio del artículo cuyo código coincide con el valor ingresado por teclado:
Si el resultado del comando select recuperó un registro de la tabla articulos luego la llamada al método Read
se verifica verdadero y procedemos a mostrar el resultado por pantalla:
if (registro.Read())
{
label4.Text = registro["descripcion"].ToString();
label5.Text = registro["precio"].ToString();
}
En el caso que hayamos ingresado un código inexistente procedemos a mostrar un mensaje por el else:
else
MessageBox.Show("No existe un artículo con el código ingresado");
Retornar
Problema
Implementar una aplicación que permita consultar un artículo por su código. Luego de mostrarlo activar un
botón para poder eliminarlo.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos5
{
public partial class Form1 : Form
{
private SqlConnection conexion = new SqlConnection("server=DIEGO-PC\\SQLEXPRESS
; database=base1 ; integrated security = true");
public Form1()
{
InitializeComponent();
}
using System.Data.SqlClient;
Como en los dos métodos utilizamos el objeto SqlConnection lo definimos como un atributo de la clase
Form1:
El algoritmo de consulta es idéntico al concepto que vimos anteriormente (solo modificamos la propiedad
Enabled del button2):
Lo nuevo aparece en el borrado del artículo que acabamos de consultar, primer confeccionamos un string
con el comando delete:
string cod = textBox1.Text;
string cadena = "delete from articulos where codigo=" + cod;
Luego procedemos a crear un objeto de la clase SqlCommand pasando el comando SQL de borrado con la
referencia de la conexión:
Llamamos seguidamente al método ExecuteNonQuery que se comunica con el SQL Server para que ejecute
el comando configurado previamente y retorna la cantidad de registros afectados (en este caso retorna la
cantidad de registros borrados):
int cant;
cant = comando.ExecuteNonQuery();
if (cant==1)
{
label4.Text = "";
label5.Text = "";
MessageBox.Show("Se borró el artículo");
}
Retornar
83 - SqlCommand (update): Modificar registro
Vimos hasta ahora el alta (carga), consulta (búsqueda) y baja (borrado). Ahora veremos como modificar
datos de una fila de la tabla.
Problema
Implementar una aplicación que permita consultar un artículo por su código. Luego mostrar en dos TextBox
la descripción y el precio actual. Permitir modificar la descripción y precio dejando registrado dicho cambio
en la tabla de la base de datos al presionar un botón.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos6
{
public partial class Form1 : Form
{
private SqlConnection conexion = new SqlConnection("server=DIEGO-PC\\SQLEXPRESS
; database=base1 ; integrated security = true");
public Form1()
{
InitializeComponent();
}
Primero indicamos el uso del espacio de nombres relacionado a la comunicación con SQL Server:
using System.Data.SqlClient;
Como en los dos métodos utilizamos el objeto SqlConnection lo definimos como un atributo de la clase
Form1:
En el evento Click de la modificación procedemos a configurar un string con el comando SQL update
rescatando los datos de los tres TextBox:
Seguidamente creamos un objeto de la clase SqlCommand pasando al constructor el string con el comando
update y la referencia a la conexión:
Llamamos al método ExecuteNonQuery para que se ejecute el comando update por parte del SQL Server y
nos retorne la cantidad de registros afectados:
int cant;
cant = comando.ExecuteNonQuery();
if (cant == 1)
{
MessageBox.Show("Se modificaron los datos del artículo");
textBox1.Text = "";
textBox2.Text = "";
textBox3.Text = "";
}
Esta metodología tiene el inconveniente cuando hay que elaborar comandos SQL con muchos campos
(imaginemos que tenemos que elaborar un insert con una tabla que tiene más de 10 campos)
Para solventar esta dificultad se introduce el concepto de parámetros dentro del comando SQL.
Problema 1
Implementar una interfaz visual para el alta de registros en la tabla artículos utilizando parámetros en el
comando insert.
Implementar una interfaz visual para el alta o carga de registros en la tabla artículos.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos7
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
Para trabajar con parámetros dentro de comandos SQL debemos disponer el caracter @ y un nombre en los
lugares donde se sustituye en tiempo de ejecución con el dato que carga el operador.
Es importante notar que no disponemos las camillas simple cuando un parámetro es de tipo varchar. Solo
indicamos nombres de parámetros y no tipo de parámetros.
Luego si debemos llamar al método Add de la propiedad Parameters del objeto de la clase SqlCommand
indicando cada parámetro y de que tipo se trata (con esto la clase SqlCommand conocerá si tiene que
disponer las comillas simples o no):
comando.Parameters.Add("@descripcion", SqlDbType.VarChar);
comando.Parameters.Add("@precio", SqlDbType.Float);
Luego que creamos los dos parámetros debemos inicializar sus valores mediante la propiedad Parameters y
accediendo como subíndice al nombre de parámetro respectivo:
comando.Parameters["@descripcion"].Value = textBox1.Text;
comando.Parameters["@precio"].Value = float.Parse(textBox2.Text);
Finalmente llamamos al método ExecuteNonQuery para enviar al Sql Server el comando Sql respectivo
(previamente el objeto comando de la clase SqlCommand se encarga de sustituir los parámetros por los
valores asignados y según el tipo de dato del parámetros añadir las comillas simples.
Problema 2
En el evento Click procedemos a buscar el código del artículo ingresado en el TextBox. El código fuente es:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos8
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
comando.Parameters.Add("@codigo", SqlDbType.Int);
comando.Parameters["@codigo"].Value = int.Parse(textBox1.Text);
Problema 3
Implementar una aplicación que permita consultar un artículo por su código. Luego de mostrarlo activar un
botón para poder eliminarlo. Definir parámetros tanto para la consulta como para el borrado.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Data.SqlClient;
namespace PruebaBaseDatos9
{
public partial class Form1 : Form
{
private SqlConnection conexion = new SqlConnection("server=DIEGO-PC\\SQLEXPRESS
; database=base1 ; integrated security = true");
public Form1()
{
InitializeComponent();
}
Problema 4
Implementar una aplicación que permita consultar un artículo por su código. Luego mostrar en dos TextBox
la descripción y el precio actual. Permitir modificar la descripción y precio dejando registrado dicho cambio
en la tabla de la base de datos al presionar un botón. Utilizar parámetros en los comandos SQL en la
consulta y borrado
namespace PruebaBaseDatos10
{
public partial class Form1 : Form
{
private SqlConnection conexion = new SqlConnection("server=DIEGO-PC\\SQLEXPRESS
; database=base1 ; integrated security = true");
public Form1()
{
InitializeComponent();
}
Retornar
Pero hay situaciones donde definimos un atributo y luego una propiedad que accede a dicho atributo y no
definimos ningún algoritmo tanto en la sección del get como del set. En estos casos nos cuestionamos el
porque no definir public el atributo y no definir una propiedad para accederlo. Debido a estos casos en C# se
ha agregado la posibilidad de definir propiedades automáticas.
Una propiedad automáticas crea en forma transparente un atributo donde se almacena el valor y
posteriormente se consulta.
Veamos un ejemplo como definimos una propiedad en la forma tradicional y como la definimos con esta
técnica.
Problema 1:
Plantear una clase llamada Cuadrado definir una propiedad llamada Lado. Calcular la superficie.
Veamos primero el código que ya conocemos para la definición de un atributo y la propiedad que la accede:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades1
{
class Cuadrado
{
private int lado;
class Program
{
static void Main(string[] args)
{
Cuadrado cuadrado1 = new Cuadrado();
cuadrado1.Lado = 30;
Console.WriteLine("Su superficie es:" + cuadrado1.RetornarSuperficie());
Console.ReadKey();
}
}
}
Veamos la sintaxis que reduce la cantidad de líneas que debe escribir el programador para los casos en que
una propiedad tiene el objetivo de fijar y recuperar el valor de un atributo atributo:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades1
{
class Cuadrado
{
public int Lado { get; set; }
class Program
{
static void Main(string[] args)
{
Cuadrado cuadrado1 = new Cuadrado();
cuadrado1.Lado = 30;
Console.WriteLine("Su superficie es:" + cuadrado1.RetornarSuperficie());
Console.ReadKey();
}
}
}
Como podemos ver podemos definir una propiedad automática con la sintaxis:
Podemos definir modificadores de acceso para el get y el set, por ejemplo la sintaxis:
No permite extraer el valor de la propiedad desde fuera de la clase, por ejemplo sería un error la siguiente
sintaxis:
class Program
{
static void Main(string[] args)
{
Cuadrado cuadrado1 = new Cuadrado();
cuadrado1.Lado = 30;
//La siguiente línea genera un error de compilación ya que Lado define el
'get' en forma privada
int x = cuadrado1.Lado;
Console.WriteLine("Su superficie es:" + cuadrado1.RetornarSuperficie());
Console.ReadKey();
}
}
Dentro de la clase no hay problema de acceder a la propiedad 'Lado' por más que sea private:
En la última versión de C# se permite inicializar la propiedad automática con un valor cuando se la define,
en lugar de tener que hacerlo en el constructor:
Retornar
86 - Creando un objeto e iniciando sus
propiedades inmediatamente
Hemos visto en conceptos anteriores que cuando se crea un objeto lo primero que sucede es la ejecución del
constructor.
Veremos ahora que en el momento de crear un objeto podemos iniciar una o varias de sus propiedades.
Problema 1:
Plantear una clase llamada Triangulo. Por defecto el valor de cada lado del triángulo es 5. Calcular el
perímetro.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades2
{
class Triangulo
{
public int Lado1 { get; set; } = 5;
public int Lado2 { get; set; } = 5;
public int Lado3 { get; set; } = 5;
class Program
{
static void Main(string[] args)
{
Triangulo triangulo1 = new Triangulo { Lado1 = 10, Lado2 = 20, Lado3 = 30
};
Console.WriteLine(triangulo1.RetornarPerimetro());
Triangulo triangulo2 = new Triangulo { Lado3 = 30 };
Console.WriteLine(triangulo2.RetornarPerimetro());
Console.ReadKey();
}
}
}
Luego cuando creamos un objeto de la clase Triangulo debemos indicar entre llaves los nombres de las
propiedades y los valores que se deben asignar, no es obligatorio tener que inicializar todas las propiedades:
Triangulo triangulo1 = new Triangulo { Lado1 = 10, Lado2 = 20, Lado3 = 30
};
Console.WriteLine(triangulo1.RetornarPerimetro());
Triangulo triangulo2 = new Triangulo { Lado3 = 30 };
Console.WriteLine(triangulo2.RetornarPerimetro());
Es importante notar que si la clase tiene constructor primero se ejecuta este y luego se inician las
propiedades.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Propiedades2
{
class Triangulo
{
public int Lado1 { get; set; } = 5;
public int Lado2 { get; set; } = 5;
public int Lado3 { get; set; } = 5;
class Program
{
static void Main(string[] args)
{
Triangulo triangulo1 = new Triangulo("Perimetro triangulo 1") { Lado1 = 10,
Lado2 = 20, Lado3 = 30 };
Console.WriteLine(triangulo1.RetornarPerimetro());
Triangulo triangulo2 = new Triangulo("Perimetro triangulo 2") { Lado3 = 30
};
Console.WriteLine(triangulo2.RetornarPerimetro());
Console.ReadKey();
}
}
}
Primero se ejecuta el contructor donde se muestra el string que le pasamos como parámetro y seguidamente
se inician las propiedades Lado1, Lado2 y Lado3:
Si bien en el constructor iniciamos las propiedades con el valor 1, luego le asignamos los valores 10,20 y 30
(estos son los que quedan en definitiva)
Retornar
87 - Clases genéricas
C# permite crear clases que administren distintos tipos de datos.
Se utilizan mucho para la administración de colecciones de datos (pilas, colas, listas, árboles etc.)
Para entender las ventajas de definir clases genéricas implementaremos los algoritmos para administrar una
pila de enteros y una pila de string. Primero lo haremos utilizando clases tradicionales y luego mediante una
clase genérica.
Para concentrarnos en la sintaxis plantearemos la pila utilizando un vector de 5 elementos y definiremos los
dos métodos fundamentales de insertar y extraer (no haremos ningún tipo de validaciones por simplicidad)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace SinGenericos
{
class PilaEnteros
{
private int[] vec = new int[5];
private int tope = 0;
class PilaString
{
private string[] vec = new string[5];
private int tope = 0;
class Program
{
static void Main(string[] args)
{
PilaEnteros pila1 = new PilaEnteros();
pila1.Insertar(20);
pila1.Insertar(40);
pila1.Insertar(17);
Console.WriteLine(pila1.Extraer());
Console.ReadKey();
}
}
}
Como podemos analizar hemos planteado dos clases, una para administrar una pila con tipos de dato
enteros:
class PilaEnteros
{
private int[] vec = new int[5];
private int tope = 0;
Y por otro lado otra clase para administrar una pila de tipo de dato string:
class PilaString
{
private string[] vec = new string[5];
private int tope = 0;
En la main para probar estas dos clases definimos un objeto de la clase PilaEnteros e insertamos tres valores
y luego extraemos uno:
Hasta este momento no hemos presentado ninguna novedad con respecto a lo que conocemos. Veamos ahora
como podemos resolver este problema pero empleando una clase genérica:
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Genericos1
{
class Pila<T>
{
private T[] vec = new T[5];
private int tope = 0;
public T Extraer()
{
tope--;
return vec[tope];
}
}
class Program
{
static void Main(string[] args)
{
Pila <int>pila1 = new Pila<int>();
pila1.Insertar(20);
pila1.Insertar(40);
pila1.Insertar(17);
Console.WriteLine(pila1.Extraer());
Console.ReadKey();
}
}
}
Como vemos hemos declarado una sola clase llamada Pila y hemos sustituido en los lugares donde hacíamos
referencia a int o string por el tipo 'T' que también tenemos que hacer referencia en la primer línea:
class Pila<T>
{
private T[] vec = new T[5];
private int tope = 0;
public T Extraer()
{
tope--;
return vec[tope];
}
}
Luego en la main cuando creamos un objeto de la clase Pila debemos indicar cuando la creamos el tipo de
datos que administrará nuestra pila:
Podemos crear objetos de la clase Pila con cualquier tipo de dato primitivo (int, char, float, double etc.) o de
otra clase.
Modifiquemos el ejercicio anterior para crear una pila de la clase Persona (almacena el nombre y la edad):
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Genericos2
{
class Persona
{
public string Nombre { get; set; }
public int Edad { get; set; }
}
class Pila<T>
{
private T[] vec = new T[5];
private int tope = 0;
public T Extraer()
{
tope--;
return vec[tope];
}
}
class Program
{
static void Main(string[] args)
{
Pila<Persona> pila1 = new Pila<Persona>();
Persona persona1 = new Persona { Nombre = "Juan", Edad = 22 };
Persona persona2 = new Persona { Nombre = "Ana", Edad = 34 };
Persona persona3 = new Persona { Nombre = "Carlos", Edad = 47 };
pila1.Insertar(persona1);
pila1.Insertar(persona2);
pila1.Insertar(persona3);
Persona p = pila1.Extraer();
Console.WriteLine(p.Nombre + " " + p.Edad);
Console.ReadKey();
}
}
}
class Persona
{
public string Nombre { get; set; }
public int Edad { get; set; }
}
class Pila<T>
{
private T[] vec = new T[5];
private int tope = 0;
public T Extraer()
{
tope--;
return vec[tope];
}
}
En la main creamos un objeto de la clase Pila e indicamos que almacenará objetos de la clase Persona:
Extraemos un elemento de la pila y guardamos su referencia en la variable "p" que debe ser de la clase
Persona:
Persona p = pila1.Extraer();
Console.WriteLine(p.Nombre + " " + p.Edad);
Como vemos el planteo de clases genéricas nos reduce tener que crear múltiples clases para administrar
distintos tipos de datos.
Problema:
Plantear una clase para administrar una lísta de datos utilizando genéricos. Implemente los métodos para
insertar, extraer, cantidad e imprimir.
Crear luego tres objetos de la clase Lista uno con enteros, otros con string y finalmente otro de tipo Persona
(declarar una clase Persona con dos propiedades: Nombre y Edad)
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Genericos2
{
class Persona
{
public string Nombre { get; set; }
public int Edad { get; set; }
class ListaGenerica<T>
{
class Nodo
{
public T Info { get; set; }
public Nodo Sig { get; set; }
}
public ListaGenerica()
{
raiz = null;
}
class Program
{
static void Main(string[] args)
{
ListaGenerica<int> lista1 = new ListaGenerica<int>();
lista1.Insertar(1, 10);
lista1.Insertar(2, 50);
lista1.Insertar(3, 70);
lista1.Imprimir();
Console.ReadKey();
}
}
}
Declaramos la clase indicando el tipo 'T' (podemos disponer cualquier nombre en lugar de 'T', pero por
convención se suele utilizar este caracter):
class ListaGenerica<T>
La estructura del nodo define una propiedad llamada Info de tipo 'T':
class Nodo
{
public T Info { get; set; }
public Nodo Sig { get; set; }
}
Luego podemos insertar un conjunto de enteros en distintas posiciones en la lista e imprimir la lista:
lista1.Insertar(1, 10);
lista1.Insertar(2, 50);
lista1.Insertar(3, 70);
lista1.Imprimir();
De la misma forma creamos ahora una lista con componentes de tipo string:
Para trabajar una lista de tipo Persona debemos por un lado declarar la clase Persona:
class Persona
{
public string Nombre { get; set; }
public int Edad { get; set; }
Y en la Main creamos una lista de tipo Persona e insertamos objetos de dicha clase:
ListaGenerica<Persona> lista3 = new ListaGenerica<Persona>();
lista3.Insertar(1, new Persona { Nombre = "juan", Edad = 20 });
lista3.Insertar(2, new Persona { Nombre = "ana", Edad = 12 });
lista3.Insertar(3, new Persona { Nombre = "luis", Edad = 40 });
lista3.Imprimir();
Console.Write(reco.Info + "-");
Utilizar genéricos en C# nos reduce mucho el código a implementar, podemos utilizar la misma clase
administrando distintos tipos de datos.
Retornar
88 - Colecciones
Hemos visto en conceptos anteriores como administrar distintas estructuras de datos estáticas (vectores,
matrices) y dinámicas (listas y árboles)
Aprendimos a crear clases en C# para administrar listas tipo pila, cola y genéricas. Desarrollamos todos los
algoritmos internos para su administración utilizando punteros.
Veremos ahora que en Microsoft .Net tenemos un conjunto de clases que nos facilitan la creación de pilas,
colas y listas genéricas.
En muchas situaciones el empleo de esta librería de clases nos reducen el tiempo de desarrollo de un
programa.
Para trabajar con estas clases debemos importar el espacio de nombres 'System.Collections.Generic' donde
se encuentran las mismas.
Todas estas clases están implementadas con el concepto de genéricos para poder almacenar cualquier tipo de
datos.
Las colecciones fundamentales que podemos hacer uso en nuestros proyectos son:
Queue<T> : Implementa el concepto de una cola (FIFO - Fist In First Out - Primero en entrar
primero en salir)
Stack<T> : Implementa el concepto de una pila (LIFO - Last In First Out - Ultimo en entrar primero
en salir)
List<T> : Administra una lista de elementos accesible mediante un subíndice como los vectores pero
con la ventaja de que puede crecer en forma dinámica.
LinkedList<T> : Administra una lista doblemente encadenada con las facilidades de insertar y
extraer elementos en cualquier parte de la lista en forma muy eficiente.
Dictionary<TKey, TValue> Facilita administrar una lista de valores accesibles mediante una clave.
Retornar
Queue<T> : Implementa el concepto de una cola (FIFO - Fist In First Out - Primero en entrar
primero en salir)
Stack<T> : Implementa el concepto de una pila (LIFO - Last In First Out - Ultimo en entrar primero
en salir)
Veamos un programa que defina objetos de estas clases y llame a sus diferentes métodos.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PilaCola1
{
class Program
{
static void Main(string[] args)
{
Stack<int> pila1 = new Stack<int>();
Console.WriteLine("Insertamos tres elementos en la pila:10, 25 y 70");
pila1.Push(10);
pila1.Push(25);
pila1.Push(70);
Console.WriteLine("Cantidad de elementos en la pila:"+pila1.Count);
Console.WriteLine("Extraemos un elemento de la pila:" + pila1.Pop());
Console.WriteLine("Cantidad de elementos en la pila:" + pila1.Count);
Debemos importar el espacio de nombres donde están definidas las clases Queue<T> y Stack<T>:
using System.Collections.Generic;
Creamos un objeto de la clase Stack<T>, debemos indicar el tipo de datos que almacenará la pila:
pila1.Push(10);
pila1.Push(25);
pila1.Push(70);
Para conocer la cantidad de elementos que almacena la colección de tipo Stack<T> accedemos a la
propiedad Count:
cola1.Enqueue("ana");
cola1.Enqueue("juan");
cola1.Enqueue("pedro");
También esta clase tiene una propiedad Count que indica la cantidad de elementos de la cola:
Este práctico ya lo resolvimos cuando vimos el concepto de pilas e implementamos manualmente una clase
Pila y sus métodos.
Todo compilador o intérprete de un lenguaje tiene un módulo dedicado a analizar si una expresión está
correctamente codificada, es decir que los paréntesis estén abiertos y cerrados en un orden lógico y bien
balanceados.
Se debe desarrollar un programa que tenga las siguientes responsabilidades (clase Formula):
Veamos como nos puede ayudar el empleo de una pila para solucionar este problema.
Primero cargaremos la fórmula en un TextBox.
}(2+[3-12]*{8/3})
Incorrecta: llega una } de cerrado y la pila está vacía.
{[2+4}]
Incorrecta: llega una llave } y en el tope de la pila hay un corchete [.
{[2+4]
Incorrecta: al finalizar el análisis del último caracter en la pila queda pendiente una
llave {.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace ProblemaPila1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
En el método Click verifica si la fórmula están correctos los parentesis, corchetes y llaves.
Stack<char> pila1;
pila1 = new Stack<char>();
string cadena = textBox1.Text;
En caso de ser un ) cerrado debemos verificar si la pila está vacía o en la pila no coincide con el paréntesis
de apertura '(' la fórmula está incorrecta:
if (cadena.ElementAt(f) == ')')
{
if (pila1.Count==0 || pila1.Pop() != '(')
{
Text = "Incorrecta";
return;
}
}
Al finalizar el análisis de toda la cadena si la pila está vacía podemos afirmar que la fórmula está
correctamente balanceada, en caso contrario quiere decir que faltan símbolos de cerrado y es incorrecta:
if (pila1.Count==0)
{
Text = "Correcta";
}
else
{
Text = "Incorrecta";
}
Es importante entender que la clase Form utiliza un objeto de la clase Stack<T> para resolver el algoritmo
de verificar el balanceo de la fórmula.
Problema 2:
Este práctico tiene por objetivo mostrar la importancia de las colas en las Ciencias de la Computación y más
precisamente en las simulaciones.
Las simulaciones permiten analizar situaciones de la realidad sin la necesidad de ejecutarlas realmente.
Tiene el beneficio que su costo es muy inferior a hacer pruebas en la realidad.
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace ProblemaCola1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
La clase Form1 define tres objetos de la clase Label para mostrar los resultados de la simulación.
El método más importante es el click del botón, veamos las distintas partes de dicho método:
La variable estado almacena un cero si el cajero está libre y un uno cuando está ocupado.
La variable llegada almacena en que minuto llegará el próximo cliente (debemos generar un valor entre 2 y
3)
La variable salida almacenará en que minuto terminará el cliente de ser atendido (como al principio el cajero
está vacío inicializamos esta variable con -1)
Creamos un objeto de la clase Random para poder utilizar el método Next que nos retorna un valor aleatorio
en el rango que le pasamos como parámetros (si pasamos un 0 y 2 luego nos puede retornar un 0 o un 1)
Disponemos un for que se repita 600 veces (600 minutos o lo que es lo mismo 10 horas)
Dentro del for hay dos if fundamentales que verifican que sucede cuando llega una persona o cuando una
persona se retira:
if (llegada == minuto)
{
............
}
if (salida == minuto)
{
............
}
Cuando llega una persona al cajero primero verificamos si el cajero está desocupado:
if (llegada == minuto)
{
if (estado==0)
{
Si está desocupado lo ocupamos cambiando el valor de la variable estado y generando en que minuto esta
persona dejará el cajero (un valor aleatorio entre 2 y 4 minutos):
estado = 1;
salida = minuto + 2 + ale.Next(0, 3);
Si el cajero está ocupado procedemos a cargar dicha persona en la cola (insertamos el minuto que llega):
else
{
cola.Enqueue(minuto);
}
El otro if importante es ver que sucede cuando sale la persona del cajero:
if (salida == minuto) {
Si sale una persona del cajero cambiamos el valor de la variable estado, incrementamos en uno el contador
cantAtendidos y si la cola no está vacía extraemos una persona, cambiamos a uno la variable estado y
generamos en que minuto dejará esta persona el cajero:
estado = 0;
cantAtendidas++;
if (cola.Count!=0)
{
cola.Dequeue();
estado = 1;
salida = minuto + 2 + ale.Next(0, 3);
}
label1.Text="Atendidos:" + cantAtendidas.ToString();
label2.Text="En cola" + cola.Cantidad().ToString();
label3.Text="Minuto llegada:" + cola.Extraer().ToString();
Retornar
90 - Colección : List<T>
La clase List<T> nos facilita administrar un conjunto de datos en forma similar a un vector por medio de
subíndices.
Esta clase trae una serie de ventajas con respecto a los vectores:
Como desventaja podemos hacer notar que los tiempos de redimensionar la colección son costosos.
Problema 1:
Confeccionar un programa que cree un objeto de la clase List<T> y llamar a su métodos principales para
insertar, consultar y eliminar elementos de la colección.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ProblemaList1
{
class Program
{
private static void ImprimirColeccion(List<int> lista)
{
for (int x = 0; x < lista.Count; x++)
Console.Write(lista[x] + "-");
Console.WriteLine();
}
El objetivo de cada uno de los métodos está indicado mediante un mensaje en el mismo programa:
Retornar
91 - Colección : LinkedList<T>
La clase LinkedList<T> nos facilita administrar una lista doblemente encadenada. A diferencia de la clase
List<T> es más eficiente cuando tenemos que insertar y borrar elementos del medio de la colección.
Hay que tener en cuenta que con la clase List<T> si insertamos un elemento en la primer posición se deben
desplazar todos los elementos una posición hacia adelante, en cambio con la clase LinkedList<T> solo se
modifican punteros.
Como desventaja no podremos acceder a los elementos por medio de un subíndice como lo hacemos con la
clase List<T> y deberemos recorrerla mediante una estructura repetitiva.
Problema 1:
Confeccionar un programa que cree un objeto de la clase LinkedList<T> y llamar a sus métodos y
propiedades principales.
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace ProblemaLinkedList1
{
class Program
{
private static void ImprimirLista(LinkedList<int> lista)
{
LinkedListNode<int> reco = lista.First;
while (reco!=null)
{
Console.Write(reco.Value + "-");
reco = reco.Next;
}
Console.WriteLine();
}
lista1.AddFirst(30);
lista1.AddFirst(20);
lista1.AddFirst(10);
lista1.AddLast(1);
Para imprimir la lista la clase LinkedList<T> tiene una propiedad llamada First que almacena la referencia
al primer nodo de la lista (los nodos son de la clase LinkedListNode<int>), para avanzar al siguiente nodo
accedemos a la propiedad Next que guarda la referencia del siguiente nodo:
Para imprimir la lista del final al principio solo debemos obtener la referencia del último nodo y
retrocedemos al nodo anterior mediante la propiedad Previous:
Problema 2:
Programa:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace Vibora
{
public partial class Form1 : Form
{
private enum TDireccion { izquierda, derecha, arriba, abajo };
private TDireccion Direccion { get; set; } = TDireccion.derecha;
private LinkedList<Punto> lista1 = new LinkedList<Punto>();
private Punto Fruta { get; set; } = new Punto(10, 10);
private int Pendientes { get; set; } = 0;
public Form1()
{
InitializeComponent();
lista1.AddFirst(new Punto(0, 0));
}
Analicemos un poco nuestro problema, primero debemos definir un Timer para que avance la serpiente
varias veces por segundo:
Definimos que el Timer se dispare cada 100 milisegundos y lo activamos con el valor true en la propiedad
Enabled.
Definimos un atributo de tipo LinkedList que almacenará cada trozo de la víbora. Cada trozo son de tipo
Punto que almacena la coordenada X e Y del segmento:
Creamos el tipo de dato enum para que nuestro programa sea más legible cuando necesitemos saber la
dirección actual de la víbora.
La propiedad fruta almacena la coordenada donde aparece la primera fruta que puede comer la víbora (la
primera fruta aparece en la coordenada 10,10):
Por último la propiedad Pendientes representa cuantos segmentos debemos agregarle a la víbora cuando
coma una fruta:
En el constructor agregamos el primer segmento a la víbora indicando que arranca en la coordenada 0,0:
public Form1()
{
InitializeComponent();
lista1.AddFirst(new Punto(0, 0));
}
El método timer1_Tick se dispara cada 100 milisegundos y lo primero que hacemos es verificar cual es la
dirección que tiene actualmente la víbora y según este valor añadimos un nodo al LinkedList al principio.
También tenemos que controlar si pierde llamando a los métodos: SaleMapa y SePisa. Verificamos si come
la fruta llamando a TocaFruta. Removemos la cola de la víbora llamando RemoverCola y repintamos la
pantalla llamando al método Invalidate:
El método Form1_KeyDown se dispara cuando el operador presiona alguna tecla del teclado y en la misma
verificamos cual de las teclas de flechas presionó. Cambiamos el valor de la propiedad Direccion por el
valor respectivo:
El método Form1_Paint se ejecuta cuando aparece el Form en pantalla y cada vez que se llama al método
Invalidate. Aquí dibujamos todos los segmentos de la víbora y también dibujamos la fruta:
En el método TocaFruta verificamos si la cabeza de la víbora (que está representada por el primer nodo del
LinkedList) coincide con las coordenadas de la fruta, en el caso de coincidir iniciamos la propiedad
Pendientes con el valor 5 (indica que se agregaran 5 segmentos a la víbora) y generamos otra coordenada
aleatoria para la fruta:
Cada vez que se dispara el Timer agregamos un nodo a la lista al principio y borramos el último, con esto
logramos que se desplace la víbora. Si la propiedad Pendientes es cero se borra el nodo en caso que la
propiedad sea distinta a cero dejamos el nodo del final con lo que logramos que la víbora crezca en un
segmento (cada vez que come una fruta crece 5 segmentos):
El método SaleMapa se ejecuta cada vez que se dispara el Timer y controlamos si la cabeza de la víbora sale
del área (el área permitida son las coordenadas en X de 0 a 39 y en Y de 0 a 39):
Retornar
Para cada clave tenemos un valor asociado. Podemos después buscar fácilmente un valor para una
determinada clave. Las claves en el diccionario no pueden repetirse.
Guardar en la clave las extensiones de archivos y en el valor los nombres de archivos que lo pueden
abrir
En una agenda podemos guardar como 'clave' la fecha y hora y las actividades en el 'valor'.
Problema:
Almacenar un diccionario las palabras en castellano como 'clave' y las traducciones de las mismas en el
'valor'. Probar los métodos más significativos de la clase Dictionary<TKey, TValue>
Programa:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Diccionario1
{
class Program
{
static void Main(string[] args)
{
Dictionary <string,string> dicc1 = new Dictionary<string,string> ();
dicc1["rojo"]="red";
dicc1["verde"]="green";
dicc1["azul"]="blue";
dicc1["blanco"]="whilte";
foreach( KeyValuePair<string,string> elemento in dicc1)
Console.WriteLine(elemento.Key + "=" + elemento.Value);
if (dicc1.ContainsKey("rojo"))
Console.WriteLine(dicc1["rojo"]);
dicc1.Remove("rojo");
if (dicc1.ContainsKey("rojo"))
Console.WriteLine(dicc1["rojo"]);
else
Console.WriteLine("No existe la clave 'rojo'");
Console.ReadKey();
}
}
}
En este problema tanto la clave como el valor son de tipo string, de todos modos pueden ser de otro tipo
dependiendo el problema, la creación del objeto es:
Para cargar elementos en el diccionario podemos indicar en el subíndice la clave y le asignamos el valor a
guardar para dicho subíndice:
dicc1["rojo"]="red";
dicc1["verde"]="green";
dicc1["azul"]="blue";
dicc1["blanco"]="whilte";
También podríamos haber utilizado la llamada al método Add para obtener el mismo resultado:
dicc1.Add("rojo", "red");
dicc1.Add("verde", "green");
dicc1.Add("azul", "blue");
dicc1.Add("blanco", "white");
Para recorrer el diccionario en forma completa podemos utilizar el foreach y en cada ciclo nos retorna un
objeto de tipo KeyValuePair<string,string>:
Para verificar si existe una determinada clave en el diccionario podemos hacerlo mediante el método
ContainsKey y luego accedemos al valor de dicha clave mediante subíndice:
if (dicc1.ContainsKey("rojo"))
Console.WriteLine(dicc1["rojo"]);
Si necesitamos eliminar un elemento del diccionario lo hacemos con el método Remove pasando como
parámetro la clave:
dicc1.Remove("rojo");
Retornar