关于利用maven搭建ssm的博客,我们一起来探讨下问的最多的问题

  • 时间:
  • 浏览:0
  • 来源:极速快3_快3计划三期必中_极速快3计划三期必中

前言

  开心一刻

    有个同学去非洲援建,刚到工地接待他的施工员是个黑人,他就用英语跟人家交流,黑人没做声。 你你这个 他又用法语,黑人还是没说话。 你你这个 他用手去比划。黑人终于开口了:瞎比划嘎哈,整个工地都中国人

前提背景

  在利用maven/eclipse搭建ssm(spring+spring mvc+mybatis)一文的现象反馈中,大体分一有有有另1个:404和页面无数据;至于800,此人 认为比较好处里,按照提示进行处里就好,本文就不讨论800了

  404

    主要也是一种生活

    1、webapp未发布

      相关资源未部署,累似 webapp未发布部署,累似 如下

      不你你这个 你你这个 webapp,main下的java、resources、webapp,maven依赖不会须要部署到tomcat,不然就不完全,就会居于各种各样的少内容的现象;

    2、请求URL不对

      你你这个 并不一定是你你这个 你你这个 新入行的小伙伴容易经常出现的现象

      肯能工程正常部署,请求URL经常出现404,很有肯能是亲戚亲戚人们请求的URL不对;亲戚亲戚人们到tomcat的home目录下看看工程不是正常部署,累似 如下

      还能不到看看工程发布的内容(现象1中须要发布的内容)不是不会;肯能工程部署正常,而请求的URL又经常出现404,那不不不,就在等你的URL写错了

    404的处里方案你你这个 你你这个 :确认工程不是正确部署到tomcat,确认请求的URL不是正确,基本或者我确认这两点也就能找到现象了;后文不不再完全的讲404,亲戚亲戚人们将重点放进去下面你你这个 现象上

  页面无数据

    具体的现象应该是原先的:当亲戚亲戚人们请求:http://localhost:端口/工程名/personController/showPerson时,数据正常显示如下

    当亲戚亲戚人们直接请求jsp时,不到title如此 数据,如下

    这是为哪此?

    对于你你这个 现象一时候开始了并不一定没太在意,你你这个 你你这个 提示小伙伴去看servlet的四大作用域和jsp的九大内置对象,上边陆陆续续你你这个 你你这个 小伙伴都问了我,包括评论区留言、站内消息、QQ私聊等

    站内信

    评论区

    QQ

    我发现你你这个 现象好像不会个别小伙伴的现象,你你这个 你你这个 新入门的小伙伴都居于原先的现象,下面亲戚亲戚人们就对此次现象就行一有有有另1个完全的探究;后续篇幅较长,基础铺垫较多,希望亲戚亲戚人们耐心想看

