java--web

JAVA-WEB
一、Servlet
1,jsp或html 是客户端,用来发送请求,通过表单或其他形式
请求方法按一般有get和post。
2,servlet用来处理请求;收到请求之后,调用service方法,处理请求
service方法中的参数是:
ServletRequest,子接口HttpServletRequest:封装了所有请求信息,包括请求方法、请求参数等;
--获取请求参数:1)String getParameter(String name)[根据前端参数名获取参数值]
2)String[] getParameterValues(String name)[根据前端参数名获取多个参数值,例如用于多选]
3)Enumeration getParameterNames():返回参数名对应的Enumeration对象
ServletResponse,子接口HttpServletResponse:封装了所有响应信息,用该接口的方法来给用户响应
3,GernericServlet是一个Servlet,是Servlet和ServletConfig的实现类,并且是一个抽象类,使用这个抽象类,会使开发更加简洁
4,HttpServlet:因为使用Http的request和response在GenericServlet需要强转,所以定义该接口,该Servlet继承自GenericServlet
实际开发中直接使用HttpServlet,会根据请求方式直接复写doXXX方法即可

二、JSP
在web.xml中指定出错的代码,然后配置指定的跳转页面
没有JSP的时候,是在Servlet中发响应的,可以通过拼接的方式返回一个HTML页面,也可以直接返回一些数据
在JSP中写JAVA代码的话,需要加符号<% %>;

1,JSP的九大隐含对象(没有声明就可以使用的对象)
比较容易理解的是,JSP继承了HttpServlet,那么它本质也是一个Servlet类,JSP通常是前端页面,然后Servlet是Controller控制器
所以里边的对象基本通用,也就是前后端的参数数据可以通过这些对象传递
1)request:(因为JSP继承了HttpServlet,所以有request等对象,里面的方法自然可以使用)
2)response:基本不用
3)session:代表浏览器和服务器的一次会话
4)pageContext:页面上下问对象,该对象可以获取到其他八个隐含对象,代表当前页面的对象
5)application:代表当前WEB应用,是servletContext对象
6)config:当前jsp对应的servlet的一个servletConfig对象(几乎不用),常用于获取初始化参数(初始化参数是在web-xml中配置的initParam)
7)out:jspWriter对象,调用out.println()方法 --->换行 out。println('<br/>')
8)page:基本不用
9)exception:只有声明页面为 <%@isErrorPage=true %>时才可以使用该对象

域对象:(对属性作用域的范围从小到大)
pageContext :属性作用范围仅限于当前页面
request:属性作用范围仅限于一次请求(每刷新一次页面或者点击一个超链接就是一次请求)
session:属性作用范围仅限于一次会话(浏览器打开直到关闭算一次会话【在此期间会话不失效】)
application:属性作用范围限于当前web应用,范围最大的属性作用范围

2,和属性相关的方法(*):pageContext ,request,session,application都有以下方法
Object getAttribute(String name) : 获取指定的属性
Enumeration getAttributeNames():
void setAttribute(String name,Object o):
removeAttribute(String name):

3,请求转发和重定向
请求转发和重定向的本质区别:
请求转发只发了一次请求:地址栏是初次发出请求的地址
重定向发出了两次请求:地址栏不再是初次发出的请求的地址,地址栏为最后响应的地址

重定向可以防止表单的重复提交

区别二:
请求转发的资源只能是当前Web应用的资源
重定向可以重定向到任意资源

4,JSP指令
JSP指令是为了JSP引擎设计的,并不产生任何直接的输出,而只是告诉JSP引擎该如何处理JSP页面的其余部分
JSP引擎是将JSP转化为Servlet的玩意
1)<page>
.contentType:指定当前JSP页面的响应类型,实际调用的是response.setContentType("text/html;charset=UTF-8");
通常情况下都为text/html;charset=UTF-8,charset是返回页面的字符编码
.pageContentType:指定当前页面的字符编码格式
2)<include>
该指令可以静态引入其他jsp,然后JSP引擎将其翻译为同一个Servlet (同一个.java文件)
3)<taglib>

5,jsp标签
<jsp:include>标签
动态引入:一个jsp引入另一个jsp,生成的是两个.java文件
<jsp:forward>:相当于在页面写了一个请求转发

