viernes, 25 de noviembre de 2011

TDD… ejemplos básicos



Es una técnica de programación que emplea pruebas unitarias y refactorización de código.


El ciclo de vida de TDD comprende tres etapas

  1. Escribir una prueba que falle (rojo)
  2. Escribir una prueba que pase (verde)
  3. Refactorización, mejorar el código sin afectar su funcionamiento interno


Problema. Obtener el número de pulsaciones de una persona con 15 segundos de ejercitación.

Entrada: edad de la persona
Salida: número de pulsaciones
Restricciones o casos posibles:
  1. la edad no debe ser cero o un valor negativo,
  2. un valor flotante (ej. 4.5)
  3. o un carácter.

Fórmula. Num_pulsaciones = (220 – edad)/15

El primer paso es crear una prueba que falle, existen libros sobre TDD y Refactorización que indican la forma realizar los test, pero bien lo trataré de hacer según mi entendimiento.

PulsacionTest. Java

@Test
public void testObtenerPulsaciones() {
Pulsacion pulsacion = new Pulsacion();
System.out.println("obtenerPulsaciones: entero positivo");
int edad = 26;
double expResult = 12.0;
assertEquals(expResult, pulsacion.obtenerPulsaciones(edad), 0.0);
}

Obvio es que falle (no tengo que probar). Ahora el segundo paso es crear el código necesario (Pulsacion.java) para pasar la prueba.


//Código inicial
Pulsacion.java

public double obtenerPulsaciones(int edad){
return (220-edad)/15;
}

Listo pasa la primer prueba (valor entero no negativo). Ahora otro problema, es necesario realizar pruebas para los tres casos posibles (valor cero o negativo, valor flotante, o carácter).

Para ahorrar líneas de código defino “pulsación” como un atributo de la clase PulsacionTest.java para que sea accesible para los demás métodos.

He aquí el código aplicando TDD y un poco de Refactorización.

PulsacionTest. Java

Pulsacion pulsacion = new Pulsacion();

//para un valor entero no negativo
@Test
public void testObtenerPulsaciones() {
System.out.println("obtenerPulsaciones: entero positivo");
int edad = 26;
double expResult = 12.0;
assertEquals(expResult, pulsacion.obtenerPulsaciones(edad), 0.0);
}

// para un valor cero o negativo
@Test
public void testObtenerPulsacionesNegativoCero(){
System.out.println("obtenenerPulsaciones: cero o negativo");
int edad_neg=-26;
double expResult1=14.0;
assertEquals(expResult1, pulsacion.obtenerPulsaciones(edad_neg), 0.0);
}

// para un valor flotante
@Test
public void testObtenerPulsacionesReal(){
System.out.println("obtenenerPulsaciones: real");
double edad_real=26.0;
double expResult2=12.0;
assertEquals(expResult2, pulsacion.obtenerPulsaciones((int)edad_real), 0.0);
}

//para un carácter
@Test
public void testObtenerPulsacionesCaracter(){
System.out.println("obtenenerPulsaciones: caracter");
String edad_cad="26";
double expResult3=12.0;
assertEquals(expResult3, pulsacion.obtenerPulsaciones(Integer.parseInt(edad_cad)), 0.0);
}

En este método se podría agregar otro método que compruebe si la cadena introducida es numérica o no.


public bolean esNumerico (String cad){
try{
Integer.parseInt (cad);
return true;

}catch(NumberFormatException ex){
return false;}
}
//Código final
Pulsacion.java

public double obtenerPulsaciones(int edad){
if((int)edad<=0){
return (220-0)/15;
}
else if((int)edad>0){
return (220-(int)edad)/15;
}
else{
return (220-edad)/15;
}
}

jueves, 17 de noviembre de 2011

¿Cómo introducir/mostrar datos en Java, C, Groovy & Python?



En Java/Groovy

1. Uso de java.util.Scanner y java.io
package control.pruebas;


import java.util.Scanner;
import java.io.*;

