Apache目录结构
Web 目录结构
WEB-INF下包括下列文件
WEB-INF目录结构
1. WEB-INF是java的WEB应用的安全目录。所谓安全目录就是客户端无法访问,只有服务器端可以访问的目录。
2. Web.xml,项目部署文件。
3. Classes文件夹,用以放置*.class文件
4. Lib文件夹,用以存放需要的jar包。
MyEclipseJavaWeb项目目录结构
Web项目虚拟路径
http://localhost:8080/MyFirstWebApp/index.jsp MyFirstWebApp 叫做Web虚拟路径
可以在下面这个地方配置
访问路径就变成了http://localhost:8080/HelloWorld/index.jsp
修改tomcat默认的端口号
在conf/server.xml文件
<Connector port=”8080”
Protocol=”HTTP/1.1”
connectionTimeout=”20000”
redirecPort=”8443”
/>
JSP:
JSP全名为Java Server Pages 其根本是一个简化的Servlet设计,他实现了在Java当中使用标准HTML标签。Jsp是一种动态网页技术标准,也是JavaEE标准。JSP和Servlet一样,是在服务器端执行的。
常见动态网站开发技术对比
JSP |
Java平台,安全性高,适合开发大型的,企业级的Web应用程序。 |
Asp.net |
.Net平台,简单易学。但是安全性以及跨平台性差。 |
Php |
简单,高效,成本低开发周期短,特别适合中小型企业Web应用开发。(LAMP:Linux+Apache+MySQL+PHP) |
Jsp基础语法
Page指令 通常位于jsp页面的顶端,同一个页面可以有多个page指令。
Include指令 将一个外部文件嵌入到JSP文件中,同时解析这个页面中的JSP语句。
Taglib指令 使用标签库定义新的自定义标签,在JSP页面中启用定制行为。
Page指令语法
<%@ page 属性1=“属性值” 属性2=“属性值1,属性值2…” 属性n=“属性值n”%>
属性 |
描述 |
默认值 |
Language |
指定JSP页面使用的脚本语言 |
Java |
Import |
通过该属性来引用脚本语言中使用到的类文件 |
无 |
ContentType |
指定JSP页面所采用的编码方式 |
Text/html ISO-8859-1 |
JSP 注释
HTML的注释 |
<!—html注释--> //客户端可见 |
JSP注释 |
<%--html注释--%> //客户端不可见 |
JSP脚本注释 |
//单行注释 /**/多行注释 |
JSP脚本
<% Java代码 %> |
Jsp声明
<%! Java代码%> <%! String s="ISI"; //声明了一个变量 int add(int x,int y){ //声明了一个方法 return x+y; } %> |
JSP表达式
<% =表达式%> //注意:表达式不以分号结束 你好,<%=s%> <%=add(3,2) %> |
JSP页面声明周期
JspService()方法被调用来处理客户端的请求。对每一个请求,JSP引擎创建一个新的线程来处理该请求。如果有多个客户端同时请求该JSP文件,则JSP引擎会创建多个线程。每个客户端请求对应一个线程。以多线程方式执行可以大大降低对系统的资源需求,提高系统的并发量和响应时间。但要注意多线程的编程带来的程序的同步问题,由于该servlet始终常驻与内存,所以响应时间非常快。
打印九九乘法表
<%! String printCalculate(){ String s=""; for(int i = 1;i<=9;i++){ for(int j=1;j<=i;j++){
s+=(i+"*"+j+"="+(j*i)+" "); } s+="<br/>"; } return s; }
//使用脚本打印乘法表 void printCalculate1(JspWriter out)throws IOException{ String s=""; for(int i = 1;i<=9;i++){ for(int j=1;j<=i;j++){
out.println(i+"*"+j+"="+(j*i)+" "); } out.println("<br/>"); } } %> <%=printCalculate()%> <%--脚本方式调用要加分号 --%> <%printCalculate1(out); %> |
Get |
以明文的方式通过URL提交数据,数据在URL中可以看到。提交的数据最多不超过2KB。安全性较低但效率比post方式高。适合提交数据量不大,安全性不高的数据。比如:搜索、查询等 |
Post |
将用户提交信息封装在HTML HEADER中。适合提交数据量大,安全性高的用户信息。比如:注册、修改、上传等。 |
JSP内置对象
JSP内置对象是Web容器创建的一组对象,不使用new关键字就可以使用的内置对象。
JSP九大内置对象 |
介绍 |
Out对象 |
Out对象是JspWriter类的实例,是向客户端输出内容常用的对象。 常用方法: 1. Void println() 向客户端打印字符串 2. Void clear()清除缓冲区的内容,如果在flush之后调用会抛出异常。 3. Void clearBuffer();清除缓冲区的内容,如果在flush之后调用不会抛出异常。 4. Void flush() 将缓冲区内容输出到客户端。 5. Int getBufferSize() 返回缓冲区以字节数的大小,如不设缓冲则为0 6. Int getRemaining() 返回缓冲区还剩余多少可用。 7. Boolean isAutoFlush() 返回缓冲区满时,是自动清空还是抛出异常。 8. Void close() 关闭输出流。 |
Request对象 |
客户端的请求信息被封装在request对象中,通过它才能了解客户的需求,然后做出响应。他是HttpServletRequest类的实例。Request对象具有请求域,即完成客户的请求之前,该对象一直有效。 常用方法: 1. String getParameter(String name) 返回name指定参数的字符。 2. String[] getParameterValues(String name) 返回包含参数name的所有值得数组。 3. Void setAttribute(String,Object);存储此请求中的属性。 4. Object getAttribute(String name) 返回指定属性的属性值。 5. String getContentType() 得到请求体的MIME类型。 6. String getProtocol() 返回请求用的协议类型及版本号。 7. String getServerName() 返回接受请求的服务器主机名。 8. int getServerPort()返回服务器接收此请求所用的端口号 9. String getCharacterEncoding()返回字符编码方式 10. Void setCharacterEncoding()设置请求的字符编码方式。 11. Int getContentLength() 返回请求体的长度(以字节数) 12. String getRemoteAddr()返回发送此请求的客户端IP地址 13. String getRealPath(String path) 返回一虚拟路径的真实路径 14. String request.getContextPath()返回上下文路径 |
Response对象 |
Response 对象包含了响应客户请求的有关信息,但在JSP中很少用到。它是HttpServletReponse类的实例。Response对象具有页面作用域,即访问一个页面时,该页面内的response对象只能对这次的访问有效,其他页面的response对象对当前页面无效。 1. String getCharacterEncoding() 返回响应用的是何种字符编码。 2. Void setContentType(String type) 设置响应的MIME类型。 3. PrintWriter getWriter() 返回可以向客户端输出字符的一个对象(注意比较:PrintWriter与内置out的区别)PrintWriter输出总是提前于内置out对象。 4. SendRedirect(java.lang.String location) 重新定向客户端请求 |
Session对象 |
Session 表示客户端与服务器的一次会话。 Web中的session指的是用户在浏览某个网站时,从进入网站到浏览器关闭所经过的这段时间,也就是用户浏览这个网站所花费的时间,session实际上是一个特定的时间概念。 在服务器的内存中保存着不同用户的session。 1. Session对象在第一个JSP页面被装载时自动创建,完成会话期管理。 2. 从一个客户打开浏览器并连接到服务器开始,到客户端关闭浏览器离开这个服务器结束,被称为一个会话。 3. 当一个客户访问一个服务器时,可能会在服务器的几个页面之间切换,服务器应当通过某种方法指导这是一个客户,就需要session对象。 4. Session对象是HttpSession类的实例。 Session对象的方法 1. Long getCreationTime() 返回session创建时间。 2. Public String getId(); 返回session创建时JSP引擎为它设的唯一ID号。 3. Public Object setAttribute(String name,Object value) 使用指定名称将对象绑定到此会话。 4. Public Object getAttribute(String name) 返回此会话中的指定名称绑定在一起的对象,如果没有对象绑定在该名称下,则返回null。 5. String[] getValueNames() 返回一个包含此session N种可用属性的数组。 6. Int getMaxInctiveInterval(); 返回两次请求间隔多长时间此session被取消。(单位秒) 7. SetMaxInctiveInterval(); 设置生存时间。 Session的生命周期 创建: 当客户端第一次访问某个jsp或者servlet时候,服务器会为当前会话创建一个SessionId,每次客户端向服务器发送请求时,都会将此SessionId携带过去,服务器会对此SessionID进行校验。 活动:某次会话当中通过超链接打开的新页面属于同一次会话。 只要当前会话页面没有全部关闭,重新打开新的浏览器窗口访问同一项目资源时属于同一次会话。除非本次会话的所有页面都关闭后再重新访问某个Jsp或者Servlet将会创建新的会话。 销毁:1.session.invalidate()方法 2.Session过期(超时) 3. 服务器重启 |
application对象 |
Application 对象实现了用户间数据的共享,可存放全部变量。 Application 开始于服务器的启动,终止于服务器的关闭。 在用户的前后连接或不同用户之间的连接中,可以对Application对象的同一属性进行操作。 在任何地方对Application对象属性的操作,都将影响到其他用户对此的访问。 服务器的启动和关闭决定了Application对象的生命。 Application对象是ServletContext类的实例 常用方法: 1. Public void setAttribute(String name,Object value)使用指定名称将对象绑定到此会话。 2. Public Object getAttribute(String name) 返回与此会话中的指定名称绑定在一起的对象,如果没有对象绑定在该名称下,则返回null 3. Enumeration getAttributeNames()返回所有可用属性名的枚举 4. String getServerInfo() 返回JSP(SERVLET)引擎名及版本号 |
Page内置对象 |
Page对象就是指向当前jsp页面本身,有点像类中的this指针,它是java.lang.Object 类的实例。常用方法如下: 1. Class getClass() 返回此Object的类 2. Int hashCode() 返回此Object的hash码 3. Boolean equals(Object obj) 判断此Object是否与指定的Object对象相等。 4. Void copy(Object obj) 把此Object拷贝到指定的Object对象中。 5. Object clone() 克隆此Object对象 6. String toString() 把此Object对象转换成String类的对象 7. Void notify() 唤醒一个等待的线程 8. Void nofityAll() 唤醒所有等待的线程 9. Void wait(int timeout) 使一个线程处于等待直到timeout结束或被唤醒。 10. Void wait() 使一个线程处于等待直到被唤醒 |
PageContext对象 |
PageContext 对象提供了对JSP页面内所有的对象及名字空间的访问。 PageContext 对象可以访问到本页所在的session,也可以取本页所在的Application的某一属性值。 PageContext对象相当于页面中所有功能的集大成者。 PageContext 对象的本类名也叫pageContext. 常用方法: 1. JspWriter getOut() 返回当前客户端响应被使用的JspWriter流(out) 2. HttpSession getSession() 返回当前页中的HttpSession对象(session) 3. Object getPage() 返回当前页的Object对象(page) 4. ServletRequest getRequest() 返回当前页的ServletRequest对象(request) 5. ServletResponse getResponse() 返回当前页的ServletResponse对象(response) 6. Void setAttribute(String name,int scope) 设置属性及属性值。 7. Object getAttribute(String name,int scope) 在指定范围内取属性的值。 8. Int getAttributeScope(String name) 返回某属性的作用范围。 9. Void forward(String relativeUrlPath) 使当前页面重导到另一页面 10. Void include(String relativeUrlPath) 在当前位置包含另一文件 |
Config对象 |
Config对象是在一个Servlet初始化时,JSP引擎向它传递信息用的,此信息包括Servlet初始化时所要用到的参数(通过属性名和属性值构成)以及服务器的有关信息(通过传递一个ServletContext对象),常用方法: 1. ServletContext getServletContext() 返回含有服务器相关信息的ServletContext对象。 2. String getInitParameter(String name) 返回初始化参数的值。 3. Enumeration getInitParameterNames() 返回Servlet初始化所需所有参数的枚举。 |
Exception 对象 |
Exception对象是一个异常对象,当一个页面在运行过程中发生了异常,就产生这个对象。如果一个JSP页面要应用此对象,就必须把isErrorPage设为true,否则无法编译。他实际上是java.lang.Throwable的对象,常用方法如下: 1. String getMessage() 返回描述异常的信息 2. String toString() 返回关于异常的简短描述消息 3. Void printStackTrace() 显示异常及其栈轨迹 4. Throwable FillInStackTrace() 重写异常的执行栈轨迹 <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8" errorPage="exception.jsp"%> <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8" isErrorPage="true"%> |
使用URL传参中文乱码问题
修改tomcat配置文件server.xml
<Connector port=”8080”protocol=”HTTP/1.1”
connectionTimeout=”20000”
redirectPort=”8443”
URIEncoding=”utf-8”>
重启tomcat服务器
请求转发与请求重定向
请求重定向:客户端行为,response.sendRedirect(),从本质上讲等同于两次请求,前一次的请求对象不会保存,传递的参数丢失,地址栏的URL地址会改变。
请求转发:服务器行为,reqeust.getRequestDispatcher().forward(req,resp);是一次请求,转发后请求对象会保存,地址栏的URL地址不会改变。
Session超时时间
Tomcat默认session超时时间为30分钟。
设置session超时有两种方式。
1. session.setMaxInactiveInterval();//单位是秒
2. 在web.xml中配置
<session-config>
<session-timeout>
10
</session-timeout>
</session-config> //单位是分钟
Javabean
Javabeans 就是符合某种特定的规范的Java类。使用Javabeans的好处是解决代码重复编写,减少代码冗余,功能区分明确,提高代码的维护性。
Javabean设计原则
- Public class Students{
- Private String name;
- Private int age;
- Public Students(){}
- Public void setName(String name){this.name=name;}
- Public String getName()(return this.name;)
- Public void setAge(int age){this.age=age;}
- Public int getAge(){return this.age;}
- }<strong>
- </strong>
Jsp动作
JSP动作元素(action elements),动作元素为请求处理阶段提供信息。动作元素遵循XML元素的语法,有一个包含元素名的开始标签,可以有属性、可选的内容、与开始标签匹配的结束标签。
Jsp动作元素包括五类
第一类是与存取Javabean 有关的,包括:
<jsp:userBean><jsp:setProperty><jsp:getProperty>
第二类是JSP1.2就开始有的基本元素,包括6个动作元素
<jsp:include> <jsp:forward><jsp:param> <jsp:plugin> <jsp:params> <jsp:fallback>
第三类是JSP2.0新增加的元素,主要与JSP Document相关,包括六个元素
<jsp:root> <jsp:declaration> <jsp:scriptlet><jsp:expression> <jsp:text> <jsp:output>
第四类是JSP2.0新增的动作元素,主要是用来动态生成XML元素标签的值,包括3个动作
<jsp:attribute> <jsp:body><jsp:element>
第五类是JSP2.0新增的动作元素,主要是用在Tag File中,有2个元素
<jsp:invoke> <jsp:dobody>
在JSP页面中使用Javabeans
创建一个JavaBeans类
package com.po; /** * 用户类 */ public class Users { private String username; private String password; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } public Users(){} } |
1. 像使用普通java类一样,创建javabean实例。
<%@ page import="com.po.Users" %> <% Users user = new Users(); user.setPassword("123"); user.setUsername("lisi"); %> <h1>使用普通方式创建javaBean实例</h1> 用户名:<%=user.getUsername() %><br/> 密码:<%=user.getPassword() %> |
1. 在Jsp页面中通常使用jsp动作标签使用javabean
UserBeans动作
setProperty动作
getProperty动作
<jsp:useBean> 作用:在jsp页面中实例化或者在指定范围内使用javabean <jsp:useBean id=”标识符” class=”java类名” scope=”作用范围”/> 例: <jsp:useBean id="myUsers" class="com.po.Users" scope="page"></jsp:useBean> <h1>使用useBean创建javaBean实例</h1> 用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> 这里获取的值都为null |
<jsp:setProperty> 作用:给已经实例化的Javabean对象的属性赋值,一共有四种形式 1. <jsp:setProperty name=”javaBean实例名” property=”*”/> (跟表单关联) <jsp:useBean id="myUsers" class="com.po.Users" scope="page"></jsp:useBean> <h1>setProperty用法</h1><hr> <!-- 根据表单名字和Javabean名字自动匹配所有的属性 --> <jsp:setProperty property="*" name="myUsers"/> 用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> 2. <jsp:setProperty name=”javaBean实例名” property=”JavaBean属性名”/> (跟表单关联) 对部分属性进行匹配,这里只匹配用户名,密码显示为空 <jsp:useBean id="myUsers" class="com.po.Users" scope="page"></jsp:useBean> <h1>setProperty用法</h1><hr> <jsp:setProperty property="username" name="myUsers"/> 用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> 3. <jsp:setProperty name=”JavaBean实例名” property=”JavaBean属性名” value=”BeanValue”/>(手工设置) 和表单无关,通过手工赋值的方法。 <jsp:setProperty property="username" name="myUsers" value="lisi"/> <jsp:setProperty property="password" name="myUsers" value="123"/> 用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> 4. <jsp:setProperty name=”JavaBean实例名” property=”propertyName” param=”request对象中的参数名”/>(跟request参数关联) Param 可以是url地址传参的方式给javabean赋值,密码显示666 <form action="setProperty.jsp?mypass=666" method="post"> <jsp:setProperty property="password" name="myUsers" param="mypass"/> 用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> |
<jsp:getProperty> 作用: 获取指定Javabean对象的属性值 <jsp:getProperty name=”JavaBean实例名” property=”属性名”/> 用户名:<jsp:getProperty property="username" name="myUsers"/><br/> 密码:<jsp:getProperty property="password" name="myUsers"/> <%--用户名:<%=myUsers.getUsername() %><br/> 密码:<%=myUsers.getPassword() %> --%> |
Javabean的四个作用域范围
说明:使用useBeans的scope属性可以用来指定javabean的作用范围
Page |
仅在当前页面有效 |
Request |
可以通过HttpRequest.getAttribute()方法获取JavaBean对象 |
Session |
可以通过HttpSession.getAttribute()方法获取JavaBean对象 |
Application |
可以通过application.getAttribute()方法取得JavaBean对象 <jsp:useBean id="myUsers" class="com.po.Users" scope="application"></jsp:useBean> 用户名:<jsp:getProperty property="username" name="myUsers"/><br/> 密码:<jsp:getProperty property="password" name="myUsers"/><hr> <!-- 通过内置对象方式获取 --> 用户名<%=((Users)application.getAttribute("myUsers")).getUsername()%><br/> 密码:<%=((Users)application.getAttribute("myUsers")).getPassword()%> |
Model1简介
在Model 1 出现之前,整个Web应用的情况:几乎全部由JSP页面组成,JSP页面接收处理客户端的请求,对请求处理后直接作出响应。
Javabean的出现可以使jsp页面中使用Javabean封装的数据或者调用 Javabean的业务逻辑代码,这样大大提升了程序的可维护性。
例子:Model 1登录案例
Login.jsp 登录界面
- <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'index.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <body>
- <form action="dologin.jsp" method="post">
- 用户名:<input type="text" name="username"/><br/>
- 密码:<input type="password" name="password" value=""/><br/>
- <input type="submit" value="登录">
- </form>
- </body>
- </html>
Users.java Javabean
- package com.po;
- public class Users {
- private String username;
- private String password;
- public Users(){
- }
- public String getUsername() {
- return username;
- }
- public void setUsername(String username) {
- this.username = username;
- }
- public String getPassword() {
- return password;
- }
- public void setPassword(String password) {
- this.password = password;
- }
- }
UsersDAO.java 处理数据操作
- package com.dao;
- import com.po.Users;
- //用户的业务逻辑
- public class UsersDAO {
- //用户登录方法
- public boolean usersLogin(Users u){
- if("admin".equals(u.getUsername())&&"admin".equals(u.getPassword())){
- return true;
- }else{
- return false;
- }
- }
- }
dologin.jsp 处理登录,登录成功或失败分别跳转到不同页面
- <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%>
- <jsp:useBean id="loginUser" class="com.po.Users" scope="page"></jsp:useBean>
- <jsp:useBean id="userDao" class="com.dao.UsersDAO" scope="page"></jsp:useBean>
- <jsp:setProperty property="*" name="loginUser"/>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- if(userDao.usersLogin(loginUser)){
- session.setAttribute("loginUser",loginUser.getUsername() );
- request.getRequestDispatcher("login_success.jsp").forward(request, response);
- }else{
- response.sendRedirect("login_fail.jsp");
- }
- %>
Jsp状态管理
http协议的无状态性
无状态是指,当浏览器发送请求给服务器的时候,服务器响应客户端的请求。
但是当同一个浏览器再次发送请求给服务器的时候,服务器并不知道它就是刚才那个浏览器。
简单地说,就是服务器不会记得你,所以就是无状态协议。
Cookie
Cookie: 是Web服务器保存在客户端的一系列文本信息。
典型应用一:判断注册用户是否已经登录网站。
典型应用二:“购物车”的处理。
Cookie的作用:
对特定对象的追踪。
保存用户网页浏览记录与习惯。
简化登录。
安全风险:容易泄露用户的信息。
Jsp中创建与使用Cookie
创建Cookie对象 Cookie newCookie = new Cookie(String key ,Object value); 写入Cookie对象 Response.addCookie(new Cookie); 读取Cookie对象 Cookie[] cookies= request.getCookies(); |
Cookie的常用方法
Void setMaxAge(int expiry) |
设置cookie的有效期,以秒为单位 |
Void setValue(String value) |
在cookie创建后,对cookie进行赋值 |
String getName() |
获取cookie的名称 |
String getValue() |
获取cookie的值 |
Int getMaxAge() |
获取cookie的有效时间,以秒为单位 |
案例:记住用户的用户名和密码的登录状态。
login.jsp
- <%@ page language="java" import="java.util.*,java.net.*" contentType="text/html; charset=utf-8"%>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'index.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <body>
- <%
- String username="";
- String password="";
- Cookie[] cookies = request.getCookies();
- if(cookies != null && cookies.length>0){
- for(Cookie c: cookies){
- if(c.getName().equals("username")){
- username = URLDecoder.decode(c.getValue(),"utf-8");
- }
- if(c.getName().equals("password")){
- password = c.getValue();
- }
- }
- }
- %>
- <form action="dologin.jsp" method="post">
- 用户名:<input type="text" name="username" value="<%=username%>"/><br/>
- 密码:<input type="password" name="password" value="<%=password %>"/><br/>
- 记住我的登录状态<input type="checkbox" name="isUseCookie" checked="checked"/><br/>
- <input type="submit" value="登录"> <input type="reset" value="取消">
- </form>
- </body>
- </html>
dologin.jsp 处理登录逻辑,向客户端写Cookie
- <%@page import="java.net.CookieStore"%>
- <%@ page language="java" import="java.util.*,java.net.*" contentType="text/html; charset=utf-8"%>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'users.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <!--
- Cookie 中文乱码,需要使用 java.net.* 包里的类来处理
- -->
- <body>
- <h1>用户登录</h1>
- <%
- request.setCharacterEncoding("utf-8");
- //判断用户是否点击了记住登录状态
- String[] isUserCookies = request.getParameterValues("isUseCookie");
- if(isUserCookies != null && isUserCookies.length >0){
- //把用户名和密码保存在Cookie对象里面
- //使用URLEncoder 解决无法再Cookie中存储中文问题
- String username= URLEncoder.encode(request.getParameter("username"),"utf-8");
- String password=request.getParameter("password");
- Cookie usernameCookie = new Cookie("username",username);
- Cookie passwordCookie = new Cookie("password",password);
- //设置最大生存期限
- usernameCookie.setMaxAge(864000);
- passwordCookie.setMaxAge(864000);
- response.addCookie(usernameCookie);
- response.addCookie(passwordCookie);
- }else{
- //把已经保存的cookie 清空
- Cookie[] cookies = request.getCookies();
- if(cookies != null && cookies.length>0){
- for(Cookie c : cookies){
- if(c.getName().equals("username") || c.getName().equals("password")){
- c.setMaxAge(0);//设置最大时间为0,使其失效
- response.addCookie(c);
- }
- }
- }
- }
- %>
- <a href="users.jsp" target="_blank">查看用户名和密码</a>
- <hr>
- </body>
- </html>
users.jsp 读取cookie中用户的登录信息
- <%@ page language="java" import="java.util.*,java.net.*" contentType="text/html; charset=utf-8"%>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'users.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <body>
- <h1>查看用户信息</h1>
- <%
- String username="";
- String password="";
- Cookie[] cookies = request.getCookies();
- if(cookies != null && cookies.length>0){
- for(Cookie c: cookies){
- if(c.getName().equals("username")){
- username = URLDecoder.decode(c.getValue(),"utf-8");
- }
- if(c.getName().equals("password")){
- password = c.getValue();
- }
- }
- }
- %>
- <hr>
- 用户名:<%=username %><br/>
- 密码:<%=password %>
- </body>
- </html>
Session与Cookie对比
Session |
Cookie |
在服务器端保存用户信息 Session中保存的是Object类型 随会话的结束而将其存储的数据销毁 保存重要的信息 |
在客户端保存用户信息 Cookie保存的是String类型 Cookie可以长期保存在客户端 保存不重要的用户信息 |
Include指令
语法:
<%@ include file=”URL”%>
Include动作
语法:
<jsp:include page=”URL”flush=”true|false”/>
Page 表示要包含的页面
Flush 表示是否从缓冲区中读取页面
Include指令与include动作比较
|
Include指令 |
Jsp:include 动作 |
语法格式 |
<%@ include file=” ..”%> |
<jsp:include page=”..”> |
发生作用的时间 |
页面编译期间 |
请求期间 |
包含的内容 |
文件的实际内容 |
页面的输出 |
转换成Servlet |
主页面和包含页面转换为一个Servlet |
主页面和包含页面转换为独立的Servlet |
编译时间 |
较慢—资源必须被解析 |
较快 |
执行时间 |
稍快 |
较慢—每次资源必须被解析 |
Forward动作
语法:
<jsp:forward page=”URL”/>
等同于服务器转发:
Request.getRequestDispatcher(“/url”).forward(request,response);
Param动作
语法:
<jsp:param name=”参数名” value=”参数值”>
常常与 <jsp:forward> 一起使用,作为其子标签。
<jsp:forward page=”xxx.jsp”> <jsp:param name=”email” value=”lisi@163.com”> </jsp:forward> |
小案例:商品浏览记录
1.首先,打开mysql数据库,创建表和插入数据
- DROP TABLE IF EXISTS `items`;
- CREATE TABLE `items` (
- `id` int(11) NOT NULL auto_increment,
- `name` varchar(50) default NULL,
- `city` varchar(50) default NULL,
- `price` int(11) default NULL,
- `number` int(11) default NULL,
- `picture` varchar(500) default NULL,
- PRIMARY KEY (`id`)
- ) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8;
- -- ----------------------------
- -- ----------------------------
- INSERT INTO `items` VALUES ('1', '沃特篮球鞋', '佛山', '180', '500', '001.jpg');
- INSERT INTO `items` VALUES ('2', '安踏运动鞋', '福州', '120', '800', '002.jpg');
- INSERT INTO `items` VALUES ('3', '耐克运动鞋', '广州', '500', '1000', '003.jpg');
- INSERT INTO `items` VALUES ('4', '阿迪达斯T血衫', '上海', '388', '600', '004.jpg');
- INSERT INTO `items` VALUES ('5', '李宁文化衫', '广州', '180', '900', '005.jpg');
- INSERT INTO `items` VALUES ('6', '小米3', '北京', '1999', '3000', '006.jpg');
- INSERT INTO `items` VALUES ('7', '小米2S', '北京', '1299', '1000', '007.jpg');
- INSERT INTO `items` VALUES ('8', 'thinkpad笔记本', '北京', '6999', '500', '008.jpg');
- INSERT INTO `items` VALUES ('9', 'dell笔记本', '北京', '3999', '500', '009.jpg');
- INSERT INTO `items` VALUES ('10', 'ipad5', '北京', '5999', '500', '010.jpg');
2. 创建数据库帮助类,获取数据库连接对象 DBHelper.java
- package util;
- import java.sql.Connection;
- import java.sql.DriverManager;
- public class DBHelper {
- private static final String driver ="com.mysql.jdbc.Driver";//加载数据库驱动
- //连接数据库的URL地址
- private static final String url = "jdbc:mysql://localhost:3306/shopping?useUnicode=true&characterEncoding=UTF-8";
- private static final String username="root";//数据库用户名
- private static final String password="limeng";//数据库密码
- private static Connection conn= null;
- static{
- try {
- Class.forName(driver);
- } catch (ClassNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- //单例模式返回数据库连接对象
- public static Connection getConnection() throws Exception{
- if(conn== null){
- conn = DriverManager.getConnection(url,username,password);
- return conn;
- }
- return conn;
- }
- }
3.创建实体类Items.java 和数据库的字段对应
- package entity;
- //商品类
- public class Items {
- private int id;
- private String name;//名字
- private String city;//产地
- private int price;//价格
- private int number;//库存
- private String picture;//图片
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getCity() {
- return city;
- }
- public void setCity(String city) {
- this.city = city;
- }
- public int getPrice() {
- return price;
- }
- public void setPrice(int price) {
- this.price = price;
- }
- public int getNumber() {
- return number;
- }
- public void setNumber(int number) {
- this.number = number;
- }
- public String getPicture() {
- return picture;
- }
- public void setPicture(String picture) {
- this.picture = picture;
- }
- }
4.创建业务逻辑类 ItemsDAO.java 处理不同的业务逻辑
- package dao;
- import java.awt.event.ItemListener;
- import java.sql.Connection;
- import java.sql.PreparedStatement;
- import java.sql.ResultSet;
- import java.sql.SQLException;
- import java.util.ArrayList;
- import java.util.LinkedList;
- import java.util.List;
- import util.DBHelper;
- import entity.Items;
- //商品业务逻辑类
- public class ItemsDAO {
- //获取所有的商品信息
- public ArrayList<Items> getAllItems(){
- Connection conn = null;
- PreparedStatement stmt = null;
- ResultSet rs = null;
- ArrayList<Items> list = new ArrayList<Items>();
- try {
- conn = DBHelper.getConnection();
- String sql = "select * from items;";
- stmt = conn.prepareStatement(sql);
- rs = stmt.executeQuery();
- while(rs.next()){
- Items item = new Items();
- item.setId(rs.getInt("id"));
- item.setName(rs.getString("name"));
- item.setCity(rs.getString("city"));
- item.setNumber(rs.getInt("number"));
- item.setPrice(rs.getInt("price"));
- item.setPicture(rs.getString("picture"));
- list.add(item);
- }
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }finally{
- try {
- if(rs != null){
- rs.close();
- }
- if(stmt != null){
- stmt.close();
- }
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- return list;
- }
- //根据商品编号获取商品资料
- public Items getItemById(int id){
- Connection conn = null;
- PreparedStatement stmt = null;
- ResultSet rs = null;
- try {
- conn = DBHelper.getConnection();
- String sql = "select * from items where id=?;";
- stmt = conn.prepareStatement(sql);
- stmt.setInt(1, id);
- rs = stmt.executeQuery();
- if(rs.next()){
- Items item = new Items();
- item.setId(rs.getInt("id"));
- item.setName(rs.getString("name"));
- item.setCity(rs.getString("city"));
- item.setNumber(rs.getInt("number"));
- item.setPrice(rs.getInt("price"));
- item.setPicture(rs.getString("picture"));
- return item;
- }else{
- return null;
- }
- } catch (Exception e) {
- e.printStackTrace();
- return null;
- }finally{
- try {
- if(rs != null){
- rs.close();
- }
- if(stmt != null){
- stmt.close();
- }
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- //获取最近浏览信息
- public ArrayList<Items> getViewList(String list){
- String[] ids = list.split("#");
- ids = array_unique(ids);//去除重复的id
- ArrayList<Items> itemsList = new ArrayList<Items>();
- if(ids != null && ids.length>0){
- if(ids.length>=5){
- for(int i = 0;i<5;i++){
- Items items = getItemById(Integer.parseInt(ids[i]));
- itemsList.add(items);
- }
- return itemsList;
- }else{
- for(int i= 0;i<ids.length;i++){
- Items items = getItemById(Integer.parseInt(ids[i]));
- itemsList.add(items);
- }
- return itemsList;
- }
- }
- return null;
- }
- //去除数组中重复的记录
- public static String[] array_unique(String[] a) {
- // array_unique
- List<String> list = new LinkedList<String>();
- for(int i = a.length-1; i >=0; i--) {
- if(!list.contains(a[i])) {
- list.add(a[i]);
- }
- }
- return (String[])list.toArray(new String[list.size()]);
- }
- }
5. index.jsp 显示所有商品列表
- <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%>
- <%@ page import="dao.ItemsDAO,entity.Items" %>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <style type="text/css">
- div{
- float:left;
- margin: 10px;
- }
- div dd{
- margin:0px;
- font-size:10pt;
- }
- div dd.dd_name
- {
- color:blue;
- }
- div dd.dd_city
- {
- color:#000;
- }
- </style>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'index.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <body>
- <h1>商品展示</h1>
- <hr>
- <center>
- <table width="770" height="60" cellpadding="0" cellspacing="0" border="0">
- <tr>
- <td>
- <!-- 商品循环开始 -->
- <%
- ItemsDAO itemsDAO = new ItemsDAO();
- ArrayList<Items> list = itemsDAO.getAllItems();
- //开始循环
- if(list != null && list.size()>0){
- for(int i= 0;i<list.size();i++){
- Items item = list.get(i);
- %>
- <div>
- <dl>
- <dt>
- <a href="detail.jsp?id=<%=item.getId()%>"><img src="images/<%=item.getPicture()%>" width="120" height="90" border="1"></a>
- </dt>
- <dd class="dd_name"><%=item.getName() %></dd>
- <dd class="dd_city">产地<%=item.getCity() %> 价格¥<%=item.getPrice() %></dd>
- </dl>
- </div>
- <!-- 商品循环结束 -->
- <%
- }
- }
- %>
- </td>
- </tr>
- </table>
- </center>
- </body>
- </html>
6. detail.jsp 显示商品详情和最近浏览记录(最多显示5条)
- <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%>
- <%@ page import="dao.ItemsDAO,entity.Items" %>
- <%
- String path = request.getContextPath();
- String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
- %>
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html>
- <head>
- <base href="<%=basePath%>">
- <title>My JSP 'detail.jsp' starting page</title>
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
- <!--
- <link rel="stylesheet" type="text/css" href="styles.css">
- -->
- </head>
- <body>
- <h1>商品详情列表</h1><hr>
- <center>
- <table width="770" height="60" cellpadding="0" cellspacing="0" border="0">
- <tr>
- <!-- 商品详情 -->
- <%
- ItemsDAO dao = new ItemsDAO();
- Items item = dao.getItemById(Integer.parseInt(request.getParameter("id")));
- if(item != null){
- %>
- <td width="70%" valign="top">
- <table>
- <tr>
- <td rowspan="4"><img src="images/<%=item.getPicture()%>" width="200" height="160"/></td>
- </tr>
- <tr>
- <td><B><%=item.getName() %></B></td>
- </tr>
- <tr>
- <td>产地:<%=item.getCity()%></td>
- </tr>
- <tr>
- <td>价格:<%=item.getPrice() %>¥</td>
- </tr>
- </table>
- </td>
- <%
- }
- %>
- <%
- String list ="";
- //从客户端获得Cookies集合
- Cookie[] cookies = request.getCookies();
- //遍历这个Cookies集合
- if(cookies!=null&&cookies.length>0)
- {
- for(Cookie c:cookies)
- {
- if(c.getName().equals("ListViewCookie"))
- {
- list = c.getValue();
- }
- }
- }
- list+=request.getParameter("id")+"#";
- //如果浏览记录超过1000条,清零.
- String[] arr = list.split("#");
- if(arr!=null&&arr.length>0)
- {
- if(arr.length>=1000)
- {
- list = "";
- //清空后前900条
- for(int i =arr.length-101;i<arr.length;i++){
- list+= arr[i];
- }
- }
- }
- Cookie cookie = new Cookie("ListViewCookie",list);
- response.addCookie(cookie);
- %>
- <!-- 浏览过的商品 -->
- <td width="30%" bgcolor="#EEE" align="center">
- <br>
- <b>您浏览过的商品</b><br>
- <!-- 循环开始 -->
- <%
- ArrayList<Items> itemlist = dao.getViewList(list);
- if(itemlist!=null&&itemlist.size()>0 )
- {
- System.out.println("itemlist.size="+itemlist.size());
- for(Items i:itemlist)
- {
- %>
- <div>
- <dl>
- <dt>
- <a href="details.jsp?id=<%=i.getId()%>"><img src="images/<%=i.getPicture() %>" width="120" height="90" border="1"/></a>
- </dt>
- <dd class="dd_name"><%=i.getName() %></dd>
- <dd class="dd_city">产地:<%=i.getCity() %> 价格:<%=i.getPrice() %> ¥ </dd>
- </dl>
- </div>
- <%
- }
- }
- %>
- <!-- 循环结束 -->
- </td>
- </tr>
- </table>
- </center>
- </body>
- </html>