***** String转Integer:Integer.parseInt(String str);
***** Integer转String:1、toString 2,String.valueOf(Integer i);
-------------------------
MVC流程:
Model View Controller
model:数据层
view:视图层
controller:控制器
控制器是用来处理请求的。具体的做法是:1、Controller收到请求数据
2、调用数据库访问层(DAO)的方法进行处理,DAO处理完之后返回结果给Controller,DAO可以看作是model层
3、Controller收到返回的处理结果,可能会把DAO的返回值放到request中,然后转发或者重定向给View
4、view根据在controller中放入request域对象拿到相应数据,然后进行展示
JSP--->Servlet--->DAO--->DB :只能左边的依赖右边的,不能右边的依赖左边的,是为了保证DAO还可以重用
controller到jsp,可以请求转发或者重定向,但是要注意,请求转发的域对象只能是request,不然拿不动。重定向的域对象是session或application


-------------------------------------
6,案例
query:

查询的时候必须有返回,其余操作看业务
查询返回单个对象SingleObj
查询返回所有对象List

模糊查询
场景:页面任意条件模糊查询
SQL:“SELECT id,name,address,phone FROM customer WHERE” + "name LIKE ? AND address LIKE ? AND phone LIKE ?"

delete:
JSP上的jQuery提示信息:
--“确定要删除吗?”
$(function(){
$(".delete").click(function(){

var flag = confirm("确定要删除吗?");
return false;
});
});
-----自测confirm 函数

insert:

public void addCustomer(request,response){
//获取表单参数
String id = request.getParameter("id");
String name = request.getParameter("name");
String address = request.getParameter("address");
//校验名字是否被占用(****)
//首先调用方法查数据库,例如,查询该名字在数据库的总数,返回值为数值型
//若返回值大于0,说明该名字被占用;否则可以继续添加
//1,如果名字被占用,则在页面显示一个错误信息提示:用户名name已经被占用,请重新输入!
1)具体实现:当该名字被占用时,可以在request作用域中添加属性message,值为“:用户名name已经被占用,请重新输入!”
//return 结束方法
//-->request.setAttribute("message",":用户名name已经被占用,请重新输入!")
//然后在页面获取message的值,并显示(request.getAttribute("message");) 页面显示的方法:out.print();
//将参数交给DAO处理,可以线封装成一个请求对象
Customer customer = new Customer();
customer.setId(id);
customer.setAddress(address);
customer.setName(name);
customerDao.add(customer);
//跳转道成功页面
response.senRedirect("/success.jsp");
}
success.jsp 中一般有一个返回按钮,因为新增成功之后可以跳转到首页

EL(Expression Language) 可以代替原本在JSP中冗长的java代码


update:
update页面必须有一个隐藏域来保存要修改的对象的id,不然就不知道改的是谁
先显示要修改的页面(SELECT)
再修改(UPDATE)


7,会话与状态管理(cookie ,session)

1),cookie:
cookie机制是在客户端保持状态的机制,当浏览器访问服务器时,服务器响应时会在响应头中加入一个小文本文件(cookie,实际上在服务器执行:Cookie cookie = new cookie("name","value")
然后response.addCookie()),

一旦浏览器保存了这个文本文件(没有禁用cookie),那么在下次请求服务器时,请求头中就会带上这个cookie回传给服务器

2),session:
session是在服务器端保持状态:服务器首先检查客户端请求是否具又sessionId,如果有的话,则不为其创建;否则为请求创建sessionId
这个sessionId将会响应给客户端保存(在cookie中保存),但是由于cookie经常被禁用,这时候就找不到sessionId了,这是可以采用URL重写的方法:即把sessionId拼接在URL后面
1,Cookie禁用时怎么用session:URL重写
将sessionId作为URL的一个参数
方法:encodeURL();
encodeRedirectURL();

2,HttpSession相关API
1)获取session对象:request.getSession()
2)属性相关的:setAttribute,getAttribute,removeAttribute
3)使其失效:invalidate()
4)设置最大失效时间:setMaxInactiveInterval

session实现购物车:
String path = request.getContextPath();这个是获取资源的绝对路径

session 防止表单重复提交:

在服务器端,生成一个唯一的标识符,将它存入session,同时将它写入表单的隐藏字段中,然后将表单页面发给浏览器,