public class Entrada {

/**
* @param args
*/
public static void main(String[] args) {
BufferedReader teclado= new BufferedReader(new InputStreamReader(System.in));
Scanner tecla=new Scanner(System.in);


System.out.println("Introduce entero");
int entero;
try {
entero = Integer.parseInt(teclado.readLine());
System.out.printf("%d",entero);
System.out.println(entero);
} catch (NumberFormatException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}



System.out.println("Introduce real");
double real;
try {
real = Double.parseDouble(teclado.readLine());
System.out.printf("%f",real);
System.out.println(real);
} catch (NumberFormatException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

System.out.println("Introduce cadena");
String cadena;
try {
cadena = teclado.readLine();
System.out.printf("%s",cadena);
System.out.println(cadena);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}


System.out.println("Introduce entero");
int enterox= tecla.nextInt();

System.out.printf("%d",enterox);
System.out.println(enterox);
System.out.println("Introduce real");
double realx= tecla.nextInt();

System.out.printf("%f",realx);
System.out.println(realx);
System.out.println("Introduce cadena");
String cadenax= tecla.nextLine();

System.out.printf("%s",cadenax);
System.out.println(cadenax);

}

}

En Python
entero=int(raw_input('Introduce entero:'))
print "entero: %d"%(entero)
print "entero: ",entero

real=float(raw_input('Introduce real:'))
print "real: %f"%(real)
print "real: ",real

cadena=raw_input('Introduce cadena')
print "cadena: ",cadena

En C
int entero;
float real;

printf("Introduce entero:"); scanf("%d",&entero);
printf("entero: %d",entero);
printf("Introduce real:"); scanf("%d",&real);
printf("entero: %f",real);

char cadena[23];
puts("Introduce cadena");
gets(cadena);

puts(cadena);


domingo, 13 de noviembre de 2011

¿Cómo aprender a programar?

  • Conoce gente que sepa de programación 
  • Busca información, libros, apuntes, blogs, sitios especializados,etc.
  • Comparte tus conocimientos (aunque sean limitados) siempre alguien te podrá corregir si te equivocas
  • Siempre es bueno tener tu propio "banco de datos" donde almacenar tus apuntes, tus notas, etc.
  • Leer ayuda de mucho, también la lógica y las matemáticas
  • Haz notas de lo que no entiendas, busca términos que te confundan o que no entiendas claramente
  • Preguntar es bueno, aunque no siempre esperes que te contesten y resuelvan tus problemas
  • El Face y Twitter no son solo para ver mensajes sin contenido ni sustancia, también pueden servir para encontrar buena información y conocer gente
  • Si encuentras código no es bueno tan solo copiar y pegar,estudialo no todo el código que encuentras en  internet sirve
  • Y lo más importante, el programador se hace con la práctica no en las aulas

Un ejemplo muy sencillo. Sumar todos los valores de un arreglo
//defino un arreglo con números
int[] numeros={3,4,5,6,7,8,9};
int i,suma=0;
//bucle que va sumando todos los valores del arreglo
for(i=0;i<numeros.length;i++){
   suma+=numeros[i];
}
//muestro la suma de todos los valores
System.out.println("Suma de todos los números: "+suma);

    sábado, 12 de noviembre de 2011

    ¿XML ... eso con que se come?



    Java aparte de bonito, barato y eficiente es un lenguaje que permite procesar (con ayuda de algunas herramientas como:JAXB , XStream, JDom, etc.) los archivos *.xml.
    Los documentos xml tienen varios usos:
    1. Intercambo de informaciön entre aplicaciones (Web o de escritorio)
    2. XML como "base de datos"
    3. Ahorro de recursos de los servidores
    4. etc.
    Permite crear etiquetas personalizadas, sin embargo, es necesario que los documentos xml tengan un significado coherente.
    Un ejemplo. Crear un xml que represente la entidad Programador y sus atributos (nombre,edad y email).
    Programador.xml
    <?xml version=”1.0” encoding=”UTF-8” standalone=”yes”?>
    <Programador>
    <Nombre>Daniel</Nombre>
    <Edad>27</Edad>
    <Email>daniel.2132@servidor.com</Email>
    </Programador>
    Es un archivo xml básico, pero todavia falta como darle validez.

    ¿Cómo se le da validez a los archivos xml?

