1.目前主流的技术是前后端分离(比如: Spring Boot + Vue/React),我们会讲的.[看一下]
2. JSP技术使用在逐渐减少,但使用少和没有使用是两个意思,一些老项目和中小公司还在使用JSP,工作期间,你很有可能遇到JSP
3.老韩小结: JSP使用在减少(但是现阶段还是有必要学一下, 让我们的技术栈更加全面),我们最佳
技术策略:JSP (能基本使用,能看懂,能维护相关项目,不用深入)
1.servlet做页面过于繁琐,html又无法动态获得,所以引入JSP技术
2.jsp= html+java代码+标签+ javascript+css
1. JSP全称是Java Server Pages, Java的服务器页面,就是服务器端渲染技术
2. JSP这门技术的最大的特点在于,写JSP就像在写HTML
相比html而言,htmI只能为用户提供静态数据,而JSP技术允许在页面中嵌套java代码,为用户提供动态数
相比Servlet而言,Servlet很难对数据进行排版,而jsp除了可以用java代码产生动态数据的同时,也很容易对数据进行排版。
3. jsp技术基于Servlet,你可以理解成JSP就是对Servlet的包装(本质JSP就是Servlet)。
4.会使用JSP的程序员,再使用thymeleaf是非常容易的事情,几乎是无缝接轨。
<%--
Created by IntelliJ IDEA.
User: 64301
Date: 2023/12/20
Time: 16:31
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
jsp的简单求和计算器
简单计算器
<%
//1.在jsp的 % 标签中 可以写java代码
int i = 10;
int j = 20;
int res = i + j;
//2.jsp内置对象可以直接使用 比如out
out.println(i + "+" + j + "=" + res);
%>
注意:
1. jsp页面不能像HTML页面,直接用浏览器运行。只能通过浏览器访问Tomcat来访问jsp页面
2.如何设置jsp模板
1. jsp页面本质是一个Servlet程序(本质就是java程序),其性能是和java关联的,只是长得丑.
2.第1次访问jsp页面的时候。Tomcat服务器会把jsp页面解析成为一个java源文件。并且对它进行编译成为.class字节码程序。看下Cal.jsp对应的cal_ jsp.java和cal_ jsp.class文件
1. language表示jsp翻译后是什么语言文件,只支持java
2. contentType表示jsp返回的数据类型,对应源码中response.setContentType()参数值
3. pageEncoding属性表示当前jsp页面文件本身的字符集
4. import属性跟java源代码中一样。用于导包,导类
1.声明脚本的格式是: <%! 声明java代码%>
2.作用:定义jsp的需要属性、方法、静态代码块和内部类等
<%--
Created by IntelliJ IDEA.
User: 银小海
Date: 2023/12/20 Time: 16:59
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
jsp声明脚本
<%!
//这里我们可以声明该jsp需要使用的属性,方法,静态代码块, 内部类
//也就是给 statement.jsp 对应的 statement_jsp 类定义成员
//1. 属性
private String name = "jack";
private int age;
private static String company;
//2 方法
public String getName() {
return name;
}
//3. 静态代码块
static {
company = "字节跳动";
}
%>
1.表达式脚本的格式是:<%=表达式%>
2.表达式脚本的作用是:在jsp 页面上输出数据
3.脚本中的表达式不能以分号结束。
<%--
Created by IntelliJ IDEA.
User: 银小海
Date: 2023/12/20 Time: 17:05
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
表达式脚本的使用
个人信息
<%
String name = "yinhai";
String email = request.getParameter("email");
%>
用户名: <%=name%>
工作是: <%="java工程师"%>
年龄: <%=request.getParameter("age")%>
电邮: <%=email%>
1.代码脚本的语法是:<% java代码%>
2.代码脚本的作用是:可以在jsp页面中,编写我们需要的功能(使用java )
3.可以由多个代码脚本块组合完成一个完整的 java语句。
4.代码脚本还可以和表达式脚本一起组合使用,在jsp页面上输出数据
<%@ page import="com.yinhai.domain.Monster" %>
<%@ page import="java.util.ArrayList" %><%--
Created by IntelliJ IDEA.
User: 银小海
Date: 2023/12/20 Time: 17:10
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
演示代码脚本
演示代码脚本
<%
//创建ArrayList ,放入两个monster对象
ArrayList monsterList = new ArrayList<>();
monsterList.add(new Monster(1, "牛魔王", "芭蕉扇"));
monsterList.add(new Monster(2, "蜘蛛精", "吐口水"));
%>
id
名字
技能
<%
for (int i = 0; i < monsterList.size(); i++) {
//先取出monster对象
Monster monster = monsterList.get(i);
%>
<%=monster.getId()%>
<%=monster.getName()%>
<%=monster.getSkill()%>
<%
}
%>
//在java片段中,仍然是java的注释
String email = "[email protected]";
/*
多行注释
*/
1、JSP内置对象Q),是指Tomcat在翻译jsp页面成为Servlet后,内部提供的九大对象,叫内置对象
2、内置对象,可以直接使用,不需要手动定义
1. out向客户端输出数据,out.println("); ;
2. request客户端的http请求
3. response响应对象
4. session会话对象
5. application对应ServletContext
6. pageContext jsp页面的上下文,是一个域对象,可以setAttribue(), 作用范围只是本页面
7. exception异常对象, getMessage()
8. page代表jsp这个实例本身
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
jsp内置对象
jsp内置对象
<%
//梳理jsp的内置对象
//out 类型是 JspWriter 父类就是 Writer.
out.println("jsp out");
//request是HttpServletRequest
request.getParameter("age");
//response就是 HttpServletResponse
//response.sendRedirect("http://www.baidu.com");
//session 就是 HttpSession
session.setAttribute("job", "PHP工程师");
//application类型就是ServletContext
application.setAttribute("name", "老韩老师");
//pageContext 可以存放数据(属性), 但是该数据只能在本页面使用
pageContext.setAttribute("age", 100);
//exception 异常对象 使用比较少
//page 内置对象,类似 this
out.println("page=" + page);
//config 内置对象的类型就是ServletConfig
String pwd = config.getInitParameter("pwd");
%>
age: <%=pageContext.getAttribute("age")%>
1. pageContext (域对象,存放的数据只能在当前页面使用)
2. request (域对象,存放的数据在一次request请求有效)
3. session(域对象,存放的数据在一次会话有效)
4. application(域对象,存放的数据在整个web应用运行期间有效)
1.scope文件
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
scope文件
<%
//在不同的域对象中,放入数据
//1. 因为四个域对象,是不同的对象,因此name(key) 相同时,并不会冲突
pageContext.setAttribute("k1", "pageContext数据(k1)");
request.setAttribute("k1", "request数据(k1)");
session.setAttribute("k1", "session数据(k1)");
application.setAttribute("k1", "application数据(k1)");
//做一个请求转发的操作
//路径应该怎么写,请不清楚地小伙伴,去看韩老师讲的web路径专题
//request.getRequestDispatcher("/scope2.jsp").forward(request, response);
//做一个重定向
String contextPath = request.getContextPath();//返回的就是 web路径=>/jsp
//response.sendRedirect("/jsp/scope2.jsp");
response.sendRedirect(contextPath + "/scope2.jsp");
%>
四个域对象,在本页面获取数据的情况
pageContext-k1: <%=pageContext.getAttribute("k1")%>
request-k1: <%=request.getAttribute("k1")%>
session-k1: <%=session.getAttribute("k1")%>
application-k1: <%=application.getAttribute("k1")%>
2.scope2
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
scope2.js
在scope2页面获取数据的情况
pageContext-k1: <%=pageContext.getAttribute("k1")%>
request-k1: <%=request.getAttribute("k1")%>
session-k1: <%=session.getAttribute("k1")%>
application-k1: <%=application.getAttribute("k1")%>
1.域对象是可以像Map一样存取数据的对象。四个域对象功能一样。不同的是它们对数据的存储范围
2.从存储范围(作用域范围看) pageContext < request < session < application
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
aa.jsp
aa.jsp
<%--
1. jsp提供了很多标签,但是因为jsp不是重点,老韩就讲一个常用forward
2. jsp:forward 本质就是 等价 request.getRequestDispatcher("/bb.jsp").for...
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
bb.jsp
bb.jsp页面
需求分析:使用jsp完成-一个简单的计算器,需求如图
1)要求在前端页面对输入的num1和num2进行校验[提示:正则表达式],必须是整数
2)验证成功,提交数据给服务器,能够显示结果
3)点击超链接,可以返回界面
4)其它完整考虑[思考题]
1.calUI,前端判断是否满足数的要求并创建对应的ui
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
JSP计算器
JSP计算器
2.calServlet
@WebServlet(name = "CalServlet",urlPatterns = "/calServlet")
public class CalServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
System.out.println("CalServlet 被调用...");
//思路 ==> 代码
//1. 是Servlet
//2. 接收数据
//String num1 = request.getParameter("num1");
//String num2 = request.getParameter("num2");
//进行转换->int
double num1 = WebUtils.parseDouble(request.getParameter("num1"), 0);
double num2 = WebUtils.parseDouble(request.getParameter("num2"), 0);
String oper = request.getParameter("oper");
double res = 0; //使用变量来接收运算结果
//3. 完成计算
if ("+".equals(oper)) {
res = num1 + num2;
} else if ("-".equals(oper)) {
res = num1 - num2;
} else if ("*".equals(oper)) {
res = num1 * num2;
} else if ("/".equals(oper)) {
res = num1 / num2;
} else {
System.out.println(oper + " 不正确...");
}
//4. 把结果保存到域对象[request, session, servletContext]
// 因为一次请求对应一次计算, 所以我建议将结果保存到request
// 把结果组织到一个字符串中., 方便我们在下一个页面显示
// java基础: String.format ,可以格式化字符串
String formatRes = String.format("%s %s %s = %s", num1, oper, num2, res);
request.setAttribute("res", formatRes);
//System.out.println("formatRes= " + formatRes);
//5. 转发到显示页面 calRes.jsp
request.getRequestDispatcher("/cal/calRes.jsp").forward(request, response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}
3.calRes
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
计算结果
计算结果
<%=request.getAttribute("res")%>
<%--返回重新来玩一把--%>
返回重新来玩一把~
拓展
1. EL表达式全称: Expression Language,是表达式语言
2. EL表达式主要是代替jsp页面的表达式脚本<% = request.getAttribute("xx")%>
3. EL表达式输出数据的时,比jsp的表达式脚本简洁
4. EL表达式基本语法: ${key1},可以理解就是一个语法糖
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
el表达式的快速入门
el表达式的快速入门
<%
request.setAttribute("name", "hello EL表达式");
%>
<%--
1. 如果name是 null, request.getAttribute() 返回的是null字符串
2. 如果name是 null, ${name}, 返回的""
--%>
jsp表达式脚本
名字= <%=request.getAttribute("name") == null ? "": request.getAttribute("name")%>
el表达式
名字= ${name}
EL表达式常用输出Bean的普通属性、数组属性、List 集合属性和map集合属性
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
el 表达式输出数据演示
el 表达式输出数据演示
<%
//创建Book 对象,放入相关的属性
//private String name;//书名
//private String[] writer;//作者
//private List reader;//读者
//private Map topics;//评讲
Book book = new Book();
book.setName("昆虫总动员");
book.setWriter(new String[]{"jack", "tom"});
ArrayList readers = new ArrayList<>();
readers.add("老韩");
readers.add("老李");
book.setReader(readers);//放入readers
//创建topics
HashMap topics = new HashMap<>();
topics.put("topic1", "这是我看过的最好的动画片");
topics.put("topic2", "不错的电影~~");
book.setTopics(topics);
//把book 放入到request域对象
request.setAttribute("bookkey", book);
%>
book对象: ${bookkey}
.
book.name= ${bookkey.name}
book.writer= ${bookkey.writer}
book.writer[0]= ${bookkey.writer[0]}
book.readers= ${bookkey.reader}
book.readers第2个= ${bookkey.reader.get(1)}
book.readers第2个= ${bookkey.reader[1]}
book.topics= ${bookkey.topics}
book.topics第一个评论= ${bookkey.topics["topic1"]}
1.基本语法语法: ${运算表达式}
语法:${运算表达式}
2.关系运算
3.逻辑运算
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
el的运算符
el的运算符
<%
request.setAttribute("num1", 90);
request.setAttribute("num2", 30);
%>
num1+num2=${num1 + num2}
num1>num2=${num1 > num2}
1. empty运算可以判断-个数据是否为空,如果为空,返回true,否则返回false
2.以下几种情况为空
值为null
值为空串的时
值是Object类型数组,长度为零
list集合,元素个数为零
map集合,元素个数为零
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
el empty的运算
el empty的运算
<%
request.setAttribute("k1", null);
request.setAttribute("k2", "");
request.setAttribute("k3", new Object[]{});
request.setAttribute("k4", new ArrayList<>());
request.setAttribute("k5", new HashMap<>());
request.setAttribute("score", 70);
%>
k1是否为空= ${empty k1}
k2是否为空= ${empty k2}
k3是否为空= ${empty k3}
k4是否为空= ${empty k4}
k5是否为空= ${empty k5}
是否及格= ${score >= 60 ? "及格": "不及格"}
1.表达式1?表达式2:表达式3
2.如果表达式1的值为真,返回表达式2的值,反之,返回表达式3的值。.
EL四个域对象
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
演示el的四个常用的隐藏对象(域对象)
演示el的四个常用的隐藏对象(域对象)
<%
request.setAttribute("k1", "request-k1数据");
pageContext.setAttribute("k1", "pageContext-k1数据");
session.setAttribute("k1", "session-k1数据");
application.setAttribute("k1", "application-k1数据");
%>
jsp脚本方式获取
request域中的k1= <%=request.getAttribute("k1")%>
el方式来获取域对象的数据
request域中的k1= ${requestScope.k1}
pageContext域中的k1= ${pageScope.k1}
session域中的k1= ${sessionScope.k1}
application域中的k1= ${applicationScope.k1}
pageContext对象的使用
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
pageContext 对象的使用
pageContext 对象的使用
<%--
//通过request对象来获取和HTTP协议相关的数据
request.getScheme() 它可以获取请求的协议
request.getServerName() 获取请求的服务器 ip 或域名
request.getServerPort() 获取请求的服务器端口号
getContextPath() 获取当前工程路径
request.getMethod() 获取请求的方式(GET 或 POST)
request.getRemoteHost() 获取客户端的 ip 地址
session.getId() 获取会话的唯一标识
--%>
<%--
1.我们可以通过pageContext.request.xx 俩获取和http协议相关的信息
2.相当于替代 request.getMethod()....
--%>
协议: ${ pageContext.request.scheme }
服务器 ip:${ pageContext.request.serverName }
服务器端口:${ pageContext.request.serverPort }
工程路径:${ pageContext.request.contextPath }
请求方法:${ pageContext.request.method }
客户端 ip 地址:${ pageContext.request.remoteHost }
会话id :${ pageContext.session.id }
使用jsp表达式脚本获取如上信息
ip地址: <%=request.getRemoteHost() %>
使用el表达式形式获取信息-简化写法
<%
pageContext.setAttribute("req", request);
%>
ip地址(简化获取): ${req.remoteHost}
获取请求方法(简化获取): ${req.method}
1. JSTL标签库是指JSP Standard Tag Library JSP标准标签库
2. EL表达式是为了替换jsp中的表达式脚本,JSTL是为了替换代码脚本。这样jsp页面变得更佳洁
3. JSTL由五个标签库组成
4.使用前需要导入对应的jar包
底层仍然是被翻译成JAVA编译的
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
jstl的快速入门
jstl的快速入门
<%--
1. c:if ... 类似
2. if(10>2){
out.println("10 > 2 成立~
")
}
--%>
10 > 2 成立~
1.介绍: < c;setrscope= "request" var= "username" value="hello~"/>
1.等价 域对象.setAttribute(key,value);
2.scope属性设置保存到哪个域
page表示PageContext域( 默认值)
request表示Request域
session表示Session域
application表示ServletContext域
3. var属性设置key是什么
4. value属性设置值
1.介绍:
如果为真输出标签中间的值
1. if标签用来做if判断。
2. test属性表示判断的条件(用EL表达式输出)
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
c:if标签使用
${num1} > ${num2}
1.介绍:多路判断。跟switch... case... default...非常接近
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
c:choose标签的使用
c:choose标签的使用
<%
request.setAttribute("score", 50);
//request.setAttribute("k1", "request-k1的值");
//session.setAttribute("k1", "session-k1的值");
//application.setAttribute("k1", "application-k1的值");
//pageContext.setAttribute("k1", "pageContext-k1的值~");
%>
<%--
1. 如果${requestScope.score} 那么就明确的指定从request域对象取出数据
2. 如果${score}, 这是就按照从小到大的域范围去获取 pageContext->request->session->application
--%>
k1=${k1}
${score}-成绩优秀
${score}-成绩一般, 及格了
${score}-没有及格,下次努力~
1.介绍: c:forEach标签用来遍历输出,主要有4种形式
普通遍历输出i到j
遍历数组
遍历Map
遍历List
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
c:forEach 标签
c:forEach 标签
第1种遍历方式从i到j
<%--
1.遍历 1 到 5,
2. 输出 begin 属性设置开始的索引 end 属性设置结束的索引
3. var 属性表示循环的变量(也是当前正在遍历到的数据)
4. 等价 for (int i = 1; i <= 5; i++) {}
5. 在默认情况下, i 每次会递增1
--%>
- 排名=${i}
第2种遍历方式:遍历数组
<%
request.setAttribute("sports", new String[]{"打篮球", "乒乓球"});
%>
<%--
1. items 遍历的集合/数组
2. var 遍历到的数据
3. 等价 for (Object item: arr) {}
--%>
运动名称= ${sport}
第3种遍历方式:遍历Map
<%
Map map = new HashMap<>();
map.put("key1", "北京");
map.put("key2", "上海");
map.put("key3", "天津");
request.setAttribute("cities", map);
%>
<%--
1. items 遍历的map集合
2. var 遍历到的数据
3. entry.key 取出key
4. entry.value 取出值
--%>
城市信息: ${city.key}--${city.value}
第4种遍历方式:遍历List
<%
List monsters = new ArrayList<>();
monsters.add(new Monster(100, "小妖怪", "巡山的"));
monsters.add(new Monster(200, "大妖怪", "做饭的"));
monsters.add(new Monster(300, "老妖怪", "打扫位置的"));
request.setAttribute("monsters", monsters);
%>
<%--
items 表示遍历的集合
var 表示遍历到的数据
begin 表示遍历的开始索引值 ,从0开始计算
end 表示结束的索引值
step 属性表示遍历的步长值
varStatus 属性表示当前遍历到的数据的状态,可以得到step,begin,end等属性值
//老师提示, 对于jstl标签,能看懂,会使用即可
--%>
妖怪的信息: ${monster.id}-${monster.name}-${monster.skill}
需求分析:使用jsp+servlet+jstl+el完成查询 - 显示案例,需求如图
1)点击超链接,可以显示所有的妖怪信息
2)要求显示的数据在Servlet准备,并放入到request域对象
3)扩展,如果要求增加根据id条件过滤,怎么处理? 一定要自己先思考
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
显示所有的妖怪
显示所有的妖怪
id
name
skill
<%-- 使用c:foreach循环显示即可 显示id > 200--%>
${monster.id}
${monster.name}
${monster.skill}
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
查询妖怪
查询妖怪
点击查询所有的妖怪
public class QueryServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1. 准备要显示的数据--> 从DB
ArrayList list = new ArrayList<>();
list.add(new Monster(100, "牛魔王", "芭蕉扇"));
list.add(new Monster(200, "狐狸精", "美人计"));
list.add(new Monster(300, "白骨精", "吃人骨头"));
//2. 把list放入到request域, 供jsp页面使用
request.setAttribute("monsters", list);
//3. 请求转发 list.jsp
request.getRequestDispatcher("/hm/list.jsp")
.forward(request,response);
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doPost(request, response);
}
}