现象探究

  servlet与servlet容器

    狭义上来讲,servlet指的你你这个 你你这个 接口:javax.servlet.Servlet,广义上来讲,servlet指的是servlet规范:Java Servlet API 标准;javax.servlet.Servlet与servlet容器不会servlet规范下的产物。Java Servlet API是Servlet容器和Servlet之间的接口,它定义了Servlet的各种妙招,还定义了Servlet容器传送给Servlet的对象类,其中最重要的是请求对象ServletRequest和响应对象ServletResponseo你你这个 有有有另1个对象不会由Servlet容器在客户端调用Servlet时产生的,Servlet容器把客户请求信息封放进ServletRequest对象中,你你这个 把你你这个 有有有另1个对象都传送给要调用的Servlet,Servlet处里就让把响应结果写入ServletResponse,你你这个 由Servlet容器把响应结果发送到客户端。

    Servlet与Servlet容器的关系有点硬像枪和子弹的关系,枪是为子弹而生,而子弹又让枪有了杀伤力。并不一定它们是彼此依存的,你你这个 又相互独立发展,你你这个 切不会为了适应工业化生产的结果。从技术高度来说是为了解耦,通过标准化接口来相互商务相互合作。Servlet 容器作为一有有有另1个独立发展的标准化产品,目前它的种类你你这个 你你这个 ,包括Jetty、tomcat、resin、JBoss、WebSphere、Weblogic等,哪此不会成熟期期的句子的句子图片 图片 的句子图片 的句子是什么图片 是什么的句子是什么的产品,有专门的公司肯能组织进行维护,亲戚亲戚人们直接拿来用就好。

    亲戚亲戚人们约定下,下文中的servet指的不会servlet接口:javax.servlet.Servlet,servlet容器指的是:Tomcat,Web服务器与Servlet容器是同一有有有另1个内容(实际是有区别的,具体区别亲戚亲戚人们自行去查阅)

    Tomcat容器模型如下

    Tomcat响应客户请求过程

      其中,①处表示Web服务器接收到客户端发出的HTTP请求后,转发给Servlet容器,再由Servlet容器转发给具体的Servlet实例进行请求的处里;②处表示Servlet实例将处里结果封放进去ServletResponse中,再由Servlet容器把ServletResponse发给Web服务器,通知Web服务器以HTTP响应的妙招把结果发送到客户端。也你你这个 你你这个 说,与客户端直接打交道的是tomcat(servlet容器),而不会亲戚亲戚人们的Servlet实例,而真正处里请求的才是亲戚亲戚人们的Servlet实例。

    说的简单点,亲戚亲戚人们自定义的Servlet,并不一定是对servlet容器在业务层面的拓展,合适业务定制一样;亲戚亲戚人们能不到原先理解,servlet容器对servlet提供技术支持,而servlet对servlet容器提供业务拓展,两者缺一不可,缺了技术支持,业务拓展实施不起来,缺了业务拓展,技术支持如此 现实意义。Servlet容器封装了底层错综复杂的技术实现,使亲戚亲戚人们能不到专注于业务实现,而Servlet容器与业务实现之间的纽带你你这个 你你这个 Servlet接口,它是亲戚亲戚人们对Servlet容器进行业务拓展的标准,你你这个 你你这个 亲戚亲戚人们的业务须要实现Servlet接口。套用阿基米德的杠杆原理:给Servlet容器多个servlet实例,Servlet容器还你充足的web服务。

  JSP

    示例代码:our-servlet

    亲戚亲戚人们先来看看在jsp经常出现就让,servlet如可输出页面,HelloServlet如下

    不仅仅是业务数据,还包括静态页面的内容,通通在servlet返回,肯能页面简单,如此 处里不不会 接受,你你这个 肯能页面像淘宝、京东那样非常错综复杂,你能想象吗?太容易出错了,一旦静态页面的元素少了肯能多了内容,都真不知道如可排查,面对茫茫多的out.write,就不到哭的份了。你你这个 你你这个 jsp就应运而生了。

    JSP全称:Java Server Pages,允许在传统静态网页HTML中插入Java代码片段(Scriptlet)和JSP标签,以错综复杂页面静态内容的开发。但须要注意的是,JSP文件的本质还是Servlet,只不过与Servlet不同的是,JSP是专门用于进行数据展示的Servlet;JSP最终会被Tomcat解析成Servlet,在Tomcat内置了一有有有另1个JSP解析引擎,当第一次访问该JSP页面时,解析引擎会将JSP页面解析成Servlet,你你这个 再由Servlet将动态数据、静态内容完全输出到浏览器供展示。亲戚亲戚人们来看看jsp解析后的文件在哪里、内容是哪此,以示例中的index.jsp为例。路径如下图

    index_jsp.java

/*
 * Generated by the Jasper component of Apache Tomcat
 * Version: Apache Tomcat/7.0.47
 * Generated at: 2019-04-08 13:14:31 UTC
 * Note: The last modified time of this file was set to
 *       the last modified time of the source file after
 *       generation to assist with modification tracking.
 */
package org.apache.jsp;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;

