domingo, 4 de febrero de 2018

Tutorial Java 1

Objetivos del curso y nociones básicas indispensables

 

El curso está ideado para ser desarrollado por una persona que no conoce nada de programación y se utilice Java como primer lenguaje.
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.
Para la resolución de un problema hay que plantear un algoritmo.
Algoritmo: Son los pasos a seguir para resolver un problema.

Diagrama de flujo

Un diagrama de flujo es la representación gráfica de un ALGORITMO.
Los símbolos gráficos a utilizar para el planteo de diagramas de flujo son:
simbolos de diagrama de flujo
Estos son los elementos esenciales que intervienen en el desarrollo de un diagrama de flujo.

Planteo de un problema utilizando diagramas de flujo.

Para plantear un diagrama de flujo debemos tener muy en claro el problema a resolver.
Ejemplo : Calcular el sueldo mensual de un operario conociendo la cantidad de horas trabajadas y el pago por hora.
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.
Si hacemos un análisis todo problema está constituido por:
- Datos conocidos: Datos con los que se cuenta al plantear el problema.
- Proceso: Operaciones a realizar con los datos conocidos.
- Información resultante: Es la información que resuelve el problema.
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.
diagrama de flujo calculo sueldo
Resulta mucho más fácil entender un gráfico que un texto.
El diagrama de flujo nos identifica claramente los datos de entrada, operaciones y datos de salida.
En el ejemplo tenemos dos datos de entrada: horasTrabajadas y costoHora, a las entradas las representamos con un paralelogramo y hacemos un paralelogramo por cada dato de entrada.
La operación se representa con un rectángulo, debemos hacer un rectángulo por cada operación. A la salida la representamos con la hoja rota.
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.

Codificación del problema con el lenguaje Java.

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 Java.
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.

Conceptos básicos para codificar un programa.

Variable: Es un depósito donde hay un valor. Consta de un nombre y pertenece a un tipo.
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.)
Elección del nombre de una variable:
Debemos elegir nombres de variables representativas. En el ejemplo el nombre horasTrabajadas es lo suficientemente claro para darnos una idea acabada sobre su contenido. Podemos darle otros buenos nombres. Otros no son tan representativos, por ejemplo hTr. Posiblemente cuando estemos resolviendo un problema dicho nombre nos recuerde que almacenamos las horas trabajadas por el operario pero cuando pase el tiempo y leamos el diagrama probablemente no recordemos ni entendamos qué significa hTr.

Consideraciones a tener en cuenta en cada proyecto.

Hay que tener en cuenta que el entorno de programación "Eclipse" no a sido desarrollado pensando en un principiante de la programación. Lo mismo ocurre con el propio lenguaje Java, es decir su origen no tiene como principio 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.
Codificaremos el problema propuesto para repasar los pasos para la creación de un proyecto en Eclipse, creación de la clase principal, definición de la función main y el posterior desarrollo del algoritmo del problema.

Pasos.

1 - Creación del proyecto (tema visto anteriormente). Podemos asignarle como nombre: SueldoOperario (normalmente uno busca un nombre representativo al programa que desarrolla)
creación proyecto java
diálogo proyecto java
2 - Creación de la clase. Definiremos como nombre el mismo que le asignamos al proyecto (esto no es obligatorio como veremos más adelante un proyecto puede contener varias clases)
Es decir disponemos como nombre de la clase: SueldoOperario.
clase java
Inicializamos el campo que solicita el "Name" con "SueldoOperario".
3 - Codificamos el algoritmo en la clase:SueldoOperario.
import java.util.Scanner;

public class SueldoOperario {