用户录入信息后点击提交,在服务器端,获取表单中隐藏字段的值,与session中的唯一标识符比较,相等说明是首次提交

,就处理本次请求,然后将session中的唯一标识符移除;不相等说明是重复提交,就不再处理。

原文:https://blog.csdn.net/myhuashengmi/article/details/52694077

使用HttpSession实现验证码
--->1,在原表单页面生成一个验证码的图片,生成图片的同时,需要把该图片中的文字放入到session中
--->2,在原表单页面,定义一个文本框,用于输入验证码
--->3,在目标servlet中获取session和表单域中验证码的值
--->4,比较两个值是否一致,若一致,受理请求,同时把session中的值清空;若不一致,则直接通过重定向的方式返回原表单,并提示用户”验证码错误!“
//验证码servlet
//将验证码的值放入session中
StringBuilder = sb;
sb.append("checkCode");
request.getSession().setAttribute("checkCode",sb);
//目标servlet
//1,获取验证码的值
String sessionCode = request.getSession().getAttribute("checkCode");
//2,获取表单域文本框的验证码的值
String requestCode = request.getAttribute("checkCode");
//3,比对,如果一致,则显示,不一致,重定向到index.jsp输入页面
if(!(requestCode != null && sessionCode.equals(requestCode))){
  request.getSession().setAttribute("message","验证码不一致!");
  response.sendRedirect(request.getContextPath() + "/index.jsp");
  return;
}
System.out.println("受理请求");

 javabean

有无参构造器的类就是Javabean,为了反射能用,必须有无参构造器


8,EL表达式
1),基本语法:${域对象范围.getAttribute().getProperties()}

EL 全名为Expression Language

EL 语法很简单,它最大的特点就是使用上很方便。接下来介绍EL主要的语法结构:

${sessionScope.user.sex}

所有EL都是以${为起始、以}为结尾的。上述EL范例的意思是:从Session的范围中,取得

用户的性别。假若依照之前JSP Scriptlet的写法如下:

User user =(User)session.getAttribute("user");

String sex =user.getSex( );

两者相比较之下,可以发现EL 的语法比传统JSP Scriptlet 更为方便、简洁。

  2),.与 [ ] 运算符

EL 提供 . 和 [ ] 两种运算符来导航数据。下列两者所代表的意思是一样的:

${sessionScope.user.sex}等于${sessionScope.user["sex"]}

和 [ ] 也可以同时混合使用,如下:

${sessionScope.shoppingCart[0].price}

回传结果为shoppingCart中第一项物品的价格。

不过,以下两种情况,两者会有差异:

(1) 当要存取的属性名称中包含一些特殊字符,如. 或 – 等并非字母或数字的符号,就一定要使用 [ ],

例如:${user.My-Name }

上述是不正确的方式,应当改为:${user["My-Name"] }

(2) 我们来考虑下列情况:

${sessionScope.user[data]}

此时,data 是一个变量,假若data的值为"sex"时,那上述的例子等于${sessionScope.user.sex};

假若data 的值为"name"时,它就等于${sessionScope.user.name}。

因此,如果要动态取值时,就可以用上述的方法来做,但无法做到动态取值。

3), EL 变量

EL 存取变量数据的方法很简单,例如:${username}。它的意思是取出某一范围中名称为username的变量。

因为我们并没有指定哪一个范围的username,所以它的默认值会先从Page 范围找,假如找不到,

再依序到Request、Session、Application范围。假如途中找到username,就直接回传,不再继续找下去,

但是假如全部的范围都没有找到时,就回传null,当然EL表达式还会做出优化,页面上显示空白,而不是打印输出NULL。

属性范围(jstl名称)

EL中的名称

Page

PageScope

Request

RequestScope

Session

SessionScope

Application

ApplicationScope

我们也可以指定要取出哪一个范围的变量:

范例

说明

${pageScope.username}

取出Page范围的username变量

${requestScope.username}

取出Request范围的username变量

${sessionScope.username}

取出Session范围的username变量

${applicationScope.username}

取出Application范围的username变量

其中,pageScope、requestScope、sessionScope和applicationScope都是EL 的隐含对象,

由它们的名称可以很容易猜出它们所代表的意思,