public final class index_jsp extends org.apache.jasper.runtime.HttpJspBase
    implements org.apache.jasper.runtime.JspSourceDependent {

  private static final javax.servlet.jsp.JspFactory _jspxFactory =
          javax.servlet.jsp.JspFactory.getDefaultFactory();

  private static java.util.Map<java.lang.String,java.lang.Long> _jspx_dependants;

  private javax.el.ExpressionFactory _el_expressionfactory;
  private org.apache.tomcat.InstanceManager _jsp_instancemanager;

  public java.util.Map<java.lang.String,java.lang.Long> getDependants() {
    return _jspx_dependants;
  }

  public void _jspInit() {
    _el_expressionfactory = _jspxFactory.getJspApplicationContext(getServletConfig().getServletContext()).getExpressionFactory();
    _jsp_instancemanager = org.apache.jasper.runtime.InstanceManagerFactory.getInstanceManager(getServletConfig());
  }

  public void _jspDestroy() {
  }

  public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
        throws java.io.IOException, javax.servlet.ServletException {

    final javax.servlet.jsp.PageContext pageContext;
    javax.servlet.http.HttpSession session = null;
    final javax.servlet.ServletContext application;
    final javax.servlet.ServletConfig config;
    javax.servlet.jsp.JspWriter out = null;
    final java.lang.Object page = this;
    javax.servlet.jsp.JspWriter _jspx_out = null;
    javax.servlet.jsp.PageContext _jspx_page_context = null;


    try {
      response.setContentType("text/html; charset=utf-8");
      pageContext = _jspxFactory.getPageContext(this, request, response,
                  null, true, 8192, true);
      _jspx_page_context = pageContext;
      application = pageContext.getServletContext();
      config = pageContext.getServletConfig();
      session = pageContext.getSession();
      out = pageContext.getOut();
      _jspx_out = out;

      out.write("\n");
      out.write("<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\" \"http://www.w3.org/TR/html4/loose.dtd\">\n");
      out.write("<html>\n");
      out.write("<head>\n");
      out.write("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n");
      out.write("<title>index</title>\n");
      out.write("</head>\n");
      out.write("<body>\n");
      out.write("    <div>\n");
      out.write("        Welcome, my friend!\n");
      out.write("    </div>\n");
      out.write("</body>\n");
      out.write("</html>");
    } catch (java.lang.Throwable t) {
      if (!(t instanceof javax.servlet.jsp.SkipPageException)){
        out = _jspx_out;
        if (out != null && out.getBufferSize() != 0)
          try { out.clearBuffer(); } catch (java.io.IOException e) {}
        if (_jspx_page_context != null) _jspx_page_context.handlePageException(t);
        else throw new ServletException(t);
      }
    } finally {
      _jspxFactory.releasePageContext(_jspx_page_context);
    }
  }
}
View Code

    发现熟悉的out.write又回来了,你你这个 你你这个 此时的out.write不会亲戚亲戚人们手动写的,你你这个 你你这个 Tomcat解析jsp后生成的;肯能jsp没变动,jsp只会在第一次被调用时解析、编译一次,后续的请求不会由编译后的servlet处里,亲戚亲戚人们来验证下,如可验证了? 不变index.jsp内容再请求index.jsp,看看上图中文件的修改时间会不不变

    发现文件的修改时间如此 变动,也你你这个 你你这个 说上边的的结论:肯能jsp没变动,jsp只会在第一次被调用时解析、编译一次是对的。感兴趣的亲戚人们能不到去看下Tomcat的源码,看看具体的实现细节。

    人们肯能会问:为哪此不将jsp的内容直接返回给浏览器?亲戚亲戚人们要明白你你这个 :浏览器不到解析html、css、js,除此之外的内容它解析不了,如此 亲戚亲戚人们能直接将jsp的内容返回给浏览器吗?你你这个 你你这个 上边有处里过程,最终由servlet将静态内容返回给浏览器。你你这个 爱问的小伙伴肯能又会问了:浏览器为哪此不到解析:html、css、js,这涉及到浏览器规范的现象,除非你有能力改变你你这个 规范,让浏览器支持你不不的内容,你你这个 现象不做陷得的讨论,亲戚亲戚人们姑且认为这是浏览器的限制,既然亲戚亲戚人们改变不了你你这个 限制,那就适应你你这个 限制。

  Servlet四大作用域与JSP九大内置对象

    Servlet四大作用域包括:page域、request域、session域、application域,作用域指的是变量的有效期限,具体如下

      当变量的作用域是page,它的有效范围只在当前jsp页面里有效;

      当变量的作用域是request,它的有效范围是当前请求周期,所谓请求周期,你你这个 你你这个 指从http请求发起,到服务器处里时候开始了,返回响应的整个过程,在你你这个 过程中肯能使用forward的妙招跳转了多个jsp页面,在哪此页面里你都能不到使用你你这个 变量;

      当变量的作用域是session,它的有效范围是当不会话,何为当不会话,你你这个 你你这个 指从用户打开浏览器时候开始了,到用户关闭浏览器的整个过程,你你这个 过程肯能富含多个请求响应;

      当变量的作用域是application,它的有效范围是整个应用,何为整个应用,你你这个 你你这个 指从应用启动,到应用时候开始了;

    JSP九大内置对象包括:page、request 、response、pageContext、session、application、out、config、exception,内置对象指的是Servlet容器创建的一组对象,不需使用new关键字就能不到直接使用的内置对象。

    四大作用域与九大内置对象对应关系如下

    更多详情须要亲戚亲戚人们此人 去查阅资料了

  EL表达式与JSTL标签

    亲戚亲戚人们知道jsp中能不到插入Java代码片段,累似 如下

