domingo, 12 de mayo de 2013

Sound in Java

Ver en Español



For placing sound to our java desktop Application, we have at our disposal multiple alternatives, however I have done a simple class for that endeavor, which has served me for many projects.

The Class is the next:


public class soundthread extends Thread{
        
           
            private Clip sound;
            private boolean continuex;
            
        

         public soundthread(String filepath){
           
            continuex=true;
            try {
                sound = AudioSystem.getClip();
                sound.open(AudioSystem.getAudioInputStream(new File(filepath)));
                
            } catch (UnsupportedAudioFileException ex) {
              
                Logger.getLogger(soundthread.class.getName()).log(Level.SEVERE, null, ex);
            } catch (IOException ex) {
               
                Logger.getLogger(soundthread.class.getName()).log(Level.SEVERE, null, ex);
            } catch (LineUnavailableException ex) {
               
                Logger.getLogger(soundthread.class.getName()).log(Level.SEVERE, null, ex);
            }
         }
         
        @Override
         public void run(){
              
            sound.start();
              
            
            // Wait while it is playing
            do{
                try {
                Thread.sleep(500);
                } catch (InterruptedException ex) {
                Logger.getLogger(soundthread.class.getName()).log(Level.SEVERE, null, ex);
                }
                
            }while (continuex && sound.isActive());
            
            if(sound.isActive()){
                sound.stop();
            }
            
            // Close the clip.

            sound.close();
          
         
         }
        
        public void stop(){
            continuex=false;
        }
         
     }  



For use this class, we should only do something similar or equal to the following:



//...


// Example of path in Windows: C:\path\audio_file.wav
// Example of path in Linux: /home/user/audio_file.wav

soundthread obj = new soundthread("/home/user/audio_file.wav");
obj.start();

//...




Sonido en Java

See in English



Para colocar sonido a nuestra aplicación de escritorio en java, tenemos a nuestra disposición varias alternativas, sin embargo yo he hecho una clase sencilla para esa labor, la cual me ha servido para muchos proyectos.

Dicha clase es la siguiente:


  public class hiloSonido extends Thread{
        
           
            private Clip sonido;
            private boolean seguir;
            
        

         public hiloSonido(String rutaArchivo){
           
            seguir=true;
            try {
                sonido = AudioSystem.getClip();
                sonido.open(AudioSystem.getAudioInputStream(new File(rutaArchivo)));
                
            } catch (UnsupportedAudioFileException ex) {
              
                Logger.getLogger(ReproduceSonido.class.getName()).log(Level.SEVERE, null, ex);
            } catch (IOException ex) {
               
                Logger.getLogger(ReproduceSonido.class.getName()).log(Level.SEVERE, null, ex);
            } catch (LineUnavailableException ex) {
               
                Logger.getLogger(ReproduceSonido.class.getName()).log(Level.SEVERE, null, ex);
            }
         }
         
        @Override
         public void run(){
              
            sonido.start();
              
            // Espera mientras se esté reproduciendo.
            do{
                try {
                Thread.sleep(500);
                } catch (InterruptedException ex) {
                Logger.getLogger(ReproduceSonido.class.getName()).log(Level.SEVERE, null, ex);
                }
                
            }while (seguir && sonido.isActive());
            
            if(sonido.isActive()){
                sonido.stop();
            }
            
            // Se cierra el clip.
            sonido.close();
          
         
         }
        
        public void parar(){
            seguir=false;
        }
         
     }  



Para utilizar esta clase solo debemos hacer algo similar o igual a lo siguiente:


//...


// Ejemplo de ruta en Windows: C:\ruta\archivo_de_audio.wav
// Ejemplo de ruta en Linux: /home/usuario/archivo_de_audio.wav

hiloSonido obj = new hiloSonido("/home/usuario/archivo_de_audio.wav");
obj.start();

//...




domingo, 29 de enero de 2012

Clase Scanner en Java

La clase Scanner nos facilita mucho la tarea de obtención de datos desde diferentes fuentes.


Obtener datos desde el teclado

Una de las utilidades de la clase Scanner es la obtención de datos tecleados finalizando con un "enter".

Ejemplo:

import java.util.Scanner; // se importa la clase Scanner



 class Ejemplo {

 
 public static void main(String arg[]){
  
  String variableString;
  
  Scanner entrada=new Scanner(System.in);// se declara e inicializa una instancia  de la clase Scanner.
  
  System.out.print("Ingrese un texto: ");
  
  variableString = entrada.next();
  
  System.out.println("Texto ingresado: "+variableString);
  
 }
 
}