例如:${sessionScope.username}是取出Session范围的username 变量。这种写法是不是比之前JSP 的写法:

String username =(String) session.getAttribute("username");容易、简洁许多.


4),自动转变类型

EL 除了提供方便存取变量的语法之外,它另外一个方便的功能就是:自动转变类型,我们来看下面这个范例:

${param.count + 20}

假若窗体传来count的值为10时,那么上面的结果为30。之前没接触过JSP 的读者可能会认为上面的例子是理所当然的,

但是在JSP 1.2 之中不能这样做,原因是从窗体所传来的值,它们的类型一律是String,所以当你接收之后,必须再将它转为其他类型,

如:int、float 等等,然后才能执行一些数学运算,下面是之前的做法:

String str_count =request.getParameter("count");

int count =Integer.parseInt(str_count);

count = count + 20;
所以,注意不要和java的语法(当字符串和数字用“+”链接时会把数字转换为字符串)搞混淆喽。


5),EL 隐含对象

JSP有9个隐含对象,而EL也有自己的隐含对象。EL隐含对象总共有11 个

隐含对象

类型

说明

PageContext

javax.servlet.ServletContext

表示此JSP的PageContext

PageScope

java.util.Map

取得Page范围的属性名称所对应的值

RequestScope

java.util.Map

取得Request范围的属性名称所对应的值

sessionScope

java.util.Map

取得Session范围的属性名称所对应的值

applicationScope

java.util.Map

取得Application范围的属性名称所对应的值

param

java.util.Map

如同ServletRequest.getParameter(String name)。回传String类型的值

paramValues

java.util.Map

如同ServletRequest.getParameterValues(String name)。回传String[]类型的值

header

java.util.Map

如同ServletRequest.getHeader(String name)。回传String类型的值

headerValues

java.util.Map

如同ServletRequest.getHeaders(String name)。回传String[]类型的值

cookie

java.util.Map

如同HttpServletRequest.getCookies()

initParam

java.util.Map

如同ServletContext.getInitParameter(String name)。回传String类型的值

 

不过有一点要注意的是如果你要用EL输出一个常量的话,字符串要加双引号,不然的话EL会默认把你认为的常量当做一个变量来处理,

这时如果这个变量在4个声明范围不存在的话会输出空,如果存在则输出该变量的值。

 

属性(Attribute)与范围(Scope)

与范围有关的EL 隐含对象包含以下四个:pageScope、requestScope、sessionScope 和

applicationScope,它们基本上就和JSP的pageContext、request、session和application一样,所以笔者在这里只稍略说明。

不过必须注意的是,这四个隐含对象只能用来取得范围属性值,即JSP中的getAttribute(String name),却不能取得其他相关信息,

例如:JSP中的request对象除可以存取属性之外,还可以取得用户的请求参数或表头信息等等。

但是在EL中,它就只能单纯用来取得对应范围的属性值,

例如:我们要在session 中储存一个属性,它的名称为username,在JSP 中使用session.getAttribute("username")来取得username 的值,

 但是在EL中,则是使用${sessionScope.username}来取得其值的。

 

cookie

所谓的cookie是一个小小的文本文件,它是以key、value的方式将Session Tracking的内容记录在这个文本文件内,这个文本文件通常存在于浏览器的暂存区内。

JSTL并没有提供设定cookie的动作,因为这个动作通常都是后端开发者必须去做的事情,而不是交给前端的开发者。

假若我们在cookie 中设定一个名称为userCountry的值,那么可以使用${cookie.userCountry}来取得它。

 

header 和headerValues

header 储存用户浏览器和服务端用来沟通的数据,当用户要求服务端的网页时,会送出一个记载要求信息的标头文件,例如:用户浏览器的版本、用户计算机所设定的区域等其他相关数据。假若要取得用户浏览器的版本,即${header["User-Agent"]}。另外在鲜少机会下,有可能同一标头名称拥有不同的值,此时必须改为使用headerValues 来取得这些值。

 

注意:因为User-Agent 中包含“-”这个特殊字符,所以必须使用“[]”,而不能写成

$(header.User-Agent)。

 

initParam

就像其他属性一样,我们可以自行设定web 站台的环境参数(Context),当我们想取得这些参数initParam就像其他属性一样,