<%pageContext.setAttribute("sex", "男"); %>  <!-- 设置值,作用域是当前jsp页面 -->
    <div>
        <%=pageContext.getAttribute("sex") %>   <!-- 注意去看解析后的el_jsp.java,被解析成了out.print(pageContext.getAttribute("sex") ); -->
         Welcome, my friend!
    </div>

    其中<% %>邮邮寄邮邮寄包裹的你你这个 你你这个 java片段,<%= %>输出表达式值到页面;能不到想看 缺陷简洁,阅读性你你这个 你你这个 太友好,你你这个 你你这个 EL表达式就应运而生了,上述代码能不到替再加如下代码

<%pageContext.setAttribute("sex", "男"); %>  <!-- 设置值,作用域是当前jsp页面 -->
<div>
         ${sex} Welcome, my friend! <!-- ${expression} EL的语法形态 -->
    </div>

    EL不不不会 访问页面的上下文以及不同作用域中的对象 ,取得对象属性的值,或执行简单的运算或判断操作,用来错综复杂JSP中的java代码。EL表达式是JSP1.2就让内置支持的,能不到直接在JSP中使用,它从servlet四大作用域(范围servletContext > session > request > pageContext)中取值,这1个域不会setAttribute("",object)妙招和getAttribute("")妙招, EL表达式会自动按作用范围从小到大的顺序从四大作用域中寻找对应名字的值,找到了就立即返回不再继续寻找,其內部调用的你你这个 你你这个 pageContext的findAttribute("")妙招。

    EL并不一定能错综复杂JSP中的java代码,你你这个 它功能非常简单,不到满足你你这个 错综复杂的代码逻辑,你你这个 你你这个 就诞生了JSTL。JSP标准标签库(JSTL)是一有有有另1个JSP标签集合,它封装了JSP应用的通用核心功能,支持通用的、形态化的任务,比如迭代,条件判断,XML文档操作,国际化标签,SQL标签,另外还支持自定义标签,它实现了JSP页面中的代码复用、错综复杂了代码的书写,共同也保证了JSP的可读性更强。JSTL功能比较充足,但它不会JSP内置支持的,你你这个 你你这个 须要导入标签库到JSP页面(须要再加jstl的jar包依赖)。JSTL往往会集合EL表达式来使用,简单示累似 下

<%@ page language="java" contentType="text/html; charset=utf-8"
    pageEncoding="utf-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%> <!-- 引入JSTL标签库,c表示标签库别名,能不到任意命名,一般而言用c -->
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>index</title>
</head>
<body>
    <c:set var="sex" value="女" scope="page"></c:set> <!-- scope指定作用域,page/request/session/application -->
    <c:if test="${sex == '男' }">
        Hello, ${sex}
    </c:if>
    <c:if test="${sex != '男' }">
        Hi, girl
    </c:if>
</body>
</html>

    这代码看起来就清爽多了,如此 java代码,前端开发者也很容易看懂;关于EL表达式与JSTL标签更完全信息,须要亲戚亲戚人们自行去查阅资料了,本文篇幅有限,不做越多的讲解了。

  重定向与请求转发

    如此 可想而知,重定向的request作用域的变量是会失效的,而转发则不不

  Spring MVC

    还记得亲戚亲戚人们是如可配置Spring MVC的吗, 亲戚亲戚人们会在web.xml中配置如下代码

<servlet>
        <servlet-name>springDispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:spring-mvc.xml</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>springDispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

    原先就配置了Spring MVC;亲戚亲戚人们能不到留意下DispatcherServlet,去看他的类图会发现,他你你这个 你你这个 一有有有另1个Servlet的实现,也你你这个 你你这个 说Sprinv MVC你你这个 你你这个 基于Servlet的拓展。

    亲戚亲戚人们在Spring MVC基础上进行开发的就让,将数据绑定到作用域的就让,一般用的是SpringMVC的数据模型:Model肯能ModelMap,累似 原先