En el código anterior se importa la clase Scanner, posteriormente se declara una instancia de la misma llamada "entrada" y se inicializa pasando como parámetro un objeto InputStream ,
el cual es devuelto por la clase System con ayuda de su método in. Luego vemos como se imprime el aviso correspondiente del tipo de datos que se va a pedir y se guarda todo lo que se digíte en la variable "variableString", hasta que se teclee un enter.
Y todo se logra por medio del objeto Scanner llamado "entrada" y su método next().

De igual manera un objeto Scanner ofrece otros métodos que permiten recibir diferentes tipos:

import java.util.Scanner;



 class Ejemplo {

 
 public static void main(String arg[]){
  
  int variableEntero;
  float variableFloat;
  
  
  Scanner entrada=new Scanner(System.in);
  
  System.out.print("Ingrese un entero: ");
  
  variableEntero = entrada.nextInt();
  
  System.out.println("Entero ingresado: "+variableEntero);
  
  System.out.print("Ingrese un float: ");
  
  variableFloat = entrada.nextFloat();
  
  System.out.println("float ingresado: "+variableFloat);
  
 }
 
}





Obtener datos desde un archivo



La forma de obtener el contenido de un archivo con la clase Scanner es sin duda muy limpia, lo que cambia son los parámetros que se pasan al constructor, esta vez sera un objeto File:

    Scanner entrada=new Scanner(System.in);
    long numero=-1;
  
    try
    {
         entrada = new Scanner(new File("ruta_a_miArchivo/miArchivo"));

         while (entrada.hasNextLong()) 
         {
            numero = entrada.nextLong();
         }
        
        entrada.close();        
    
    } 
    catch (FileNotFoundException e)
    {
        e.printStackTrace();
    }
       
   System.out.println(" numero: "+numero);




En el código anterior se accede al contenido del archivo "miArchivo", el cual es un número desconocido.
El bloque try captura la excepción resultante en caso de que no exista dicho archivo,
y el while se encarga de asignar el contenido del archivo a la variable "numero", pero solo si ese archivo no esta vacío, es decir evita una excepción del tipo "NoSuchElementException".
Cabe aclarar que si el archivo tiene extensión como por ejemplo .txt , esta también se debe especificar en la ruta.



Ahora, en caso de nuestro archivo de ejemplo "miArchivo" tuviese varias lineas, como por ejemplo:

nombre: stivenson
lugar: Cúcuta Colombia
telefono: 256254

entonces, se podría utilizar los métodos nextLine y hasNextine, de la siguiente manera:

   
           
       Scanner entrada=new Scanner(System.in);
       String[] datos=new String[3];
       int indiceVector=0;
    
 try 
 {
    entrada = new Scanner(new File( "ruta_a_miArchivo/miArchivo"));

    while (entrada.hasNextLine()) 
    {
       datos[indiceVector++] = entrada.nextLine();
    }

    entrada.close();

 } 
 catch (FileNotFoundException e)
        {
    e.printStackTrace();
        }
       
       
 for(int i = 0 ; i < 3 ; i++ )
 {
    System.out.println(datos[i]);  
 }
       
       


En el código anterior se guarda cada una de las lineas del archivo en las posiciones del vector "datos" que posteriormente se imprimen en un for.


Y aquí no acaba todo, en el siguiente ejemplo vamos a obtener datos pero especificando un delimitador, supongamos que ahora nuestro archivo "miArchivo" tiene este contenido:

cantidad: 26565 cantidad: 3262625 cantidad: 2x10^4 cantidad: 121400001


y supongamos que solo queremos traer los números, para ello vamos a fijar delimitadores, algo que es posible y sin duda muy útil:

   Scanner entrada=new Scanner(System.in);
   String[] datos=new String[4];
   int indiceVector=0;
    
  try 
  {
   entrada =
   new Scanner(new File("ruta_a_miArchivo/miArchivo"))
   .useDelimiter("\\s*cantidad:\\s*");

 while (entrada.hasNext()) 
 {
     datos[indiceVector++] = entrada.next();
 }

   entrada.close();
  }
  catch (FileNotFoundException e)
  {
   e.printStackTrace();
  }
       
    
  for(int i = 0 ; i < 4 ; i++ )
  {
   System.out.println(datos[i]);  
  }
       
       


En el código anterior se obtienen los cuatro números del archivo y cada uno de estos se almacena en una posición del vector. Al delimitador (método useDelimiter) se le da como parámetro: "\\s*cantidad:\\s*".
Este parametro indica muchas cosas, primero están los signos "\\s*" los cuales indican que el delimitador puede empezar con cero o muchos espacios después se indica que va a estar seguido de la palabra "cantidad:" que a su vez va a estar seguida por cero o muchos espacios.