我们可以自行设定web 站台的环境参数(Context),当我们想取得这些参数

<?xml version="1.0"encoding="ISO-8859-1"?>

<web-appxmlns="http://java.sun.com/xml/ns/j2ee"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"

version="2.4">:

<context-param>

<param-name>userid</param-name>

<param-value>mike</param-value>

</context-param>:

</web-app>

那么我们就可以直接使用${initParam.userid}来取得名称为userid,其值为mike 的参数。

下面是之前的做法:String userid =(String)application.getInitParameter("userid");

param和paramValues
在取得用户参数时通常使用一下方法:

request.getParameter(String name)
request.getParameterValues(String name)

 

在 EL中则可以使用param和paramValues两者来取得数据。
${param.name}
${paramValues.name}

这里param 的功能和request.getParameter(String name)相同,而paramValues和
request.getParameterValues(String name)相同。如果用户填了一个表格,表格名称为username,则我们就可以使用${param.username}来取得用户填入的值。

 

看到这里,大家应该很明确EL表达式只能通过内置对象取值,也就是只读操作,如果想进行写操作的话就让后台代码去完成,毕竟EL表达式仅仅是视图上的输出标签罢了。

pageContext

我们可以使用 ${pageContext}来取得其他有关用户要求或页面的详细信息。下表列出了几个比较常用的部分

Expression

说明

${pageContext.request.queryString}

取得请求的参数字符串

${pageContext.request.requestURL}

取得请求的URL,但不包括请求之参数字符串,即servlet的HTTP地址。

${pageContext.request.contextPath}

服务的webapplication的名称

${pageContext.request.method}

取得HTTP的方法(GET、POST)

${pageContext.request.protocol}

取得使用的协议(HTTP/1.1、HTTP/1.0)

${pageContext.request.remoteUser}

取得用户名称

${pageContext.request.remoteAddr}

取得用户的IP地址

${pageContext.session.new}

判断session是否为新的,所谓新的session,表示刚由server产生而client尚未使用

${pageContext.session.id}

取得session的ID

${pageContext.servletContext.serverInfo}

取得主机端的服务信息

 这个对象可有效地改善代码的硬编码问题,如页面中有一A标签链接访问一个SERVLET,如果写死了该SERVLET的HTTP地址

那么如果当该SERVLET的SERVLET-MAPPING改变的时候必须要修改源代码,这样维护性会大打折扣。


6),EL算术运算

表达式语言支持的算术运算符和逻辑运算符非常多,所有在Java语言里支持的算术运算符,表达式语言都可以使用;

甚至Java语言不支持的一些算术运算符和逻辑运算符,表达式语言也支持。

 

Code
<%@ page contentType="text/html; charset=gb2312"%>
<html>
<head>
<title>表达式语言 - 算术运算符</title>
</head>
<body>
<h2>表达式语言 - 算术运算符</h2>
<hr>
<table border="1" bgcolor="aaaadd">
<tr>
<td><b>表达式语言</b></td>
<td><b>计算结果</b></td>
</tr>
<!-- 直接输出常量 -->
<tr>
<td>\${1}</td>
<td>${1}</td>
</tr>
<!-- 计算加法 -->
<tr>
<td>\${1.2 + 2.3}</td>
<td>${1.2 + 2.3}</td>
</tr>
<!-- 计算加法 -->
<tr>
<td>\${1.2E4 + 1.4}</td>
<td>${1.2E4 + 1.4}</td>
</tr>
<!-- 计算减法 -->
<tr>
<td>\${-4 - 2}</td>
<td>${-4 - 2}</td>
</tr>
<!-- 计算乘法 -->
<tr>
<td>\${21 * 2}</td>
<td>${21 * 2}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3/4}</td>
<td>${3/4}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3 div 4}</td>
<td>${3 div 4}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3/0}</td>
<td>${3/0}</td>
</tr>
<!-- 计算求余 -->
<tr>
<td>\${10%4}</td>
<td>${10%4}</td>
</tr>
<!-- 计算求余 -->
<tr>
<td>\${10 mod 4}</td>
<td>${10 mod 4}</td>
</tr>
<!-- 计算三目运算符 -->
<tr>
<td>\${(1==2) ? 3 : 4}</td>
<td>${(1==2) ? 3 : 4}</td>
</tr>
</table>
</body>
</html>