@RequestMapping("/showPerson")
public String showPersons(Model model){
    List<Person> persons = personService.loadPersons();
    model.addAttribute("persons", persons);        // 绑定数据到视图
    return "showperson";
}

    而不会显示的直接绑定到Servlet四大作用域,数据难道如此 绑定到四大作用域? 亲戚亲戚人们说过,EL表达式不到在四大作用域中取值,你你这个 取不到,你你这个 你你这个 SpringMVC中的数据绑定最终还是会到四大作用域的某一有有有另1个中,至于是啥时候、何地、如可将Model中的属性绑定到哪个作用域,你你这个 不会本文要说的了,篇幅越多了,有兴趣的能不到去看看这篇博客:springmvc的工作原理,亲戚亲戚人们来看看其源码实现。这里给个结论:在默认情况下,Model中的属性作用域是request级别

  现象解答

    你你这个 小伙伴会抱怨了:上边哔哔了如此 多,缘何你你这个 你你这个 不讲答案,净说你你这个 没用的

    肯能亲戚亲戚人们坚持想看 这了,再坚持会,答案马上揭晓,上边铺垫了如此 多,绝对是有用的。

    亲戚亲戚人们回到现象:当亲戚亲戚人们请求http://localhost:端口/工程名/personController/showPerson时,数据正常显示,而当亲戚亲戚人们直接请求jsp时,不到title却如此 数据,这是为哪此?title是静态页面内容,你你这个 不不管,那为哪此直接请求jsp为哪此如此 数据库的person列表呢? jsp源代码如下

<%@ page language="java" contentType="text/html; charset=utf-8"
     pageEncoding="utf-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
        <title>person list</title>
    </head>
    <body>
        <table>
            <tr>
                <th>姓名</th>
                <th>年龄</th>
            </tr>
            <c:forEach items="${persons}" var="person">
                <tr>
                    <td>${person.name }</td>
                    <td>${person.age }</td>
                </tr>
            </c:forEach>
        </table>
    </body>
</html>

    上边用到了<c:forEach>和EL表达式,解释下你你这个 流程:EL表达式先从四大作用域获取名为persons的集合,你你这个 <c:forEach>遍历该集合,每次遍历的结果放进去page作用域,并取叫安person,最后通过EL表达式输出person的name和age到页面。如此 请问:直接访问JSP,四大作用域富含叫安persons的属性吗?很显然如此 ,persons不居于,遍历它会有结果输出吗?这你你这个 你你这个 为哪此直接访问jsp如此 数据的答案。

    亲戚亲戚人们再回到Controller层

@RequestMapping("/showPerson")
public String showPersons(Model model){
    List<Person> persons = personService.loadPersons();        // 从数据获取person列表,并存放进去了persons集合中
    model.addAttribute("persons", persons);                    // 将persons集合再加到model的persons属性中
    return "showperson";                                    // 转发到showperson.jsp
}

    代码也非常简单,先从数据库获取person集合,你你这个 将该集合设置到了model的属性persons中,亲戚亲戚人们知道model的属性默认情况下会设置到request作用域;你你这个 将请求转发到showperson.jsp,转发过程中,request作用域的变量仍然有效,你你这个 你你这个 jsp中EL表达式不不不会 读取到persons变量,你你这个 你你这个 不会数据输出到页面了。

总结

  1、Servlet与Servlet容器的关系比较暧昧,两者相互作用,实现web服务;简单点说,亲戚亲戚人们自定义的Servlet你你这个 你你这个 对Servlet容器的业务拓展,而Servlet容器是对Servlet的支撑;

  2、JSP的经常出现时为了错综复杂静态页面的开发,EL表达式与JSTL的经常出现则是为了错综复杂JSP页面的Java代码;JSP本质还是Servlet,在第一次被访问的过不会被Servlet容器解析成Servlet、编译Servlet,最终还是有Servlet将页面内容out.write到浏览器;

  3、Spring MVC本质还是Servlet,它的经常出现是为了错综复杂web开发,共同能不到与spring无缝对接,享受spring带来的好处;Spring MVC的数据绑定,依托的还是Servlet的的四大作用域,你你这个 你你这个 上边居于转换过程;

  4、EL表达式的取值须要居于于四大作用域中,在jsp中用EL表达式时,一定要保证数据正确地再加到了四大作用域中,不然,EL表达式会取不到值;

参考

  《深入分析JavaWeb技术内幕》

  《Tomcat 系统架构与模式设计分析》

  Java Web(一) Servlet详解!!