Manual 2012-II 04 Lenguaje de Programación II (0480)
Manual 2012-II 04 Lenguaje de Programación II (0480)
Manual 2012-II 04 Lenguaje de Programación II (0480)
Faltan Crditos
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
NDICE
Presentacin Red de contenidos Unidad de Aprendizaje 1: Serviet y JSP SEMANA 1 SEMANA 2 SEMANA 3 SEMANA 4 SEMANA 5 SEMANA 6 SEMANA 7 SEMANA 8 : Arquitectura JEE y Tomcat : Servlet y JSP : Servlet y JSP HttpSession Patrones de Diseo de Software Patrn Data Access Object : Examen parcial de teora : Examen parcial de laboratorio 7 21 33 53 67 85 5 6
Unidad de Aprendizaje 2: Seguridad de Aplicaciones SEMANA 9 : Manejo de Seguridad con Java 103
Unidad de Aprendizaje 3: Standard Actions SEMANA 10 : Manejo de Etiquetas Standard Actions Unidad de Aprendizaje 4: Custom Tag SEMANA 11 : Custom Tag I SEMANA 12 : Custom Tag II Unidad de Aprendizaje 5: JSTL y Lenguaje de Expresiones SEMANA 13 : Etiquetas estndares para Java (JSTL) SEMANA 14 : Lenguaje de Expresiones (EL) SEMANA 15 : Examen final de laboratorio SEMANA 16 : Sustentacin comercial de sistemas SEMANA 17 : Examen final de teora 139 126 113
CIBERTEC
CARRERAS PROFESIONALES
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
PRESENTACIN
Lenguaje de Programacin II pertenece a la lnea de cursos de programacin y se dicta en la carrera de Computacin e Informtica. El curso brinda un conjunto de herramientas del lenguaje Java que permite a los alumnos utilizar para la implementacin de aplicaciones web en Java.
El manual para el curso ha sido diseado bajo la modalidad de unidades de aprendizaje, las que se desarrollan durante semanas determinadas. En cada una de ellas, hallar los logros, que debe alcanzar al final de la unidad; el tema tratado, el cual ser ampliamente desarrollado; y los contenidos, que debe desarrollar, es decir, los subtemas. Asimismo, encontrar las actividades que deber desarrollar en cada sesin, que le permitirn reforzar lo aprendido en la clase.
El curso es eminentemente prctico: consiste en desarrollar una aplicacin web. En primer lugar, se inicia con crear programas utilizando la tecnologa de Servlet y JSP. Contina con la presentacin de nuevas tecnologas como el patrn Data Access Object. Luego, se desarrollan los componentes reutilizables mediante el CustomTag. Finalmente, se concluye con el uso del lenguaje de expresiones (EL).
CIBERTEC
CARRERAS PROFESIONALES
RED DE CONTENIDOS
Lenguaje de Programacin II
Servlet y JSP
Seguridad de Aplicaciones
Standard Accions
Custom Tag
JSTL y EL
Seguridad
JSTL EL
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
UNIDAD DE APRENDIZAJE
1
SEMANA
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
Arquitectura Java EE y el contenedor Tomcat
ACTIVIDADES PROPUESTAS
Los alumnos reconocen la estructura de carpetas del contenedor de Servlet llamado Tomcat. Los alumnos reconocen la estructura de una aplicacin web.
CIBERTEC
CARRERAS PROFESIONALES
1. JEE
Java Platform, Enterprise Edition o Java EE (anteriormente conocido como Java 2 Platform, Enterprise Edition o J2EE hasta la versin 1.4), es una plataforma de programacinparte de la Plataforma Javapara desarrollar y ejecutar software de aplicaciones en Lenguaje de programacin Java con arquitectura de N niveles distribuida, basndose ampliamente en componentes de software modulares ejecutndose sobre un servidor de aplicaciones. La plataforma Java EE est definida por una especificacin. Similar a otras especificaciones del Java Community Process, Java EE es tambin considerada informalmente como un estndar debido a que los suministradores deben cumplir ciertos requisitos de conformidad para declarar que sus productos son conformes a Java EE; estandarizado por The Java Community Process / JCP. Java EE incluye varias especificaciones de API, tales como JDBC, RMI, e-mail, JMS, Servicios Web, XML, etc. y define cmo coordinarlos. Java EE tambin configura algunas especificaciones nicas para Java EE para componentes. Estas incluyen Enterprise JavaBeans, servlets, portlets (siguiendo la especificacin de Portlets Java), JavaServer Pages y varias tecnologas de servicios web. Esto permite al desarrollador crear una Aplicacin de Empresa portable entre plataformas y escalable, a la vez que integrable con tecnologas anteriores. Otros beneficios aadidos son, por ejemplo, que el servidor de aplicaciones puede manejar transacciones, la seguridad, escalabilidad, concurrencia y gestin de los componentes desplegados, significando que los desarrolladores pueden concentrarse ms en la lgica de negocio de los componentes en lugar de en tareas de mantenimiento de bajo nivel. Las razones que empujan a la creacin de la plataforma JEE son las siguientes:
Programacin eficiente. Para conseguir productividad es importante que los equipos de desarrollo tengan una forma estndar de construir mltiples aplicaciones en diversas capas (cliente, servidor web, etc.). En cada capa necesitaremos diversas herramientas; por ejemplo, en la capa cliente tenemos applets, aplicaciones Java, etc. En la capa web, tenemos servlets, pginas JSP, etc. Con JEE tenemos una tecnologa estndar, un nico modelo de aplicaciones, que incluye diversas herramientas; en contraposicin al desarrollo tradicional con HTML, Javascript, CGI, servidor web, etc. que implicaba numerosos modelos para la creacin de contenidos dinmicos, con los lgicos inconvenientes para la integracin. Extensibilidad frente a la demanda del negocio. En un contexto de crecimiento de nmero de usuarios, es precisa la gestin de recursos, como conexiones a bases de datos, transacciones o balanceo de carga. Adems, los equipos de desarrollo deben aplicar un estndar que les permita abstraerse de la implementacin del servidor, con aplicaciones que puedan ejecutarse en mltiples servidores, desde un simple servidor hasta una arquitectura de alta disponibilidad y balanceo de carga entre diversas mquinas. Integracin. Los equipos de ingeniera precisan estndares que favorezcan la integracin entre diversas capas de software.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
La plataforma JEE implica una forma de implementar y desplegar aplicaciones empresariales. La plataforma se ha abierto a numerosos fabricantes de software para conseguir satisfacer una amplia variedad de requisitos empresariales. La arquitectura JEE implica un modelo de aplicaciones distribuidas en diversas capas o niveles (tier). La capa cliente admite diversas tipos de clientes (HTML, Applet, aplicaciones Java, etc.). La capa intermedia (middle tier) contiene subcapas (el contenedor web y el contenedor EJB). La tercera capa dentro de esta visin sinttica es la de de aplicaciones 'backend' como ERP, EIS, bases de datos, etc. Como se puede ver, un concepto clave de la arquitectura es el de contenedor, que dicho de forma genrica no es ms que un entorno de ejecucin estandarizado que ofrece unos servicios por medio de componentes. Los componentes externos al contenedor tienen una forma estndar de acceder a los servicios de dicho contenedor, con independencia del fabricante.
Contenedor Web, tambin denominado contenedor Servlet/JSP, maneja la ejecucin de los servlets y pginas JSP. Estos componentes se ejecutan sobre un servidor Enterprise Edition. Contenedor Enterprise JavaBeans, que gestiona la ejecucin de los EJB. Esta ejecucin requiere de un server EE.
Los contenedores incluyen descriptores de despliegue (deployment descriptors), que son archivos XML que nos sirven para configurar el entorno de ejecucin: rutas de acceso a aplicaciones, control de transacciones, parmetros de inicializacin, etc. La plataforma JEE incluye APIs para el acceso a sistemas empresariales:
CIBERTEC
CARRERAS PROFESIONALES
10
Java Transaction API (JTA) es el API para manejo de transacciones a travs de sistemas heterogneos. Java Naming and Directory Interface (JNDI) es el API para acceso a servicios de nombres y directorios. Java Message Service (JMS) es el API para el envo y recepcin de mensajes por medio de sistemas de mensajera empresarial como IBM MQ Series. JavaMail es el API para envo y recepcin de email. Java IDL es el API para llamar a servicios CORBA.
Servidor HTTP (tambin denominado servidor Web o servidor de pginas). Un ejemplo es el servidor Apache. Contenedor de aplicaciones o contenedor Servlet/JSP. Un ejemplo, Tomcat (que incluye el servicio anterior sobre pginas) Contenedor Enterprise Java Beans, que contiene aplicativos Java de interaccin con bases de datos o sistemas empresariales. Un ejemplo es JBoss, que contiene a los anteriores (servidor de pginas web y contenedor de aplicacione web).
Sin embargo, conviene empezar por el principio, es decir, el lenguaje bsico de interconexin: el protocolo HTTP. Este es un protocolo de aplicacin, generalmente implementado sobre TCP/IP. Es un protocolo sin estado basado en solicitudes (request) y respuestas (response), que usa por defecto el puerto 8080:
"Basado en peticiones y respuestas": significa que el cliente (por ejemplo, un navegador) inicia siempre la conexin (por ejemplo, para pedir una pgina). No hay posibilidad de que el servidor realice una llamada de respuesta al cliente (retrollamada). El servidor ofrece la respueta (la pgina) y cierra la conexin. En la siguiente peticin del cliente, se abre una conexin y el ciclo vuelve e empezar: el servidor devuelve el recurso y cierra conexin. "Sin estado": el servidor cierra la conexin una vez realizada la respuesta. No se mantienen los datos asociados a la conexin. Ms adelante veremos que hay una forma de persistencia de datos asociada a la "sesin".
Qu ocurre cuando un navegador invoca una aplicacin? El cliente (el navegador) no invoca directamente el contenedor de aplicaciones, sino que llama al servidor web por medio de HTTP. El servidor web se interpone en la solicitud o invocacin; siendo el servidor web el responsable de trasladar la solicitud al contenedor de aplicaciones.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
11
El cliente (normalmente por medio de un navegador, aunque podra ser una aplicacin Swing) solicita un recurso por medio de HTTP. Para localizar el recurso al cliente especifica una URL (Uniform Resource Locator), como por ejemplo http://www.host.es/aplicacion/recurso.html. El URI (Uniform Resource Identifier) es el URL excluyendo protocolo y host. Existen diversos mtodos de invocacin, aunque los ms comunes son POST y GET. Los veremos ms adelante. Sobre una misma mquina podemos tener diversas instancias de un AS (Application Server), procurando que trabajen sobre puertos diferentes, para que no se produzcan colisiones (por defecto HTTP trabaja con 8080). Un servicio crucial es la capacidad de recibir peticiones HTTP, para lo cual tenemos un HTTP Listener (aunque puede tener listeners para otros protocolos como IIOP). La solicitud llega el servidor de pginas web, que tiene que descifrar si el recurso solicitado es un recurso esttico o una aplicacin. Si es una aplicacin delega la solicitud en el contenedor web (contenedor Servlet/JSP). El contenedor web gestiona la localizacin y ejecucin de Servlets y JSP, que no son ms que pequeos programas. El contenedor web o contenedor Servlet/JSp recibe la solicitud. Su mquina Java (JVM) invoca al objeto Servlet/JSP, por tanto nos encontramos ante un tipo de aplicaciones que se ejecutan en el servidor, no en el cliente. No conviene olvidar que un Servlet o un JSP no es ms que una clase Java. Lo ms interesante en este sentido es que: o La JVM (generalmente) no crea una instancia de la clase por cada solicitud, sino que con una nica instancia de un Servlet/JSP se da servicio a mltiples solicitudes HTTP. Esto hace que el consumo de recursos sea pequeo en comparacin con otras opciones, como el uso de CGIs, en donde cada solicitud se resuelve en un proceso. o Para cada solicitud se genera un hilo (thread) para resolverla (pero con una nica instancia de la clase, como hemos dicho).
CIBERTEC
CARRERAS PROFESIONALES
12
El contenedor necesita conectores que sirven de intermediarios para comunicarse con elementos externos. Los conectores capacitan al AS para acceder a sistemas empresariales (backends). Por ejemplo: o El Java Message Service ofrece conectividad con sistemas de mensajeria como MQSeries. o El API JDBC da la capacidad de gestionar bases de datos internas al AS, pero adems permite ofrecer servicios como un pool de conexiones. Es necesario una gestin de hilos, ya que ser necesario controlar la situacin en la que tenemos una instancia de un componente (por ejemplo, un servlet) que da respuesta a varias peticiones, donde cada peticin se resuelve en un hilo.
Capa de cliente, tambin conocida como capa de presentacin o de aplicacin. Nos encontramos con componentes Java (applets o aplicaciones) y no-Java (HTML, JavaScript, etc.). Capa Web. Intermediario entre el cliente y otras capas. Sus componentes principales son los servlets y las JSP. Aunque componentes de capa cliente (applets o aplicaciones) pueden acceder directamente a la capa EJB, lo normal es que Los servlets/JSPs pueden llamar a los EJB. Capa Enterprise JavaBeans. Permite a mltiples aplicaciones tener acceso de forma concurrente a datos y lgica de negocio. Los EJB se encuentran en un servidor EJB, que no es ms que un servidor de objetos distribuidos. Un EJB puede conectarse a cualquier capa, aunque su misin esencial es conectarse con los sistemas de informacin empresarial (un gestor de base de datos, ERP, etc.) Capa de sistemas de informacin empresarial.
La visin de la arquitectura es un esquema lgico, no fsico. Cuando hablamos de capas nos referimos sobre todo a servicios diferentes (que pueden estar fsicamente dentro de la misma mquina e incluso compartir servidor de aplicaciones y JVM).
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
13
CIBERTEC
CARRERAS PROFESIONALES
14
XML utiliza HTTP como su soporte para el transporte. Una pregunta muy comn es cundo usar servlets y cundo usar pginas JSP. La pregunta es lgica, al fin y al cabo ambos mecanismos permiten generar contenidos dinmicos y adems las JSP son servlets generados por el servidor de aplicaciones. La norma es que la mayor parte de las interacciones con el usuario se realizarn en las JSP debido a su flexibilidad, ya que integran de forma natural etiquetas HTML, XML, JSF, etc. Los servlets sern la excepcin (un ejemplo tpico es usar un servlet como controlador: un controlador recibe peticiones o eventos desde el interfaz de cliente y "sabe" el componente que debe invocar).
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
15
En este tipo de escenario, la capa web implica tanto lgica de presentacin como lgica de negocio. Pero lo deseable es no mezclar todas las cosas, planteando un diseo modular. Para ello, las JSP y servlets no suelen acceder de forma directa a la base de datos, sino que lo hacen por medio de un servicio de acceso a datos:
CIBERTEC
CARRERAS PROFESIONALES
16
Ya dijimos en el escenario web-centric que en bastantes ocasiones las aplicaciones no requieren acceder a mltiples sistemas empresariales, es decir, la capa de lgica de negocio no requiere fuerte conectividad distribuida con sistemas empresariales. No se contemplan EJBs, pero esto no significa que desaparezcan los componentes del modelo, sino que los servicios del modelo se implementan en JavaBeans (no Enterprise JavaBeans) para ser utilizados por Servlets/JSP dentro de la capa web:
En una aplicacin centrada en la capa Web, sigue existiendo, aunque sea ligera, un modelo que contiene entidades y reglas de negocio; es decir, el que no sean necesarios los EJB no implica no modularizar, mezclarlo todo y eliminar los componentes del modelo. Nota: La especificacin JEE no considera como componentes JEE a los Java Beans ya que son diferentes de los EJB (no confundirlos). La arquitectura de componentes JavaBeans se pueden utilizar tanto en la capa de cliente como de servidor, mientras que los componentes Enterprise JavaBeans slo se utilizan en la capa de negocio como parte de los servicios del servidor.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
17
Borland Enterprise Server Se puede trabajar con Tomcat como contenedor de JSPs y Servlets y algn otro servidor de aplicaciones como contenedor de EJBs. A continuacin, se listan ejemplos de servidores de aplicaciones que soportan EJBs: Oracle 9iAS JBoss (gratuito) JOnAS (gratuito) Web Logic IBM WebSphere
Tomcat se constituye en uno de los proyectos ms interesantes de cdigo abierto (open source) liderado por la Apache Software Foundation y es distribudo junto con el servidor web Apache. Tomcat cumple parcialmente (no soporta EJBs) con los estndares JEE e implementa las especificaciones Servlet 2.3 y JavaServer Pages 1.2.
miwebapp
WEB-INF login.jsp home.jsp index.html
web.xml classes
CIBERTEC
CARRERAS PROFESIONALES
18
Principales directorios de una aplicacin web: /miwebapp. Directorio raz de la aplicacin web en donde se colocan todos los archivos HTML, JSP, GIF, JPG que utiliza la aplicacin. Se pueden crear subdirectorios adicionales para mantener cualquier otro recurso de tipo esttico que forme parte de la aplicacin web. /miwebapp/WEB-INF. Directorio que contiene todos los recursos relacionados con la aplicacin web que no son de acceso directo para un cliente (browser). En este directorio, se coloca el archivo descriptor web.xml, donde se define la configuracin de la aplicacin web. /miwebapp/WEB-INF/classes. Directorio que contiene todos los Servlets y cualquier otra clase de utilidad o complementaria que se necesite para la ejecucin de la aplicacin web. Por lo general, contiene solo archivos compilados .class Archivo de configuracin web.xml Conocido como archivo descriptor de la aplicacin web, este archivo xml, ubicado dentro del directorio WEB-INF, contiene la descripcin de la configuracin correspondiente a la aplicacin web. La informacin que contiene puede incluir lo siguiente:
Configuracin de la sesin Definiciones de Servlets y Registro de Servlets Registro de tipos MIME Pginas de error Pginas de bienvenida (tag <welcome-file-list>)
Archivos War Un archivo WAR (web Archive) es la representacin de una aplicacin web en una unidad nica distribuible. Es el mtodo estndar empleado para empaquetar una aplicacin web y dejarla lista para su distribucin y acceso a travs de servidores web con soporte para Servlets y pginas JSP. No importa el nmero o tipo de recursos (Servlets, JSPs, HTMLs, etc.); un archivo WAR agrupa una aplicacin web completa, en una nica unidad de distribucin, en un nico archivo.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
19
Resumen
miwebapp
web.xml
WEB-INF
classes lib
Servidor HTTP (tambin denominado servidor Web o servidor de pginas); por ejemplo, el servidor Apache Contenedor de aplicaciones o contenedor Servlet/JSP; por ejemplo, Tomcat (que incluye el servicio anterior sobre pginas) Contenedor Enterprise Java Beans, que contiene aplicativos Java de interaccin con bases de datos o sistemas empresariales. Un ejemplo es JBoss, que contiene a los anteriores (servidor de pginas web y contenedor de aplicacione web).
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://pdf.coreservlets.com/ Aqu hallar lo referente a Servlets.
CIBERTEC
CARRERAS PROFESIONALES
20
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
21
UNIDAD DE APRENDIZAJE
1
SEMANA
2
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
Servlets Objeto RequestDispatcher
CIBERTEC
CARRERAS PROFESIONALES
22
1. SERVLETS
Un servlet es un componente JEE que extiende la capacidad de proceso de un servidor que emplea el paradigma request response. Por lo tanto, Un servlet es una clase Java que recibe requerimientos de un cliente para cumplir con un servicio; luego de cumplir con el servicio, enva la respuesta hacia el cliente.
Peticin (Request)
Servidor web
Los Servlets son cargados en memoria en la primera peticin de un cliente o cuando el servidor de aplicaciones arranca.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
23
Cada peticin de cliente es servida sobre un diferente hilo (thread); por lo tanto, muchos clientes pueden acceder al mismo cdigo en paralelo. Es responsabilidad del desarrollador sincronizar los accesos a los recursos compartidos. Se pueden crear Servlets de un solo hilo (SingleThread) implementando la interfaz: javax.servlet.SingleThreadModel
3. La clase HTTPSERVLET
Los clientes web (browsers) interactan con los Servlets usando el protocolo HTTP (Request-Response); por lo tanto, para crear Servlets que soporten este protocolo, se debe heredar de la clase javax.servlet.http.HttpServlet. La clase HttpServlet provee una estructura de trabajo adecuada para manipular el protocolo HTTP junto con los mtodos GET y POST. El servlet debe ser creado dentro de la carpeta classes del web application con el que se est trabajando. Ejemplo: Webapps Miwebapp
WEB-INF classes
CIBERTEC
CARRERAS PROFESIONALES
24
protected doPut(HttpServletRequest req, HttpServletResponse resp) void Called by the server (via the service method) to allow a servlet to
...
<!--=======================================================--> <!-- La etiqueta <servlet> define el nombre fsico de un servlet --> <!-- =======================================================--> <servlet> <servlet-name> SampleServlet </servlet-name> <servlet-class> edu.cibertec.servlet.SampleServlet </servlet-class> </servlet> <!--==========================================================--> <!-- La etiqueta <servlet-mapping> define el nombre lgico de un servlet --> <!--==========================================================--> <servlet-mapping> <servlet-name> SampleServlet </servlet-name> <url-pattern> /servlet1 </url-pattern> </servlet-mapping>
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
25
Returns the value of the named attribute as an Object, or null if no attribute of the given name exists.
Enumeration getAttributeNames()
Returns an Enumeration containing the names of the attributes available to this request.
String getCharacterEncoding()
Returns the name of the character encoding used in the body of this request.
int getContentLength()
Returns the length, in bytes, of the request body and made available by the input stream, or -1 if the length is not known.
String getContentType()
Returns the MIME type of the body of the request, or null if the type is not known.
ServletInputStream getInputStream()
Returns the Internet Protocol (IP) address of the interface on which the request was received.
Locale getLocale()
Returns the preferred Locale that the client will accept content in, based on the Accept-Language header.
Enumeration getLocales()
Returns an Enumeration of Locale objects indicating, in decreasing order starting with the preferred locale, the locales that are acceptable to the client based on the AcceptLanguage header.
String getLocalName()
Returns the host name of the Internet Protocol (IP) interface on which the request was received.
int getLocalPort()
Returns the Internet Protocol (IP) port number of the interface on which the request was received.
String getParameter(String name)
Returns the value of a request parameter as a String, or null if the parameter does not exist.
Map getParameterMap()
Returns an Enumeration of String objects containing the names of the parameters contained in this request.
CIBERTEC
CARRERAS PROFESIONALES
26
values the given request parameter has, or null if the parameter does not exist.
String getProtocol()
Returns the name and version of the protocol the request uses in the form protocol/majorVersion.minorVersion, for example, HTTP/1.1.
BufferedReader getReader()
Deprecated. As of Version 2.1 of the Java Servlet API, use ServletContext.getRealPath(java.lang.String) instead.
String getRemoteAddr()
Returns the Internet Protocol (IP) address of the client or last proxy that sent the request.
String getRemoteHost()
Returns the fully qualified name of the client or the last proxy that sent the request.
int getRemotePort()
Returns the Internet Protocol (IP) source port of the client or last proxy that sent the request.
RequestDispatcher getRequestDispatcher(String path)
Returns a RequestDispatcher object that acts as a wrapper for the resource located at the given path.
String getScheme()
Returns the name of the scheme used to make this request, for example, http, https, or ftp.
String getServerName()
Returns the host name of the server to which the request was sent.
int getServerPort()
Returns a boolean indicating whether this request was made using a secure channel, such as HTTPS.
void removeAttribute(String name)
Overrides the name of the character encoding used in the body of this request.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
27
the servlet.
String getContextPath()
Returns the portion of the request URI that indicates the context of the request.
Cookie[] getCookies()
Returns an array containing all of the Cookie objects the client sent with this request.
long getDateHeader(String name)
Returns the value of the specified request header as a long value that represents a Date object.
String getHeader(String name)
Returns all the values of the specified request header as an Enumeration of String objects.
int getIntHeader(String name)
Returns the name of the HTTP method with which this request was made, for example, GET, POST, or PUT.
String getPathInfo()
Returns any extra path information associated with the URL the client sent when it made this request.
String getPathTranslated()
Returns any extra path information after the servlet name but before the query string, and translates it to a real path.
String getQueryString()
Returns the query string that is contained in the request URL after the path.
String getRemoteUser()
Returns the login of the user making this request, if the user has been authenticated, or null if the user has not been authenticated.
String getRequestedSessionId()
Returns the part of this request's URL from the protocol name up to the query string in the first line of the HTTP request.
StringBuffer getRequestURL()
Returns the part of this request's URL that calls the servlet.
HttpSession getSession()
Returns the current session associated with this request, or if the request does not have a session, creates one.
HttpSession getSession(boolean create)
CIBERTEC
CARRERAS PROFESIONALES
28
Returns the current HttpSession associated with this request or, if there is no current session and create is true, returns a new session.
Principal getUserPrincipal()
Returns a java.security.Principal object containing the name of the current authenticated user.
boolean isRequestedSessionIdFromCookie()
Checks whether the requested session ID came in as part of the request URL.
boolean isRequestedSessionIdValid()
Returns a boolean indicating whether the authenticated user is included in the specified logical "role".
Una vez obtenido el objeto RequestDispatcher, invocamos al otro recurso (servlet) envindole el objeto request. rd.forward (request, response) Hay que registrar el servlet MyForwardServlet para lo cual debemos modificar el archivo web.xml.
8. COMPARTIENDO OBJETOS
Existen dos formas de compartir objetos entre Servlets: El objeto ServletContext y El objeto HttpServletRequest
El ServletContext es usado cuando un grupo de Servlets necesitan trabajar con un mismo objeto. El servlet que desea compartir el objeto usa el mtodo setAttribute del objeto ServletContext y otro servlet dentro del contexto de Servlets que desea tomar el objeto usa el mtodo getAttribute. Ejemplo:
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
29
Servlet 1 (compartiendo objeto) : MiClase elObjeto = new MiClase(); getServletContext().setAttribute(objeto,elObjeto); Servlet 2 (tomando objeto): MiClase elObjeto; ElObjeto =(MiClase)getServletContext().getAttribute(objeto);
HttpServletRequest es usado cuando al hacer uso de una remisin (forward) se necesita compartir un objeto entre Servlets. El servlet que desea compartir el objeto usa el mtodo setAttribute del objeto HttpServletRequest y el otro servlet usa el mtodo getAttribute. Ejemplo:
ElObjeto =(MiClase)request.getAttribute(objeto);
CIBERTEC
CARRERAS PROFESIONALES
30
Preguntas de Certificacin
1. Cules mtodos en la clase HttpServlet sirve el pedido del HTTP POST? (Seleccione uno) a. b. c. d. doPost(ServletRequest, ServletResponse) doPOST(ServletRequest, ServletResponse) servicePost(HttpServletRequest, HttpServletResponse) doPost(HttpServletRequest, HttpServletResponse)
2. Considere el siguiente cdigo de la pgina HTML: <html><body> <a href="/servlet/HelloServlet">POST</a> </body></html> Cul mtodo de la clase HelloServlet fue invocado cuando el enlace mostrado es pulsado? (Seleccione uno) a. b. c. d. e. doGet doPost doForm doHref serviceGet
3. Considere el siguiente cdigo: public void doGet(HttpServletRequest req, HttpServletResponse res) { HttpSession session = req.getSession(); ServletContext ctx = this.getServletContext(); if(req.getParameter("userid") != null) { String userid = req.getParameter("userid"); //1 } } Si desea usar el parmetro userid para que est disponible en los pedidos del mismo usuario, cul de las siguientes lneas se deber insertar en //1? (Seleccione uno) a. b. c. d. e. session.setAttribute("userid", userid); req.setAttribute("userid", userid); ctx.addAttribute("userid", userid); session.addAttribute("userid", userid); this.addParameter("userid", userid);
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
31
f.
this.setAttribute("userid", userid);
4. Cul de las siguientes lneas se deber usar en la salida del DataServlet para invocar algn otro servlet? (Seleccione uno) a. RequestDispatcher rd = request.getRequestDispatcher("/servlet/DataServlet"); rd.include(request, response); RequestDispatcher rd = request.getRequestDispatcher("/servlet/DataServlet"); rd.include(response); RequestDispatcher rd = request.getRequestDispatcher(); rd.include("/servlet/DataServlet", request, response); RequestDispatcher rd = request.getRequestDispatcher(); rd.include("/servlet/DataServlet", response); RequestDispatcher rd = request.getRequestDispatcher(); rd.include("/servlet/DataServlet");
b.
c.
d.
e.
CIBERTEC
CARRERAS PROFESIONALES
32
Resumen
Para registrar un Servlet, es necesario escribir un conjunto de tags en el archivo de web.xml. <servlet> <servlet-name>NombreServlet</servlet-name> <servlet-class>rutaDeLaClase </servlet-class> </servlet> <servlet-mapping> <servlet-name>NombreServlet </servlet-name> <url-pattern>/Alias</url-pattern> </servlet-mapping> Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://courses.coreservlets.com/Course-Materials/scwcd.html Aqu hallar ejercicios sobre JSP.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
33
UNIDAD DE APRENDIZAJE
1
SEMANA
3
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
Java Server Pages Eventos (Listener)
ACTIVIDADES PROPUESTAS Los alumnos realizarn programas utilizando la tecnologia de Servlets y JSPs.
CIBERTEC
CARRERAS PROFESIONALES
34
No es muy frecuente encontrar programadores Java buenos en diseo html o diseadores de html buenos programando en Java; por ello, JSP es la solucin para separar roles: Un jsp tiene poco cdigo Java que puede ser fcilmente entendido por un diseador html. Ejemplo:
<HTML> <HEAD><TITLE>Saludo</TITLE><HEAD> <BODY> <% String miAlias=request.getParameter(miAlias); %> <H1>Hola <%=miAlias%></H1> </BODY> </HTML>
El servlet resultante procesa las peticiones http. El proceso de traducido y compilado es realizado una sola vez antes de procesar la primera peticin http; luego, el servlet resultante tiene el mismo ciclo de vida que cualquier servlet. Los equivalentes a los mtodos init, service y destroy son los siguientes: jspInit, _jspService y jspDestroy
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
35
3. DIRECTIVAS JSP
Las directivas JSP son usadas para definir informacin al traductor java acerca de la pgina. La sintaxis es la siguiente: <%@ directiva[ atributo=valor ] %>
Directiva include. Se usa para definir el contenido de otro archivo en el JSP. Ejemplo: <HTML> <HEAD><TITLE>Mi primer loguito<TITLE></HEAD> <BODY> <%@ include file=loguito.html%>
Directiva page. Se usa para definir las caractersticas de las cuales va a depender la pgina. La directiva aplica a todo el JSP incluso los archivos incluidos con la directiva include. Directiva page info. Define una cadena de texto que es ubicada en el mtodo Servlet.getServletInfo() del cdigo traducido. Ejemplo: <%@ page info=En el mar, la vida es ms sabrosa %>
Directiva page import. Se usa para importar una lista de nombres de paquetes separados por comas. Ejemplo: <%@ page import=java.math.*,java.util.* %>
Directiva page errorPage. Se usa para redireccionar un cliente a un URL especfico cuando ocurre una excepcin que no ha sido capturada en la pgina. Ejemplo: <%@ page errorPage=/jsps/error.jsp %>
CIBERTEC
CARRERAS PROFESIONALES
36
Directiva page isErrorPage. Se usa para indicar si la pgina es un target vlido (destino) de una directiva page errorPage. El valor por defecto es false. Ejemplo: <%@ page isErrorPage=true %> <%@ page isErrorPage=false %>
4. JSP SCRIPTING
El scripting es utilizado para codificar el JSP. El scripting est conformado de: Declarations Scriplets Expressions
a)
Declarations. Son utilizadas para declarar mtodos y variables de instancia en el servlet JSP. Sintaxis: <%! Declaracin %>
b)
Scriplets. Se utilizan para escribir cdigo Java en el JSP. Sintaxis: <% cdigo_java %>
Ejemplo: <% String sexo = request.getParameter(sexo); If(sexo.equals(M)){ %> <H2>Sr.</H2> <% }else{ %> <H2>Sr.</H2> <% } %>
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
37
c)
Expressions. Las expresiones son para incluir directamente dentro de la salida de la pgina cadenas (Strings), que son el resultado de evaluar una expresin de cdigo Java y luego convertirla en una cadena. Sintaxis: Ejemplo: La fecha actual es <%=new java.util.Date() %> <%= expression %>
b) Un JSP puede ser invocado desde un servlet usando el mtodo forward del objeto RequestDispatcher. Ejemplo: String miRuta = /jsps/consultas.jsp; RequestDispatcher rd = getServletContext().getRequestDispatcher(miRuta); Rd.forward(request,response);
c) Un Servlet o un JSP pueden ser invocados desde un JSP usando la etiqueta <FORM> o la etiqueta <A HREF>. Ejemplo: <FORM ACTION=/aplicacionWeb/URLServlet> <A HREF=/aplicacionWeb/URLServlet> texto </A>
CIBERTEC
CARRERAS PROFESIONALES
38
b) Servlet Compilation. Si se falla al escribir cdigo Java o cuando se omite alguna directiva de pgina que sea obligatoria. <%= new java.util.Dat() % >
a) Request Scope. Este mbito se constituye en la va ms adecuada para que un servlet pase referencias de objetos al JSP. Las referencias a objetos con mbito Request son almacenadas en el objeto request. Se usa lo siguiente: setAttribute(String,Object) para cargar (setear) el objeto en el request y getAttribute(String) para recuperar el objeto
b) Session Scope. Se puede acceder a este mbito desde Servlets y pginas JSP que estn procesando peticiones que se encuentran en la misma sesin.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
39
Las referencias a los objetos son perdidas despus que la sesin asociada es finalizada. Las referencias a los objetos con mbito Session son almacenadas en el objeto session.
Se usa lo siguiente: setAttribute(String,Object) para cargar(setear) el objeto en la sesin y getAttribute(String) para recuperar el objeto
c) Application Scope. Se puede acceder a este mbito desde Servlets y pginas JSP que estn procesando peticiones que se encuentran en la misma aplicacin web. (El mismo contexto) Las referencias a los objetos con mbito Application son almacenadas en el objeto application. Se usa lo siguiente: setAttribute(String,Object) para cargar(setear) el objeto en el contexto y getAttribute(String) para recuperar el objeto
Significado El objeto HttpServletRequest asociado con la peticin El objeto HttpServletResponse asociado con la respuesta El Writer empleado para enviar la salida al cliente. La salida de los JSP emplea un buffer que permite que se enven cabeceras HTTP o cdigos de estado aunque ya se haya empezado a escribir en la salida (out no es un PrintWriter, sino un objeto de la clase especial JspWriter). El objeto HttpSession asociado con la peticin actual. En JSP, las sesiones se crean automticamente, de modo que este objeto est instanciado aunque no se cree explcitamente una sesin. El objeto ServletContext, comn a todos los Servlets de la aplicacin web El objeto ServletConfig, empleado para leer parmetros de inicializacin permite acceder desde un nico objeto a todos los dems objetos implcitos
out
session
CIBERTEC
CARRERAS PROFESIONALES
40
page
Referencia al propio servlet generado (tiene el mismo valor que this).Como tal, en Java no tiene demasiado sentido utilizarla, pero est pensada para el caso en que se utilizara un lenguaje de programacin distinto.
exception
Representa un error producido en la aplicacin. Solo es accesible si la pgina se ha designado como pgina de error (mediante la directiva
page isErrorPage)
Como sus nombres indican, la aplicacin de inicio de evento se produce cuando su aplicacin web se carga por primera vez y comenz por el contenedor de Servlets; y solicitud de cierre se produce cuando la aplicacin web se cierra. El perodo de sesiones se produce en la creacin de una nueva sesin cada vez que se crea en el servidor y de manera similar el perodo de sesiones se origina con la invalidacin de una sesin cada vez que se anula. Para hacer uso de estas aplicaciones web y eventos para hacer algo til, tendr que crear y hacer uso de "clases oyentes". De aqu en adelante, vamos a implementar clases oyentes y cmo se pueden utilizar.
La implementacin de ServletContextListener permite escuchar la creacin o destruccin del contexto. La implementacin de HttpSessionListener le permite escuchar la creacin de una sesin. Veamos cules son los diferentes mtodos de esta interfaz, que se tendrn que aplicar.
10.2 ServletContextListener:
Esta interfaz contiene dos mtodos:
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
41
Un ejemplo de esa clase es el siguiente: javax.servlet.ServletContextListener de importacin; javax.servlet.ServletContextEvent de importacin; clase pblica ApplicationWatch implementa ServletContextListener ( public static applicationInitialized largo = 0L; / * Aplicacin de inicio del evento * / public void contextInitialized (ServletContextEvent ce) ( applicationInitialized = System.currentTimeMillis (); ) / * Aplicacin del evento de apagado * / public void contextDestroyed (ServletContextEvent ce) () ) En el cdigo anterior, una clase Java ApplicationWatch implementa ServletContextListener. Al implementar sus dos mtodos realmente slo usa uno de ellos y el segundo mtodo sigue teniendo el cuerpo vaco. En esta categora, se observa en el momento de la solicitud de inicio en public static que puede ser llamado desde otra aplicacin para saber qu clase fue la ltima vez que esta solicitud se inici. Se explicar cmo decirle al servidor de aplicaciones que tienen esta clase de oyente; adems, se quiere que se les diga la aplicacin de estos acontecimientos en un momento, pero primero vamos a ver cules son los diferentes mtodos de HttpSessionListener interfaz.
10.3 HttpSessionListener
Esta interfaz te permite poder escuchar al crear o destruir una sesin.
public void sessionCreated (HttpSessionEvent se); public void sessionDestroyed (HttpSessionEvent se);
Al igual que lo que hicimos en el caso de ApplicationWatch anterior, tendr que crear una clase Java y aplicar HttpSessionListener interfaz. Un ejemplo de esa clase es el siguiente:
/*
Expediente: SessionCounter.java */
CIBERTEC
CARRERAS PROFESIONALES
42
javax.servlet.http.HttpSessionListener de importacin; javax.servlet.http.HttpSessionEvent de importacin; clase pblica SessionCounter implementa HttpSessionListener ( private static int activeSessions = 0; / * Creacin de eventos de sesin * / public void sessionCreated (HttpSessionEvent se) ( activeSessions + +; ) / * Sesin de invalidacin del evento * / public void sessionDestroyed (HttpSessionEvent se) ( if (activeSessions> 0) activeSessions -; ) public static int getActiveSessions () ( activeSessions retorno; ) ) En el cdigo anterior, SessionCounter clase implementa HttpSessionListener para contar el nmero de sesiones activas. Se ha aprendido cules son los eventos de aplicaciones web, interfaz de lo que est disponible para nosotros y tambin se han visto ejemplos de la aplicacin de la interfaz de las clases. Veamos cmo decirle al servidor de aplicaciones acerca de estas clases oyentes.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
43
</ oyente de clase> </ oyente> <listener> <listener-class> com.stardeveloper.web.listener.ApplicationWatch </ oyente de clase> </ oyente> </ web-app> Como se indica anteriormente, es muy fcil declarar oyentes en el archivo web.xml. Ahora, cada vez que el servidor se inicia o cierre, una sesin se crea o destruye.
11.1 Ventajas
Son componentes reutilizables. Son parte del estndar.
CIBERTEC
CARRERAS PROFESIONALES
44
Son fciles de implementar. Se pueden incorporar y retirar de forma sencilla. Pueden ofrecer gran variedad de servicios.
11.2 Funcionamiento
Cada filtro responde a un Servlet o un URL que represente uno o ms recursos
La interfaz javax.servlet.FilterConfig
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
45
Define cuatro mtodos: o getFilterName() : Retorna el nombre asociado al filtro en la configuracin en un String o getInitParameter(String): Retorna el valor de un parmetro de configuracin o getInitParameterNames(): Retorna los nombres de los parmetros de configuracin. o getServletContext() : Retorna la referencia al ServletContext en el que trabajar el filtro Define un nico mtodo: o doFilter() throws ServletException o El mtodo doFilter recibe como argumentos ServletRequest y ServletResponse. Es invocado por el contenedor cuando el filtro forma parte de la cadena creada ante la peticin/respuesta de un recurso web.
public class MiFiltro implements Filter{ private FilterConfig filterConfig = null; public void init(FilterConfig filterConfig) throws ServletExceptin { this.filterConfig = filterConfig; } public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws ServletException { chain.doFilter(request,response); } public void destroy(){ this.filterConfig =null; } } Definir el filtro y sus parmetros
<filter> <filter-name>Filtro Aniversario Java</filter-name> <filter-class>MiFiltro</filter-class> <init-param> <param-name>aniversarioNumero</param-name> <param-value>10</param-value> </init-param> </filter> Definir el patrn de recursos para el filtro
CIBERTEC
CARRERAS PROFESIONALES
46
<filter-mapping> <filter-name>Filtro Aniversario Java</filter-name> <url-pattern>/miPagina.jsp</url-pattern> </filter-mapping> Solo miPagina.jsp <filter-mapping> <filter-name>Filtro Aniversario Java</filter-name> <url-pattern>/*.jsp</url-pattern> </filter-mapping Mapping a un Servlet
La regla para seleccionar los miembros de la cadena es el siguiente: o Primero obtener los filtros cuyo patrn URL coincida con el recurso solicitado, o Luego los que el nombre Servlet coincida con el Servlet solicitado. o Una misma implementacin, puede ser utilizada para distintos recursos empleado nombres diferentes e incluso parmetros diferentes
<filter> <filter-name>FiltroJSP</filter-name> <filter-class>MiFiltro</filter-class> <init-param> <param-name>param</param-name> <param-value>10</param-value> </init-param> </filter> <filter-mapping> <filter-name>FiltroJSP</filter-name> <url-pattern>/*.jsp</url-pattern> </filter-mapping>
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
47
<filter> <filter-name>FiltroHTM</filter-name> <filter-class>MiFiltro</filter-class> <init-param> <param-name>param</param-name> <param-value>20</param-value> </init-param> </filter> <filter-mapping> <filter-name>FiltroHTM</filter-name> <url-pattern>/*.htm</url-pattern> </filter-mapping> Un elemento a considerar es la longitud de la cadena para un recurso o grupo de recursos especfico. Debe evitarse el uso excesivo de filtros ya que impactaran en el tiempo de respuesta Uno o dos filtros suelen ser suficientes en la mayora de los casos Al utilizar la misma clase en diferentes filtros: o Cada definicin debe tener un nombre diferente en <filter-name> o El contenedor crear una instancia por cada definicin <filter> en el web.xml o Las llamadas concurrentes activaran mltiples Threads para la instancia del filtro asociado o La ejecucin de un filtro se divide en dos momentos: antes y despus de invocar la cadena
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws ServletException { //Cdigo previo a la cadena //Es el mejor lugar para modificar el request chain.doFilter(request,response); //Entrega el control al prximo de la cadena //Cdigo posterior a la cadena //Este es el mejor lugar para modificar la respuesta. }
CIBERTEC
CARRERAS PROFESIONALES
48
Por qu un filtro? Premisas de trabajo: Debe ser desarrollado en Java La seguridad de ser controlada por el contenedor. La solucin debe ser fcil de implantar y retirar sin afectar el funcionamiento de la aplicacin. La solucin debe ser reutilizable.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
49
Acciones realizadas por el FCC Verifica si el usuario que intenta iniciar sesin pertenece al grupo de control. Si pertenece al grupo, desva la peticin a una segunda pgina de login. Si no, contina la cadena. Intenta obtener las credenciales contra el segundo repositorio. Si tiene xito, la credencial se almacena para uso futuro en la aplicacin. Si falla, lo enva al inicio. Pasa el control a la cadena de filtros.
Mapping Control de Credenciales <filter> <filter-name>CtrlCredenciales</filter-name> <filter-class>FilterClassName</filter-class> </filter> <filter-mapping> <filter-name>CtrlCredenciales</filter-name> <url-pattern>/j_security_check</url-pattern> </filter-mapping>
CIBERTEC
CARRERAS PROFESIONALES
50
Preguntas de Certificacin
1 Considere el siguiente cdigo y seleccione la sentencia correcta acerca del siguiente supuesto. (Seleccione uno.) <html><body> <%! int aNum=5 %> The value of aNum is <%= aNum %> </body></html> a It will print "The value of aNum is 5" to the output. b It will flag a compile-time error because of an incorrect declaration. c It will throw a runtime exception while executing the expression. d It will not flag any compile time or runtime errors and will not print anything to the output. 2. Cules de los siguientes tags se pueden usar para imprimir un valor? (Seleccione dos.) a <%@ %> b <%! %> c <% %> d <%= %> e <%-- --%> 3. Cul de los siguientes mtodos es definido por la JSP engine? (Seleccione uno.) a jspInit() b _jspService() c _jspService(ServletRequest, ServletResponse) d _jspService(HttpServletRequest, HttpServletResponse) e jspDestroy() 4. Cul de las siguientes exceptions puede ser disparado por _jspService() mtodos? (Seleccione uno.) a javax.servlet.ServletException b javax.servlet.jsp.JSPException c javax.servlet.ServletException and javax.servlet.jsp.JSPException d javax.servlet.ServletException and java.io.IOException e javax.servlet.jsp.JSPException and java.io.IOException
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
51
Resumen
Existe tres mbitos: request (es el objeto HttpServletRequest actual) session (es el objeto HttpSession actual) application (es el objeto ServletContext actual)
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://courses.coreservlets.com/Course-Materials/scwcd.html Aqu hallar ejercicios sobre JSPs.
CIBERTEC
CARRERAS PROFESIONALES
52
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
53
UNIDAD DE APRENDIZAJE
1
SEMANA
4
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
HttpSession
CIBERTEC
CARRERAS PROFESIONALES
54
1.
HTTPSESSION
En una transaccin http, cada conexin entre un cliente y un servidor es muy breve. Esta es una caracterstica tpica de cualquier protocolo sin estado (staless). El protocolo http no mantiene el estado, es decir, no tiene un mecanismo para saber que una serie de peticiones (request) provienen de un mismo cliente. Ante esta limitacin, surge la sesin como el mecanismo adecuado para identificar un usuario que est interactuando con el sitio web. La sesin puede ser implementada a travs de diferentes opciones: HttpSession Campos ocultos HTML(hidden fields) Cookies URL Rewriting
La interfaz HttpSession es parte de la API de Servlets y proporciona una alternativa para manejar el estado de una aplicacin en un servidor web.
Sintaxis: request.getSession() o request.getSession(true). Retornan la sesin actual si existe; de lo contrario, crean una nueva. request.getSession(false). Retorna la sesin actual si existe; de lo contrario, retorna un objeto nulo.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
55
3. CONTENIDO DE LA SESIN
El objeto HttpSession contiene informacin acerca de la sesin. Para acceder a esta informacin, existen una serie de mtodos getters de los cuales mostraremos los ms utilizados:
session.getLastAccessedTime(). La ltima vez que el cliente envi una peticin asociada con la sesin
session.getMaxInactiveInterval(). El intervalo de tiempo mximo (en segundos) que el Servlet Container mantiene abierta la sesin entre accesos del cliente
3. ...
FINALIZANDO LA SESION
session.setAttribute("miNombre",nombre);
CIBERTEC
CARRERAS PROFESIONALES
56
La sesin puede ser finalizada de diferentes formas: Programticamente Por tiempo de expiracin Cuando el usuario cierra el browser
Programticamente: a) Usando el mtodo invalidate() del objeto session b) Removiendo todos los objetos de la sesin con el mtodo removeAttribute() Sintaxis : removeAttribute(String nombreAtributo) nombreAtributo es el nombre del atributo a ser removido.
Por tiempo de expiracin: La sesin es invalidada cuando se cumple un tiempo de inactividad de la sesin. Podemos definir el tiempo de inactividad mximo de dos formas: a) utilizando el mtodo setMaxInactiveInterval(int) Ejemplo: session.setMaxInactiveInterval(int)
b) Utilizando la etiqueta <session-config> dentro del archivo web.xml : ... <session-config> <session-timeout>30</session-timeout> </session-config> ...
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
57
5 Cookies
Las sesiones vistas anteriormente basan su funcionamiento en los cookies. Cuando se hace uso de la interfaz HttpSession de forma interna y totalmente transparente al programador se est haciendo uso de los cookies. De hecho, cuando a travs de una pgina JSP se comienza una sesin, se crea un cookie llamado JSSESIONID. La diferencia es que este cookie es temporal y durar el tiempo que permanezca el navegador ejecutndose, siendo borrada cuando el usuario cierre el navegador. El objetivo de utilizar cookies es poder reconocer al usuario en el momento en el que se conecta al servidor. Una de las pginas que recoge la peticin del usuario puede comprobar si existe un cookie que ha dejado anteriormente; si es as, sabe que ese usuario ya ha visitado ese website y, por lo tanto, puede leer valores que le identifiquen. Otro de los usos de los cookies es ofrecer una personalizacin al usuario. En muchos sitos web es posible elegir el color de fondo, el tipo de letra utilizado, etc. Estos valores pueden ser almacenados en cookies de forma que cuando acceda de nuevo a la web y se compruebe la existencia de esos valores, sern recuperados para utilizarlos en la personalizacin de la pgina tal y cmo el usuario estableci en su momento. Un ejemplo que se ha podido encontrar en muchas webs es en el momento de realizar un registro o solicitar el alta en un rea restringida, ya que en muchas ocasiones existe un checkbox que cuando se selecciona permite recordar el nombre de usuario a falta de que slo se escriba la clave. Utilizando tambin el identificador idSession que se genera en una sesin como ya hemos visto y guardndolo en el cookie, se pueden mostrar mensajes personalizados en el momento en el que el usuario acceda de nuevo al website. Para trabajar con cookies, se utiliza la clase Cookie que est disponible en paquete javax.servlet.http. Por medio de esta clase, se pueden crear cookies, establecer sus valores y nombres, alguna de sus propiedades, eliminarlas, leer valores que almacenan, etc.
Tambin, es posible crear cookies con contenido que se genere de forma dinmica. El siguiente cdigo muestra un cookie que guarda un texto que est concatenado a la fecha/hora en ese momento:
CIBERTEC
CARRERAS PROFESIONALES
58
<%@page contentType="text/html; charset=iso-8859-1" session="true" language="java" import="java.util.*" %> <% Cookie miCookie=null; Date fecha=new Date(); String texto= Este es el texto que vamos a guardar en el cookie+fecha; miCookie=new Cookie(nombre,texto); %> En esta ocasin, el contenido del valor a guardar en el cookie est en la variable texto. Tambin, se pueden guardar valores o datos que provengan de pginas anteriores y que hayan sido introducidas a travs de un formulario:
<% Cookie miCookie=null; String ciudad= request.getParameter(formCiudad); miCookie=new Cookie(ciudadFavorita,ciudad); %> Una vez que se ha creado un cookie, es necesario establecer una serie de atributos para poder ser utilizado. El primero de esos atributos es el que se conoce como tiempo de vida. Por defecto, cuando creamos un cookie, se mantiene mientras dura la ejecucin del navegador. Si el usuario cierra el navegador, los cookies que no tengan establecido un tiempo de vida sern destruidos. Por tanto, si se quiere que un cookie dure ms tiempo y est disponible para otras situaciones, es necesario establecer un valor de tiempo (en segundos) que ser la duracin o tiempo de vida del cookie. Para establecer este atributo, se utiliza el mtodo setMaxAge(). El siguiente ejemplo establece un tiempo de 31 das de vida para el cookie unCookie:
<% unCookie.setMaxAge(60*60*24*31); %> Si se utiliza un valor positivo, el cookie ser destruido despus de haber pasado ese tiempo; si el valor es negativo, el cookie no ser almacenado y se borrar cuando el usuario cierre el navegador. Por ltimo, si el valor que se establece como tiempo es cero, el cookie ser borrado. Otro de los atributos que se incluye cuando se crea un cookie es el path desde el que ser visto, es decir, si el valor del path es / (raz), quiere decir que en todo el site se podr utilizar ese cookie, pero si el valor es /datos, quiere decir que el valor del cookie slo ser visible dentro del directorio datos. Este atributo se establece mediante el mtodo setPath().
<% unCookie.setPath(/);
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
59
<% out.println(cookie visible en: +unCookie.getPath()); %> Existe un mtodo dentro de la clase Cookie que permite establecer el dominio desde el cual se ha generado el cookie. Este mtodo tiene su significado, porque un navegador slo enva al servidor los cookies que coinciden con el dominio del servidor que los envi. Si en alguna ocasin se requiere que estn disponibles desde otros subdominios, se especifica con el mtodo setDomain(), Loa anterior se aplica si existe el servidor web en la pgina www.paginasjsp.com, pero al mismo tiempo tambin existen otros subdominios como usuario1.paginasjsp.com, usuario2.paginasjsp.com, etc. Si no se establece la propiedad domain, se entiende que el cookie ser visto slo desde el dominio que lo cre;sin embargo, si se especifica un nombre de dominio, se entender que el cookie ser visto en aquellos dominios que contengan el nombre especificado. En el siguiente ejemplo, se hace que el cookie definido en el objeto unCookie est disponible para todos los dominios que contengan el nombre .paginasjsp.com. Un nombre de dominio debe comenzar por un punto.
<% unCookie.setDomain(.paginasjsp.com); %> Igualmente, para conocer el dominio sobre el que acta el cookie, basta con utilizar el mtodo getDomain() para obtener esa informacin. Una vez que se ha creado el objeto Cookie, y se ha establecido todos los atributos necesarios es el momento de crear realmente, ya que hasta ahora slo se tena un objeto que representa ese cookie. Para crear el archivo cookie real, se utiliza el mtodo addCookie() de la interfaz HttpServletResponse:
<% response.addCookie(unCookie); %> Una vez ejecutada esta lnea es cuando el cookie existe en el disco del cliente que ha accedido a la pgina JSP. Es importante sealar que si no se ejecuta esta ltima lnea, el cookie no habr sido grabado en el disco y, por lo tanto, cualquier aplicacin o pgina que espere encontrar dicho cookie no lo encontrar.
CIBERTEC
CARRERAS PROFESIONALES
60
<% Cookie [] todosLosCookies=request.getCookies(); /* El siguiente paso es crear un bucle que vaya leyendo todos los cookies. */ for(int i=0;i<todosLosCookies.length;i++) { Cookie unCookie=todosLosCookies[i]; /* A continuacin se compara los nombres de cada uno de los cookies con el que se est buscando. Si se encuentra un cookie con ese nombre se ha dado con el que se est buscando, de forma que se sale del bucle mediante break. */ if(unCookie.getName().equals(nombre)) break; } /* Una vez localizado tan slo queda utilizar los mtodos apropiados para obtener la informacin necesaria que contiene. */ out.println(Nombre: +unCookie.getName()+<BR>); out.println(Valor: +unCookie.getValue()+<BR>); out.println(Path: +unCookie.getPath()+<BR>); out.println(Tiempo de vida:+unCookie.getMaxAge()+<BR>); out.println(Dominio: +unCookie.getDomain()+<BR>); %>
<%@ page session="true" import="java.util.*"%> <% String usuario = ""; String fechaUltimoAcceso = ""; /*Bsqueda del posible cookie si existe para recuperar su valor y ser mostrado en el campo usuario */
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
61
Cookie[] todosLosCookies = request.getCookies(); for (int i=0; i<todosLosCookies.length; i++) { Cookie unCookie = todosLosCookies[i]; if (unCookie.getName().equals("cokieUsu")) { usuario = unCookie.getValue(); } } /* Para mostrar la fecha del ltimo acceso a la pgina. Para ver si el cookie que almacena la fecha existe, se busca en los cookies existentes. */ for (int i=0; i<todosLosCookies.length; i++) { Cookie unCookie = todosLosCookies[i]; if (unCookie.getName().equals("ultimoAcceso")) { fechaUltimoAcceso = unCookie.getValue(); } } /* Se comprueba que la variable es igual a vaco, es decir no hay ningn cookie llamado ultimoAcceso, por lo que se recupera la fecha, y se guarda en un nuevo cookie. */ if (fechaUltimoAcceso.equals("")) { Date fechaActual = new Date(); fechaUltimoAcceso = fechaActual.toString(); Cookie cookieFecha = new Cookie("ultimoAcceso",fechaUltimoAcceso); cookieFecha.setPath("/"); cookieFecha.setMaxAge(60*60*24); response.addCookie(cookieFecha); } %>
<html> <head><title>Proceso de login</title> </head> <body> <b>PROCESO DE IDENTIFICACIN</b> <br>ltima vez que accedi a esta pgina:<br><%=fechaUltimoAcceso%> <p> <% if (request.getParameter("error") != null) { out.println(request.getParameter("error")); } %> <form action="checklogin.jsp" method="post"> usuario: <input type="text" name="usuario" size="20" value="<%=usuario%>"><br> clave: <input type="password" name="clave" size="20"><br> Recordar mi usuario: <input type="checkbox" name="recordarUsuario" value="on"><br>
CIBERTEC
CARRERAS PROFESIONALES
62
El siguiente paso es modificar la pgina checklogin.jsp que recoge el usuario y clave introducidos y, por lo tanto, ahora tambin la nueva opcin de Recordar mi usuario. Dentro de la condicin que se cumple si el usuario y la clave son correctos, y despus de crear la sesin, escribimos el cdigo que crear el cookie con el usuario. El primer paso es comprobar que el usuario ha activado esta opcin, es decir, ha seleccionado el checkbox. Tambin, se realiza la comprobacin de que el campo recordarUsuario no llegue con el valor nulo y produzca un error en la aplicacin, en caso de que el usuario deje sin seleccionar el checkbox:
<%@ page session="true" import="java.util.*"%> <% String usuario = ""; String clave = ""; if (request.getParameter("usuario") != null) usuario = request.getParameter("usuario"); if (request.getParameter("clave") != null) clave = request.getParameter("clave"); if (usuario.equals("spiderman") && clave.equals("librojsp")) { out.println("checkbox: " + request.getParameter("recordarUsuario") + "<br>"); HttpSession sesionOk = request.getSession(); sesionOk.setAttribute("usuario",usuario); if ((request.getParameter("recordarUsuario") != null) && (request.getParameter("recordarUsuario").equals("on"))) { out.println("entra"); Cookie cookieUsuario = new Cookie ("cokieUsu",usuario); cookieUsuario.setPath("/"); cookieUsuario.setMaxAge(60*60*24); response.addCookie(cookieUsuario); } /* Se realiza un proceso similar a la creacin de cookie de recordar el usuario. En este caso se trata de crear un nuevo cookie con el nuevo valor de la fecha y guardarlo con el mismo nombre. De esta forma ser borrado el anterior y prevalecer el valor del ltimo. */ Date fechaActual = new Date(); String fechaUltimoAcceso = fechaActual.toString(); Cookie cookieFecha = new
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
63
Cookie("ultimoAcceso",fechaUltimoAcceso); cookieFecha.setPath("/"); cookieFecha.setMaxAge(60*60*24); response.addCookie(cookieFecha); %> <jsp:forward page="menu.jsp" /> <% } else { %> <jsp:forward page="login.jsp"> <jsp:param name="error" value="Usuario y/o clave incorrectos.<br>Vuelve a intentarlo."/> </jsp:forward> <% } %>
Se seala errneamente que los cookies estn envueltos con la intromisin de la privacidad de los usuarios por parte de determinados websites e incluso con la seguridad. Estos son pequeos archivos de texto que poco dao pueden causar, y que como es lgico no pueden contener ningn cdigo ejecutable. Desde el punto de vista es cierto que determinados sites pueden saber qu clase de palabras se consultan en un buscador o cul es nuestra frecuencia de visita a un determinado web. En cualquier caso, son datos que no descubren la verdadera identidad del usuario, aunque si bien por este motivo, muchos de ellos deciden desactivar la recepcin de cookies. Si esto es as, las pginas en las que se utilicen cookies no funcionarn de forma correcta al no permitir ser recibidas por los navegadores.
CIBERTEC
CARRERAS PROFESIONALES
64
Preguntas de Certificacin
1 Cul de las siguientes interfaz o clases son usadas para recibir una sesin asociado con un usuario? (Seleccione uno.) a GenericServlet b ServletConfig c ServletContext d HttpServlet e HttpServletRequest f HttpServletResponse 2. Cul de los siguientes bloques de cdigos es insertado que permitir contar correctamente la cantidad de pedidos que ha hecho un usuario? (Seleccione uno.) a HttpSession session = request.getSession(); int count = session.getAttribute("count"); session.setAttribute("count", count++); b HttpSession session = request.getSession(); int count = (int) session.getAttribute("count"); session.setAttribute("count", count++); c HttpSession session = request.getSession(); int count = ((Integer) session.getAttribute("count")).intValue(); session.setAttribute("count", count++); d HttpSession session = request.getSession(); int count = ((Integer) session.getAttribute("count")).intValue(); session.setAttribute("count", new Integer(count++)); 3. Cul de los siguientes mtodos se invoca cuando un atributo implementa HttpSessionBindingListener al destruitr la sesin? (Seleccione uno) a sessionDestroyed b valueUnbound c attributeRemoved d sessionInvalidated de una sesin
4 Cul de los siguientes mtodos es invocado dentro de un atributo de una sesin que implementa apropiadamente las interfaz al destruir la sesin? (Seleccione uno.) a sessionDestroyed of HttpSessionListener b attributeRemoved of HttpSessionAttributeListener c valueUnbound of HttpSessionBindingListener d sessionWillPassivate of HttpSessionActivationListener
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
65
Resumen
El tiempo que durar la sesin de inactividad en el Tomcat se registra en el web.xml en minutos. <session-config> <session-timeout>30</session-timeout> </session-config> Mtodo para destruir una sesin programaticamente session.sinvalidate(); Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina.
http://java.sun.com/products/servlet/2.2/javadoc/javax/servlet/http/HttpSession.html
CIBERTEC
CARRERAS PROFESIONALES
66
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
67
UNIDAD DE APRENDIZAJE
1
SEMANA
5
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
Patrones de Diseo de Software
ACTIVIDADES PROPUESTAS
Los alumnos realizarn un carrito de compras en web.
CIBERTEC
CARRERAS PROFESIONALES
68
1.1 Historia
El concepto de "patrn de diseo" que tenemos en Ingeniera del Software se ha tomado prestado de la arquitectura. En 1977, se publica el libro "A Pattern Language: Towns/Building/Construction", de Christopher Alexander, Sara Ishikawa, Murray Silverstein, Max Jacobson, Ingrid Fiksdahl-King y Shlomo Angel, Oxford University Press. Contiene numerosos patrones con una notacin especfica de Alexander. Alexander comenta que cada patrn describe un problema que ocurre una y otra vez en nuestro entorno, para describir despus el ncleo de la solucin a ese problema, de tal manera que esa solucin pueda ser usada ms de un milln de veces sin hacerlo siquiera dos veces de la misma forma. El patrn es un esquema de solucin que se aplica a un tipo de problema. Esta aplicacin del patrn no es mecnica, sino que requiere de adaptacin y matices. Por ello, dice Alexander que los numerosos usos de un patrn no se repiten dos veces de la misma forma. La idea de patrones de diseo estaba "en el aire", la prueba es que numerosos diseadores se dirigieron a aplicar las ideas de Alexander a su contexto. El catlogo ms famoso de patrones se encuentra en Design Patterns: Elements of Reusable Object-Oriented Software, de Erich Gamma, Richard Helm, Ralph Johnson y John Vlissides, 1995, Addison-Wesley, tambin conocido como el libro GOF (Gang-OfFour). Siguiendo el libro de GOF, los patrones se clasifican segn el propsito para el que han sido definidos:
Creacionales: solucionan problemas de creacin de instancias. Nos ayudan a encapsular y abstraer dicha creacin. Estructurales: solucionan problemas de composicin (agregacin) de clases y objetos. De Comportamiento: soluciones respecto a la interaccin y responsabilidades entre clases y objetos, as como los algoritmos que encapsulan.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
69
De Conducta Interprete
Plantilla
Objeto
Cadena de Responsabilidad, Adaptador Comando (orden), (objetos), Puente, Fbrica, Iterador, Composicin, Constructor, Intermediario, Prototipo, Singleton Decorador, Observador, Estado, Fachada, Flyweight Estrategia, Visitante, Memoria
CIBERTEC
CARRERAS PROFESIONALES
70
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
71
Capa de Presentacin
Decorating Filter / Intercepting Filter Un objeto que est entre el cliente y los componentes web. Este procesa las peticiones y las respuestas. Un objeto que acepta todos los requerimientos de un cliente y los direcciona a manejadores apropiados. El patrn Front Controller podra dividir la funcionalidad en 2 diferentes objetos: el Front Controller y el Dispatcher. En ese caso, El Front Controller acepta todos los requerimientos de un cliente y realiza la autenticacin, y el Dispatcher direcciona los requerimientos a manejadores apropiada. Un objeto helper que encapsula la lgica de acceso a datos en beneficio de los componentes de la presentacin. Por ejemplo, los JavaBeans pueden ser usados como patrn View Helper para las pginas JSP. Un objeto vista que est compuesto de otros objetos vista. Por ejemplo, una pgina JSP que incluye otras pginas JSP y HTML usando la directiva include o el action include es un patrn Composite View. Es como el patrn de diseo MVC con el Controlador actuando como Front Controller pero con una cosa importante: aqu el Dispatcher (el cual es parte del Front Controller) usa View Helpers a gran escala y ayuda en el manejo de la vista. Es como el patrn de diseo MVC con el controlador actuando como Front Controller pero con un asunto importante: aqu el Dispatcher (el cual es parte del Front Controller) no usa View Helpers y realiza muy poco trabajo en el manejo de la vista. El manejo de la vista es manejado por los mismos componentes de la Vista.
View Helper
Composite view
Service To Worker
Dispatcher View
Capa de Negocios
Un objeto que reside en la capa de presentacin y en beneficio de los otros componentes de la capa de presentacin llama a mtodos remotos en los objetos de la capa de negocios.
Business Delegate
CIBERTEC
CARRERAS PROFESIONALES
72
Replicate Object
datos sobre la red. El uso de un bean de sesin como una fachada (facade) para encapsular la complejidad de las interacciones entre los Entity objetos de negocio y participantes en un flujo de trabajo. El Session Faade maneja los objetos de negocio y proporciona un servicio de acceso uniforme a los clientes. Un bean entidad que es construido o es agregado a otros beans de entidad Un objeto que reside en la capa de negocios y crea Value Objets cuando es requerido Es un objeto que maneja la ejecucin de Page-by-Page consultas SQL, cach y procesamiento del resultado. Usualmente implementado como beans de sesin Consiste en utilizar un objeto Service Locutor para abstraer toda la utilizacin JNDI y para ocultar las complejidades de la creacin del contexto inicial, de bsqueda de objetos home EJB y recreacin de objetos EJB. Varios clientes pueden reutilizar el objeto Service Locutor para reducir la complejidad del cdigo, proporcionando un punto de control.
Service Locator
Capa de Integracin
Consiste en utilizar un objeto de acceso a datos para abstraer y encapsular todos los accesos a la fuente de datos. El DAO maneja la conexin con la fuente de datos para obtener y almacenar datos. Se utiliza para recibir peticiones y mensajes asncronos de los clientes. Cuando se recibe un mensaje, el Service Activator localiza e invoca a los mtodos de los componentes de negocio necesarios para cumplir la peticin de forma asncrona.
Service Activator
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
73
Nombre Propsito o finalidad Sinnimos (otros nombres por los que puede ser conocido) Problema al que es aplicable Estructura (diagrama de clases) Participantes (responsabilidad de cada clase) Colaboraciones (diagrama de interacciones) Implementacin (consejos, notas y ejemplos) Otros patrones con los que est relacionado
Ventajas de los patrones: La clave para la reutilizacin es anticiparse a los nuevos requisitos y cambios, de modo que los sistemas evolucionen de forma adecuada. Cada patrn permite que algunos aspectos de la estructura del sistema puedan cambiar independientemente de otros aspectos. Facilitan la reusabilidad, extensibilidad y mantenimiento. Un patrn es un esquema o microarquitectura que supone una solucin a problemas (dominios de aplicacin) semejantes (aunque los dominios de problema pueden ser muy diferentes e ir desde una aplicacin CAD a un cuadro de mando empresarial). Interesa constatar una vez ms la vieja distincin entre dominio del problema (donde aparecen las clases propias del dominio, como cuenta, empleado, coche o beneficiario) y el dominio de la solucin o aplicacin (donde adems aparecen clases como ventana, men, contenedor o listener). Los patrones son patrones del dominio de la solucin. Tambin, conviene distinguir entre un patrn y una arquitectura global del sistema. Por decirlo en breve, es la misma distancia que hay entre el diseo de un componente (o mdulo) y el anlisis del sistema. Es la diferencia que hay entre el aspecto micro y el macro; por ello, en ocasiones, se denomina a los patrones como "microarquitecturas". En resumen, un patrn es el denominador comn, una estructura comn que tienen aplicaciones semejantes. Esto tambin ocurre en otros rdenes de la vida. Por ejemplo, en nuestra vida cotidiana, aplicamos a menudo el esquema saludopresentacin-mensaje-despedida en ocasiones diversas.
CIBERTEC
CARRERAS PROFESIONALES
74
Un ejemplo: supongamos que un objeto debe ordenar una estructura de datos. Puede delegar en otro objeto el mtodo de comparacin. En Java, tenemos un caso: la clase Collections tiene el mtodo esttico sort(). Desde este mtodo se delega en un comparador para establecer el orden:
Collections tiene el mtodo sort() con un algoritmo de ordenacin. Comparador tiene el mtodo compare() que implementa el orden de comparacin.
import java.util.Comparator;
public class comparador implements Comparator { public int compare( Object o1, Object o2 ) { if ( ((ente)o1).obt_id() < ((ente)o2).obt_id() ) return 1; if ( ((ente)o1).obt_id() > ((ente)o2).obt_id() ) return 1; return 0; }
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
75
entonces la consecuencia ser que, cuando necesitemos cambiar la interfaz, tendremos que modificar trabajosamente los componentes de negocio. Mayor trabajo y ms riesgo de error. Se trata de realizar un diseo que desacople la vista del modelo, con la finalidad de mejorar la reusabilidad. De esta forma, las modificaciones en las vistas impactan en menor medida en la lgica de negocio o de datos. Elementos del patrn:
Modelo: datos y reglas de negocio Vista: muestra la informacin del modelo al usuario Controlador: gestiona las entradas del usuario
Un modelo puede tener diversas vistas, cada una con su correspondiente controlador. Un ejemplo clsico es el de la informacin de una base de datos, que se puede presentar de diversas formas: diagrama de tarta, de barras, tabular, etc. Veamos cada componente: 1. El modelo es el responsable de lo siguiente: o Acceder a la capa de almacenamiento de datos. Lo ideal es que el modelo sea independiente del sistema de almacenamiento. o Define las reglas de negocio (la funcionalidad del sistema). Un ejemplo de regla puede ser la siguiente: "Si la mercanca pedida no est en el almacn, consultar el tiempo de entrega estndar del proveedor". o Lleva un registro de las vistas y controladores del sistema. o Si estamos ante un modelo activo, notificar a las vistas los cambios que en los datos pueda producir un agente externo (por ejemplo, un archivo bacth que actualiza los datos, un temporizador que desencadena una insercin, etc).
2. El controlador es responsable de lo siguiente: o Recibe los eventos de entrada (un clic, un cambio en un campo de texto, etc.). o Contiene reglas de gestin de eventos, del tipo "SI Evento Z, entonces Accin W". Estas acciones pueden suponer peticiones al modelo o a las vistas. Una de estas peticiones a las vistas puede ser una llamada al mtodo "Actualizar()". Una peticin al modelo puede ser "Obtener_tiempo_de_entrega( nueva_orden_de_venta )".
CIBERTEC
CARRERAS PROFESIONALES
76
3. Las vistas son responsables de lo siguiente: o Recibir datos del modelo y los muestra al usuario. o Tienen un registro de su controlador asociado (normalmente porque adems lo instancia). o Pueden dar el servicio de "Actualizacin()", para que sea invocado por el controlador o por el modelo (cuando es un modelo activo que informa de los cambios en los datos producidos por otros agentes). Un ejemplo de MVC con un modelo pasivo (aquel que no notifica cambios en los datos) es la navegacin web, que responde a las entradas del usuario, pero no detecta los cambios en datos del servidor. El diagrama de secuencia:
Pasos: 1. El usuario introduce el evento. 2. El Controlador recibe el evento y lo traduce en una peticin al Modelo (aunque tambin puede llamar directamente a la vista). 3. El modelo (si es necesario) llama a la vista para su actualizacin. 4. Para cumplir con la actualizacin, la Vista puede solicitar datos al Modelo. 5. El Controlador recibe el control. Sin embargo, en su implementacin, existe una pequea dificultad: la mayor parte de las herramientas de desarrollo incorporan en las clases de la vista gran parte o todo el procesamiento de eventos, con lo que el controlador queda semioculto dentro de la vista. A pesar de ello, podemos acercarnos bastante al patrn. En el siguiente ejemplo en Java, el objeto vista es un Applet AWT. El controlador (controlador.java) puede gestionar el clic en un botn, de tal forma que recoge datos por medio del Modelo (model.cargar_texto()) y los manda a la Vista (el applet) para su actualizacin (vista.mostrar_texto( )):
/*************************************************************** * Responde al click en botn "abrir" La respuesta al evento es hacer que se abra en la vista el archivo correspondiente a la referencia seleccionada en el combo box **************************************************************** /
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
77
void b_abrir_actionPerformed(ActionEvent e) { String texto_archivo = model.cargar_texto( indice_ref ); // Obtener texto de archivo /*** Si la carga de archivo es ok, lo muestro. Si no, aviso de error ****/ if (texto_archivo != null) { vista.mostrar_texto(texto_archivo); // Mostrar texto vista.mostrar_aviso("Carga de " + path + " completada."); } else vista.mostrar_aviso("Error en la carga de " + path); }
Factora simple: una clase que crea objetos de otras clases. No delega en otras subclases y sus mtodos pueden ser estticos. Factory Method: Se define como una interfaz para crear objetos, como en el Abstract Factory, pero se delega a las subclases implementar la creacin en concreto Abstract Factory: Nos da una interfaz para crear objetos de alguna familia, sin especificar la clase en concreto.
CIBERTEC
CARRERAS PROFESIONALES
78
Estos patrones entran en la categora de patrones de creacin [GoF95], la cual comparten con otros patrones tales como el Singleton, Builder y Prototype [GoF95]. Tienen la responsabilidad de crear instancias de objetos de otras clases. Tienen, adems, la responsabilidad y el conocimiento necesario para encapsular la forma en que se crean determinados tipos de objetos en una aplicacin. Existen diferentes patrones de tipo Factory. Factoria simple Como en todas las factoras, tenemos las clases instanciadas (JuegoDelDado y JuegoDeMoneda) que se relacionan con una clase madre (extends) o con una interfaz lgica (implements). En nuestro caso, usamos interfaz. A continuacin, puede ver un sencillo ejemplo en el que cada juego implementa el mtodo lanzar(): el juego del dado muestra un nmero aleatorio del 1 al 6 y el de la moneda 'Cara' o 'Cruz':
public interface Juego { void lanzar(); } import java.util.Random; public class JuegoDelDado implements Juego { public void lanzar() { Random ran = new Random(); int resul = ran.nextInt(6) + 1; System.out.println( resul );
// Ale
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
79
} } import java.util.Random; public class JuegoDeMoneda implements Juego { public void lanzar() { Random ran = new Random(); if ( ran.nextBoolean() ) System.out.println( "Cara" ); else System.out.println( "Cruz" ); } }
La clase FactoriaJuegos es nica. No delega en una subclase la creacin de instancias (a diferencia de Factory Method). Esta factoria es muy sencilla: en funcin del argumento, crea un juego u otro:
public class FactoriaJuegos { public static Juego getJuego( String nombreJuego ) { if ( nombreJuego.equals("JuegoDelDado") ) return new JuegoDelDado(); else { if ( nombreJuego.equals("JuegoDeMoneda") ) return new JuegoDeMoneda(); else return null; } }
CIBERTEC
CARRERAS PROFESIONALES
80
Lo esencial de la clase Factora es que oculta a la clase cliente (Inicio.java) la complejidad de crear un objeto. Encapsula la creacion de la instancia. La clase cliente (Inicio.java) llama al mtodo esttico getJuego() de la factora, para que le devuelva el juego sealado en el argumento. Introduce todos los juegos en un vector y, a continuacin, le dice a cada juego que juegue. El mtodo jugar() es un ejemplo de patrn 'estrategia': el metodo contiene un comportamiento genrico (en nuestro ejemplo, realiza dos lanzamientos para cada juego). El comportamiento especfico se define en funcin del objeto que se pasa como argumento. La parte que vara es el argumento, esta es la estrategia.
public class Inicio { public static void main(String[] args) { //// Crea un vector de juegos Vector vec = new Vector(); vec.add( FactoriaJuegos.getJuego( "JuegoDelDado") ); vec.add( FactoriaJuegos.getJuego( "JuegoDeMoneda") ); vec.add( FactoriaJuegos.getJuego( "JuegoDelDado") ); //// A cada juego del vector le dice que juegue for ( int i = 0; i < vec.size(); i++ ) { Juego j = (Juego) vec.get(i); if ( j != null ) jugar( j ); else System.out.println("Juego no encontrado"); } }
/*************************************************************** ********************* * Lanza dos veces **************************************************************** ******************/ public static void jugar( Juego juego ) { juego.lanzar(); juego.lanzar(); } }
Al recorrer el vector de juegos, vemos un ejemplo tpico de polimorfismo: a cada referencia del tipo Juego (usamos el mismo interfaz) le decimos que juegue, pero cada juego implementa su forma especfica de jugar (ms en concreto de lanzar). Es la idea de polimorfismo: una interfaz y mltiples implementaciones.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
81
Aunque la forma ms abstracta y profesional es usar en la factora newInstance() en funcin de los valores de un archivo .properties:
import java.io.*; public class FactoriaJuegos { // true=Carga de Properties desde archivo private static boolean propiedadesCargadas = false; // Propiedades private static java.util.Properties prop = new java.util.Properties(); /***************************************************** * Crea y devuelve el juego ***************************************************************/ public static Juego getJuego( String nombreJuego ) { try { //// La clase se consigue leyendo del archivo properties Class clase = Class.forName( getClase( nombreJuego ) ); //// Creo una instancia return (Juego) clase.newInstance(); } catch (ClassNotFoundException e) { // No existe la clase e.printStackTrace(); return null; } catch (Exception e) { // No puedo instanciar la clase e.printStackTrace(); return null; } } /*************************************************************** *********** * Lee un archivo properties donde se indica la clase que debe ser instanciada **************************************************************** ******************/
private static String getClase( String nombrePropiedad ) { try { //// Carga de propiedades desde archivo if ( !propiedadesCargadas ) {
CIBERTEC
CARRERAS PROFESIONALES
82
FileInputStream archivo = new FileInputStream( "src/factoriaJuegosNewInstanceProperties/propiedades.properties" ); prop.load( archivo ); // Cargo propiedades propiedadesCargadas = true; } //// Lectura de propiedad String nombreClase = prop.getProperty( nombrePropiedad, ""); if ( nombreClase.length() > 0) return nombreClase; return null; } catch ( FileNotFoundException e) { // No se puede encontrar archivo e.printStackTrace(); return null; } catch ( IOException e) { // Falla load() e.printStackTrace(); return null; } catch (Exception e) { e.printStackTrace(); return null; } }}
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
83
Resumen
El catlogo ms famoso de patrones se encuentra en Design Patterns: Elements of Reusable Object-Oriented Software, de Erich Gamma, Richard Helm, Ralph Johnson y John Vlissides, 1995, Addison-Wesley, tambin conocido como el libro GOF (Gang-Of-Four). Patrn Factory "Factora" En realidad, son una familia de patrones:
Factora simple: una clase que crea objetos de otras clases. No delega en otras subclases y sus mtodos pueden ser estticos. Factory Method: se define una interfaz para crear objetos, como en el Abstract Factory, pero se delega a las subclases implementar la creacin en concreto. Abstract Factory: da una interfaz para crear objetos de alguna familia, sin especificar la clase en concreto.
CIBERTEC
CARRERAS PROFESIONALES
84
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
85
UNIDAD DE APRENDIZAJE
1
SEMANA
6
SERVLET Y JSP
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos implementarn una aplicacin web en Java que realiza operaciones SELECT, INSERT, UPDATE, DELETE a la base de datos que contenga Servlets, JSP, JavaBeans y procese los objetos request y response.
TEMARIO
Patrn Data Access Object Carrito de Compras
ACTIVIDADES PROPUESTAS
Los alumnos realizarn programas utilizando el patrn DAO en la capa del Modelo de Datos.
CIBERTEC
CARRERAS PROFESIONALES
86
El acceso a los datos vara dependiendo de la fuente de los datos. El acceso al almacenamiento persistente, como una base de datos, vara en gran medida dependiendo del tipo de almacenamiento (bases de datos relacionales, bases de datos orientadas a objetos, archivos de texto, etc.) y de la implementacin del vendedor.
1.2
Problema
Muchas aplicaciones de la plataforma JEE en el mundo real necesitan utilizar datos persistentes en algn momento. Para muchas de ellas, este almacenamiento persistente se implementa utilizando diferentes mecanismos, y hay marcadas diferencias en los APIS utilizados para acceder a esos mecanismos de almacenamiento diferentes o datos residentes en sistemas diferentes. Tpicamente, las aplicaciones utilizan componentes distribuidos compartidos persistentes como los entity bean para representar la data persistente. Una aplicacin es considerada que emplear un bean managed persistence para sus entity beans cuando estas entity beans explcitamente accesen al almacn persistente, es decir, cuando el entity bean incluya cdigo para acceder directamente al almacn persistente. Una aplicacin con requerimientos simples no utilizara entity beans y, en vez de ellos, utilizara session beans o Servlets para acceder directamente al almacn persistente para recuperar y modificar datos. Las aplicaciones pueden usar la API-JDBC para acceder a los datos que residen en un RDBMS. Esta API permite el acceso estndar y la manipulacin de datos en un almacn persistente como una base de datos relacional. JDBC permite a las aplicaciones JEE utilizar sentencias SQL estndar. Sin embargo, las sentencias SQL podran variar dependiendo del DBMS.
1.3
Solucin
Utilizar un Data Access Object (DAO) para abstraer y encapsular todos los accesos a la fuente de datos. El DAO maneja la conexin con la fuente de datos para obtener y almacenar datos. El DAO implementa el mecanismo de acceso requerido para trabajar con la fuente de datos.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
87
El Objeto de Acceso a Datos (DAO) es el objeto principal de este patrn. La fuente de datos podra ser un almacn persistente como un RDBMS, un servicio externo como B2B, un servicio de negocios accesado va CORBA. Los componentes del negocio que cuentan con los objetos DAO utilizan una interfaz simple expuesta por el DAO para sus clientes. El DAO completamente oculta la implementacin de la fuente de datos y lo aparta de los clientes, debido a que la interfaz expuesta por el DAO no cambia cuando la implementacin de la fuente de datos cambia. Este patrn permite al DAO adaptarse a diferentes esquemas de almacenamiento sin afectar a sus clientes o componentes de negocio. DAO esencialmente acta como un adaptador entre el componente de negocio y la fuente de datos (data source).
1.4
Diagrama de Clases
CIBERTEC
CARRERAS PROFESIONALES
88
DAOs que necesita la aplicacin. En la siguiente figura, podemos ver el diagrama de clases para este caso:
En la siguiente figura, podemos ver el diagrama de clases para esta estrategia. En l vemos una fbrica base de DAOs, que es una clase abstracta que extienden e implementan las diferentes factoras concretas de DAOs para soportar el acceso especfico a la implementacin del almacenamiento.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
89
El cliente puede obtener una implementacin de la factora concreta del DAO como una RdbDAOFactory y utilizarla para obtener los DAOs concretos que funcionan en la implementacin del almacenamiento. Por ejemplo, el cliente puede obtener una RdbDAOFactory y utilizarlas para obtener DAOs especfcios como RdbCustomerDAO, RdbAccountDAO, etc. Los DAOs pueden extender e implementar una clase base genrica (mostradas como DAO1 y DAO2) que describa especficamente los requerimientos del DAO para el objeto de negocio que soporta. Cada DAO concreto es responsable de conectar con la fuente de datos y de obtener y manipular los datos para el objeto de negocio que soporta.
CIBERTEC
CARRERAS PROFESIONALES
90
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
91
Catlogo de Productos
CIBERTEC
CARRERAS PROFESIONALES
92
Carrito de Compras
venta_productomain.jsp
JSP que visualiza el catlogo de productos y a partir del cual se accede al carrito de compras.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
93
<%@page import="edu.cibertec.bean.*" %> <%@page import="edu.cibertec.util.*" %> <%@page import="edu.cibertec.admin.dao.*" %> <% DAOFactory daoFactory=DAOFactory.getDAOFactory(DAOFactory.MYSQL); ProductoDAO objProductoDAO =daoFactory.getProductoDAO(); //Para el Nro. de Pagina. int intPagina=0; //Si la pgina es nula entonces debe ser 1; intPagina=(request.getParameter("pagina")==null?1:Integer.parseInt(request.getPar ameter("pagina"))); Lista objLista=objProductoDAO.obtenerProductos(); //Nmero de registro por pginas. int iTamanoPagina=4; //iTamanoPagina=Integer.parseInt(String.valueOf(application.getAttribute("AppTaman oPagina"))); objLista.setTamPagina(iTamanoPagina); %> <html> <head> <meta http-equiv="Content-Language" content="es"> <meta http-equiv="Content-Type" content="text/html; charset=windows-1252"> <title>Carrito de Compras</title> </head> <body> <table border="1" width="100%" bordercolor='#000000' cellspacing="0" cellpadding="0"> <tr> <td width="100%" height="62"> <p><font size="5" face="GulimChe" color="#008000"><b><u>Portal de Ventas</u></b></font></td> </tr> <tr> <td width="100%" height="19"><b><font color="#800000">Lista de productos</font></b></td> </tr>
CIBERTEC
CARRERAS PROFESIONALES
94
<tr> <td width="100%" height="19"> <font size="1" face="Verdana">Pag: <%for(int i=1;i<=objLista.getNumPagina();i++){ if(i==intPagina){ out.print(i+" "); } else{%> <a href="venta_productomain.jsp?pagina=<%=i%>"><%=i%></a> <% } } %> </font> </td> </tr> <tr> <td width="100%" height="19"> <table border="1" bordercolor='#3399CC' width="100%" cellspacing="0" cellpadding="0"> <tr> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Cdigo</font></td> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Nombre</font></td> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Precio</font></td> <td width="25%" bgcolor="#000000"> </td> </tr> <%for(int i=objLista.getFirstElementPage(intPagina)1;i<=objLista.getLastElementPage(intPagina)-1;i++){%> <tr> <td width="25%"><%=((Producto)objLista.getElemento(i)).getCodigo()%></td> <td width="25%"><%=((Producto)objLista.getElemento(i)).getNombre()%></td> <td width="25%"><%=((Producto)objLista.getElemento(i)).getPrecio()%></td> <td width="25%"><A href="Carrito?accion=comprar&codigo=<%=((Producto)objLista.getElemento(i)).getC odigo()%>&nombre=<%=((Producto)objLista.getElemento(i)).getNombre()%>&precio =<%=((Producto)objLista.getElemento(i)).getPrecio()%>&cantidad=1">Comprar</A>< /td> </tr> <%}%> </table> </td> </tr> </table> </body> </html>
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
95
ServletCarrito.java
Implementa toda la funcionalidad base del Carrito de compras.
package edu.cibertec.admin.servlet; import javax.servlet.http.*; import java.util.*; public class ServletCarrito extends javax.servlet.http.HttpServlet { public void elimina( javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, java.io.IOException { // --- Para cdigo de producto. String sCodigo; // --- Para manejo de la sesin. HttpSession se = request.getSession(); sCodigo = request.getParameter("optCodigo"); // --- Para eliminar. if (sCodigo != null) { // --- Declaramos una tabla. Hashtable htCarrito; // --- Obtenemos la tabla a partir de la sesin. htCarrito = (Hashtable) (se.getAttribute("objCarrito")); // --- Elimino el elemento. htCarrito.remove(sCodigo); // --- Guardamos la tabla actualizada en la sesin. se.setAttribute("objCarrito", htCarrito); } // --- Mostramos el contenido del carrito. response.sendRedirect("venta_productocarrito.jsp"); } public void limpia( javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, java.io.IOException { // --- Para manejo de la sesin. HttpSession se = request.getSession(); // --- Declaramos una tabla. Hashtable htCarrito;
CIBERTEC
CARRERAS PROFESIONALES
96
public void service( javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, java.io.IOException { if (request.getParameter("accion").equals("eliminar")) { elimina(request, response); } else if (request.getParameter("accion").equals("limpiar")) { limpia(request, response); } else if (request.getParameter("accion").equals("recalcular")) { recalcular(request, response); } else { update(request, response); } } public void update( javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, java.io.IOException { // --- Declaramos variables para almacenar ls parmetros. String sCodigo = null; String sNombre = null; String sPrecio = null; String sCantidad = null; // --- Creamos una matriz para guardar valores. String saValores[] = new String[3]; // --- Recibimos los parmetros. sCodigo = request.getParameter("codigo"); sNombre = request.getParameter("nombre"); sPrecio = request.getParameter("precio"); sCantidad = request.getParameter("cantidad"); // --- Ingresamos valores a la matriz. saValores[0] = sNombre; saValores[1] = sPrecio; saValores[2] = sCantidad; // --- Para manejar la sesin. HttpSession se = request.getSession(); // --- Verificamos si existe el carrito if (se.getAttribute("objCarrito") == null) { // --- Como no existe creamos la tabla. Hashtable htCarrito = new Hashtable(); // --- Guardamos valores en la tabla. htCarrito.put(sCodigo, saValores);
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
97
// --- Guardamos la tabla con nuevas filas en la sesin. se.setAttribute("objCarrito", htCarrito); } //Mostramos el contenido del carrito. response.sendRedirect("venta_productocarrito.jsp"); } public void recalcular( javax.servlet.http.HttpServletRequest request, javax.servlet.http.HttpServletResponse response) throws javax.servlet.ServletException, java.io.IOException { // --- Para manejo de la sesin. HttpSession se = request.getSession(); // --- Declaramos una tabla. Hashtable htCarrito; // --- Obtenemos la tabla a partir de la sesin. htCarrito = (Hashtable) (se.getAttribute("objCarrito")); int valor = 0; Enumeration productos = htCarrito.keys(); while (productos.hasMoreElements()) { String sClave = (String) productos.nextElement(); String saValores[] = (String[]) htCarrito.get(sClave); // --- obtenemos la cantidad ingresada en la caja de texto String nuevaCantidad = request.getParameter("txt" + sClave); boolean flag = false; try { int x = Integer.valueOf(nuevaCantidad).intValue(); flag = true; } catch (Exception e) { } // --- si no se ingreso cantidad, se ingresaron letras o se ingreso el nmero cero // --- se coloca por defecto el nmero uno en el campo cantidad if (nuevaCantidad.equals("") || nuevaCantidad.equals("0") || flag == false) valor = 1; else valor = Integer.valueOf(nuevaCantidad).intValue(); // --- cambiamos el valor del campo cantidad saValores[2] = "" + valor;
CIBERTEC
CARRERAS PROFESIONALES
98
// --- Guardamos los nuevos valores en la tabla. htCarrito.put(sClave, saValores); } // --- Guardamos la tabla actualizada en la sesin. se.setAttribute("objCarrito", htCarrito); // --- Mostramos el contenido del carrito. response.sendRedirect("venta_productocarrito.jsp"); } }
venta_productocarrito.jsp
JSP que visualiza el carrito de compras e invoca al ServletCarrito para administrar el carrito. <%@page import="java.util.*"%> <% //Para las claves. String sClave; //Para los datos. String saValores[]; //Valores Iniciales int iCantidad=0;double dPrecio=0;double dSubTotal=0;double dTotal=0;int iContador=0; String sColor=null; //Obtenemos el carrito lleno Hashtable htCarrito=(Hashtable)session.getAttribute("objCarrito"); HttpSession se= request.getSession(); if (se.getAttribute("objCarrito") == null) { response.sendRedirect("venta_productomain.jsp"); } Enumeration enIndices=htCarrito.keys(); %> <html> <head> <meta http-equiv="Content-Language" content="es"> <meta http-equiv="Content-Type" content="text/html; charset=windows-1252"> <title>Carrito de Compras</title> <script language="JavaScript"> <!-window.name='main' function f_eliminar() { document.forms[0].accion.value='eliminar'; document.forms[0].action='Carrito'; document.forms[0].target='main'; document.forms[0].submit(); }
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
99
dSubTotal= iCantidad * dPrecio; dTotal = dTotal + dSubTotal; iContador++; if (iContador% 2 == 0) sColor="white"; else sColor="#EFEFEF"; %> <tr bgcolor=<%=sColor%>> <td width="16%" align="center"><input type="radio" name="optCodigo" value="<%=sClave%>"></td> <td width="34%"><%=sClave%></td> <td width="25%"><%=saValores[0]%></td> <td width="25%"><%=saValores[1]%></td> <td width="25%"><input type="text" name="txt<%=sClave%>" value="<%=saValores[2]%>"></td> </tr> <%}%> </table> </form> </td> </tr> <tr> <td width="100%"> <table border="1" bordercolor='#000000' width="100%" cellspacing="0" cellpadding="0"> <tr> <td width="84%"> <p align="right">Total=</td> <td width="16%"><%=dTotal%></td> </tr> </table> </td> </tr> <tr><td width="100%"> <table border="1" bordercolor='#000000' width="100%" cellspacing="0" cellpadding="0"> <tr> <td width="25%"><a href="javascript:f_recalcular();">Recalcular</a></td> <td width="25%"><a href="javascript:f_eliminar();">Eliminar Producto</a></td> <td width="25%"><a href="Carrito?accion=limpiar">Limpiar Carrito</a></td> <td width="25%"><a href="venta_productomain.jsp"> Continuar Compra</a></td> <td width="25%"> Terminar Pedido</td> </tr> </table> </td> </tr><tr> <td width="100%"> </td> </tr> </table> </body> </html>
CIBERTEC
CARRERAS PROFESIONALES
100
function f_recalcular() { document.forms[0].accion.value='recalcular'; document.forms[0].action='Carrito'; document.forms[0].target='main'; document.forms[0].submit(); } // --> </script> </head> <body> <table border="1" bordercolor='#000000' width="100%" cellspacing="0" cellpadding="0"> <tr> <td width="100%" height="62"> <p><font size="5" face="GulimChe" color="#008000"><b><u>Portal de Ventas</u></b></font></td> </tr> <tr> <td width="100%" height="19"><b><font color="#800000">Carrito de Compras</font></b></td> </tr> <tr> <td width="100%" height="19"> <table border="1" bordercolor='#000000' width="100%" cellspacing="0" cellpadding="0"> <tr> <td width="100%"> <form > <input type="hidden" name="accion"> <table border="1" bordercolor='#3399CC' cellspacing="0" cellpadding="0" width="100%"> <tr> <td width="16%" bgcolor="#000000" align="center"><font color="#FFFFFF">Eliminar</font></td> <td width="34%" bgcolor="#000000"><font color="#FFFFFF">Cdigo</font></td> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Nombre</font></td> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Precio</font></td> <td width="25%" bgcolor="#000000"><font color="#FFFFFF">Cantidad</font></td> </tr> <% while(enIndices.hasMoreElements()) { sClave=(String)enIndices.nextElement(); saValores=(String[])htCarrito.get(sClave); iCantidad = Integer.parseInt(saValores[2]); dPrecio = Double.parseDouble(saValores[1]);
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
101
Resumen
Si desea saber ms acerca de estos temas, puede consultar las siguientes pginas. http://java.sun.com/j2se/1.4.2/docs/api/java/util/Hashtable.html Aqu hallar la descripcin de los mtodos del HashTable. http://es.wikipedia.org/wiki/Data_Access_Object Aqu hallar definiciones del patrn DAO.
CIBERTEC
CARRERAS PROFESIONALES
102
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
103
UNIDAD DE APRENDIZAJE
2
SEMANA
9
SEGURIDAD DE APLICACIONES WEB
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos crearn aplicaciones web en Java que utilice para seguridad: BASIC, DIGEST, FORM, and CLIENT-CERT, en funcin de un requerimiento dado.
TEMARIO
Autenticacin, Autorizacin, Confidencialidad, Integridad de la data Autenticacin basado en BASIC, DIGEST, FORM, and CLIENT-CER
ACTIVIDADES PROPUESTAS
Los alumnos crearn aplicaciones que permitan utilizar una seguridad adecuada.
CIBERTEC
CARRERAS PROFESIONALES
104
1 Mecanismos de seguridad
La seguridad en los Servlets se reduce a cuatro conceptos principales:
authentication: El proceso de autenticacin es el encargado de determinar que el interesado en acceder a los recursos es realmente quien dice que es. Lo ms comn es hacer que el usuario o la aplicacin enve un password que lo identifica. authorization: Es el proceso por el cual se restringe el acceso a ciertos recursos a un conjunto de clientes. En realidad, lo que se define es un conjunto de roles que pueden acceder a ciertos recursos y se asocia cada usuario a uno o ms roles. data integrity: Es el proceso que asegura que los datos que se reciben (tanto el cliente como el servidor) no han sido corrompidos. Generalmente se usan algoritmos de encriptacin para lograr esto. confidentiality: Es el proceso que intenta asegurar que solamente los clientes autorizados reciban la informacin. La manera de lograr esto puede ser a travs del uso de claves pblicas/privadas.
BASIC: Este es el mtodo ms simple, pero el ms inseguro. Cuando usamos este tipo de autenticacin, se abre una ventana en el browser del cliente para que este ingrese el usuario y el password. El password se enva al servidor codificado en Base64. FORM: Este mtodo es similar a BASIC, pero utiliza un formulario provisto por el programador. Este formulario tiene que cumplir los siguientes requisitos:
1. El atributo action debe ser j_security_check 2. El casillero de ingreso del usuario debe llamarse j_username 3. El casillero de ingreso del password debe llamarse j_password
<form method="post" action="j_security_check"> Usuario: <input type="text" name="j_username"> Password: <input type="text" name="j_password"> </form>
CLIENT-CERT: En este mtodo se usan certificados digitales. El browser debe tener instalado el certificado para que la comunicacin se pueda establecer.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
105
DIGEST: La autenticacin tambin es hecha utilizando usuario y password, pero el password es enviado encriptado de una forma mucho ms segura (por ejemplo, utilizando HTTPS client authentication).
3 Autorizacin
El primer paso es definir los roles que se encontrarn definidos dentro de nuestro sistema. Para el caso de Tomcat, estos se definen en el archivo tomcat-users.xml. Dentro de este archivo, cada usuario tiene asignado un usuario y password; y puede estar asociado a uno o ms roles.
<tomcat-users> <role rolename="Admin/"> <role rolename="Member/"> <role rolename="Guest/"> <user username="jose" password="duke" roles="Admin,"> </tomcat-users>
El siguiente paso es relacionar los roles definidos en el archivo tomcat-users.xml, con los roles a utilizar dentro del sistema. Para esto deberemos utilizar los elementos <security-role> y <role-name> dentro del archivo web.xml.
Una vez definidos los roles, se procede a establecer declarativamente el acceso a un conjunto de recursos en combinacin con los mtodos, a los cuales tendrn accesos determinados roles. El elemento principal es el <security-constraint>.
CIBERTEC
CARRERAS PROFESIONALES
106
El elemento <web-resource-name> es obligatorio y utilizado principalmente por los IDEs. El elemento <url-pattern> define el recurso al cual se limitar el acceso. Al menos se debe definir un <url-pattern>. El elemento <http-method> describe el mtodo HTTP que ser restringido para acceder al recurso definido en los <url-pattern>. Si no se define ningn <http-method>, entonces se restringir el acceso a todos los mtodos HTTP. El elemento <auth-constraint> es opcional y permite definir cules roles pueden llamar a los mtodos HTTP restringidos por los elementos <httpmethod>
Para permitir el acceso a todos los roles, se debe definir el elemento <authconstraint> de la siguiente forma:
<auth-constraint> <role-name>*</role-name> </auth-constraint> <auth-constraint></auth-constraint> En el caso de que se quiera bloquear el acceso a todos los roles, se deber definir el <auth-constraint> de la siguiente forma: <auth-constraint/> Es importante tener en cuenta que se puede tener ms de un elemento <webresource-collection> dentro de un <security-constraint>. En este caso, el elemento <auth-constraint> se aplica a todos los <web-resource-collection> definidos dentro de un <security-constraint>.
4 Autentificacin
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
107
Anteriormente mencionamos que la especificacin de servlet ofrece cuatro tipos de autentificacin: BASIC, FORM, CLIENT-CERT y DIGEST. Cada uno de estos tipos se puede definir en el web.xml de la siguiente forma:
<login-config> <auth-method>BASIC</auth-method> </login-config> <login-config> <auth-method>FORM</auth-method> <form-login-config> <form-login-page>/loginUsuario.html</form-login-page> <form-error-page>/loginError.html</form-error-page> </form-login-config> </login-config> <login-config> <auth-method>CLIENT-CERT</auth-method> </login-config> <login-config> <auth-method>DIGEST</auth-method> </login-config>
Un punto importante es que de los cuatro tipos de autentificacin, el tipo DIGEST es el nico que es opcional para los contenedores JEE.
5 Modos de transporte
La especificacin tambin define tres modos para indicar la seguridad del transporte de los datos a travs de la web.
NONE: Ningn tipo de cifrado, simplemente HTTP. CONFIDENTIAL: Define la confidencialidad de los datos. Es decir, que nadie pueda ver ni modificar los datos que se envan. INTEGRAL: Define la integridad de los datos; es decir que, aunque alguien pueda ver lo que se enva, se garantice que los datos no se modifiquen por el camino.
CIBERTEC
CARRERAS PROFESIONALES
108
<http-method>GET</http-method> <http-method>POST</http-method> </web-resource-collection> <auth-constraint> <role-name>Admin</role-name> <role-name>Member</role-name> </auth-constraint> <user-data-constraint> <transport-guarantee>CONFIDENTIAL</transportguarantee> </user-data-constraint> </security-constraint>
6 El mtodo isUserInRole()
La clase HTTPServletRequest tiene tres mtodos que estn asociados al manejo de la seguridad de manera programtica. Uno de estos es el mtodo isUserInRole. A travs del mtodo isUserInRole(), en vez de manejar el acceso a nivel de mtodo HTTP, ahora podremos autorizar el acceso a una parte del cdigo de nuestro mtodo basados en los roles del usuario. Antes de que se llame al mtodo isUserInRole(), el usuario necesita estar autenticado. En el caso de que llame al mtodo sobre un usuario que no ha autenticado, el contenedor retornar false. En el caso de que est autenticado, el contenedor toma el argumento del mtodo y compara este con los roles definidos para el usuario en el request. Si el usuario es ubicado a este rol, el contenedor retornar true.Por ejemplo, dentro del cdigo en el servlet:
if( request.isUserInRole(DukeChile)) { // Permiso para actualizar la informacion } else { // permiso solo para consultar la informacion }
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
109
El elemento <security-role-ref> permite relacionar un <role-name> definido programticamente (en el servlet) con uno que existe en el web.xml.
CIBERTEC
CARRERAS PROFESIONALES
110
Preguntas de Certificacin
1. Cul de las siguientes sentencias sigue los conceptos de integridad? (Seleccione uno.) a It guarantees that information is accessible only to certain users. b It guarantees that the information is kept in encrypted form on the server. c It guarantees that unintended parties cannot read the information during transmission between the client and the server. d It guarantees that the information is not altered during transmission between the client and the server. 2. Cul de los siguientes procesos se utiliza cuando un usuario ha accedido a un recurso en particular? (Seleccione uno.) a Authorization b Authentication c Confidentiality d Secrecy 3. Cul de los siguientes procesos se realiza despus de haber hecho la autorizacin? (Seleccione uno.) a Data validation b User authentication c Data encryption d Data compression 4. Cules de las siguientes acciones debern ejecutarse para prevenir que su sitio web sea atacado? (Seleccione tres.) a Block network traffic at all the ports except the HTTP port. b Audit the usage pattern of your server. c Audit the Servlet/JSP code. d Use HTTPS instead of HTTP.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
111
Resumen
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://courses.coreservlets.com/Course-Materials/scwcd.html En esta pgina, hallar descripcin sobre seguridad en Java web.
CIBERTEC
CARRERAS PROFESIONALES
112
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
113
UNIDAD DE APRENDIZAJE
3
SEMANA
10
STANDARD ACTIONS
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos crearn aplicaciones web en Java que utilice en los JSP la tecnologia Standard Actions, que utilice jsp:useBean, jsp:setProperty, jsp:getProperty, jsp:atribute en la implemetacin de cada JSP.
TEMARIO
jsp:useBean, jsp:setProperty, jsp:getProperty, jsp:atribute jsp:include, jsp:forward
ACTIVIDADES PROPUESTAS
Los alumnos crearn aplicaciones que permitan utilizar los tag estndares.
CIBERTEC
CARRERAS PROFESIONALES
114
1 La accin <jsp:include>
Esta accin incluye en una pgina la salida generada por otra perteneciente a la misma aplicacin web. La peticin se redirige a la pgina incluida, y la respuesta que genera se incluye en la generada por la principal. Su sintaxis es la siguiente:
El atributo flush especfica si el flujo de salida de la pgina principal debera ser enviado al cliente antes de enviar el de la pgina incluida. En JSP 1.2 este atributo es optativo, y su valor por defecto es false. En JSP 1.1 es obligatorio y siempre deba valer true (el forzar el vaciado de buffer era problemtico, porque una vez que ha sucedido esto no se pueden hacer redirecciones ni ir a pginas de error, ya que ya se han terminado de escribir las cabeceras). Esta accin presenta la ventaja sobre la directiva del mismo nombre de qu cambios en la pgina incluida no obligan a recompilar la "principal". No obstante, la pgina incluida solo tiene acceso al JspWriter de la "principal" y no puede generar cabeceras (por ejemplo, no puede crear cookies). Por defecto, la peticin que se le pasa a la pgina incluida es la original, pero se le pueden agregar parmetros adicionales, mediante la etiqueta jsp:param. Por ejemplo:
La accin <jsp:plugin>
Esta accin sirve para incluir, de manera portable e independiente del navegador, applets que utilicen alguna librera de Java 2 (Swing, colecciones, Java 2D), ya que las mquinas virtuales Java distribuidas con algunos navegadores relativamente antiguos (Explorer 5.x, Netscape 4.x) son de una versin anterior a Java 2.
La accin <jsp:forward>
Esta accin se utiliza para redirigir la peticin hacia otra pgina JSP que est en la misma aplicacin web que la actual. Un ejemplo de su sintaxis bsica es la siguiente:
<jsp:forward page="principal.jsp"/>
La salida generada hasta el momento por la pgina actual se descarta (se borra el buffer). En caso de que no se utilizara buffer de salida, se producira una excepcin.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
115
Al igual que en el caso de <jsp:include>, se pueden aadir parmetros a la peticin original para que los reciba la nueva pgina JSP:
2 Accin jsp:include
Esta accin nos permite insertar archivos en una pgina que est siendo generada. La sintaxis se parece a esto:
Al contrario que la directiva include, que inserta el archivo en el momento de la conversin de la pgina JSP a un Servlet, esta accin inserta el archivo en el momento en que la pgina es solicitada. Esto se paga un poco en la eficiencia, e imposibilita a la pgina incluida de contener cdigo JSP general (no puede seleccionar cabeceras HTTP, por ejemplo), pero se obtiene una significante flexibilidad. Por ejemplo, aqu tenemos una pgina JSP que inserta cuatro puntos diferentes dentro de una pgina web "What's New?". Cada vez que cambian las lneas de cabeceras, los autores slo tienen que actualizar los cuatro archivos, pero pueden dejar como estaba la pgina JSP principal.
WhatsNew.jsp
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML> <HEAD> <TITLE>What's New</TITLE> <LINK REL=STYLESHEET HREF="My-Style-Sheet.css" TYPE="text/css"> </HEAD> <BODY BGCOLOR="#FDF5E6" TEXT="#000000" LINK="#0000EE" VLINK="#551A8B" ALINK="#FF0000"> <CENTER> <table BORDER=5 BGCOLOR="#EF8429"> <TR><TH CLASS="TITLE"> What's New at JspNews.com</table> </CENTER>
CIBERTEC
CARRERAS PROFESIONALES
116
<P>
Accin jsp:useBean Esta accin nos permite cargar y utilizar un JavaBean en la pgina JSP. Esta es una capacidad muy til porque nos permite utilizar la reusabilidad de las clases Java sin sacrificar la conveniencia de aadir JSP sobre Servlets solitarios. La sintaxis ms simple para especificar que se debera usar un Bean es la siguiente:
Esto normalmente significa "ejemplariza un objeto de la clase especificada por class, y nelo a una variable con el nombre especificado por id". Sin embargo, tambin podemos especifar un atributo scope que hace que ese bean se asocie con ms de una sla pgina. En este caso, es til obtener referencias a los beans existentes, y la accin jsp:useBean especifica que se ejemplarizar un nuevo objeto si no existe uno con el mismo nombre y mbito. Ahora, una vez que tenemos un bean, podemos modificar sus propiedades mediante jsp:setProperty, o usando un scriptlet y llamando a un mtodo explcitamente sobre el objeto con el nombre de la variable especificada anteriormente mediante el atributo id. Recuerde que con los beans, cuando decimos "este bean tiene una propiedad del tipo X llamada foo", realmente queremos decir "Esta clase tiene un mtodo getFoo que devuelve algo del tipo X, y otro mtodo llamado setFoo que toma un X como un argumento". La accin jsp:setProperty se describe con ms detalle en la siguiente seccin, pero ahora observemos que podemos suministrar un valor explcito, dando un atributo param para decir que el valor est derivado del parmetro de la peticin nombrado, o slo lista las propiedades para indicar que el valor debera derivarse de los parmetros de la peticin con el mismo nombre que la propiedad. Leemos las propiedades existentes en una expresin o scriptlet JSP llamando al mtodo getXxx, o ms comnmente, usando la accin jsp:getProperty. Observe que la clase especificada por el bean debe estar en el path normal del servidor, no en la parte reservada que obtiene la recarga automtica cuando se modifican. Por ejemplo, en el Java web Server, l y todas las clases que usa deben ir en el directorio classes o estar en un archivo JAR en el directorio lib, no en el directorio Servlets.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
117
Aqu tenemos un ejemplo muy sencillo que carga un bean y selecciona y obtiene un sencillo parmetro String. BeanTest.jsp
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML> <HEAD> <TITLE>Reusing JavaBeans in JSP</TITLE> <LINK REL=STYLESHEET HREF="My-Style-Sheet.css" TYPE="text/css"> </HEAD> <BODY> <CENTER> <table BORDER=5> <TR><TH CLASS="TITLE"> Reusing JavaBeans in JSP</table> </CENTER> <P> <jsp:useBean id="test" class="hall.SimpleBean" /> <jsp:setProperty name="test" property="message" value="Hello WWW" /> <H1>Message: <I> <jsp:getProperty name="test" property="message" /> </I></H1> </BODY> </HTML>
SimpleBean.java Aqu est el cdigo fuente empleado para el bean usado en la pgina BeanTest. Tambin, se puede emplear as:
package hall; public class SimpleBean { private String message = "No message specified"; public String getMessage() { return(message); }
CIBERTEC
CARRERAS PROFESIONALES
118
Ms detalles de jsp:useBean
La forma ms sencilla de usar un bean es usar
para cargar el bean, luego usar jsp:setProperty y jsp:getProperty para modificar y recuperar propiedades del bean. Sin embargo, tenemos dos opciones. Primero, podemos usar un formato de contenedor, llamado
para indicar que la porcin Body slo se debera ejecutar cuando el bean es ejemplarizado por primera vez, no cuando un bean existente se encuentre y se utilice. Como se explica abajo, los bean pueden ser compartidos, por eso no todas las sentencias jsp:useBean resultan en la ejemplarizacin de un bean. Segundo, adems de id y class, hay otros tres atributos que podemos usar: scope, type, y beanName. Atributo id class Uso Da un nombre a la variable que referenciar el bean. Se usar un objeto bean anterior en lugar de ejemplarizar uno nuevo si se puede encontrar uno con el mismo id y scope. Designa el nombre completo del paquete del bean. Indica el contexto en el que el bean debera estar disponible. Hay cuatro posibles valores: page, request, session, y application. El valor por defecto, page, indica que el bean estar slo disponible para la pgina actual (almacenado en el PageContext de la pgina actual). Un valor de request indica que el bean slo est disponible para la peticin actual del cliente (almacenado en el objeto ServletRequest). Un valor de session indica que el objeto est disponible para todas las pginas durante el tiempo de vida de la HttpSession actual. Finalmente, un valor de application indica que est disponible para todas las pginas que compartan el mismo ServletContext. La razn de la importancia del mbito
scope
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
119
es que una entrada jsp:useBean slo resultar en la ejemplarizacin de un nuevo objeto si no haba objetos anteriores con el mismo id y scope. De otra forma, se usarn los objetos existentes, y cualquier elemento jsp:setParameter u otras entradas entre las etiquetas de inicio jsp:useBean y la etiqueta de final, sern ignoradas. type Especifica el tipo de la variable a la que se referir el objeto. Este debe corresponder con el nombre de la clase o ser una superclase o una interfaz que implemente la clase. Recuerde que el nombre de la variable se designa mediante el atributo id.
Da el nombre del bean, como lo suministraramos en el mtodo instantiate beanName de beans. Est permitido suministrar un type y un beanname, y omitir el atributo class.
3 Accin jsp:setProperty
Usamos jsp:setProperty para obtener valores de propiedades de los beans que se han referenciado anteriormente. Podemos hacer esto en dos contextos. Primero, podemos usar antes jsp:setProperty, pero fuera de un elemento jsp:useBean, de esta forma:
<jsp:useBean id="myName" ... /> ... <jsp:setProperty name="myName" property="someProperty" ... />
En este caso, el jsp:setProperty se ejecuta sin importar si se ha ejemplarizado un nuevo bean o se ha encontrado uno ya existente. Un segundo contexto en el que jsp:setProperty puede aparecer dentro del cuerpo de un elemento jsp:useBean, de esta forma:
<jsp:useBean id="myName" ... > ... <jsp:setProperty name="myName" property="someProperty" ... /> </jsp:useBean>
Aqu, el jsp:setProperty slo se ejecuta si se ha ejemplarizado un nuevo objeto, no si se encontr uno ya existente. Aqu tenemos los cuatro atributos posibles de jsp:setProperty:
Atributo name
Uso Este atibuto requerido designa el bean cuya propiedad va a ser seleccionada. El elemento jsp:useBean debe aparecer antes del elemento jsp:setProperty.
property Este atributo requerido indica la propiedad que queremos seleccionar. Sin
CIBERTEC
CARRERAS PROFESIONALES
120
embargo, hay un caso especial: un valor de "*" significa que todos los parmetros de la peticin cuyos nombres correspondan con nombres de propiedades del Bean sern pasados a los mtodos de seleccin apropiados. Este atributo opcional especifica el valor para la propiedad. Los valores string son convertidos automticamente a nmeros, boolean, Boolean, byte, Byte, char, y Character mediante el mtodo estndard valueOf en la fuente o la clase envolvente. Por ejemplo, un valor de "true" para una propiedad boolean o Boolean ser convertido mediante Boolean.valueOf, y un valor de "42" para una propiedad int o Integer ser convertido con Integer.valueOf. No podemos usar value y param juntos, pero s est permitido no usar ninguna. Este parmetro opcional designa el parmetro de la peticin del que se debera derivar la propiedad. Si la peticin actual no tiene dicho parmetro, no se hace nada: el sistema no pasa null al mtodo seleccionador de la propiedad. As, podemos dejar que el bean suministre los valores por defecto, sobreescribindolos slo cuando el parmetro dice que lo haga. Por ejemplo, el siguiente cdigo dice "selecciona el valor de la propiedad numberOfItems a cualquier valor que tenga el parmetro numItems de la peticin, si existe dicho parmetro; si no existe, no se hace nada" <jsp:setProperty name="orderBean" property="numberOfItems" param="numItems" /> Si omitimos tanto value como param, es lo mismo que si suministramos un nombre de parmetro que corresponde con el nombre de una propiedad. Podremos tomar esta idea de automaticidad usando el parmetro de la peticin cuyo nombre corresponde con la propiedad suministrada un nombre de propiedad de "*" y omitir tanto value como param. En este caso, el servidor itera sobre las propiedades disponibles y los parmetros de la peticin, correspondiendo aquellas con nombres idnticos. Aqu tenemos un ejemplo que usa un bean para crear una tabla de nmeros primos. Si hay un parmetro llamado numDigits en los datos de la peticin, se pasa dentro del bean a la propiedad numDigits. Al igual que en numPrimes. JspPrimes.jsp
value
param
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML> <HEAD> <TITLE>Reusing JavaBeans in JSP</TITLE> <LINK REL=STYLESHEET HREF="My-Style-Sheet.css" TYPE="text/css"> </HEAD> <BODY> <CENTER> <table BORDER=5> <TR><TH CLASS="TITLE">
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
121
Reusing JavaBeans in JSP</table> </CENTER> <P> <jsp:useBean id="primetable" class="hall.NumberedPrimes" /> <jsp:setProperty name="primetable" property="numDigits" /> <jsp:setProperty name="primetable" property="numPrimes" /> Some <jsp:getProperty name="primetable" property="numDigits" /> digit primes: <jsp:getProperty name="primetable" property="numberedList" /> </BODY> </HTML>
Accin jsp:getProperty
Este elemento recupera el valor de una propiedad del bean, lo convierte a un string e inserta el valor en la salida. Los dos atributos requeridos son name, el nombre de un bean referenciado anteriormente mediante jsp:useBean, y property, la propiedad cuyo valor debera ser insertado. Aqu tenemos un ejemplo:
<jsp:useBean id="itemBean" ... /> ... <UL> <LI>Number of items: <jsp:getProperty name="itemBean" property="numItems" /> <LI>Cost of each: <jsp:getProperty name="itemBean" property="unitCost" /> </UL>
Accin jsp:forward
Esta accin nos permite reenviar la peticin a otra pgina. Tiene un slo atributo, page, que debera consistir en una URL relativa. Este podra ser un valor esttico o podra ser calculado en el momento de la peticin, como en estos dos ejemplos:
CIBERTEC
CARRERAS PROFESIONALES
122
Accin jsp:plugin
Esta accin nos permite insertar un elemento OBJECT o EMBED especfico del navegador para especificar que el navegador debera ejecutar un applet usando el Plug-in Java.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
123
Preguntas de Certificacin
1. Cul de las siguientes sentencias es vlida para usar la accin <jsp:useBean>? (Seleccione una opcin.) a. <jsp:useBean id="address" class="AddressBean" /> b. <jsp:useBean name="address" class="AddressBean"/> c. <jsp:useBean bean="address" class="AddressBean" /> d. <jsp:useBean beanName="address" class="AddressBean" /> 2. Cul de las siguientes sentencias es el camino vlido para obtener el atributo de los beans? (Seleccione una opcin.) a. <jsp:useBean action="get" id="address" property="city" /> b. <jsp:getProperty id="address" property="city" /> c. <jsp:getProperty name="address" property="city" /> d. <jsp:getProperty bean="address" property="*" /> 3. Cules de los siguientes son usos vlidos de la accin <jsp:useBean>? (Seleccione dos opciones.) a. <jsp:useBean id="address" class="AddressBean" name="address" /> b. <jsp:useBean id="address" class="AddressBean" type="AddressBean" /> c. <jsp:useBean id="address" beanName="AddressBean" class="AddressBean" /> d. <jsp:useBean id="address" beanName="AddressBean" type="AddressBean" /> 4. Cul de los siguientes gets or sets del bean es vlido en el contenedor ServletContext? (Seleccione una opcin.) a. <jsp:useBean id="address" class="AddressBean" /> b. <jsp:useBean id="address" class="AddressBean" scope="application" /> c. <jsp:useBean id="address" class="AddressBean" scope="servlet" /> d. <jsp:useBean id="address" class="AddressBean" scope="session" /> e. Ninguna de las anteriores 5. Considere el siguiente cdigo: <html><body> <jsp:useBean id="address" class="AddressBean" scope="session" /> state = <jsp:getProperty name="address" property="state" /> </body></html> Cules de las siguientes lneas es vlida para remplazar la primera lnea de cdigo Java? (Seleccione tres lneas.) a. <% state = address.getState(); %> b. <% out.write("state = "); out.print(address.getState()); %> c. <% out.write("state = "); out.print(address.getstate()); %> d. <% out.print("state = " + address.getState()); %> e. state = <%= address.getState() %> f. state = <%! address.getState(); %>
CIBERTEC
CARRERAS PROFESIONALES
124
Resumen
Aqu est el cdigo fuente usado para el Bean usado en la pgina BeanTest. package hall; public class SimpleBean { private String message = "No message specified"; public String getMessage() { return(message); } public void setMessage(String message) { this.message = message; } }
La forma ms sencilla de usar un Bean es usar lo siguiente: <jsp:useBean id="name" class="package.class" />
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://www.java2s.com/Code/Java/JSP/JSPStandardActionssetproperty.htm Aqu hallar teora sobre JSP Standard Actions.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
125
UNIDAD DE APRENDIZAJE
4
SEMANA
11-12
CUSTOM TAG
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos crearn componentes mediante etiquetas personalizadas en la capa vista, que permitan mostrar de manera reutilizable la presentacin de la informacin.
TEMARIO
Custom Tag
ACTIVIDADES PROPUESTAS
Los alumnos crearn aplicaciones que permitan utilizar los tag personalizados.
CIBERTEC
CARRERAS PROFESIONALES
126
1 CustomTag
La tecnologa JSP nos permite introducir cdigo Java en pginas HTML. Este documento supone que est familiarizado con la creacin de pginas JSP y que tiene instalado Apache-Tomcat (en realidad, valdria cualquier servidor de JSP que cumpla la especificacion JSP 1.1, pero los ejemplos se han probado en Tomcat). Las JSP estn compuestas por varios tags o, como las denominaremos en adelante, etiquetas. Ejemplos de etiquetas JSP son setProperty, include, forward, usebean. Todas ellas encapsulan comportamientos de una manera sencilla. Nos detendremos en la etiqueta <jsp:useBean>. Esta etiqueta crea una instancia de una clase definida como un JavaBean. El uso de la etiqueta <jsp:useBean> en conjuncin con los JavaBeans hace que podamos separar la lgica de presentacin de la de proceso y consigue unos archivos JSP ms claros, y lo que es ms importante, el que escribe la JSP no tiene por qu tener unos conocimientos avanzados de programacin para crear pginas JSP de funcionalidades avanzadas. Pues bien, las JSP Custom Tags son un paso ms all. Con ellas se pueden aadir nuevas etiquetas a las ya existentes en JSP (eso s a partir de la especificacion JSP 1.1) que encapsulen comportamientos ms o menos complicados. Antes de seguir, vamos a revisar algunos conceptos. Hay dos tipos principales de etiquetas: etiquetas que no tienen cuerpo y etiquetas que s lo tienen. Ejemplos son la etiqueta <img src="asdfa.gif"> que no tiene cuerpo y no tiene etiqueta de final; y <title>manolo</title> que tiene cuerpo y etiqueta de final. Estos son tambin los dos tipos de etiquetas JSP que nos podemos crear a medida.
public inteface Tag { public final static public final static public final static public final static public public public public
int doEndTag() throws JspException; int doStartTag() throws JspException; Tag getParent(); void release();
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
127
setPageContext: Este mtodo lo invoca el motor de JSP para definir el contexto. setParent: Lo invoca el motor de JSP para pasarle un manejador de etiqueta a la etiqueta padre. getParent: Devuelve una instancia de la etiqueta padre. doStartTag: Se procesa la etiqueta de apertura. doEndTag: Se ejecuta despues de doStartTag. release: Lo invoca el motor de JSP para hacer limpieza. La etiqueta deber implementar todos los mtodos y ejecutar la complicada y novedosa tarea de mostrarnos un Hola mundo. Crearemos ahora lo que se llama un Tag Handler. Primero el cdigo y luego comentamos:
// HolaMundoTag.java import java.io.*; import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; public class HolaMundoTag implements Tag { private PageContext pc; private Tag parent; public HolaMundoTag() { super(); } public int doStartTag() throws JspException { try { pc.getOut().print("Hola mundo"); } catch (IOException e){ throw new JspException("Error al enviar al cliente" + e.getMessage()); } return SKIP_BODY; } public int doEndTag() throws JspException { return SKIP_PAGE; } public void release(){
CIBERTEC
CARRERAS PROFESIONALES
128
} public void setPageContext(PageContext pc){ this.pc = pc; } public void setParent(Tag parent){ this.parent = parent; } public Tag getParent(){ return parent;
HolaMundoTag.java
Vemos que la funcionalidad de la etiqueta es mnima, la cantidad de mtodos a implementar es bastante grande y la mayora de ellos triviales. La especificacin JSP 1.1 nos provee de una clase que implementa una Custom Tag con la funcionalidad mnima y de la que podemos heredar para sobreescribir los mtodos que necesitemos. Esta clase es TagSupport, el cdigo anterior se puede reescribir de la siguiente manera:
// HolaMundoTag.java import java.io.*; import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; public class HolaMundoTag extends TagSupport{ public int doStartTag() throws JspException { try{ pageContext.getOut().print("Hola Mundo"); } catch (IOException e) { throw new JspException ("Error: IOException" + e.getMessage()); } return SKIP_BODY; } public int doEndTag() throws JspException { return EVAL_PAGE; } }
HolaMundoTag.java
Se necesita importar java.io.* para escribir en el cliente. Los otros dos import son los tpicos para una JSP Custom Tag. El mtodo doStartTag se invoca cuando se encuentra la etiqueta de apertura, el cdigo tiene que ir incluido en un try-catch porque el mtodo print() puede generar una IOException, al final se devuelve el valor
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
129
SKIP_BODY que le indica al motor de JSP que la etiqueta no tiene cuerpo (en realidad que no queremos evaluar el cuerpo). El mtodo doEndtag se invoca cuando se encuentra la etiqueta de cierre. En este ejemplo, se devuelve EVAL_PAGE para que siga evaluando el resto de la pgina. Si no queremos evaluar el resto de la pgina, el valor por devolver ser SKIP_PAGE. Para compilar debemos incluir en el classpath los paquetes de JSP, suponiendo que Tomcat est instalado en c:\tomcat: javac -classpath c:\tomcat\lib\common\servlet.jar HolaMundoTag.java Ya tenemos la etiqueta compilada. Ahora tenemos que decirle a Tomcat cmo y cundo utilizarla. Para esto tenemos que crear un archivo que describa a esta etiqueta, en concreto que describa una librera de etiquetas. Este archivo se llama Tag Library Descriptor y sigue una sintaxis xml:
<!-- Ejemplos.tld --> <?xml version="1.0" encoding="ISO-8859-1" ?> <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN" "http://java.sun.com/JEE/dtds/web-jsptaglibrary_1_1.dtd"> <taglib> <tlibversion>1.0</tlibversion> <jspversion>1.1</jspversion> <shortname>ejemplos</shortname> <uri></uri> <info>Etiquetas de ejemplo</info> <tag> <name>holamundo</name> <tagclass>HolaMundoTag</tagclass> <bodycontent>empty</bodycontent> <info>Saludo</info> </tag> </taglib>
Ejemplos.tld
El documento empieza con la etiqueta que declara que es un documento xml y la declaracin DOCTYPE. Luego, el resto estar encerrado entre <taglib> y </taglib>. Las etiquetas <tlibversion> y <jspversion> definen las versiones de la librera de etiquetas y de la especificacin JSP, respectivamente. Vemos que cada etiqueta va encerrada entre <tag> y </tag> y lleva el nombre de la etiqueta (<name>), la clase que implementa la etiqueta (<tagclass>), si tiene cuerpo (<bodycontent>, en este caso empty para indicar que no lleva) y la etiqueta <info> que almacena una descripcin de la etiqueta. Ya tenemos la clase que implementa la etiqueta (que por cierto, para probarla, la guardaremos en c:\tomcat\webapps\ejemplos\WEB-INF\classes\HolaMundoTag.class, suponiendo que el Tomcat est en c:\tomcat), el tag library descriptor (que
CIBERTEC
CARRERAS PROFESIONALES
130
guardaremos en c:\tomcat\webapps\ejemplos\Ejemplos.tld) y necesitamos una pgina JSP para probar nuestra primera etiqueta personalizada; por ejemplo, esta:
<!-- HolaMundo.jsp --> <%@ taglib uri="Ejemplos.tld" prefix="ejemplos" %> <HTML> <HEAD> <TITLE>Tag Hola Mundo</TITLE> </HEAD> <ejemplos:holamundo/> </BODY> </HTML>
HolaMundo.jsp
Vemos que es una JSP normal a excepcin de nuestra nueva etiqueta. Obsrvese que no tiene etiqueta de cierre porque no tiene cuerpo. En la directiva taglib definimos dnde se encuentra la librera de etiquetas y el nombre por el que nos referiremos a esta librera durante todo el documento. Si situamos el archivo HolaMundo.jsp en c:\tomcat\webapps\ejemplos\HolaMundo.jsp, arrancamos el Tomcat y visitando la direccin http://localhost:8080/ejemplos/HolaMundo.jsp (sustituyendo el host y el puerto de acuerdo con nuestra configuracin) veremos nuestra primera etiqueta en accin.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
131
import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; public class SumaTag extends TagSupport{ private int num1,num2; public void setNum1(int num1){ this.num1 = num1; } public void setNum2(int num2){ this.num2 = num2; } public int doStartTag() throws JspException { try{ pageContext.getOut().print("Suma: " + (num1+num2)); } catch (IOException e) { throw new JspException ("Error: IOException" + e.getMessage()); } return SKIP_BODY; } public int doEndTag() throws JspException { return SKIP_PAGE; } }
SumaTag.java
Tenemos que aadir la nueva etiqueta, a la librera de ejemplos, el archivo Ejemplos.tld. El esqueleto es el mismo que el anterior, solo que ahora tenemos que aadir una serie de etiquetas para cada atributo. Por cada atributo, le diremos su nombre, con la etiqueta <name>, si es necesaria o no, con la etiqueta <required> y si tiene que ser averiguada en tiempo de ejecucin con la etiqueta <rtexprvalue>. En nuestro caso, tenemos que aadir esto a Ejemplos.tld entre <taglib> y </taglib>:
<tag> <name>suma</name> <tagclass>SumaTag</tagclass> <bodycontent>empty</bodycontent> <info>Saludo</info> <attribute> <name>num1</name> <required>true</required> <rtexprvalue>false</rtexprvalue> </attribute> <attribute> <name>num2</name> <required>true</required> <rtexprvalue>false</rtexprvalue> </attribute>
CIBERTEC
CARRERAS PROFESIONALES
132
</tag>
Ejemplos.tld
El cdigo de la jsp es igual de complicado que el de la etiqueta anterior:
<%@ taglib uri="Ejemplos.tld" prefix="ejemplos" %> <HTML> <HEAD> <TITLE>Tag Suma</TITLE> </HEAD> <ejemplos:suma num1="2" num2="3"/> </BODY> </HTML>
Ejemplos.tld
Si colocamos los archivos de la misma forma que antes (c:\tomcat\webapps\ejemplos\Ejemplos.tld, c:\tomcat\webapps\ejemplos\Suma.jsp, c:\tomcat\webapps\ejemplos\classes\SumaTag.class) y visitamos http://localhost:8080/ejemplos/Suma.jsp, veremos cmo funciona nuestra segunda etiqueta personalizada.
// LongitudTag.java
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
133
import java.io.*; import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; public class LongitudTag extends BodyTagSupport{ public int doAfterBody() throws JspException { try { BodyContent bc = getBodyContent(); String cuerpo = bc.getString(); JspWriter out = bc.getEnclosingWriter(); if (cuerpo != null){ out.print("la longitud de " + cuerpo + "es: " + cuerpo.length()); } } catch (IOException e){ throw new JspException("Error: IOEXception" + e.getMessage()); } return SKIP_BODY; } }
LongitudTag.java
Primero, tomamos el contenido del cuerpo y lo almacenamos en bc. Lo tratamos como cadena y lo guardamos en cuerpo y obtenemos un canal de salida para escribir en el cliente con getEnclosingWriter(). Compilamos LogitudTag. Ahora tenemos que incluir la nueva etiqueta en nuestra librera:
Ejemplos.tld
La nueva etiqueta <bodycontent> le indica al motor de jsp cul va a ser el contenido del cuerpo.Si el contenido es texto, html u otras etiquetas, se indica JSP. Otros tipos de contenidos son tagdependent. Esto no afecta a la interpretacin de la etiqueta, solamente es informativo. La jsp de prueba es tanto o ms complicada que las anteriores:
CIBERTEC
CARRERAS PROFESIONALES
134
<HTML> <HEAD> <TITLE>Tag Longitud</TITLE> </HEAD> <ejemplos:longitud> Hola, soy una cadena </ejemplos:longitud> </BODY> </HTML>
Longitud.jsp
Como siempre, se colocan los archivos en su sitio, se ejecuta el Tomcat y se visita http://localhost:8080/ejemplos/Longitud.jsp
Hay un caso concreto de etiquetas con cuerpo, cuando se necesita que se evale el cuerpo ms de una vez. Esto se consigue devolviendo EVAL_BODY_TAG en lugar de SKIP_BODY en el mtodo doAfterBody(). Como ejemplo, vamos a programar una etiqueta personalizada que realice un bucle, que escriba un nmero determinado de veces el cuerpo de la etiqueta. El cdigo de la clase de la etiqueta bucle es el siguiente:
// BucleTag.java import java.io.*; import javax.servlet.jsp.*; import javax.servlet.jsp.tagext.*; public class BucleTag extends BodyTagSupport{ int veces = 0; BodyContent bc; public void setVeces(int veces) { this.veces = veces; } public void setBodyContent(BodyContent bc){ this.bc = bc; } public int doAfterBody() throws JspException {
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
135
if (veces-->0){ try { JspWriter out = bc.getEnclosingWriter(); out.println(bc.getString()); bc.clearBody(); } catch (IOException e) { system.out.println("Error en Tag Bucle" + e.getMessage()); } return EVAL_BODY_TAG; } else { return SKIP_BODY; } } }
BucleTag.java
Debemos aadir la descripcin de esta nueva etiqueta a la librera:
<tag> <name>bucle</name> <tagclass>BucleTag</tagclass> <bodycontent>JSP</bodycontent> <info>realiza un bucle</info> <attribute> <name>veces</name> <required>true</required> <rtexprvalue>true</rtexprvalue> </attribute> </tag>
Ejemplos.tld
Como es la ltima etiqueta de esta introduccin, la JSP va a ser la ms complicada XD.
<!-- Bucle.jsp--> <%@ taglib uri="Ejemplos.tld" prefix="ejemplos" %> <HTML> <HEAD> <TITLE>Tag Bucle</TITLE> </HEAD> <ejemplos:bucle veces="5"> Dentro del bucle<br> </ejemplos:bucle> </BODY> </HTML>
CIBERTEC
CARRERAS PROFESIONALES
136
Bucle.jsp Despus de colocar los archivos en sus lugares habituales, lanzamos el Tomcat y visitamos http://localhost:8080/ejemplos/Bucle.jsp.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
137
Resumen
La directiva taglib permite el registro de Tag Library Descriptor (TLD) <%@ taglib uri="aliasDelTLD" prefix="ejemplos" %> Mtodos de la interfaz Tag setPageContext: Este mtodo lo invoca el motor de JSP para definir el contexto. setParent: Lo invoca el motor de JSP para pasarle un manejador de etiqueta a la etiqueta padre. getParent: Devuelve una instancia de la etiqueta padre. doStartTag: Se procesa la etiqueta de apertura. doEndTag: Se ejecuta despus de doStartTag. release: Lo invoca el motor de JSP para hacer limpieza.
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://java.sun.com/JEE/tutorial/1_3-fcs/doc/JSPTags.html Aqu hallar la descripcin sobre CustomTag.
CIBERTEC
CARRERAS PROFESIONALES
138
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
139
UNIDAD DE APRENDIZAJE
5
SEMANA
13-14
JAVA STANDARD TAG LIBRARY
LOGRO DE LA UNIDAD DE APRENDIZAJE
Al trmino de la unidad, los alumnos crearn aplicaciones utilizando en los JSP las tecnologias de Lenguaje de Expresiones, JavaServer Pages Standard Tag Library y displayTag en la paginacin.
TEMARIO
JSTL Objetos Implcitos
ACTIVIDADES PROPUESTAS
Los alumnos crearn aplicaciones que permitan utilizar los tag estndares.
CIBERTEC
CARRERAS PROFESIONALES
140
La Librera Core contiene acciones para las tareas rutinarias, como incluir o excluir una parte de una pgina dependiendo de una condicin en tiempo de ejecucin, hacer un bucle sobre una coleccin de tems, manipular URLs para seguimiento de sesin, y la correcta interpretacin del recurso objetivo. Asimismo, tiene acciones para importar contenido de otros recursos y redireccionar la respuesta a una URL diferente. La Librera XML contiene acciones para procesamiento XML, incluido validar un documento XML y transformarlo usando XSLT. Tambin, proporciona acciones para extraer parte de un documento XML validado, hacer bucles sobre un conjunto de nodos y procesamiendo condicional basado en valores de nodos. La Internationalizacin (i18n) y el formateo general estn soportados por las acciones de la Librera I18N & Formatting. Podemos leer y modificar informacin almacenada en una base de datos con las acciones proporcionadas por la Librera Database Access. Ms adelante, podemos esperar que todos los contenedores web incluyan una implementacin de las libreras JSTL, as no tendremos que instalar ningn cdigo adicional. Hasta que esto suceda, podemos descargar e instalar la implementacin de referencia (RI) de JSTL. Se ha desarrollado dentro del proyecto Apache Taglibs como una librera llamada Standard. Instalar RI es fcil, slo hay que copiar los archivos JAR del directorio lib de la distribucin al directorio WEB-INF/lib de nuestra aplicacin. Obsrvese que JSTL 1.0 requiere un contenedor JSP 1.2; por ello, debemos asegurarnos de tener un contenedor compatible con JSP 1.2 antes de probar esto.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
141
Para usar una librera JSTL, tanto si la implementacin est incluida con el contenedor o con el RI, debemos declarar la librera usando una directiva taglib, como lo haramos con una librera de etiquetas personalizadas normal: <%@ taglib prefix="c" uri="http://java.sun.com/jstl/core" %> Obsrvese que siempre deberamos usar la URI por defecto, incluso aunque la especificacin JSP nos permita sobreescribirla. Un contenedor puede generar cdigo optimizado para la accin JSTL en la clase correspondiente a una pgina JSP. Esto puede resultar en un mejor rendimiento que cuando el cdigo generado llama a los manejadores de etiquetas a travs del API estndar. Sin embargo, slo cuando se usa la URI por defecto, es cuando el contenedor puede utilizar una implementacin optimizada.
Como se ve aqu, una expresin EL siempre debe estar encerrada entre los caracteres ${ y }$. Las dos primeras expresiones acceden a una propiedad llamada myProperty en un objeto representado por una variable llamada myObj. La tercera expresin accede a una propiedad con un nombre contenido en una variable. Esta sintaxis se puede utilizar con cualquier expresin que evale el nombre de la propiedad. El operador de acceso a array tambin se usa para datos representados como una coleccin de elementos indexados, como un array Java o una java.util.List:
${myList[2]}$
CIBERTEC
CARRERAS PROFESIONALES
142
${myList[aVar + 1]}$
Adems de los operadores de propiedad y elemento array, y los operadores aritmticos, relacionales y lgicos, hay un operador especial para comprobar si un objeto est "vaco" o no puede ser usado en una expresin EL. La siguiente tabla lista todos los operadores: Operador . [] () + / o div % o mod == o eq != o ne < o lt > o gt <= o le >= o gt || o or ! o not empty Accede a una propiedad. Accede a un elemento de un array/lista. Agrupa una subexpression. Suma Resta o negacin de un nmero Divisin Mdulo (resto) Comprueba Igualdad. Comprueba desigualdad. Comprueba menor que. Comprueba mayor que. Comprueba menor o igual que. Comprueba mayor o igual que. Comprueba OR lgico. Complemento binario booleano Comprueba un valor vaco (null, string vaco o una colleccin vaca) Descripcin
Lo que no encontraremos en EL son sentencias como asignaciones, if/else o while. Para este tipo de funcionalidades, en JSP se usan los elementos Action. EL no est pensado para utilizarse como un lenguaje de programacin de propsito general, slo un lenguaje de acceso a datos. Por supuesto, los literales y las variables tambin son parte del lenguaje. EL proporciona los siguientes literales, similares a los que proporcionan JavaScript, Java, y otros lenguajes similares: Tipo de Literal Descripcin Encerrado con comillas simples o dobles. Una comilla del mismo tipo dentro del string puede ser escapada con una barra invertida: (\' en un string encerrado con comillas simples; \" en un string encerrado con comillas dobles). El carcter de barra invertida debe ser escapado como \\ en ambos casos.
String
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
143
Un signo opcional (+ o -) seguido por dgitos entre 0 y 9. Lo mismo que un literal entero, excepto que usa un punto como separador de la parte fraccional y que se puede especificar un exponente con e o E, seguido por un literal entero. true o false null
Cualquier objeto en uno de los mbitos de JSP (pgina, solicitud, sesin o aplicacin) se puede utilizar como una variable en una expresin EL. Por ejemplo, si tenemos un bean con una propiedad firstName en el mbito de la solicitud bajo el nombre customer, esta expresin EL representa el valor de la propiedad firstName del bean.
${customer.firstName}
Sin embargo, EL tambin hace que la informacin de la solicitud y la informacin general del contenedor est disponible como un conjunto de variables implcitas: Variable param paramValues header headerValues cookie initParams pageContext pageScope requestScope sessionScope Descripcin Una collection de todos de los parmetros de la solicitud como un slo valor string para cada parmetro Una collection de todos los valores de los parmetros de la solicitud como un array de valores string por cada parmetro Una collection de todas las cabeceras de solicitud como un slo valor string por cada cabecera Una collection de todos los valores de cabecera de la solicitud como un array de valores string por cada cabecera Una collection con todas las cookies de la solicitud en un slo ejemplar de javax.servlet.http.Cookie por cada cokkie Una collection de todos los parmetros de inicializacin de la aplicacin en un slo valor string por cada parmetro Un ejemplar de la clase javax.servlet.jspPageContext Una collection de todos los objetos en el mbito de la pgina Una collection de todos los objetos en el mbito de la solicitud Una collection de todos los objetos en el mbito de la sesin
applicationScope Una collection de todos los objetos en el mbito de la aplicacin Las cinco primeras variables implcitas de la tabla nos ofrecen acceso a los valores de parmetros, cabeceras y cookies de la solicitud actual. Aqu hay un ejemplo de cmo acceder a un parmetro de solicitud llamado listType y a la cabecera User-Agent:
${param.listType} ${header['User-Agent']}
CIBERTEC
CARRERAS PROFESIONALES
144
Observe cmo se debe usar la sintaxis de array para la cabecera, porque el nombre incluye un guion; con la sintaxis de propiedad, sera interpretado como la expresin vairable header.User menos el valor de una variable llamada Agent. La variable initParameter proporciona acceso a los parmetros de inicializacin que se definen en el archivo web.xml de la aplicacin. La variable pageContext tiene varias propiedades que proporcionan acceso al objeto servlet que representa la solicitud, la respuesta, la sesin y la aplicacin, etc. Las cuatro ltimas variables son colecciones que contienen todos los objetos de cada mbito especfico. Podemos usarlas para limitar la bsqueda de un objeto en slo un mbito en lugar de buscar en todos ellos, lo que est por defecto si no se especifica ningn mbito. En otras palabras, si hay un objeto llamado customer en el mbito de sesin, estas dos primeras expresiones encuentran el mismo objeto, pero la tercera vuelve vaca:
Todas las acciones JSTL aceptan expresiones EL como valores de atributo, para todos los atributos excepto para var y scope, porque estos valores de atributo podran usarse para chequear el tipo en el momento de la traduccin en una futura versin. Hay un atributo de una accin JSTL adicional que no toma un valor de expresin EL, pero slo se usa en la librera XML, por eso lo ignoraremos por ahora. Se pueden usar una o ms expresiones EL en el mismo valor de atributo, y el texto fijo y las expresiones EL se pueden mezclar en el mismo valor de atributo:
First name: <c:out value="${customer.firstName}" /> <c:out value="First name: ${customer.firstName}" />
Antes de saltar a ver ejemplos de utilizacin de las acciones Core, se ver lo que dijimos anteriormente: todas las acciones JSTL de la librera EL aceptan expresiones EL. Realmente hay un conjunto parelelo de libreras JSTL, llamado conjunto de librera RT que slo acepta expresiones Java del antiguo estilo: First name: <c_rt:out value="<%= customer.getFirstName() %>" />
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
145
Veamos algunos ejemplos de cmo podemos usar el JSTL condicional y las acciones de iteraccin: <c:if>; <c:choose>, <c:when>, y un triple <c:otherwise>; y <c:forEach>. Por el camino, tambin usaremos acciones de salida bsica y de seleccin de variables: <c:out> y <c:set>. <c:if> nos permite incluir, o procesar, condicionalmente una parte de una pgina, dependiendo de la informacin durante la ejecucin. Este ejemplo incluye un saludo personal si el usuario es un visitante repetitivo, segn lo indica la presencia de una cookie con el nombre del usuario:
El valor del atributo test es una expresin EL que chequea si la cookie existe. El operador empty combinado con el operador "not" (!) significa que evala a true si el cookie no existe, haciendo que el cuerpo del elemento sea procesado. Dentro del cuerpo, la accin <c:out> aade el valor de la cookie a la respuesta. As de sencillo. Pasar a travs de una coleccin de datos es casi tan sencillo. Este fragmento itera sobre una coleccin de filas de una base de datos con informacin del tiempo de diferentes ciudades:
<c:forEach items="${forecasts.rows}" var="${city}"> City: <c:out value="${city.name}" /> Tomorrow's high: <c:out value="${city.high}" /> Tomorrow's low: <c:out value="${city.low}" /> </c:forEach>
La expresin EL para el valor items obtiene el valor de la propiedad rows desde un objeto representado por la variable forecasts. Como aprenderemos ms adelante, las acciones de bases de datos de JSTL representan un resultado de consulta como un ejemplar de una clase llamada javax.servlet.jsp.jstl.sql.Result. Esta clase se puede utilizar como un bean con varias propiedades. La propiedad rows contiene un array de ejemplares java.util.SortedMap, donde cada uno representa una fila con valores de columnas. La accin <c:forEach> procesa su cuerpo una vez por cada elemento de la coleccin especificado por el atributo tems. Adems de con arrays, la accin funciona con cualquier otro tipo de dato que represente una coleccin, como ejemplares de las clases java.util.Collection o java.util.Map. Si se especifica el atributo var, el elemento actual de la coleccin se hace disponible para las acciones del cuerpo como una variable con el nombre especificado. Aqu se llamaba city y, como la coleccin es un array de maps, esta variable contiene un nuevo map con valores de columnas cada vez que se procesa el cuerpo. Los valores de las columnas se aaden a la respuesta por el mismo tipo de acciones <c:out> que hemos visto en ejemplos anteriores. Para ilustrar el uso del resto de las acciones condicionales, extendamos el ejemplo de iteraccin para procesar slo un conjunto fijo de filas por cada pgina solicitada, aadamos enlaces "Previous" y "Next" a la misma pgina. El usuario puede entonces
CIBERTEC
CARRERAS PROFESIONALES
146
pasar sobre los resultados de la base de datos, mirando unas pocas filas cada vez, asumiendo que el objeto Result se ha grabado en el mbito de la sesin. Aqu est cmo procesar slo algunas filas:
<c:set var="noOfRows" value="10" /> <c:forEach items="${forecasts.rows}" var="${city}" begin="${param.first}" end="${param.first + noOfRows - 1}"> City: <c:out value="${city.name}" /> Tomorrow's high: <c:out value="${city.high}" /> Tomorrow's low: <c:out value="${city.low}" /> </c:forEach>
La accin <c:set> selecciona una variable con el valor especificado por el atributo value; que puede ser un valor esttico, como en este ejemplo, o una expresin EL. Tambin, podemos especificar el mbito de la variable con el atributo scope (page, request, session o application). En este ejemplo, hemos seleccionado una variable llamada noOfRows a 10 en el mbito de la pgina (por defecto). Este es el nmero de filas que mostraremos en cada solicitud. El <c:forEach>, en este ejemplo, toma los mismos valores para los atributos tems y var como antes, pero hemos aadido dos nuevos atributos:
El atributo begin toma el ndice (base 0) del primer elemento de la coleccin a procesar. Aqu se selecciona al valor de un parmetro de solicitud llamado first. Para la primera solicitud, este parmetro no est disponible, por eso la expresin se evala a 0; en otras palabtas, la primea fila. El atributo end especifica el ndice del ltimo elemento de la coleccin a procesar. Aqu lo hemos seleccionado al valor del parmetro first ms noOfRows menos uno. Para la primera solicitud, cuando no existe el parmetro de la solicitud, este resultado es 9, por eso la accin itera sobre los ndices del 0 al 9.
<c:choose> <c:when test="${param.first > 0}"> <a href="foreach.jsp?first=<c:out value="${param.first noOfRows}"/>"> Previous Page</a> </c:when> <c:otherwise> Previous Page </c:otherwise> </c:choose> <c:choose> <c:when test="${param.first + noOfRows < forecasts.rowsCount}"> <a href="foreach.jsp?first=<c:out value="${param.first + noOfRows}"/>">
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
147
El <c:choose> agrupa una o ms acciones <c:when>, cada una especificando una condicin booleana diferente. La accin <c:choose> chequea cada condicin en orden y slo permite la primera accin <c:choose> con una condicin que se evale a true para procesar su cuerpo. El cuerpo <c:choose> tambin podra contener un <c:otherwise>. Su cuerpo slo se procesa si ninguna de las condiciones de los <c:when> es true. En este ejemplo, la primera accin <c:when> comprueba si el parmetro first es mayor que cero, es decir, si la pgina muestra un subconjunto de filas distinto del primero. Si esto es cierto, el cuerpo de la accin <c:when> aade un enlace a la misma pgina con un parmetro first seleccionado al ndice del subconjunto anterior. Si no es true, se procesa el cuerpo de la accin <c:otherwise>, aadiendo slo el texto "Previous Page". El segundo bloque <c:choose> proporciona lgica similar para aadir el enlace "Next Page".
<c:url var="previous" value="foreach.jsp"> <c:param name="first" value="${param.first - noOfRows}" /> </c:url> <a href="<c:out value="${previous}"/>">Previous Page</a>
<c:url> soporta un atributo var, usado para especificar una variable para contener la URL codificada, y un atributo value para contener la URL a codificar. Se pueden especificar parmetros string para solicitar la URL usando acciones <c:param>. Los caracteres especiales en los parmetros especificados por elementos anidados son codificados (si es necesario) y luego aadidos a la URL como parmetros string de la
CIBERTEC
CARRERAS PROFESIONALES
148
consulta. El resultado final se pasa a travs del proceso de reescritura de URL, aadiendo un ID de sesin si est desactivado el seguimiento de sesin usando cookies. En este ejemplo, se utiliza la URL codificada como un valor de atributo href en un elemento de enlace HTML. <c:url> tambin realiza otro buen servicio. Las URL relativas en elementos HTML tambin deben ser relativas a la pgina que los contiene o al directorio raz del servidor (si empiezan con una barra inclinada). La primera parte del path de una URL de una pgina JSP se llama path de contexto, y podra variar de instalacin a instalacin. Por lo tanto, deberamos evitar codificar el path de contexto en las pginas JSP. Pero algunas veces realmente queremos utilizar un path de URL relativo al servidor en elemento HTML; por ejemplo, cuando necesitamos referirnos a un archivo de imagen localizado en el directorio /images compartido por todas las pginas JSP. Las buenas noticias es que si especificamos una URL con un barra inclinada como el valor <c:url>, lo convierte en un path relativo al servidor. Por ejemplo, en una aplicacin con el path de contexto /myApp, la accin <c:url> convierte el path a /myApp/images/logo.gif: <c:url value="/images/logo.gif" /> Hay unas cuantas acciones ms relacionadas con las URL en la librera corazn. La accin <c:import> es una accin ms flexible que la accin estndar <jsp:include>. Podemos usarla para incluir contenido desde los recursos dentro de la misma aplicacin web, desde otras aplicaciones web en el mismo contenedor, o desde otros servidores, usando protocolos como HTTP y FTP. La accion <c:redirect> nos permite redirigir a otro recurso en la misma aplicacin web, en otra aplicacin web o en un servidor diferente.
Escribir un conjunto de pginas para cada idioma Escribir un conjunto compartido de pginas que ponga el contenido en diferentes idiomas desde fuentes externas
Frecuentemente, terminaremos con una mezcla de estas tcnicas, usar pginas separadas para la mayora de las grandes cantidades de contenido esttico y pginas compartidas cuando la cantidad de contenido sea pequea, pero dinmica (por ejemplo, una pgina con unas pocas etiquetas fijas mostradas en diferentes idiomas y todos los dems datos que vengan desde una base de datos). Preparar una aplicacin para varios idiomas se llama internacionalizacin (comnmente abreviado i18n); y hacer que el contenido est disponible para un idioma especfico se llama localizacin (o l10n). Para hacer esto, necesitamos considerar otras cosas adems del idioma: cmo se formatean las fechas y nmeros entre los diferentes pases e, incluso, dentro de los pases. Tambin, podramos necesitar adaptar los colores, las imgenes y otro contenido no textual. El trmino localidad se refiere a un conjunto de reglas y contenido aceptable para una regin o cultura.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
149
<%@ taglib prefix="fmt" uri="http://java.sun.com/jstl/fmt" %> <html> <body> <h1>Formatting with the default locale</h1> <jsp:useBean id="now" class="java.util.Date" /> Date: <fmt:formatDate value="${now}" dateStyle="full" /> Number: <fmt:formatNumber value="${now.time}" /> </body> </html>
La primera lnea es la directiva taglib para la librera JSTL que contiene las acciones de formateo e internacionalizacin. El prefijo por defecto, usado aqu, es fmt. Para obtener un valor a formatear, entonces crea un objeto java.util.Date que representa la fecha y hora actuales, y lo graba como una variable llamada now. Ya estamos preparados para hacer algn formateo sensible a la localidad. La accin JSTL <fmt:formatDate> formatea el valor de la variable now asignado al valor del atributo value usando el tipo de expresin EL que aprendimos en la pgina anterior. El atributo dateStyle le dice a la accin cmo debe formatear la fecha. Podemos usar cualquiera de los valores default, short, medium, long, o full para el atributo dateStyle. El tipo de formateo que representa cada uno de estos tipos depende exactamente de la localidad. Para la localidad English, full resulta en un string como Thursday, August 29, 2002. En este ejemplo, slo hemos formateado la parte de la fecha, pero tambin podemos incluir la parte de la hora (o formatear slo la hora), y definir reglas de formateo personalizadas tanto para la fecha como para la hora en lugar de los estilos dependientes de la localidad:
El atributo pattern toma un patrn de formateo personalizado del mismo tipo que la clase java.text.SimpleDateFormat. El patrn usado aqu resulta en Thursday, 29 August 2002, 17:29 con la localidad English. La accin <fmt:formatNumber> soporta atributos similares para especificar cmo formatear un nmero usando estilos dependientes de la localidad para nmeros normales, valores de moneda, o un porcentaje, as como usar patrones personalizados de diferentes tipos.
CIBERTEC
CARRERAS PROFESIONALES
150
ejemplo, sin hacer nada ms, el formateo de localidad se determina comparando las localidades especificadas en un cabecera de solicitud llamada Accept-Language con las localidades soportadas por el entorno de ejecucin Java. La cabecera, si est presente, contiene una o ms especificaciones de localidades (en la forma de cdigo de idioma y posiblemente un cdigo de pas), con informacin sobre su prioridad relativa. El usuario utiliza las configuraciones del navegador para definir qu especificaciones de localidades enviar. Las localidades de la solicitud se comparan en orden de prioridad con las ofrecidas por la mquina virtual Java, y se selecciona la que mejor coincida. Si no se encuentra una correspondencia, la accin de formateo busca la llamada seleccin de configuracin de la localidad pre-definida. Una seleccin de configuracin es un valor seleccionado por un parmetro de contexto en el archivo web.xml de la aplicacin o por una accin JSTL o una sentencia Java en uno de los mbitos JSP. Para seleccionar esta localidad en el archivo web.xml, incluimos estos elementos:
Con esta seleccin, se usar la localidad German (especificada por el cdigo de idioma de como valor de parmetro) si ninguna de las localidades especificadas por la solicitud est soportada por el entorno Java. JSTL tambin nos permite seleccionar una localidad por defecto para la aplicacin que podremos sobreescribir cuando lo necesitemos. Esto nos da control total sobre la localidad a utilizar, en lugar de dejarlo a la configuracin del navegador del visitante. La localidad por defecto tambin es una seleccin de configuracin. Se puede especificar con un parmetro de contexto en el archivo web.xml de esta forma:
Esta seleccin establece English como la localidad por defecto, resultando en la anterior pgina JSP siempre formatear la fecha y los nmeros de acuerdo con las reglas inglesas.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
151
Para sobreescribir la configuracin de localidad por defecto, podemos usar la accin <fmt:setLocale>. Selecciona la localidad por defecto dentro de un mbito JSP especfico. Aqu tenemos un ejemplo que selecciona la localidad por defecto del mbito de la pgina, basndose en una cookie que sigue la pista de la localidad seleccionada por el usuario en una visita anterior:
<h1>Formatting with a locale set by setLocale</h1> <c:if test="${!empty cookie.preferredLocale}"> <fmt:setLocale value="${cookie.preferredLocale.value}" /> </c:if> <jsp:useBean id="now" class="java.util.Date" /> Date: <fmt:formatDate value="${now}" dateStyle="full" /> Number: <fmt:formatNumber value="${now.time}" />
Aqu, primero hemos usado la accin JSTL <c:if> para comprobar si con la solicitud se ha recibido una cookie llamada preferredLocale. Si es as, la accin <fmt:setLocale> sobreescribe la localidad para la pgina actual y selecciona la variable de configuracin de la localidad en el mbito de la pgina. Si lo queremos, podemos seleccionar la localidad para otro mbito usando el atributo scope. Finalmente, se formatean la fecha y el nmero de acuerdo con las reglas de la localidad especificada por la cookie, o la localidad por defecto, si la cookie no est presente. Adems de acciones para formateo, JSTL incluye acciones para interpretar (analizar) fechas y nmero de una forma sensible a la localidad: <fmt:parseDate> y <fmt:parseNumber>. Estas acciones soportan casi todos los mismos atributos que sus contrapartidas de formateo, y se pueden utilizar para convertir entradas de fechas y nmeros en sus formas nativas antes de procesarlas.
CIBERTEC
CARRERAS PROFESIONALES
152
import javax.servlet.jsp.jstl.core.Config; ... public class LoginAction extends Action { public ActionForward perform(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { String userName = request.getParameter("userName"); String password = request.getParameter("password"); ... User user = authenticate(userName, password); if (user != null) { /* * Valid user. Set the default locale in the session * to the user's preferred locale */ HttpSession session = request.getSession(); Config.set(session, Config.FMT_LOCALE, user.getLocale()); ... } } }
La clase Config proporciona cuatro mtodos set(), uno por cada mbito JSP (especificado por el primer parmetro de mtodo). Aqu hemos usado el que selecciona la variable en el mbito de sesin. El segundo parmetro es el nombre de la configuracin, tpicamente seleccionado por la constante correspondiente. El tercer parmetro es el valor de la variable. Para leer y eliminar variable, la clase proporciona mtodos get() y remove() similares, ms un mtodo find() para localizar una variable en cualquier mbito. Adems de usar estos mtodos en un Controlador, podramos querer usarlos en nuestro propio manejador de etiquetas personalizado para aprovecharnos de los mecanismos de configuracin JSTL.
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
153
el archivo del paquete de recursos anterior representa la localidad English, debera almacenarse en un archivo llamado labels_en.properties. El paquete de recursos para la localidad Swedish sera almacenado en un archivo llamado labels_sv.properties, donde sv es el cdigo de idioma. La parte fija del nombre de archivo, labels de este ejemplo, se llama el nombre base del paquete de recursos, y se combina con una especificacin de localidad (como un cdigo de idioma) para encontrar el paquete de recursos para una localidad especfica. Todos los paquetes de recursos incluyen las mismas claves; slo difiere el texto dependiendo de la localidad. Los archivos deben situarse en un directorio que sea parte del classpath del contenedor, normalmente el directorio WEB-INF/classes de la aplicacin. La accin JSTL que aade texto desde un paquete de recursos a una pgina es <fmt:message>. El paquete a utilizar puede especificarse de varias formas. Una es anidar las acciones <fmt:message> dentro del cuerpo de una accin <fmt:bundle>:
<fmt:bundle basename="labels"> Hello: <fmt:message key="hello" /> Goodbye: <fmt:message key="goodbye" /> </fmt:bundle>
En este caso, la accin <fmt:bundle> localiza el paquete de la localidad que es la correspondencia ms cercana entre la seleccin de configuracin de la localidad (o las localidades en el cabecera Accept-Language si no hay localidad por defecto) y los paquetes de recursos disponibles para el nombre base especificado. Las acciones anidadas obtienen el texto desde el paquete por la clave asociada. Al igual que con las acciones de formateo, podemos establecer un paquete por defecto, para toda la aplicacin con un parmetro de contexto o con la accin <fmt:setBundle> o la clase Config para un mbito JSP especfico: <fmt:setBundle basename="labels"/> ... Hello: <fmt:message key="hello" /> Goodbye: <fmt:message key="goodbye" /> Despus de que se haya definido un paquete por defecto, podemos usar acciones <fmt:message> independientes dentro del mbito donde se estableci el valor por defecto.
CIBERTEC
CARRERAS PROFESIONALES
154
acceso a la base de datos en clases Java, y utilizar las JSP slo para mostrar los resultados. Las acciones de bases de datos de JSTL estn basadas en el API JDBC de Java y usan las abstracciones javax.sql.DataSource presentada en JDBC 2.0 para representar una base de datos. Un DataSource proporciona conexiones a una base de datos y puede implementar una caracterstica llamada almacn de conexiones. Abrir una conexin fsica a una base de datos es una operacin que consume mucho tiempo. Con el almacenamiento de conexiones, slo necesitamos hacerlo una vez, y la misma conexin se puede reutilizar una y otra vez, sin los riesgos de problemas asociados con otras aproximaciones de comparticin de conexiones.
Se deben utilizar las herramientas de configuracin JNDI del contenedor WEB para configurar un recurso JNDI con el nombre especificado; por ejemplo, con un nombre de usuario y una password de una cuenta de usuario en una base de datos, las conexiones mnimas y mximas en el almacen, etc. La forma de realizar esto vara entre los contenedores y est fuera del mbito de este artculo. Una alternativa para los contenedores que no soportan JNDI es que un oyente del ciclo de vida de la aplicacin (contexto del servlet) cree y configure un DataSource y lo seleccione como el valor por defecto usando la clase Config de JSTL:
import javax.servlet.*; import javax.servlet.http.*; import oracle.jdbc.pool.*; public class AppListener implements ServletContextListener { private OracleConnectionCacheImpl ds =null; public void contextInitialized(ServletContextEvent sce){ ServletContext application =sce.getServletContext(); try {
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
155
ds = new OracleConnectionCacheImpl(); ds.setURL("jdbc:oracle:thin:@voyager2:1521:Oracle9i"); ds.setMaxLimit(20); ds.setUser("scott"); ds.setPassword("tiger"); } catch (Exception e){ application.log("Failed to create data source:"+ e.getMessage()); } Config.set(application, Config.SQL_DATASOURCE, ds); } ... }
La clase oyente de este ejemplo crea un DataSource con capacidades de almacn de conexiones para una base de datos Oracle9i, y la hace disponible como por defecto para las acciones JSTL usando la clase Config para seleccionar la variable de configuracin correspondiente. Una tercera forma, slo disponible para prototipos o aplicaciones que no van a utilizarse tan duramente como para necesitar el almacn de conexiones, es usar la accin <sql:setDataSource>:
Esta accin crea una sencilla fuente de datos, sin almacenamiento, para la URL JDBC especificada, con el usuario y la password, usando el driver JDBC especificado. Podramos usar esta accin para empezar, pero recomendamos la utilizacin de una de las otras alternativas para una site del mundo real. Adems de privarnos del almacn de conexiones para una fuente de datos creada de esta forma, no es una buena idea incluir informacin sensible como la URL de la base de datos, el nombre de usuario y la password en una pgina JSP, ya que sera posible para alguien acceder al cdigo fuente de la pgina.
CIBERTEC
CARRERAS PROFESIONALES
156
<%@ taglib prefix="sql" uri="http://java.sun.com/jstl/sql" %> <html> <body> <h1>Reading database data</h1> <sql:query var="emps" sql="SELECT * FROM Employee" /> ... </body> </html>
Primero, necesitamos declarar la librera JSTL que contiene las acciones de bases de datos, usando la directiva taglib de la parte superior de este ejemplo. La accin <sql:query> ejecuta la sentencia SQL SELECT especificada por el atributo sql (o como el cuerpo del elemento accin) y graba los resultados en una variable nombrada por el atributo var. El resultado de la consulta a la base de datos se devuelve como un bean del tipo javax.servlet.jsp.jstl.sql.Result con varias propiedades de solo lectura: Propiedad Tipo Java Descripcin
rows
Un array con un mapa insensible a las maysculas por cada fila con las claves java.util.SortedMap[] correspondiendo con los nombres de columna y valores correspondiendo con los valores de columna Object[][] String[] int Un array con un array por fila con valores de columna Un array con nombres de columnas El nmero de filas en el resultado true si no se han incluido todas las filas debido a que se ha alcanzado el lmite mximo de filas especificado
limitedByMaxRows boolean
Ya vimos cmo utilizar la accin <c:forEach> para mostrar todas o slo algunas filas, por eso ahora veremos cmo podemos obtener slo algunas filas para mostrarlas todas en esta parte. Los enlaces Next y Previous le permiten al usuario solicitar un conjunto diferente. Primero, aqu est cmo leer un subconjunto de filas y luego mostrar el subconjunto completo:
<c:set var="noOfRows" value="10" /> <sql:query var="emps" startRow="${param.start}" maxRows="${noOfRows}"> SELECT * FROM Employee </sql:query> <ul> <c:forEach items="${emps.rows}" var="${emp}">
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
157
El atributo startRow para la accin <sql:query> se enva a una expresin EL que lee el valor de un parmetro de solicitud llamado start. Luego veremos cmo cambia este valor cuando se pulsa sobre los enlaces Next y Previous. La primera vez que se accede a la pgina, el parmetro no existe, por eso la expresin se evala a 0. Esto significa que el resultado de la consulta contiene filas empezando con la primera que corresponda (ndice 0). El atributo maxRows limita el nmero total de filas del valor de la variable noOfRows, lo seleccionamos a 10 en este ejemplo. La accin <c:forEach> hace un bucle sobre todas las columnas del resultado y genera una lista de tems con los valores de columna por cada fila. Tambin, debemos generar los enlaces Next y Previous para permitir que el usuario seleccione un nuevo conjunto de filas:
<c:choose> <c:when test="${param.start > 0}"> <a href="emplist.jsp?start=<c:out value="${param.start - noOfRows}"/>">Previous Page</a> </c:when> <c:otherwise> Previous Page </c:otherwise> </c:choose> <c:choose> <c:when test="${emps.limitedByMaxRows}"> <a href="emplist.jsp?start=<c:out value="${param.start + noOfRows}"/>">Next Page</a> </c:when> <c:otherwise> Next Page </c:otherwise> </c:choose>
El primer bloque <c:choose> es idntico al de la pgina 1; si el parmetro de la solicitud start es mayor que cero, la pgina actual muestra un subconjunto de filas distinto del primero, por eso se aade un enlace Previous. El enlace apunta hacia la misma pgina, e incluye el parmetro start con un valor que es el valor actual menos el nmero de filas mostrado por cada pgina. El segundo bloque <c:choose> se aprovecha de la propiedad limitedByMaxRows del resultado de la consulta. Si esta propiedad es true, significa que se ha truncado el resultado al nmero de filas mostrado por cada pgina. De aqu, se genera el enlace Next con valor del parmetro start para el nuevo subconjunto de filas.
CIBERTEC
CARRERAS PROFESIONALES
158
Adems de leer datos desde una base de datos, tambin podemos usar JSTL para actualizar informacin. Este ejemplo muestra cmo insertar una nueva fila en una tabla:
<c:catch var="error"> <fmt:parseDate var="empDate" value="${param.empDate}" pattern="yyyy-MM-dd" /> </c:catch> <c:if test="${error != null}"> <jsp:useBean id="empDate" class="java.util.Date" /> </c:if> <sql:update> INSERT INTO Employee (FirstName, LastName, EmpDate) VALUES(?, ?, ?) <sql:param value="${param.firstName}" /> <sql:param value="${param.lastName}" /> <sql:dateParam value="${empDate}" type="date" /> </sql:update>
Antes de insertar la fila, este ejemplo ilustra cmo usar las acciones de validacin JSTL. La pgina espera que todos los datos para la nueva fila se enven como parmetros de solicitud (quizs introducidos en un formulario HTML), incluyendo una fecha de contratacin. Antes de que la fecha se pueda insertar en la base de datos, debemos convertirla a su forma nativa Java. Esto es lo que hace la accin <fmt:parseDate>. El atributo value contiene una expresin EL que obtiene el valor del parmetro de solicitud empDate. La accin trata de interpretarlo como una fecha escrita en el formato especificado por el atributo pattern (un ao de cuatro dgitos, seguido por dos dgitos del mes y dos dgitos del da, separados por barras inclinadas). Si tiene xito, almacena la fecha en su forma nativa con el nombre especificado por el atributo var. La accin <c:catch> tiene en cuenta los strings invlidos. Si el valor del parmetro no puede ser interpretado como una fecha, el <fmt:parseDate> lanza una excepcin, que la accin <c:catch> captura y graba en la variable especificada. Cuando esto sucede, la condicin de comprobacin de la accin <c:if> se evala a true, por lo que la fecha de contratacin es creada por la accin <jsp:useBean> anidada. Para insertar la fila, usamos la accin <sql:update>. Como la accin de consulta, la sentencia SQL se puede especificar como el cuerpo del elemento o mediante un atributo sql. La accin <sql:update> se puede usar para ejecutar sentencias INSERT, UPDATE, DELETE, asi como sentencias para crear o eliminar objetos en la base de datos, como CREATE TABLE y DROP TABLE. El nmero de filas afectado por la sentencia puede capturarse de forma opcional en una variable nombrada por el atributo var. En este ejemplo (como en las mayora de las aplicaciones del mundo real), no se conocen los nombres de las columnas en tiempo de ejecucin; vienen de los parmetros de la solicitud. Por lo tanto, la sentencia INSERT de SQL incluye una marca de interrogacin por cada valor como un contenedor y parmetros internos de la accin que seleccionan el valor dinmicamente. Las columnas FirstName y LastName
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
159
son columnas de texto y las aciones <sql:param> seleccionan sus valores al valor del parmetro de solicitud correspondiente. Sin embargo, la columna EmpDate es una columna de fecha, demandando una atencin especial. Primero de todo, debemos usar una variable que contenga la fecha en su forma nativa (un objeto java.util.Date) en lugar de usar el valor del parmetro de solicitud, usamos la variable crada por las acciones <fmt:parseDate> o <jsp:useBean>. Segundo, debemos usar la accin <sql:dateParam> para seleccionar el valor. En este ejemplo, hemos usado la parte de la fecha, por eso hemos seleccionado el atributo opcional type a date. Otros valores vlidos son time y timestamp (por defecto), para las columnas que slo toman la hora, o la fecha y la hora. Hay una accin JSTL ms que no hemos descrito hasta ahora: <sql:transaction>. Podemos usarla para agrupar varias acciones update (o incluso query) donde todas ellas se deben ejecutar como parte de la misma transaccin de la base de datos. El ejemplo estndar es transferir una cantidad de dinero de una cuenta a otra; implementada como una sentencia SQL que elimina el dinero de la primera cuenta y otra sentencia que lo aade a la segunda. Si encapsulamos todos los accesos a una base de datos en clases Java en lugar de usar las acciones de JSTL, todava hay una parte de JSTL que nos puede ser til. Es una clase llamada javax.servlet.jsp.jstl.sql.ResultSupport, con estos dos mtodos: public static Result toResult(java.sql.ResultSet rs); public static Result toResult(java.sql.ResultSet rs, int maxRows); Podemos usar esta clase para convertir un objeto ResultSet estndar JDBC en un objeto Result JSTL antes de reenviarlo a la pgina JSP para mostrarlo. Las acciones JSTL pueden acceder fcilmente a los datos de un objeto Result, como vimos anteriormente. Otra aproximacin, todava mejor, es pasar el resultado de la consulta a la pgina JSP como una estructura de datos personalizada, como una List de beans que contienen los datos de cada fila, pero el objeto Result an es un buen candidato para prototipos y pequeas aplicaciones. Funcin fn:contains(string, substring) Descripcin Devuelve true si el string contiene substring. Devuelve true si el string contiene a substring, ignorando capitalizacin. Devuelve true si el string termina con suffix. fn:escapeXml(string) Devuelve el string con todos aquellos caracteres con especial significado en XML y HTML convertidos a sus cdigos de escape pertinentes. fn:indexOf(string, substring) Devuelve la primera ocurrencia
fn:containsIgnoreCase(string, substring)
fn:endsWith(string, suffix)
CIBERTEC
CARRERAS PROFESIONALES
160
de substring en el string. fn:join(array, separator) Devuelve un string compuesto por los elementos del array, separados por separator. Devuelve el nmero de elementos en tem (si un array o coleccin), o el nmero de caracteres (es un string). Devuelve un string donde todas las ocurrencias del string before han sido reemplazadas por el string after. fn:split(string, separator) Devuelve un array donde los elementos son las partes del string separadas por separator. fn:startsWith(string, prefix) fn:substring(string, begin, end) Devuelve true si el string comienza por prefix. Devuelve la parte del string que comienza por el ndice begin y que acaba en el ndice end. Devuelve la parte del string que sigue a substring. Devuelve la parte de string que precede a substring. Devuelve un string con todos los caracteres de la entrada convertidos a minsculas. Devuelve un string con todos los caracteres de la entrada convertidos a maysculas. Devuelve un string sin espacios en sus laterales.
fn:length(item)
fn:substringAfter(string, substring)
fn:substringBefore(string, substring)
fn:toLowerCase(string)
fn:toUpperCase(string)
fn:trim(string)
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
161
Preguntas de Certificacin
1 Cul es la etiqueta que invoca al contenedor web y busca al Tag Library Descriptor? a <taglib-directory></taglib-directory> b <taglib-uri></taglib-uri> c <taglib-location></taglib-location> d <tld-directory></tld-directory> e <taglib-name></taglib-name> 2. Cules son las dos expresiones que no retornan el campo accept de la cabecera? a ${header.accept} b ${header[accept]} c ${header['accept']} d ${header["accept"]} e ${header.'accept'} 3. Cul de los siguientes mtodos permite registrar el nombre de una funcin? a <function-signature></function-signature> b <function-name></function-name> c <method-class></method-class> d <method-signature></method-signature> e <function-class></function-class> 4. Cul de los siguientes mtodos es correcto para el uso de una funcin EL? a public static expFun(void) b expFun(void) c private expFun(void) d public expFun(void) e public native expFun(void)
CIBERTEC
CARRERAS PROFESIONALES
162
CARRERAS PROFESIONALES
CIBERTEC
LENGUAJE DE PROGRAMACIN II
163
Resumen
Los usos de Lenguaje de Expresiones es obtener las propiedades de un objeto directamente a travs de estas formas: ${myObj.myProperty}$ ${myObj["myProperty"]}$ ${myObj[varWithTheName]}$
Si desea saber ms acerca de estos temas, puede consultar la siguiente pgina. http://java.sun.com/javaee/5/docs/tutorial/doc/bnahq.html Aqu hallar definiciones sobre los temas planteados.
CIBERTEC
CARRERAS PROFESIONALES