<%@ page contentType="text/html; charset=gb2312"%>
<html>
<head>
<title>表达式语言 - 算术运算符</title>
</head>
<body>
<h2>表达式语言 - 算术运算符</h2>
<hr>
<table border="1" bgcolor="aaaadd">
<tr>
<td><b>表达式语言</b></td>
<td><b>计算结果</b></td>
</tr>
<!-- 直接输出常量 -->
<tr>
<td>\${1}</td>
<td>${1}</td>
</tr>
<!-- 计算加法 -->
<tr>
<td>\${1.2 + 2.3}</td>
<td>${1.2 + 2.3}</td>
</tr>
<!-- 计算加法 -->
<tr>
<td>\${1.2E4 + 1.4}</td>
<td>${1.2E4 + 1.4}</td>
</tr>
<!-- 计算减法 -->
<tr>
<td>\${-4 - 2}</td>
<td>${-4 - 2}</td>
</tr>
<!-- 计算乘法 -->
<tr>
<td>\${21 * 2}</td>
<td>${21 * 2}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3/4}</td>
<td>${3/4}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3 div 4}</td>
<td>${3 div 4}</td>
</tr>
<!-- 计算除法 -->
<tr>
<td>\${3/0}</td>
<td>${3/0}</td>
</tr>
<!-- 计算求余 -->
<tr>
<td>\${10%4}</td>
<td>${10%4}</td>
</tr>
<!-- 计算求余 -->
<tr>
<td>\${10 mod 4}</td>
<td>${10 mod 4}</td>
</tr>
<!-- 计算三目运算符 -->
<tr>
<td>\${(1==2) ? 3 : 4}</td>
<td>${(1==2) ? 3 : 4}</td>
</tr>
</table>
</body>
</html>

 上面页面中示范了表达式语言所支持的加、减、乘、除、求余等算术运算符的功能,读者可能也发现了表达式语言还支持div、mod等运算符。

而且表达式语言把所有数值都当成浮点数处理,所以3/0的实质是3.0/0.0,得到结果应该是Infinity。

 

如果需要在支持表达式语言的页面中正常输出“$”符号,则在“$”符号前加转义字符“\”,否则系统以为“$”是表达式语言的特殊标记。


7),EL关系运算符

关系运算符

说明

范例

结果

== 或 eq

等于

${5==5}或${5eq5}

true

!= 或 ne

不等于

${5!=5}或${5ne5}

false

< 或 lt

小于

${3<5}或${3lt5}

true

> 或 gt

大于

${3>5}或{3gt5}

false

<= 或 le

小于等于

${3<=5}或${3le5}

true

>= 或 ge

大于等于

5}或${3ge5}

false

表达式语言不仅可在数字与数字之间比较,还可在字符与字符之间比较,字符串的比较是根据其对应UNICODE值来比较大小的。

注意:在使用EL 关系运算符时,不能够写成:
${param.password1} = =${param.password2}
或者
${ ${param.password1 } = = ${param.password2 } }
而应写成
${ param.password1 = =param.password2 }


8),EL逻辑运算符

逻辑运算符

范例

结果

&&或and

交集${A && B}或${A and B}

true/false

||或or

并集${A || B}或${A or B}

true/false

!或not

非${! A }或${not A}

true/false


Empty 运算符

Empty 运算符主要用来判断值是否为空(NULL,空字符串,空集合)。

 

条件运算符

${ A ? B : C}

EL 部分引用自:https://www.cnblogs.com/czs1982/p/3966748.html

9,JSTL:java标准标签库
JSTL是为了补充html的标签而编写的,方便在jsp页面中的标签里面写Java代码
常用的有
1)<c:set></c:set>
2)<c:remove></c:remove>
3)<c:choose>
<c:when test="${score eq 70}">
业务逻辑1
</c:when>
<c:when test="">
业务逻辑2
</c:when>
<c:when test="">
业务逻辑3
</c:when>
<c:otherwise>
业务逻辑4
</c:otherwise>
</c:choose>


4)<c:forEach var="user" items="${userMap}">


</c:forEach>

posted on 2019-05-13 09:44  哈皮的玩偶  阅读(209)  评论(0编辑  收藏  举报