El carácter "\s" indica que se imprima un espacio, pero nosotros no queremos imprimir solo indicar, por eso le antecedemos el otro contraes-las: "\\s"; El signo "*" es el que indica que pueden haber cero o muchos de esos espacios, si tuviéramos que tratar con archivos de varias lineas, entonces sin duda tendríamos que trabajar con: "\n" en nuestros delimitadores.
En fin, el código anterior después de su ejecución imprime lo siguiente:

26565
3262625
2x10^4
12140001


Como se puede observar el delimitador nos ayuda a omitir ciertas partes del contenido que tengan un patrón en especifico, es decir fijar un separador.


Pero que sucede si no queremos guardar estos cuatro números en un vector String, y queremos hacerlo en variables con su respectivo tipo?, entonces trabajaríamos de la siguiente manera:

        

        Scanner entrada=new Scanner(System.in);
        int cantidad1=0;
        int cantidad2=0;
        String cantidad3="";//para guardar el número en notación científica.
        String cantidad4="";//el ultimo debe ser siempre String
   
    
 try 
 {

          entrada =
          new Scanner(new File( "ruta_a_miArchivo/miArchivo"))
          .useDelimiter("\\s*cantidad:\\s*");
   
   cantidad1=entrada.nextInt();
   
   cantidad2=entrada.nextInt();
   
   cantidad3=entrada.next(); //se espera número en notación científica.
   
   cantidad4=entrada.next();// el ultimo dato del archivo siempre se debe obtener como String
   
   entrada.close();

 } 
 catch (FileNotFoundException e)
 {
    e.printStackTrace();
 }
       
        System.out.println(cantidad1);  
        System.out.println(cantidad2);
        System.out.println(cantidad3);
        System.out.println(cantidad4);


En el código anterior cada vez que se ejecuta el método nextInt o next se avanza en la obtención de los datos y lógicamente estas instrucciones deben estar en el mismo orden a como llegan los datos que provienen del archivo;
Para el caso del ejemplo se obtienen primero dos números(con nextInt) pues son los primeros en llegar desde el archivo, luego un String (con next) y finalmente se espera otro número el cual se obtiene como un String por ser lo ultimo que se encuentra en el archivo y así evitar una excepción "InputMismatchException",
para remediar este ultimo detalle faltaría hacer una conversión después de la obtención u procurar que este ultimo dato siempre sea string.


Obtener datos desde una variable String

Igualmente a un objeto Scanner se le podría pasar una variable String en vez de un archivo:

                  

    Scanner entrada=new Scanner(System.in);

    float cantidad1=0.0f;

    int cantidad2=0;

    String cantidad3="";

    String cantidad4="";
    
    String texto="cantidad: 45,14 cantidad: 3262625 cantidad: 2x10^4 cantidad: cuatro ";//float con coma


       
    entrada = new Scanner(texto).useDelimiter("\\s*cantidad:\\s*");
   
   
   
    cantidad1=entrada.nextFloat();
   
    cantidad2=entrada.nextInt();
   
    cantidad3=entrada.next(); 
   
    cantidad4=entrada.next();
   
    entrada.close();
    
       
    System.out.println(cantidad1);  
    System.out.println(cantidad2);
    System.out.println(cantidad3);
    System.out.println(cantidad4);




Especificar una base para un número dado

Otra utilidad interenzante tiene que ver con los números, algunos métodos aceptan un parámetro numérico correspondiente a la base con la que se quiere manejar un número y de cierta manera no estar siempre con los decimales, ejemplo:

  Scanner entrada=new Scanner(System.in);    
  
  
System.out.print("Ingrese un octal: ");
 
//se espera un número en base 8 es decir caracteres del 0 al 7  
int numero = entrada.nextInt(8); 

System.out.println("el número octal ingresado, en decimal es: "+numero);
   
 

  
System.out.print("Ingrese un hexadecimal: ");
      
//se espera un número en base 16 es decir caracteres del 0 al 9 y/o A a la F
numero = entrada.nextInt(16); 

System.out.println("el número hexadecimal ingresado, en decimal es: "+numero);
   
 


  
System.out.print("Ingrese un binario: ");
       
//se espera un número en base 2 es decir solo caracteres 0 y 1
numero = entrada.nextInt(2); 

System.out.println("el número binario ingresado, en decimal es: "+numero);




En código anterior se aprecia la forma como se especifica la base, y la inmediata conversión que se hace automáticamente a decimal después de recibir el número por teclado.
si se ingresan caracteres que no corresponden a los caracteres de la base que se espera, entonces se producirá una excepción "InputMismatchException".