    Existen dos alternativas (validadores) los Documentos de definición de tipos:
    1. DTD : ,describe la estructura del documento con un lenguaje que no tiene nada que ver con XML.
    2. XML Schemas: describe la estructura con XML
    Ejemplo con DTD.
    programador.dtd
    <!ELEMENT Programador(Nombre,Edad,Email)>
    <!ELEMENT Nombre (#PCDATA)>
    <!ELEMENT Edad (#PCDATA)>
    <!ELEMENT Email (#PCDATA)>
    Existen reglas para definir los elementos ( ), la cardinalidad (?,*,+), pero solo quiero poner una muestra sencilla.
    Asi quedaria el documento xml con dtd.
    <?xml version=”1.0” encoding=”UTF-8” standalone=”yes”?>
    <!DOCTYPE Programador SYSTEM "programador.dtd">
    <Programador>
    <Nombre>Daniel</Nombre>
    <Edad>27</Edad>
    <Email>daniel.2132@servidor.com</Email>
    </Programador>

    Tratamiento de documentos XML

    Básicamente existen dos alternativas (se denominan parsers o analizadores):
    1. SAX:Simple API for XML, consiste en procesar los documentos xml según se va leyendo las etiquetas y se van localzando los elementos
    2. DOM:Document Object Model, consiste en leer el documento xml y obtener y modelo basado en objetos que se mantienene en memoria para poder realizar el tratamiento conforme sea necesario sobe la información leida. Es más lento que SAX.


    He aqui un ejemplo de un programa que lee XML escrito por @javadabadoo


    Espero continuar con el tema y hacer un ejemplo con JAXB. En otro post trataréde hablar más de esa tecnología con ayuda de NetBeans y/o Eclipse.





    JAXB
    Crear una clase java básica.
    Datos1.java
    /*
     * To change this template, choose Tools | Templates
     * and open the template in the editor.
     */
    package modelo.pruebas;
    //import javax.xml.bind.annotation.XmlAttribute;
    import javax.xml.bind.annotation.XmlElement;
    import javax.xml.bind.annotation.XmlRootElement;
    /**
     *
     * @author yo
     */
    @XmlRootElement
    public class Datos1 {
       
        private double valor,tasa;
        private int periodo;     /**
         * @return the valor
         */
        public double getValor() {
            return valor;
        }
        /**
         * @param valor the valor to set
         */
        @XmlElement
        public void setValor(double valor) {
            this.valor = valor;
        }
        /**
         * @return the tasa
         */
        public double getTasa() {
            return tasa;
        }
        /**
         * @param tasa the tasa to set
         */
        @XmlElement
        public void setTasa(double tasa) {
            this.tasa = tasa;
        }
        /**
         * @return the periodo
         */
        public int getPeriodo() {
            return periodo;
        }
        /**
         * @param periodo the periodo to set
         */
        @XmlElement
        public void setPeriodo(int periodo) {
            this.periodo = periodo;
        }
       
       
    }
    Convertir esa clase a xml.
    ControlDatos1Xml.java
    /*
     * To change this template, choose Tools | Templates
     * and open the template in the editor.
     */
    package control.pruebas; import java.io.File;
    import javax.xml.bind.JAXBContext;
    import javax.xml.bind.JAXBException;
    import javax.xml.bind.Marshaller;
    import modelo.pruebas.Datos1;
    import javax.swing.JOptionPane;
    /**
     *
     * @author yo
     */
    public class ControlDatos1Xml {
       
        Datos1 datos= new Datos1();
        public static void main(String[] args){
             ControlDatos1Xml control= new ControlDatos1Xml();
           
             double v=0,t=0; int p;
             v=Double.parseDouble(JOptionPane.showInputDialog("valor"));
             t=Double.parseDouble(JOptionPane.showInputDialog("tasa"));
             p=Integer.parseInt(JOptionPane.showInputDialog("periodo"));
           
           
             control.datos.setPeriodo(p);
             control.datos.setTasa(t);
             control.datos.setValor(v);
           
             //------------------------------------
             try {

                    File file = new File("/home/user/PruebasClases/dat1.xml");
                    JAXBContext jaxbContext = JAXBContext.newInstance(Datos1.class);
                    Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

                   
                    jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

                    jaxbMarshaller.marshal(control.datos, file);
                    jaxbMarshaller.marshal(control.datos, System.out);

                  } catch (JAXBException e) {
                    e.printStackTrace();
                  }
           
             JOptionPane.showMessageDialog(null, "XML generado");
           
            //-----------------------------------
       
           
           
         
        }
       
       
       
    }
    Leer el XML como si fuera objeto.
    ControlDatos1Class.java
    /*
     * To change this template, choose Tools | Templates
     * and open the template in the editor.
     */
    package control.pruebas;
    import java.io.File;
    import javax.xml.bind.JAXBContext;
    import javax.xml.bind.JAXBException;
    import javax.xml.bind.Unmarshaller;
    import modelo.pruebas.Datos1; /**
     *
     * @author yo
     */
    public class ControlDatos1Class {
        public static void main(String[] args){
           
            try {

                    File file = new File("/home/user/PruebasClases/dat1.xml");
                    JAXBContext jaxbContext = JAXBContext.newInstance(Datos1.class);

                    Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller();
                    Datos1 datos = (Datos1) jaxbUnmarshaller.unmarshal(file);
                    System.out.println(datos);
                   
                   
                    //---------------------------------------
                    if(datos.getValor()<=0){
                      System.out.println("El valor no debe ser menor o igual a cero");
                    }
                   
                    //------------------------------------
                   
                    else{
                    System.out.println("valor -->"+datos.getValor());
                   
                   
                    System.out.println("tasa -->"+datos.getTasa());
                    System.out.println("periodo -->"+datos.getPeriodo());
                    }

              } catch (JAXBException e) {
                    e.printStackTrace();
              }
           
            //-----------------------------------
           
       
        }
       
    }



    JAXB

    Es una tecnología que permite obtener una representación XML de un objeto Java y viceversa.

    Algunas librerías necesarias para trabajar con JAXB:

    javax.xml.bind.JAXBContext
    javax.xml.bind.JAXBException
    javax.xml.bind.Mashaller
    javax.xml.bind.Unmashaller
    javax.xml.bind.annotation.XmlAttribute
    javax.xml.bind.annotation.XmlRootElement
    javax.xml.bind.annotation.XmlElement
    javax.xml.bind.annotation.XmlType
    javax.xml.bind.annotation.XmlAccessType
    javax.xml.bind.annotation.XmlAccessorType
    javax.xml.bind.annotation.XmlEnumValue
    javax.xml.bind.annotation.XmlList
    javax.xml.bind.annotation.XmlElementWrapper


    XML … XML Schemas

    XML Schemas (*.xsd): Se tratan de documentos XML que permiten definir la estructura y validez de documentos XML.

    Parecidos a los archivos *.dtd, pero permiten definir una serie de tipos como string, int, integer, etc. Y poseen un mecanismo para crear nuevos tipos, simples y complejos que representan los tipos de documentos XML.


    Tanto Eclipse como NetBeans poseen herramientas para crear estos archivos XML.


    Estructura básica de un XML Schema

    <?xml version=”1.0” encoding=”UTF-8”?>
    <xs:schema xmlns:xs=”http://www.w3.org./2009/XMLSchema”>
    <!-- aquí pones los elementos y/o descripción del Schema -->
    </xs:schema>



    Definición de tipos simples

    1.     Mediante enumeración
    2.     Mediante derivación
    3.     Mediante unión

    <xs:simpleType name=”Algo”>
    </xs:simpleType>



    Definición de tipos complejos

    <xs:complexType name=’Algo’>
    <xs:sequence>
    </xs:sequence>
    </xs:complexType>

    Ejemplo.  Crear una estructura “persona” con 4 atributos: nombre, edad, peso, y talla.

    persona.xsd

    <?xml version=”1.0” encoding=”UTF-8” standalone=”yes”?>
    <xs:schema xmlns:xs=”http://www.w3.org/2009/XMLSchema ”>
    <xs:element name=”Persona” type=”tPersona”/>
    <xs:complextType name=”tPersona”>
    <xs:sequence>
    <xs:element name=”nombre” type=”xs:string”/>
    <xs:element name=”edad” type=”xs:integer”/>
    <xs:element name=”peso” type=”xs:double”/>
    <xs:element name=”talla” type=”xs:double”/>
    </xs:sequence>
    </xs:complexType>
    </xs:schema>


    persona.xml

    <Persona xmlns:xs=”http://www.w3.org/2009/XMLSchema -instance” xsi:noNamespaceSchemaLocation=”persona.xsd”>
    <nombre>Horacio</nombre>
    <edad>20</edad>
    <peso>56</peso>
    <talla>1.65</talla>
    </Persona>

    Ejemplo con NetBeans

    1. Bajar  e instalar el pugin  XMLTools

    2. Crear el XML Schema

    3.  Crear el JAXB Binding (elegir en verbose)

    4. Crear el archivo xml

    5. Crear la clase java que lea el xml


    Ejemplo con Eclipse

    1.     Crear el XML Schema
    2.     Generar:  XML File  y/o JAXB Class
    3.     Crear la clase java que lea el xml