    public static void main(String[] ar) {
        Scanner teclado=new Scanner(System.in);
        int horasTrabajadas;
        float costoHora;
        float sueldo;
        System.out.print("Ingrese la cantidad de horas trabajadas por el empleado:");
        horasTrabajadas=teclado.nextInt();
        System.out.print("Ingrese el valor de la hora:");
        costoHora=teclado.nextFloat();
        sueldo=horasTrabajadas * costoHora;
        System.out.print("El empleado debe cobrar:");
        System.out.print(sueldo);
    }
}
4 - Ejecutamos el programa:
ejecutar programa java
5 - Si no hay errores sintácticos procedemos a activar la ventana de la "Console" con el mouse y cargamos por teclado los dos datos que se solicitan (la cantidad de horas trabajadas y el precio de la hora):
ejecutar programa java
Estos cinco pasos fundamentales debemos llevar a cabo cada vez que desarrollemos un nuevo programa en Java.

Explicación.

Ahora veremos una explicación de varias partes de nuestro programa y otras partes quedarán pendientes para más adelante ya que en este momento difícilmente se entiendan.
Conceptos que quedarán pendientes para explicar:
  1. Concepto de una clase. Veremos más adelante que en Java todo debe estar contenido en clases, por lo que hasta el problema más elemental debe estar contenido en una clase. Para declarar una clase utilizamos la sintaxis:

    public class SueldoOperario {
    
    }
    
    El nombre de la clase no puede tener espacios en blanco, comienza con una letra mayúscula y en caso de estar constituida por dos o más palabras el primer caracter va en mayúsculas, no puede empezar con un número, pero si puede llevar números a partir del segundo caracter. Toda clase debe tener una llave de apertura y una llave de cierre.
  2. Todo programa constituido por una única clase debe tener definida la función main:

        public static void main(String[] ar) {
    
        }
    
    La función main es la primera que se ejecuta y debe llevar la sintaxis indicada anteriormente (más adelante veremos que significa el parámetro ar, las palabras claves public, static y void. La función main tiene una llave de apertura y una llave de cierre (similar a la clase). La función main debe estar contenida en la clase.
  3. Cuando se requieren utilizar otras clases debemos importarlas previo a la declaración de la clase (en nuestro problema utilizamos la clase Scanner que se encuentra en el paquete java.util por lo que la importamos con la siguiente sintaxis:

    import java.util.Scanner;
    
    En la main creamos un objeto de la clase Scanner que nos permitirá ingresar por teclado los valores:

            Scanner teclado=new Scanner(System.in);
    
Conceptos que deben quedar claros:
  1. Por el momento haremos todo el algoritmo dentro de la función main. Es decir el resto siempre será lo mismo (declarar un proyecto, declarar una clase, definir una función main)
  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.

    La definición de las variables la hacemos en la main:

            int horasTrabajadas;
            float costoHora;
            float sueldo;
    
    Utilizamos la palabra clave int para definir variables enteras (en Java 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úsculas y en caso de estar constituida por dos palabras o más a partir de la segunda palabra el primer caracter se especifique con mayúsculas (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)
  3. Para mostrar mensajes en la "Console" utilizamos la siguiente sintaxis:

            System.out.print("Ingrese la cantidad de horas trabajadas por el empleado:");
    
    Con esta sintaxis todo lo que se encuentra contenido entre comillas aparecerá exactamente en la ventana de la "Console".
    Si disponemos una variable:
            System.out.print(sueldo);
    
    Aparecerá el contenido de la variable. Es decir el valor almacenado en la variable sueldo y no el mensaje "sueldo".
  4. Para hacer la entrada de datos por teclado en Java se complica. Utilizaremos una clase llamada Scanner que nos facilita el ingreso de datos. Por eso tuvimos que importar la clase Scanner que se encuentra en el paquete java.util en la primer línea de nuestro programa.
    En la función main debemos crear un objeto de la clase Scanner con la siguiente sintaxis:

            Scanner teclado=new Scanner(System.in);
    
    Luego para cargar valores enteros por teclado debemos implementar la siguiente sintaxis:
            horasTrabajadas=teclado.nextInt();
    
    Pero si el dato a cargar se trata de un valor float luego debemos utilizar la siguiente sintaxis:
            costoHora=teclado.nextFloat();
    
  5. Las operaciones que indicamos en el diagrama de flujo mediante la figura rectángulo la codificamos tal cual:

            sueldo=horasTrabajadas * costoHora;
    
Podemos ver una relación entre las instrucciones que debemos utilizar para cada símbolo del diagrama de flujo:
programa y diagrama de flujo
En el diagrama de flujo no indicamos la definición de variables:
        int horasTrabajadas;
        float costoHora;
        float sueldo;
No indicamos la creación del objeto de la clase Scanner:
        Scanner teclado=new Scanner(System.in);
No representamos con símbolos los mensajes a mostrar previo a la carga de datos por teclado:
        System.out.print("Ingrese la cantidad de horas trabajadas por el empleado:");
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.

Javascript Parte LXXXVII

El objeto String tiene métodos que reciben como parámetro una expresión regular.
El primer método es el search. Este método busca la primer ocurrencia del string según la expresión que le hemos pasado como parámetro. Devuelve un entero indicando la posición donde comienza la cadena encontrada o un -1 si no encuentra el patrón en el string.




<html>
<head>
</head>
<body>

<script type="text/javascript">
  var oracion='La noche está muy oscura y estrellada.';
  if (oracion.search(/estrella/)!=-1)
    alert('el string estrella está contenido en la variable oracion');
</script>

</body>
</html>
Hemos dispuesto un if para controlar si el string 'estrella' está contenido en la variable oración:
  if (oracion.search(/estrella/)!=-1)
    alert('el string estrella está contenido en la variable oracion');
Por supuesto que podemos emplear expresiones regulares tan complejas como necesitemos utilizando distintos metacaracteres y modificadores.
El segundo método del objeto String que veremos es el replace.
El método replace realiza una búsqueda y reemplazo. Se toma una expresión regular como primer parámetro y un string de reemplazo como su segundo argumento. Si la expresión regular tiene la bandera g, el método replace sustituye todas las coincidencias en la cadena de reemplazo, en caso contrario, se reemplaza sólo la primera coincidencia que encuentre.
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var oracion='uno dos tres uno dos tres uno dos tres';
  var cadena=oracion.replace(/uno/g,'1');
  alert(cadena); //1 dos tres 1 dos tres 1 dos tres
</script>

</body>
</html>
En este problema queremos obtener otro String sustituyendo todas las cadenas 'uno' por el caracter '1'. Para hacer que se sustituyan todas las ocurrencias debemos emplear el modificador g:
  var cadena=oracion.replace(/uno/g,'1');
Es importante tener en cuenta que la variable que llama al método replace no se modifica, sino que retorna otro String con el resultado.
El tercer método es el split que tiene por objetivo subdividir un string y retornar un vector con dichas partes. Para la división toma en cuenta la expresión regular pasada como parámetro.
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var cadena='(2,3)-(4,4)-(9,3)';
  var vec=cadena.split(/-/);
  for(var x=0;x<vec.length;x++)
  {
    document.write(vec[x]+'<br>');
  }
</script>

</body>
</html>
El vector vec almacena los valores: (2,3) (4,4) y (9,3), ya que la expresión regular indicada define el caracter - como patrón de corte.
El último método a analizar del objeto String es el match. Tiene como parámetro una expresión regular y retorna un vector con todos los string que cumplen el patrón especificado (siempre y cuando hallamos activado la bandera 'g' (global))
Problema
Se tiene un string donde tenemos almacenados los nombres de personas seguido de sus edades. Generar un vector solo con las edades de todas las personas.
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var cadena='juan 10 luis 43 pedro 21 carlos 33';
  var vec=cadena.match(/\d+/g);
  for(var x=0;x<vec.length;x++)
  {
    document.write(vec[x]+'<br>');
  }
</script>

</body>
</html>

Javascript Parte LXXXVI

El objeto RegExp se le puede aplicar tres modificadores que actúan sobre el patrón completo.
i: especifica que la búsqueda se realiza sin 
   diferenciar entre mayúsculas y minúsculas.
g: indica que los caracteres a buscar pueden 
   ser encontrados varias veces en la cadena 
   de caracteres.
m: el modificador m realiza coincidencia de
   patrones en el modo multilínea. En este modo, 
   si la cadena a buscar contiene varias líneas, 
   el metacaracter ^ y $ deben coincidir con el
   inicio y final de una línea, además de que 
   coincida con el comienzo y el final de una cadena.
Estos modificadores pueden aplicarse en forma independiente o conjunta a una expresión regular.
Veamos un ejemplo del modificador i sobre una expresión regular:
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var palabra='Administracion';
  var patron=/adm/i;
  if (patron.test(palabra))
    document.write('La palabra '+palabra+' contiene la cadena adm sin tener en cuenta mayúsculas/minúsculas');
</script>

</body>
</html>
Los modificadores se ubican luego de la segundo barra donde declaramos la expresión regular:
  var patron=/adm/i;
Ahora veamos un problema donde activamos dos modificadores:
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var oracion='El auto tiene el mejor motor que elegimos';
  var patron=/el/gi;
  var vec=patron.exec(oracion);
  document.write(vec.index);  //0
  document.write('<br>');
  vec=patron.exec(oracion);
  document.write(vec.index);  //14
  document.write('<br>');
  vec=patron.exec(oracion);
  document.write(vec.index);  //33
  document.write('<br>');
  vec=patron.exec(oracion);  
  document.write(vec);  //null
</script>

</body>
</html>
Hemos definido el patron los modificadores g (global) e i (insensitive):
  var patron=/el/gi;
Estamos buscando el patrón 'el' dentro de la oración 'El auto tiene el mejor motor que elegimos'. Como vemos la cadena 'el' se repite tres veces en la oración sin tener en cuenta mayúsculas y minúsculas:
El auto tiene el mejor motor que elegimos
Utilizamos el método exec ya que el método test solo tiene en cuenta la primera ocurrencia.
Para que tenga sentido la llamada de sucesivas veces del método exec debemos activar el modificador g (global)
Si no activamos el modificador global cada vez que llamamos al método exec la búsqueda comienza desde el principio.

Javascript Parte LXXXV

Hasta ahora siempre hemos empleado el método test que retorna un valor true si el string cumple el patrón propuesto o false en caso contrario.
La clase RegExp cuenta con otro método llamado exec que retorna un vector con más datos en caso que el string cumpla el patrón propuesto o null si no cumple el patrón.
Veamos con un problema que datos extras retorna este vector.



<html>
<head>
</head>
<body>

<script type="text/javascript">
  var oracion='saliendo de la casa que esta en la montaña';
  var patron1=/la/;
  var vector1=patron1.exec(oracion);
  document.write(vector1.index);
  document.write('<br>');
  document.write(vector1.input);
  document.write('<br>');
  var patron2=/lax/;
  var vector2=patron2.exec(oracion);
  document.write(vector2);
</script>

</body>
</html>
Hemos definido en la variable oracion el valor : 'saliendo de la casa que esta en la montaña'. El primer patrón de búsqueda es verificar si contiene los caracteres 'la' la oración:
  var oracion='saliendo de la casa que esta en la montaña';
  var patron1=/la/;
  var vector1=patron1.exec(oracion);
El método exec retornará un valor distinto a null (tener en cuenta que retorna un vector) ya que la palabra la está contenida en la oración.
El vector devuelto por el método exec tiene dos propiedades llamadas index e input:
La propiedad index almacena la posición dentro del string donde se cumple el patrón, en nuestro ejemplo retorna un 12 que es la posición donde se encuentra la l:
saliendo de la casa que esta en la montaña
La propiedad input almacena la cadena que se está analizando.
La segundo parte del problema define un segundo patrón:
  var patron2=/lax/;
  var vector2=patron2.exec(oracion);
Como no se encuentra en el string la cadena 'lax' el método exec retorna un valor null, es decir no retorna un vector.

Javascript Parte LXXXIV

Los paréntesis abiertos y cerrados pueden tener varios significados en las expresiones regulares. El fundamental es el agrupar una parte de la expresión regular y aplicarle operaciones como |, *, +, ? etc.
Problema
Elaborar una expresión regular que permita validar si se ingresó correctamente un valor entero o real y que eventualmente lleve el caracter + o - al principio. Validar que si se ingresa la coma debe ingresarse la parte real
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var nro=prompt('ingrese un número entero o real:','');
  var patron=/^[+-]?\d+(\,\d+)?$/;
  if (patron.test(nro))
    alert('Correcto');
  else
    alert('Incorrecto');
</script>

</body>
</html>
Podemos analizar esta expresión regular:
  var patron=/^[+-]?\d+(\,\d+)?$/;
Como la coma y la parte decimal son opcionales los encerramos entre paréntesis y lo finalizamos con el cuantificador '?'
Esto hace que cuando se ingresa la coma deba ingresarse uno o más dígitos obligatoriamente ya que los agrupamos mediante los paréntesis.
Problema
Ingresar una hora con el formato hh:mm:ss, plantear una expresión regular que valide si es correcta.
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var hora=prompt('ingrese una hora con el formato hh:mm:ss','');
  var patron=/^(0[1-9]|1\d|2[0-3]):([0-5]\d):([0-5]\d)$/;
  if (patron.test(hora))
    alert('Correcto');
  else
    alert('Incorrecto');
</script>

</body>
</html>
Analicemos un poco como hemos agrupado cada uno de las secciones de la hora:
  var patron=/^(0[1-9]|1\d|2[0-3]):([0-5]\d):([0-5]\d)$/;
La parte inicial puede tomar un valor entre 0 y 23:
(0[1-9]|1\d|2[0-3])
Utilizamos el metacarácter | para permitir ingresar el cero seguido de un valor entre 1 y 9, o el valor 1 y cualquier dígito, o el dos seguido del dígito 0,1,2, o 3.
Luego debe ingresarse el caracter dos puntos.
La parte de los segundos queda definida por el patrón ([0-5]\d) es decir un valor entre 0 y 5 seguido de cualquier dígito.
Finalmente los segundos es similar a los minutos: ([0-5]\d)

Javascript Parte LXXXIII

El metacarácter punto hace coincidir con cualquier carácter menos el salto de línea.
Por ejemplo si queremos validar el ingreso de cualquier cadena de 5 caracteres que comience con a y finalice con a podemos plantear la siguiente expresión regular:
  var patron=/^a...a$/;
Estamos indicando que la cadena debe comenzar con una letra a luego hemos dispuesto tres metacaracteres punto donde se verifican verdadero cualquiera sea el carácter del string y finalmente indicamos que finaliza con la letra a.
El programa para probarlo es:
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var cadena=prompt('Ingrese una cadena de cinco caracteres que comience y finalice con a','');
  var patron=/^a...a$/;
  if (patron.test(cadena))
    document.write('cadena correcta');
  else
    document.write('cadena incorrecta');
</script>

</body>
</html>
Podemos utilizar un cuantificador para no repetir tres veces el metacarácter punto:
  var patron=/^a.{3}a$/;
Es importante recordar que siempre que necesitemos utilizar alguno de los metacaracteres como dato a buscar es necesario escaparlo con la barra invertida, por ejemplo si queremos verificar si un número tiene el formato 999.99:
<html>
<head>
</head>
<body>

<script type="text/javascript">
  var nro=prompt('Ingrese un número con el formato 999.99','');
  var patron=/^\d{3}\.\d{2}$/;
  if (patron.test(nro))
    document.write('nro correcto');
  else
    document.write('nro incorrecto');
</script>

</body>
</html>
Es fundamental escapar el carácter punto \.
Esta expresión la podemos leer: debe comenzar con 3 dígitos obligatoriamente, luego un punto y finalmente dos dígitos.