mas información: API Scanner Java 7

jueves, 29 de diciembre de 2011

Etiquetas en Java

Para los que conocen algo de lenguaje ensamblador, les será familiar encontrar etiquetas en el lenguaje Java, esta forma de controlar la ejecución puede resultar útil en la optimización de ciertos códigos, y su utilización solo se justifica cuando se realiza justo antes de un ciclo que a su vez tenga otro ciclo y/o un switch anidado.

Las etiquetas en java trabajan con las palabras reservadas "break" y "continue".
Teniendo en cuenta que se deben utilizar en ciclos(for, while o do-while ), un ejemplo de su utilización podría ser el improvisado siguiente código en donde se quiere imprimir por separado cada palabra de una frase dada:


String  frase = "Empezando con Java desde cero";
int i=-1; 
 
otraPalabra: //etiqueta


  // aqui no es conveniente colocar código.


while(true)
{
  
  i++; 
  
  System.out.print("\n palabra: ");
  
     for( int a = i ; a < frase.length() ; a++ )
     {
      
        char caracter=frase.charAt(a); 
   
        if(caracter == ' ')
        {
            i = a; 
      
            continue otraPalabra;  
        }
     
        System.out.print(caracter);
   
     }
     
  break otraPalabra;
     
} 




En el código anterior:

La instrucción "otraPalabra:" que se encuentra antes del while, es la etiqueta e indica un punto especifico en la ejecución.

La instrucción "continue otraPalabra;" que se encuentra dentro del if, provocará que la ejecución continué en la etiqueta "otraPalabra", es decir por fuera del ciclo while y ademas que vuelva a entrar a dicho while.

La instrucción "break otraPalabra;" provocará que la ejecución continué en la etiqueta "otraPalabra", pero sin entrar de nuevo al while, lo que finalmente evita un ciclo infinito así como lo huviera evitado un simple "break;", el cual no se utilizo para poder ver el comportamiento de las etiquetas con el break.

El código anterior imprimirá:

palabra: Empezando
palabra: con
palabra: Java
palabra: desde
palabra: cero


viernes, 23 de diciembre de 2011

break y continue en java



Estas dos palabras reservadas se utilizan para salir de ciclos o de una parte de el. También para saltarse alguna porción de código.

break



Se puede ver la utilización del "break" en la estructura switch en donde se utiliza para saltar los casos que se encuentran debajo del que se ejecuta:



La utilización del "break" también es muy común en los ciclos (For y while), este se utiliza cuando queremos salir por algún motivo del ciclo sin importar el resto de vueltas que falten.

En el siguiente ejemplo se quiere imprimir por pantalla si la frase dada tiene alguna letra "t":


String  frase = "Anita lava la tina";
		
for( int i = 0 ; i < frase.length() ; i++ )
   {
			
	char caracter=frase.charAt(i);
			
	if( caracter == 't' )
	{
				
		System.out.println("Si hay t");
		break;
	}
			
   }



Como se puede observar, cuando el código anterior encuentra la primera "t" e imprime el respectivo aviso, no tiene sentido seguir buscando en el resto de frase pues ya se ha logrado el objetivo.
Por ello se coloca el "break", este termina el ciclo "for" y sin duda ahorra tiempo en la ejecución del programa al impedir vueltas innecesarias en este.

De igual manera funciona con el while:


	String  frase = "Anita lava la tina";
		
	int i=0;
	 
	while( i < frase.length() )
	{
			
		char caracter=frase.charAt(i);
			
			if( caracter == 't' )
			{
				
			System.out.println("Si hay t");
			break;
			}
		
		i++;
			
	}





Continue



Un ejemplo de la utilización del "continue", podría ser el siguiente, en donde se quiere imprimir los caracteres de una frase dada, pero omitiendo las letras "a" y "A":

String  frase = "Anita lava la tina";
		
for( int i = 0 ; i < frase.length() ; i++ )
   {
			
	char caracter=frase.charAt(i);
			
			
	if( (caracter == 'a') ||  (caracter == 'A') )
        {
	      continue;	
        }
				
	System.out.print(caracter);
   }



Después de su ejecutar el código anterior se habrá imprimido: nit lv l tin
y esto ocurre por que cuando se ejecuta el "continue" las instrucciones que se encuentran debajo de este, no se ejecutan, y la ejecución continua pero con la siguiente vuelta del ciclo.
En el caso del ejemplo anterior la instrucción que no se ejecuta debido a la ejecución del "continue" es: System.out.print(caracter), y a su vez el "continue" solo se ejecuta cuando la letra actual es "a" o "A".;