Sin lugar a dudas, la mensajería instantánea a través de mensajes cortos (SMS) es una de las formas de comunicación más extendida y aceptadas por la sociedad.
En este tutorial voy a intentar hacer una introducción de las características más importantes que nos proporciona Java para el envío y recepción de SMS desde aplicaciones para móviles (MIDLets). Se presupone que el lector ya posee conocimientos básicos de programación (J2ME, MIDP, CLDC), compilación e instalación de MIDLets.
Introducción a WMA
WMA, son las siglas de Wireless Message API (curiosamente también lo son de uno de los formatos de audio de Windows, Windows Media Audio), una extensión de la especificaciones del CLDC y MIDP para el envio, la recepción y la gestión de SMS desde MIDLets.
A pesar de ser una extensión opcional, la gran mayoría de los terminales la llevan instalada y lista para ser usuada desde nuestras aplicaciones J2ME.
WMA es una especificación no una implementación. Su implementación dependerá del terminal y del protocolo de comunicación que use (GSM, CDMA, etc). Por supuesto, nosotros como desarrolladores podemos abstraernos de esto último.
Versiones de la especificación WMA:
Versión 1.0: Especificación inicial. Describe las funcionales básicas para el envío y la recepción de SMS. (JSR 120)
Versión 1.1: Una ampliación de la especificación 1.0, para soportar el nuevo modelo de seguridad del MIDP 2.0.
Versión 2.0: Una ampliación a las anteriores para la gestión de mensajes multimedia (MMS). (JSR 205)
Introducción al API WMA
El API está compuesto exclusivamente de interfaces ubicadas bajo el páquetejavax.wireless.messaging
. Estas interfaces son: javax.wireless.messaging.Message: Define la funcionalidad genérica de todos los tipos de mensajes. Permite:
- Especificar el destinatario del mensaje.
public void setAddress(String addr)
- Obtener el emisor del mensaje.
public String getAddress()
- Obtener la fecha de envio del mensaje.
java.util.Date getTimestamp()
Hereda la funcionalidad de
javax.wireless.messaging.Message
añadiendo los métodos public void setPayloadText(String data)
y public String getPayloadText()
para especificar u obtener los datos del mensaje. javax.wireless.messaging.BinaryMessage: Representa a un mensaje binario.
Hereda la funcionalidad de
javax.wireless.messaging.Message
añadiendo los métodos public void setPayloadData(byte[] data)
y public byte[] getPayloadData()
para especificar u obtener los datos del mensaje. javax.wireless.messaging.MessageListener: Oyente de mensajes entrantes.
Esta interfaz es útil en
javax.wireless.messaging.MessageConnection
que funcionan en modo servidor. (Será explicada más adelante.) javax.wireless.messaging.MessageConnection: Interfaz a través de la cual se realiza el envío y la recepción de mensajes. (Será explicada más adelante.)
¿Qué pasos tengo que realizar para enviar un mensaje (SMS)?
- Obtener un MessageConnection en modo cliente (Se verá más adelante).
- Crear el mensaje a través de la interfaz MessageConnection.
- Especificar el contenido y el destinatario del mensaje.
- Usar el método send de la interfaz MessageConnection para enviar el mensaje.
¿Qué pasos tengo que realizar para recibir un mensaje (SMS) de forma asíncrona?
- Obtener un MessageConnection en modo servidor (Se verá más adelante).
- Implementar la interfaz MessageListener en una de nuestras clases.
- Asociar la clase anterior al MessageConnection.
- Cuando el método notifyIncomingMessage() de la interfaz MessageListener sea invocado, significa que hemos recibido un mensaje.
- Deberemos invocar el método receive() de la interfaz MessageConnection en un hilo independiente.
- Realizar el tratamiento del mensaje.
¿Qué pasos tengo que realizar para recibir un mensaje (SMS) de forma síncrona?
- Obtener un MessageConnection en modo servidor (Se verá más adelante).
- Invocar el método receive() de la interfaz MessageConnection en un hilo independiente (Es un método bloqueante).
- Realizar el tratamiento del mensaje.
En J2ME, todas las comunicaciones que requieren los MIDLets con el exterior (Bluetooth, Socket, Http, etc.) se obtienen a través la clase
Básicamente a través de está inteface podemos crear, enviar y recibir SMS tanto de forma síncrona como asíncrona.javax.microedition.io.Connector
que forma parte del CLDC. Esta clase devuelve una instancia de una clase que implementa la interfaz javax.microedition.io.Connection
para el modo de comunicación deseada. Pues bien, la interface javax.wireless.messaging.MessageConnection
no es más que un javax.microedition.io.Connection
para comunicación via SMS.¿Cómo obtengo javax.wireless.messaging.MessageConnection?
Losjavax.wireless.messaging.MessageConection
pueden funcionar de dos modos:- En modo cliente: Sólo sirve para enviar SMS a un destinatario.
- En modo servidor: Sirve para recibir y tratar los SMS que son dirigidos hacia el. En esté modo también se pueden enviar SMS.
javax.microedition.io.Connector
.Ejemplos de como especificar el modo cliente:
MessageConnection conn = (MessageConnection) javax.microedition.io.Connector.open("sms://+34666666666");
MessageConnection conn = (MessageConnection) javax.microedition.io.Connector.open("sms://+34666666666:5555");
En ambos casos el SMS sería enviado al teléfono 666666666 de España (por el prefijo +34), pero con una importante diferencia. En el primer caso, el SMS sería tratado por la aplicación que por defecto tiene instalada el teléfono, mientras que, en el segundo caso, el SMS sería tratado por la aplicación que esté escuchando en ese puerto.Ejemplo de como especificar el modo servidor:
Los SMS que llegen al puerto 5555 serán atendidos por el MIDLet. En realidad, "5555" NO es un puerto sino un IDENTIFICADOR que le indica a la plataforma java que desea tratar los SMS que llegen al terminal con ese identificador.
MessageConnection conn = (MessageConnection) javax.microedition.io.Connector.open("sms://:5555");
Los SMS que llegen al puerto 5555 serán atendidos por el MIDLet. En realidad, "5555" NO es un puerto sino un IDENTIFICADOR que le indica a la plataforma java que desea tratar los SMS que llegen al terminal con ese identificador.
Métodos de la clase:
public javax.wireless.messaging.Message newMessage(java.lang.String type, java.lang.String address)
Crea un nuevo mensaje para ser enviado.
En el argumento type especificamos el tipo de mensaje que deseamos enviar:
MessageConnection.TEXT_MESSAGE para mensajes de texto
MessageConnection.BINARY_MESSAGE para mensajes binarios.
En el argumento address debemos especificar la dirección del destinatario del mensaje. Normalmente, su número de teléfono.
En el argumento type especificamos el tipo de mensaje que deseamos enviar:
MessageConnection.TEXT_MESSAGE para mensajes de texto
MessageConnection.BINARY_MESSAGE para mensajes binarios.
En el argumento address debemos especificar la dirección del destinatario del mensaje. Normalmente, su número de teléfono.
public int numberOfSegments(javax.wireless.messaging.Message msg)
Devuelve el número de segmentos que son necesarios para enviar la información a través de la red.
Por ejemplo, si queremos enviar el Quijote via SMS pues seguro que son necesarios más de un segmento (un segmento es igual a un SMS como lo conocen los usuarios) y este método nos devolvería o bien un número grande o bien el valor 0 indicando que no se puede enviar la información deseada.
Por ejemplo, si queremos enviar el Quijote via SMS pues seguro que son necesarios más de un segmento (un segmento es igual a un SMS como lo conocen los usuarios) y este método nos devolvería o bien un número grande o bien el valor 0 indicando que no se puede enviar la información deseada.
public javax.wireless.messaging.Message receive() throws IOException, InterruptedIOException
Devuelve un mensaje enviado a nuestra aplicación. Hay que tener varias cosas importantes en mente:
- Es un método bloqueante, por lo que generalmente deberá ser invocado en un hilo distinto al hilo principal donde está ejecutandose el Midlet.
- Nuestra aplicación es responsable de guardar la información recibida a memoria no volátil en caso de ser necesario.
public void send(javax.wireless.messaging.Message msg) throws IOException, InterruptedIOException
Envia el mensaje al destinatario.
Este método debe ser invocado en un hilo distinto al hilo principal donde está ejecutandose el Midlet.
Para especificar el destinatario del mensaje se debe usar el método
Este método debe ser invocado en un hilo distinto al hilo principal donde está ejecutandose el Midlet.
Para especificar el destinatario del mensaje se debe usar el método
setAddress(java.lang.String addr)
definido en la interfaz javax.wireless.messaging.Message
de la que heradan javax.wireless.messaging.TextMessage
y javax.wireless.messaging.BinaryMessage
. public void setMessageListener(javax.wireless.messaging.MessageListener l) throws IOException
Registrar una clase que implemante la interfaz
Este método sólo tiene sentido para
javax.wireless.messaging.MessageListener
, el plataforma J2ME invocará el método notifyIncomingMessage()
cuando reciba un mensaje.Este método sólo tiene sentido para
javax.wireless.messaging.MessageConnection
que funcionen en modo servidorEjemplo 1. Envío de un SMS en modo texto.
En este ejemplo, vamos a hacer una aplicación que envie el texto que introduzca el usuario en un área de texto a un destinatario fijo.Para el desarrollo de aplicaciones para móviles, yo personalmente utilizo el IDE NetBeans con la extensión Mobility
- package autentia.tutoriales.wma;
- import javax.microedition.lcdui.*;
- improt javax.microedition.midlet.*;
- /**
- * MIDLet de Ejemplo del uso del API WMA (Wireless Messagin API)
- * @author Carlos García. Autentia.
- */
- public class WMAMidlet extends javax.microedition.midlet.MIDlet {
- private boolean started;
- /**
- * Constructor
- */
- public WMAMidlet(){
- this.started = false;
- }
- /*
- * @see javax.microedition.midlet.MIDlet#startApp()
- */
- protected void startApp() throws MIDletStateChangeException {
- // Este método puede ser incovado varias veces mientras la aplicación se esté ejecutando.
- // Por ejemplo, si se está ejecutando la aplicación y nos llega una llamada
- // entrante, la aplicación generalmente es pausada y esté método será
- // invocado cuando la llamada finalize.
- if (! this.started){
- this.started = true;
- Display.getDisplay(this).setCurrent(new WMAMainForm(this));
- }
- }
- /*
- * @see javax.microedition.midlet.MIDlet#destroyApp(boolean)
- */
- protected void destroyApp(boolean arg0) throws MIDletStateChangeException {
- this.destroyApp(true);
- }
- /*
- * @see javax.microedition.midlet.MIDlet#pauseApp()
- */
- protected void pauseApp() {
- // En este ejemplo no se requiere ninguna tarea cuando el MIDLet es pausado.
- }
- }
- package autentia.tutoriales.wma;
- import java.io.*;
- import javax.microedition.io.*;
- import javax.microedition.lcdui.*;
- import javax.wireless.messaging.*;
- /**
- * Ventana principal de la aplicación
- * @author Carlos García. Autentia
- */
- public class WMAMainForm extends javax.microedition.lcdui.TextBox
- implements javax.microedition.lcdui.CommandListener {
- /**
- * Referencia al MIDLet
- */
- private javax.microedition.midlet.MIDlet midlet;
- /**
- * Envia el SMS
- */
- private javax.microedition.lcdui.Command cmdSend;
- /**
- * Finaliza la aplicación
- */
- private javax.microedition.lcdui.Command cmdExit;
- /**
- * Constructor
- * Ventana principal de la aplicación
- */
- public WMAMainForm(javax.microedition.midlet.MIDlet midlet) {
- super("Mensaje a enviar", "", 166, TextField.ANY);
- this.midlet = midlet;
- this.createUI();
- }
- /**
- * Crea y configura el interfaz gráfico de la ventana.
- */
- private void createUI(){
- this.setTicker(new Ticker("Autentia Real Business Solutions"));
- this.cmdSend = new Command("Enviar", Command.OK, 1);
- this.cmdExit = new Command("Salir", Command.STOP, 1);
- this.addCommand(cmdSend);
- this.addCommand(cmdExit);
- this.setCommandListener(this);
- }
- /**
- * El usuario desea enviar el SMS
- */
- private void sendSMSClick() throws java.io.IOException {
- MessageConnection conn = null;
- TextMessage msg = null;
- try {
- // Paso 1: Obtenemos una implementación del Connection que se encargará de enviar el SMS
- conn = (MessageConnection) Connector.open("sms://+34699221570");
- // Paso 2: Creamos el SMS
- msg = (TextMessage) conn.newMessage(MessageConnection.TEXT_MESSAGE);
- // Paso 3: Establecemos el contenido del SMS
- msg.setPayloadText(this.getString());
- // Paso 4: Enviamos el SMS
- conn.send(msg);
- } finally {
- // Paso 5: IMPORTANTE Cerramos la conexión
- this.closeQuietly(conn);
- conn = null;
- }
- }
- /**
- * Cierra un Connection ignorando todas las posibles excepciones
- */
- private void closeQuietly(javax.microedition.io.Connection conn){
- try {
- conn.close();
- } catch (Exception ex){
- // Nada
- }
- }
- /*
- * Receptor de eventos del UI (User Interface)
- * @see javax.microedition.lcdui.CommandListener#commandAction(javax.microedition.lcdui.Command, javax.microedition.lcdui.Displayable)
- */
- public void commandAction(Command arg0, Displayable arg1) {
- try {
- if (arg0 == cmdSend){
- this.sendSMSClick();
- } else if (arg0 == cmdExit){
- this.midlet.notifyDestroyed();
- }
- } catch (Exception ex){
- // En caso de error modificamos el texto de la ventana con el mensaje
- this.setString(ex.toString());
- }
- }
- }
Ejemplo 2. Envío de un SMS en modo binario.
Los pasos para enviar información en modo binario son los mismos que para el envio de SMS en modo texto.Los mensajes binarios son representados bajo la clase
javax.wireless.messaging.BinaryMessage.
La principal diferencia entre esta clase y javax.wireless.messaging.TextMessage
es el método para especificar el contenido del mensaje a enviar. En este último caso, el contenido es un array de bytes y es especificado a través del método setPayloadData
.- /**
- * El usuario desea enviar el SMS en modo binario
- */
- private void sendBinarySMSClick() throws java.io.IOException {
- MessageConnection conn = null;
- BinaryMessage msg = null;
- ByteArrayOutputStream bout = null;
- try {
- // Paso 1: Obtenemos una implementación del Connection que se encargará de enviar el SMS
- conn = (javax.wireless.messaging.MessageConnection) Connector.open("sms://+34699221570");
- // Paso 2: Creamos el SMS
- msg = (BinaryMessage) conn.newMessage(MessageConnection.BINARY_MESSAGE);
- // Paso 3: Establecemos el contenido del SMS con algunos datos de prueba. Por ejemplo, datos de una persona.
- bout = new ByteArrayOutputStream();
- dout = new DataOutputStream(bout);
- dout.writeBoolean(true); // ¿Soltero?
- dout.writeByte(55); // Edad
- dout.writeUTF("Madrid"); // Provincia de nacimiento
- dout.writeUTF("España"); // País.
- dout.writeLong(888883311L); // DNI
- msg.setPayloadData(bout.toByteArray());
- // Paso 4: Enviamos el SMS
- conn.send(msg);
- } finally {
- // Paso 5: IMPORTANTE. Cerramos las objetos, liberando recursos
- this.closeQuietly(bout);
- this.closeQuietly(dout);
- this.closeQuietly(conn);
- dout = null;
- dout = null;
- conn = null;
- }
- }
- /**
- * Cierra un OutputStream ignorando todas las posibles excepciones
- */
- private void closeQuietly(java.io.OutputStream out){
- try {
- out.close();
- } catch (Exception ex){
- // Nada
- }
- }
- /**
- * Cierra un Connection ignorando todas las posibles excepciones
- */
- private void closeQuietly(javax.microedition.io.Connection conn){
- try {
- conn.close();
- } catch (Exception ex){
- // Nada
- }
- }
Introducción a Push Registry.
A partir de la especificación MIDP 2.0, se añadió una potente característica a la plataforma J2ME que consiste en que nuestras aplicaciones puedan ser iniciadas por eventos externos o temporizadores.Por ejemplo, nuestra aplicación puede ser iniciada cuando reciba un SMS en un determinado puerto (identificador).
Información interesante
A continuación os presento una tabla con información relacionada con el juego número de SMS necesarios para enviar información.Referencia: http://java.sun.com/products/wma/index.jsp Por lo general los mensajes de texto se envian con el juego de caracteres del GSM-7 bit, y sólo cuando el mensaje tiene caracteres que no pueden ser codificados con ese juego de caracteres se usará el UCS-2.
Conclusiones y reflexiones
En comparación con otros, este API es bastante simple y fácil de utilizar.
Sabiendo utilizar esta y otras tecnologías como las que os presentamos en Autentia a través de nuestros tutoriales, se pueden hacer sistemas interesantes y útiles.
No olvideis que esto es sólo una introducción, asi que si necesitais más información debeis dirigiros a las páginas oficiales de la especificación.
Espero que os haya parecido interesante este tutorial.
Hola!!!
ResponderEliminarEsta muy interesante el articulo y muy bueno!!
a mi me gustaria saber, si puedo enviar archivos con extension .doc, .pdf, etc. como un SMS?? es posible hacer eso con la api WMA??
gracias!!
saludos XD
Hola muchas gracias por la info, es muy interesante, apenas empiezo a programas en javaMe, y necesito codigo. muchas gracias
ResponderEliminar