当前位置: 首页 > news >正文

农业畜牧网站开发中企动力科技股份官网

农业畜牧网站开发,中企动力科技股份官网,青岛网站建设报价,网址浏览器MVC简介 普通的web项目每次都要进行手动的把jar包导进去#xff0c;否则会报500#xff0c;class not found [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VstjHhuz-1609824493673)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images…MVC简介 普通的web项目每次都要进行手动的把jar包导进去否则会报500class not found [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VstjHhuz-1609824493673)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104132339761.png)] [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-L9AgS48A-1609824493680)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104143509848.png)] MVC是模型(Model)、视图(View)、控制器(Controller)的简写是一种软件设计规范。 是将业务逻辑、数据、显示分离的方法来组织代码。 MVC主要作用是降低了视图与业务逻辑间的双向偶合。 MVC不是一种设计模式MVC是一种架构模式。当然不同的MVC存在差异。 **Model模型**数据模型提供要展示的数据因此包含数据和行为可以认为是领域模型或JavaBean组件包含数据和行为不过现在一般都分离开来Value Object数据Dao 和 服务层行为Service。也就是模型提供了模型数据查询和模型数据的状态更新等功能包括数据和业务。 **View视图**负责进行模型的展示一般就是我们见到的用户界面客户想看到的东西。 **Controller控制器**接收用户请求委托给模型进行处理状态改变处理完毕后把返回的模型数据返回给视图由视图负责展示。也就是说控制器做了个调度员的工作。 最典型的MVC就是JSP servlet javabean的模式。 Model1时代 在web早期的开发中通常采用的都是Model1。Model1中主要分为两层视图层和模型层。 Model1优点架构简单比较适合小型项目开发 Model1缺点JSP职责不单一职责过重不便于维护 Model2时代 Model2把一个项目分成三部分包括视图、控制、模型。 用户发请求Servlet接收请求数据并调用对应的业务逻辑方法业务处理完毕返回更新后的数据给servletservlet转向到JSP由JSP来渲染页面响应给前端更新后的页面 职责分析 Controller控制器 取得表单数据调用业务逻辑转向指定的页面 Model模型 业务逻辑保存数据的状态 View视图 显示页面 Model2这样不仅提高的代码的复用率与项目的扩展性且大大降低了项目的维护成本。Model 1模式的实现比较简单适用于快速开发小规模项目Model1中JSP页面身兼View和Controller两种角色将控制逻辑和表现逻辑混杂在一起从而导致代码的重用性非常低增加了应用的扩展性和维护的难度。Model2消除了Model1的缺点。 回顾Servlet 新建一个Maven工程当做父工程pom依赖 dependenciesdependencygroupIdjunit/groupIdartifactIdjunit/artifactIdversion4.12/version/dependencydependencygroupIdorg.springframework/groupIdartifactIdspring-webmvc/artifactIdversion5.1.9.RELEASE/version/dependencydependencygroupIdjavax.servlet/groupIdartifactIdservlet-api/artifactIdversion2.5/version/dependencydependencygroupIdjavax.servlet.jsp/groupIdartifactIdjsp-api/artifactIdversion2.2/version/dependencydependencygroupIdjavax.servlet/groupIdartifactIdjstl/artifactIdversion1.2/version/dependency /dependencies建立一个Moudlespringmvc-01-servlet 添加Web app的支持(web项目才能使用Tomcat) 导入servlet 和 jsp 的 jar 依赖 dependencygroupIdjavax.servlet/groupIdartifactIdservlet-api/artifactIdversion2.5/version /dependency dependencygroupIdjavax.servlet.jsp/groupIdartifactIdjsp-api/artifactIdversion2.2/version /dependency编写一个Servlet类用来处理用户的请求 package com.kuang.servlet;//实现Servlet接口 public class HelloServlet extends HttpServlet {Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throwsServletException, IOException {//取得参数String method req.getParameter(method);if (method.equals(add)){req.getSession().setAttribute(msg,执行了add方法);}if (method.equals(delete)){req.getSession().setAttribute(msg,执行了delete方法);}//业务逻辑//视图跳转req.getRequestDispatcher(/WEB-INF/jsp/hello.jsp).forward(req,resp);}Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throwsServletException, IOException {doGet(req,resp);} }编写Hello.jsp在WEB-INF目录下新建一个jsp的文件夹新建hello.jsp % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleKuangshen/title /head body ${msg} /body /html在web.xml中注册Servlet ?xml version1.0 encodingUTF-8? web-app xmlnshttp://xmlns.jcp.org/xml/ns/javaeexmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsdversion4.0servletservlet-nameHelloServlet/servlet-nameservlet-classcom.kuang.servlet.HelloServlet/servlet-class/servletservlet-mappingservlet-nameHelloServlet/servlet-nameurl-pattern/user/url-pattern/servlet-mapping/web-app配置Tomcat并启动测试 localhost:8080/user?methodaddlocalhost:8080/user?methoddelete MVC框架要做哪些事情 将url映射到java类或java类的方法 .封装用户提交的数据 .处理请求–调用相关的业务处理–封装响应数据 .将响应的数据进行渲染 . jsp / html 等表示层数据 . 说明 ​ 常见的服务器端MVC框架有Struts、Spring MVC、ASP.NET MVC、Zend Framework、JSF常见前端MVC框架vue、angularjs、react、backbone由MVC演化出了另外一些模式如MVP、MVVM 等等… SpringMVC简介 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bR1ePQIj-1609824493705)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210102160327108.png)] 2.1、概述 Spring MVC是Spring Framework的一部分是基于Java实现MVC的轻量级Web框架。 查看官方文档https://docs.spring.io/spring/docs/5.2.0.RELEASE/spring-framework-reference/web.html#spring-web 我们为什么要学习SpringMVC呢? Spring MVC的特点 轻量级简单易学高效 , 基于请求响应的MVC框架与Spring兼容性好无缝结合约定优于配置功能强大RESTful、数据验证、格式化、本地化、主题等简洁灵活 Spring的web框架围绕DispatcherServlet [ 调度Servlet ] 设计。 DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始使用Java 5或者以上版本的用户可以采用基于注解形式进行开发十分简洁 正因为SpringMVC好 , 简单 , 便捷 , 易学 , 天生和Spring无缝集成(使用SpringIoC和Aop) , 使用约定优于配置 . 能够进行简单的junit测试 . 支持Restful风格 .异常处理 , 本地化 , 国际化 , 数据验证 , 类型转换 , 拦截器 等等…所以我们要学习 . 最重要的一点还是用的人多 , 使用的公司多 . 2.2、中心控制器 ​ Spring的web框架围绕DispatcherServlet设计。DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始使用Java 5或者以上版本的用户可以采用基于注解的controller声明方式。 ​ Spring MVC框架像许多其他MVC框架一样, 以请求为驱动 , 围绕一个中心Servlet分派请求及提供其他功能DispatcherServlet是一个实际的Servlet (它继承自HttpServlet 基类)。 SpringMVC的原理如下图所示 ​ 当发起请求时被前置的控制器拦截到请求根据请求参数生成代理请求找到请求对应的实际控制器控制器处理请求创建数据模型访问数据库将模型响应给中心控制器控制器使用模型与视图渲染视图结果将结果返回给中心控制器再将结果返回给请求者。 2.3、SpringMVC执行原理 图为SpringMVC的一个较完整的流程图实线表示SpringMVC框架提供的技术不需要开发者实现虚线表示需要开发者实现。 简要分析执行流程 DispatcherServlet表示前置控制器是整个SpringMVC的控制中心。用户发出请求DispatcherServlet接收请求并拦截请求。 我们假设请求的url为 : http://localhost:8080/SpringMVC/hello 如上url拆分成三部分 http://localhost:8080服务器域名 SpringMVC部署在服务器上的web站点 hello表示控制器 通过分析如上url表示为请求位于服务器localhost:8080上的SpringMVC站点的hello控制器。 HandlerMapping为处理器映射。DispatcherServlet调用HandlerMapping,HandlerMapping根据请求url查找Handler。 HandlerExecution表示具体的Handler,其主要作用是根据url查找控制器如上url被查找控制器为hello。 HandlerExecution将解析后的信息传递给DispatcherServlet,如解析控制器映射等。 HandlerAdapter表示处理器适配器其按照特定的规则去执行Handler。 Handler让具体的Controller执行。 Controller将具体的执行信息返回给HandlerAdapter,如ModelAndView。 HandlerAdapter将视图逻辑名或模型传递给DispatcherServlet。 DispatcherServlet调用视图解析器(ViewResolver)来解析HandlerAdapter传递的逻辑视图名。 视图解析器将解析的逻辑视图名传给DispatcherServlet。 DispatcherServlet根据视图解析器解析的视图结果调用具体的视图。 最终视图呈现给用户。 第一个MVC程序 配置版 1、新建一个Moudle springmvc-02-hello 添加web的支持 2、确定导入了SpringMVC 的依赖 3、配置web.xml 注册DispatcherServlet ?xml version1.0 encodingUTF-8?web-app xmlnshttp://xmlns.jcp.org/xml/ns/javaee xmlns:xsihttp://www.w3.org/2001/XMLSchema-instance xsi:schemaLocationhttp://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd version4.0!--1.注册DispatcherServlet-- servlet servlet-namespringmvc/servlet-name servlet-classorg.springframework.web.servlet.DispatcherServlet/servlet-class !--关联一个springmvc的配置文件:【servlet-name】-servlet.xml-- init-param param-namecontextConfigLocation/param-name param-valueclasspath:springmvc-servlet.xml/param-value /init-param !--启动级别-1-- load-on-startup1/load-on-startup /servlet!--/ 匹配所有的请求不包括.jsp-- !--/* 匹配所有的请求包括.jsp-- servlet-mapping servlet-namespringmvc/servlet-name url-pattern//url-pattern /servlet-mapping /web-app4、编写SpringMVC 的 配置文件名称springmvc-servlet.xml : [servletname]-servlet.xml 说明这里的名称要求是按照官方来的 ?xml version1.0 encodingUTF-8?beans xmlnshttp://www.springframework.org/schema/beans xmlns:xsihttp://www.w3.org/2001/XMLSchema-instance xsi:schemaLocationhttp://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd /beans5、添加 处理映射器 bean classorg.springframework.web.servlet.handler.BeanNameUrlHandlerMapping/6、添加 处理器适配器 bean classorg.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter/7、添加 视图解析器 !--视图解析器:DispatcherServlet给他的ModelAndView--bean classorg.springframework.web.servlet.view.InternalResourceViewResolver idInternalResourceViewResolver !--前缀-- property nameprefix value/WEB-INF/jsp// !--后缀-- property namesuffix value.jsp//bean8、编写我们要操作业务Controller 要么实现Controller接口要么增加注解需要返回一个ModelAndView装数据封视图 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZABm3VoF-1609824493710)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210102164333287.png)] package com.kuang.controller;import org.springframework.web.servlet.ModelAndView; import org.springframework.web.servlet.mvc.Controller;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.lang.annotation.Annotation;//注意这里我们先导入Controller接口 public class HelloController implements Controller { public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response)throws Exception{//ModelAndView 模型和视图ModelAndView mvnew ModelAndView();//封装对象放在ModelAndView中。Modelmv.addObject(msg,HelloSpringMVC!);// 封装要跳转的视图放在ModelAndView中mv.setViewName(hello); //WEB-INF/jsp/hello.jspreturn mv;}}9、将自己的类交给SpringIOC容器注册bean !--Handler--bean id/hello classcom.kuang.controller.HelloController/10、写要跳转的jsp页面显示ModelandView存放的数据以及我们的正常页面 % page contentTypetext/html;charsetUTF-8 languagejava %htmlhead titleKuangshen/title/headbody${msg}/body/html11、配置Tomcat 启动测试 可能遇到的问题访问出现404排查步骤 查看控制台输出看一下是不是缺少了什么jar包。如果jar包存在显示无法输出就在IDEA的项目发布中添加lib依赖重启Tomcat 即可解决 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1e2lMv0E-1609824493712)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210102171102903.png)] 小结看这个估计大部分同学都能理解其中的原理了但是我们实际开发才不会这么写不然就疯了还学这个玩意干嘛我们来看个注解版实现这才是SpringMVC的精髓到底有多么简单看这个图就知道了。 注解版 1、新建一个Moudlespringmvc-03-hello-annotation 。添加web支持 2、由于Maven可能存在资源过滤的问题我们将配置完善 buildresourcesresourcedirectorysrc/main/java/directoryincludesinclude**/*.properties/includeinclude**/*.xml/include/includesfilteringfalse/filtering/resourceresourcedirectorysrc/main/resources/directoryincludesinclude**/*.properties/includeinclude**/*.xml/include/includesfilteringfalse/filtering/resource/resources/build3、在pom.xml文件引入相关的依赖主要有Spring框架核心库、Spring MVC、servlet , JSTL等。我们在父依赖中已经引入了 4、配置web.xml 注意点 注意web.xml版本问题要最新版注册DispatcherServlet关联SpringMVC的配置文件启动级别为1映射路径为 / 【不要用/*会404】 ?xml version1.0 encodingUTF-8? web-app xmlnshttp://xmlns.jcp.org/xml/ns/javaeexmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://xmlns.jcp.org/xml/ns/javaeehttp://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsdversion4.0!--1.注册servlet--servletservlet-nameSpringMVC/servlet-nameservlet-classorg.springframework.web.servlet.DispatcherServlet/servlet-class !--通过初始化参数指定SpringMVC配置文件的位置进行关联--init-paramparam-namecontextConfigLocation/param-nameparam-valueclasspath:springmvc-servlet.xml/param-value/init-param !-- 启动顺序数字越小启动越早 --load-on-startup1/load-on-startup/servlet!--所有请求都会被springmvc拦截 --servlet-mappingservlet-nameSpringMVC/servlet-nameurl-pattern//url-pattern/servlet-mapping /web-app/ 和 /* 的区别 url-pattern / / url-pattern 不会匹配到.jsp 只针对我们编写的请求即.jsp 不会进入spring的 DispatcherServlet类 。 url-pattern /* / url-pattern 会匹配 *.jsp会出现返回 jsp视图 时再次进入spring的DispatcherServlet 类导致找不到对应的controller所以报404错。 5、添加Spring MVC配置文件 在视图解析器中我们把所有的视图都存放在/WEB-INF/目录下这样可以保证视图安全因为这个目录下的文件客户端不能直接访问。 让IOC的注解生效静态资源过滤 HTML . JS . CSS . 图片 视频 …MVC的注解驱动配置视图解析器 在resource目录下添加springmvc-servlet.xml配置文件配置的形式与Spring容器配置基本类似为了支持基于注解的IOC设置了自动扫描包的功能具体配置信息如下 ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beans xmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxmlns:mvchttp://www.springframework.org/schema/mvcxsi:schemaLocationhttp://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/mvc https://www.springframework.org/schema/mvc/spring-mvc.xsd!-- 自动扫描包让指定包下的注解生效,由IOC容器统一管理 --context:component-scan base-packagecom.kuang.controller/ !-- 让Spring MVC不处理静态资源 -- mvc:default-servlet-handler/ !-- 支持mvc注解驱动 在spring中一般采用RequestMapping注解来完成映射关系 要想使RequestMapping注解生效 必须向上下文中注册DefaultAnnotationHandlerMapping 和一个AnnotationMethodHandlerAdapter实例 这两个实例分别在类级别和方法级别处理。 而annotation-driven配置帮助我们自动完成上述两个实例的注入。 --mvc:annotation-driven/!-- 视图解析器 --bean classorg.springframework.web.servlet.view.InternalResourceViewResolveridinternalResourceViewResolver !-- 前缀 --property nameprefix value/WEB-INF/jsp// !-- 后缀 --property namesuffix value.jsp//bean /beans6、创建Controller 编写一个Java控制类com.kuang.controller.HelloController , 注意编码规范 package com.kuang.controller; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping;Controller RequestMapping(/HelloController) public class HelloController { //真实访问地址 : 项目名//HelloController/helloRequestMapping(/hello)public String sayHello(Model model){//向模型中添加属性msg与值可以在JSP页面中取出并渲染model.addAttribute(msg,hello,SpringMVC);//web-inf/jsp/hello.jspreturn hello; } }Controller是为了让Spring IOC容器初始化时自动扫描到RequestMapping是为了映射请求路径这里因为类与方法上都有映射所以访问时应该是/HelloController/hello方法中声明Model类型的参数是为了把Action中的数据带到视图中方法返回的结果是视图的名称hello加上配置文件中的前后缀变成WEB-INF/jsp/hello.jsp。 7、创建视图层 在WEB-INF/ jsp目录中创建hello.jsp 视图可以直接取出并展示从Controller带回的信息 可以通过EL表示取出Model中存放的值或者对象 % page contentTypetext/html;charsetUTF-8 languagejava %htmlhead titleSpringMVC/title/headbody${msg}/body/html8、配置Tomcat运行 配置Tomcat 开启服务器 访问 对应的请求路径 OK运行成功 小结 实现步骤其实非常的简单 新建一个web项目导入相关jar包编写web.xml , 注册DispatcherServlet编写springmvc配置文件接下来就是去创建对应的控制类 , controller最后完善前端视图和controller之间的对应测试运行调试. 使用springMVC必须配置的三大件 处理器映射器、处理器适配器、视图解析器 通常我们只需要手动配置视图解析器而处理器映射器和处理器适配器只需要开启注解驱动即可而省去了大段的xml配置 再来回顾下原理吧~ 控制器Controller 控制器负责提供访问应用程序的行为通常通过接口定义或注解定义两种方法实现。 控制器负责解析用户的请求并将其转换为一个模型。 在Spring MVC中一个控制器类可以包含多个方法 在Spring MVC中对于Controller的配置方式有很多种 实现Controller接口 Controller是一个接口在org.springframework.web.servlet.mvc包下接口中只有一个方法 import org.springframework.web.servlet.mvc.Controller;//实现该接口的类获得控制器功能 public interface Controller {//处理请求且返回一个模型与视图对象ModelAndView handleRequest(HttpServletRequest var1, HttpServletResponse var2) throws Exception; }测试 新建一个Moudlespringmvc-04-controller 。将刚才的03 拷贝一份, 我们进行操作 删掉HelloControllermvc的配置文件只留下 视图解析器 编写一个Controller类ControllerTest1 //定义控制器 //注意点不要导错包实现Controller接口重写方法 public class ControllerTest1 implements Controller {public ModelAndView handleRequest(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse) throws Exception {//返回一个模型视图对象ModelAndView mv new ModelAndView();mv.addObject(msg,Test1Controller);mv.setViewName(test);return mv;} }编写完毕后去Spring配置文件中注册请求的beanname对应请求路径class对应处理请求的类 bean name/t1 classcom.kuang.controller.ControllerTest1/编写前端test.jsp注意在WEB-INF/jsp目录下编写对应我们的视图解析器 % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleKuangshen/title /head body ${msg} /body /html配置Tomcat运行测试我这里没有项目发布名配置的就是一个 / 所以请求不用加项目名OK 说明 实现接口Controller定义控制器是较老的办法 缺点是一个控制器中只有一个方法如果要多个方法则需要定义多个Controller定义的方式比较麻烦 注解Controller [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-aaVjrLB7-1609824493716)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210102201809711.png)] Controller注解类型用于声明Spring类的实例是一个控制器在讲IOC时还提到了另外3个注解 Spring可以使用扫描机制来找到应用程序中所有基于注解的控制器类为了保证Spring能找到你的控制器需要在配置文件中声明组件扫描。 !-- 自动扫描指定的包下面所有注解类交给IOC容器管理 -- context:component-scan base-packagecom.kuang.controller/增加一个ControllerTest2类使用注解实现 //Controller注解的类会自动添加到Spring上下文中 Controller public class ControllerTest2{//映射访问路径RequestMapping(/t2)public String index(Model model){//Spring MVC会自动实例化一个Model对象用于向视图中传值model.addAttribute(msg, ControllerTest2);//返回视图位置return test;} }运行tomcat测试 可以发现我们的两个请求都可以指向一个视图但是页面结果的结果是不一样的从这里可以看出视图是被复用的而控制器与视图之间是弱偶合关系。 注解方式是平时使用的最多的方式 RequestMapping RequestMapping RequestMapping注解用于映射url到控制器类或一个特定的处理程序方法。可用于类或方法上。用于类上表示类中的所有响应请求的方法都是以该地址作为父路径。 为了测试结论更加准确我们可以加上一个项目名测试 myweb 只注解在方法上面 Controller public class TestController {RequestMapping(/h1)public String test(){return test;} }访问路径http://localhost:8080 / 项目名 / h1 同时注解类与方法 Controller RequestMapping(/admin) public class TestController {RequestMapping(/h1)public String test(){return test;} }访问路径http://localhost:8080 / 项目名/ admin /h1 , 需要先指定类的路径再指定方法的路径 RestFul 风格 概念 Restful就是一个资源定位及资源操作的风格。不是标准也不是协议只是一种风格。基于这个风格设计的软件可以更简洁更有层次更易于实现缓存等机制。 功能 资源互联网所有的事物都可以被抽象为资源 资源操作使用POST、DELETE、PUT、GET使用不同方法对资源进行操作。 分别对应 添加、 删除、修改、查询。 传统方式操作资源 通过不同的参数来实现不同的效果方法单一post 和 get ​ http://127.0.0.1/item/queryItem.action?id1 查询,GET ​ http://127.0.0.1/item/saveItem.action 新增,POST ​ http://127.0.0.1/item/updateItem.action 更新,POST ​ http://127.0.0.1/item/deleteItem.action?id1 删除,GET或POST 使用RESTful操作资源 可以通过不同的请求方式来实现不同的效果如下请求地址一样但是功能可以不同 ​ http://127.0.0.1/item/1 查询,GET ​ http://127.0.0.1/item 新增,POST ​ http://127.0.0.1/item 更新,PUT ​ http://127.0.0.1/item/1 删除,DELETE 学习测试 传进来的和接收的名字必须一致 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-aLDSkwDk-1609824493718)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210102210223680.png)] 在新建一个类 RestFulController Controller public class RestFulController { }在Spring MVC中可以使用 PathVariable 注解让方法参数的值对应绑定到一个URI模板变量上。 Controller public class RestFulController {//映射访问路径RequestMapping(/commit/{p1}/{p2})public String index(PathVariable int p1, PathVariable int p2, Model model){int result p1p2;//Spring MVC会自动实例化一个Model对象用于向视图中传值model.addAttribute(msg, 结果result);//返回视图位置return test;}}我们来测试请求查看下 思考使用路径变量的好处 使路径变得更加简洁 获得参数更加方便框架会自动进行类型转换。 通过路径变量的类型可以约束访问参数如果类型不一样则访问不到对应的请求方法如这里访问是的路径是/commit/1/a则路径与方法不匹配而不会是参数转换失败。 我们来修改下对应的参数类型再次测试 //映射访问路径 RequestMapping(/commit/{p1}/{p2}) public String index(PathVariable int p1, PathVariable String p2, Model model){String result p1p2;//Spring MVC会自动实例化一个Model对象用于向视图中传值model.addAttribute(msg, 结果result);//返回视图位置return test;}使用method属性指定请求类型 用于约束请求的类型可以收窄请求范围。指定请求谓词的类型如GET, POST, HEAD, OPTIONS, PUT, PATCH, DELETE, TRACE等 我们来测试一下 增加一个方法 //映射访问路径,必须是POST请求 RequestMapping(value /hello,method {RequestMethod.POST}) public String index2(Model model){model.addAttribute(msg, hello!);return test; }我们使用浏览器地址栏进行访问默认是Get请求会报错405 如果将POST修改为GET则正常了 //映射访问路径,必须是Get请求 RequestMapping(value /hello,method {RequestMethod.GET}) public String index2(Model model){model.addAttribute(msg, hello!);return test; }小结 Spring MVC 的 RequestMapping 注解能够处理 HTTP 请求的方法, 比如 GET, PUT, POST, DELETE 以及 PATCH。 所有的地址栏请求默认都会是 HTTP GET 类型的。 方法级别的注解变体有如下几个组合注解 GetMapping PostMapping PutMapping DeleteMapping RequestMappingGetMapping 是一个组合注解平时使用的会比较多 它所扮演的是 RequestMapping(method RequestMethod.GET) 的一个快捷方式。 扩展小黄鸭调试法 场景一我们都有过向别人甚至可能向完全不会编程的人提问及解释编程问题的经历但是很多时候就在我们解释的过程中自己却想到了问题的解决方案然后对方却一脸茫然。 场景二你的同行跑来问你一个问题但是当他自己把问题说完或说到一半的时候就想出答案走了留下一脸茫然的你。 其实上面两种场景现象就是所谓的小黄鸭调试法Rubber Duck Debuging又称橡皮鸭调试法它是我们软件工程中最常使用调试方法之一。 此概念据说来自《程序员修炼之道》书中的一个故事传说程序大师随身携带一只小黄鸭在调试代码的时候会在桌上放上这只小黄鸭然后详细地向鸭子解释每行代码然后很快就将问题定位修复了。 下面我们准备研究下参数接受和结果跳转 结果跳转方式 ModelAndView 设置ModelAndView对象 , 根据view的名称 , 和视图解析器跳到指定的页面 . 页面 : {视图解析器前缀} viewName {视图解析器后缀} !-- 视图解析器 -- bean classorg.springframework.web.servlet.view.InternalResourceViewResolveridinternalResourceViewResolver!-- 前缀 --property nameprefix value/WEB-INF/jsp/ /!-- 后缀 --property namesuffix value.jsp / /bean对应的controller类 public class ControllerTest1 implements Controller {public ModelAndView handleRequest(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse) throws Exception {//返回一个模型视图对象ModelAndView mv new ModelAndView();mv.addObject(msg,ControllerTest1);mv.setViewName(test);return mv;} }ServletAPI(了解) 通过设置ServletAPI , 不需要视图解析器 . 1、通过HttpServletResponse进行输出 2、通过HttpServletResponse实现重定向 3、通过HttpServletRequset实现转发 Controller public class ResultGo {RequestMapping(/result/t1)public void test1(HttpServletRequest req, HttpServletResponse rsp) throwsIOException {rsp.getWriter().println(Hello,Spring BY servlet API);}RequestMapping(/result/t2)public void test2(HttpServletRequest req, HttpServletResponse rsp) throwsIOException {rsp.sendRedirect(/index.jsp);}RequestMapping(/result/t3)public void test3(HttpServletRequest req, HttpServletResponse rsp) throwsException {//转发req.setAttribute(msg,/result/t3);req.getRequestDispatcher(/WEB-INF/jsp/test.jsp).forward(req,rsp);}}SpringMVC 通过SpringMVC来实现转发和重定向 - 无需视图解析器 测试前需要将视图解析器注释掉 Controller public class ResultSpringMVC {RequestMapping(/rsm/t1)public String test1(){//转发return /index.jsp;}RequestMapping(/rsm/t2)public String test2(){//转发二return forward:/index.jsp;}RequestMapping(/rsm/t3)public String test3(){//重定向return redirect:/index.jsp;} }通过SpringMVC来实现转发和重定向 - 有视图解析器 重定向 , 不需要视图解析器 , 本质就是重新请求一个新地方嘛 , 所以注意路径问题. 可以重定向到另外一个请求实现 . Controller public class ResultSpringMVC2 {RequestMapping(/rsm2/t1)public String test1(){//转发return test;}RequestMapping(/rsm2/t2)public String test2(){//重定向return redirect:/index.jsp;//return redirect:hello.do; //hello.do为另一个请求/}}数据处理 处理提交数据 1、提交的域名称和处理方法的参数名一致 提交数据 : http://localhost:8080/hello?namekuangshen 处理方法 : RequestMapping(/hello) public String hello(String name){System.out.println(name);return hello; }后台输出 : kuangshen 2、提交的域名称和处理方法的参数名不一致 提交数据 : http://localhost:8080/hello?usernamekuangshen 处理方法 : //RequestParam(username) : username提交的域的名称 . RequestMapping(/hello) public String hello(RequestParam(username) String name){System.out.println(name);return hello; }后台输出 : kuangshen 3、提交的是一个对象 要求提交的表单域和对象的属性名一致 , 参数使用对象即可 1、实体类 public class User {private int id;private String name;private int age;//构造//get/set//tostring() }2、提交数据 : http://localhost:8080/mvc04/user?namekuangshenid1age15 3、处理方法 : RequestMapping(/user) public String user(User user){System.out.println(user);return hello; }后台输出 : User { id1, name‘kuangshen’, age15 } 说明如果使用对象的话前端传递的参数名和对象名必须一致否则就是null。 数据显示到前端 第一种 : 通过ModelAndView 我们前面一直都是如此 . 就不过多解释 public class ControllerTest1 implements Controller {public ModelAndView handleRequest(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse) throws Exception {//返回一个模型视图对象ModelAndView mv new ModelAndView();mv.addObject(msg,ControllerTest1);mv.setViewName(test);return mv;} }第二种 : 通过ModelMap ModelMap RequestMapping(/hello) public String hello(RequestParam(username) String name, ModelMap model){//封装要显示到视图中的数据//相当于req.setAttribute(name,name);model.addAttribute(name,name);System.out.println(name);return hello; }第三种 : 通过Model Model RequestMapping(/ct2/hello) public String hello(RequestParam(username) String name, Model model){//封装要显示到视图中的数据//相当于req.setAttribute(name,name);model.addAttribute(msg,name);System.out.println(name);return test; }对比 就对于新手而言简单来说使用区别就是 Model 只有寥寥几个方法只适合用于储存数据简化了新手对于Model对象的操作和理解ModelMap 继承了 LinkedMap 除了实现了自身的一些方法同样的继承 LinkedMap 的方法和特性ModelAndView 可以在储存数据的同时可以进行设置返回的逻辑视图进行控制展示层的跳转。当然更多的以后开发考虑的更多的是性能和优化就不能单单仅限于此的了解。 请使用80%的时间打好扎实的基础剩下18%的时间研究框架2%的时间去学点英文框架的官方文档永远是最好的教程。 乱码问题 测试步骤 1、我们可以在首页编写一个提交的表单 form action/e/t methodpostinput typetext namenameinput typesubmit /form2、后台编写对应的处理类 Controller public class Encoding {RequestMapping(/e/t)public String test(Model model,String name){model.addAttribute(msg,name); //获取表单提交的值return test; //跳转到test页面显示输入的值} }3、输入中文测试发现乱码 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xpjITslk-1609824493729)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210103003228667.png)] 不得不说乱码问题是在我们开发中十分常见的问题也是让我们程序猿比较头大的问题 以前乱码问题通过过滤器解决 , 而SpringMVC给我们提供了一个过滤器 , 可以在web.xml中配置 . 修改了xml文件需要重启服务器 filterfilter-nameencoding/filter-namefilter-classorg.springframework.web.filter.CharacterEncodingFilter/filter-classinit-paramparam-nameencoding/param-nameparam-valueutf-8/param-value/init-param /filter filter-mappingfilter-nameencoding/filter-nameurl-pattern/*/url-pattern /filter-mapping 但是我们发现 , 有些极端情况下.这个过滤器对get的支持不好 . 处理方法 : 1、修改tomcat配置文件 设置编码 Connector URIEncodingutf-8 port8080 protocolHTTP/1.1connectionTimeout20000redirectPort8443 /2、自定义过滤器 package com.kuang.filter;import javax.servlet.*; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequestWrapper; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.util.Map;/** * 解决get和post请求 全部乱码的过滤器 */ public class GenericEncodingFilter implements Filter {Overridepublic void destroy() {}Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChainchain) throws IOException, ServletException {//处理response的字符编码HttpServletResponse myResponse(HttpServletResponse) response;myResponse.setContentType(text/html;charsetUTF-8);// 转型为与协议相关对象HttpServletRequest httpServletRequest (HttpServletRequest) request;// 对request包装增强HttpServletRequest myrequest new MyRequest(httpServletRequest);chain.doFilter(myrequest, response);}Overridepublic void init(FilterConfig filterConfig) throws ServletException {}}//自定义request对象HttpServletRequest的包装类 class MyRequest extends HttpServletRequestWrapper {private HttpServletRequest request;//是否编码的标记private boolean hasEncode;//定义一个可以传入HttpServletRequest对象的构造函数以便对其进行装饰public MyRequest(HttpServletRequest request) {super(request);// super必须写this.request request;}// 对需要增强方法 进行覆盖Overridepublic Map getParameterMap() {// 先获得请求方式String method request.getMethod();if (method.equalsIgnoreCase(post)) {// post请求try {// 处理post乱码request.setCharacterEncoding(utf-8);return request.getParameterMap();} catch (UnsupportedEncodingException e) {e.printStackTrace();}} else if (method.equalsIgnoreCase(get)) {// get请求MapString, String[] parameterMap request.getParameterMap();if (!hasEncode) { // 确保get手动编码逻辑只运行一次for (String parameterName : parameterMap.keySet()) {String[] values parameterMap.get(parameterName);if (values ! null) {for (int i 0; i values.length; i) {try {// 处理get乱码values[i] new String(values[i].getBytes(ISO-8859-1), utf-8);} catch (UnsupportedEncodingException e) {e.printStackTrace();}}}}hasEncode true;}return parameterMap;}return super.getParameterMap();}//取一个值Overridepublic String getParameter(String name) {MapString, String[] parameterMap getParameterMap();String[] values parameterMap.get(name);if (values null) {return null;}return values[0]; // 取回参数的第一个值}//取所有值Overridepublic String[] getParameterValues(String name) {MapString, String[] parameterMap getParameterMap();String[] values parameterMap.get(name);return values;} }这个也是我在网上找的一些大神写的一般情况下SpringMVC默认的乱码处理就已经能够很好的解决了 然后在web.xml中配置这个过滤器即可 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hbjioweA-1609824493730)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104124235595.png)] 乱码问题需要平时多注意在尽可能能设置编码的地方都设置为统一编码 UTF-8 Json交互处理 JSON简介 JSON(JavaScript Object Notation, JS 对象标记) 是一种轻量级的数据交换格式目前使用特别广泛。采用完全独立于编程语言的文本格式来存储和表示数据。简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。易于人阅读和编写同时也易于机器解析和生成并有效地提升网络传输效率。 在 JavaScript 语言中一切都是对象。因此任何JavaScript 支持的类型都可以通过 JSON 来表示例如字符串、数字、对象、数组等。看看他的要求和语法格式 对象表示为键值对数据由逗号分隔花括号保存对象方括号保存数组 JSON 键值对是用来保存 JavaScript 对象的一种方式和 JavaScript 对象的写法也大同小异键/值对组合中的键名写在前面并用双引号 “” 包裹使用冒号 : 分隔然后紧接着值 {name: QinJiang} {age: 3} {sex: 男}很多人搞不清楚 JSON 和 JavaScript 对象的关系甚至连谁是谁都不清楚。其实可以这么理解 JSON 是 JavaScript 对象的字符串表示法它使用文本表示一个 JS 对象的信息本质是一个字符串。 var obj {a: Hello, b: World}; //这是一个对象注意键名也是可以使用引号包裹的 var json {a: Hello, b: World}; //这是一个 JSON 字符串本质是一个字符串JSON 和 JavaScript 对象互转 要实现从JSON字符串转换为JavaScript 对象使用 JSON.parse() 方法 var obj JSON.parse({a: Hello, b: World}); //结果是 {a: Hello, b: World}要实现从JavaScript 对象转换为JSON字符串使用 JSON.stringify() 方法 var json JSON.stringify({a: Hello, b: World}); //结果是 {a: Hello, b: World}代码测试 1、新建一个module springmvc-05-json 添加web的支持 2、在web目录下新建一个 json-1.html 编写测试内容 !DOCTYPE html html langen headmeta charsetUTF-8titleJSON_秦疆/title /head bodyscript typetext/javascript//编写一个js的对象var user {name:秦疆,age:3,sex:男};//将js对象转换成json字符串var str JSON.stringify(user);console.log(str);//将json字符串转换为js对象var user2 JSON.parse(str);console.log(user2.age,user2.name,user2.sex);/script/body /html3、在IDEA中使用浏览器打开查看控制台输出 Controller返回JSON数据(Jackson) Jackson应该是目前比较好的json解析工具了 当然工具不止这一个比如还有阿里巴巴的 fastjson 等等。 我们这里使用Jackson使用它需要导入它的jar包 !-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-core -- dependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-databind/artifactIdversion2.10.0/version /dependency配置SpringMVC需要的配置 web.xml ?xml version1.0 encodingUTF-8? web-app xmlnshttp://xmlns.jcp.org/xml/ns/javaeexmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsdversion4.0!--1.注册servlet--servletservlet-nameSpringMVC/servlet-nameservlet-classorg.springframework.web.servlet.DispatcherServlet/servlet-class!--通过初始化参数指定SpringMVC配置文件的位置进行关联--init-paramparam-namecontextConfigLocation/param-nameparam-valueclasspath:springmvc-servlet.xml/param-value/init-param!-- 启动顺序数字越小启动越早 --load-on-startup1/load-on-startup/servlet!--所有请求都会被springmvc拦截 --servlet-mappingservlet-nameSpringMVC/servlet-nameurl-pattern//url-pattern/servlet-mappingfilterfilter-nameencoding/filter-namefilter-classorg.springframework.web.filter.CharacterEncodingFilter/filter-classinit-paramparam-nameencoding/param-nameparam-valueutf-8/param-value/init-param/filterfilter-mappingfilter-nameencoding/filter-nameurl-pattern//url-pattern/filter-mapping/web-appspringmvc-servlet.xml ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxmlns:mvchttp://www.springframework.org/schema/mvcxsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttps://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttps://www.springframework.org/schema/mvc/spring-mvc.xsd!-- 自动扫描指定的包下面所有注解类交给IOC容器管理 --context:component-scan base-packagecom.kuang.controller/!-- 视图解析器 --beanclassorg.springframework.web.servlet.view.InternalResourceViewResolveridinternalResourceViewResolver!-- 前缀 --property nameprefix value/WEB-INF/jsp/ /!-- 后缀 --property namesuffix value.jsp //bean/beans我们随便编写一个User的实体类然后我们去编写我们的测试Controller package com.kuang.pojo;import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor;//需要导入lombok Data AllArgsConstructor NoArgsConstructor public class User {private String name;private int age;private String sex;}这里我们需要两个新东西一个是ResponseBody一个是ObjectMapper对象我们看下具体的用法 编写一个Controller Controller public class UserController {RequestMapping(/json1)ResponseBody//不会走视图解析器public String json1() throws JsonProcessingException {//创建一个jackson的对象映射器用来解析数据ObjectMapper mapper new ObjectMapper();//创建一个对象User user new User(秦疆1号, 3, 男);//将我们的对象解析成为json格式String str mapper.writeValueAsString(user);//由于ResponseBody注解这里会将str转成json格式返回十分方便return str;}}[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KpV7w7iW-1609824493731)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104132327953.png)] 配置Tomcat 启动测试一下 http://localhost:8080/json1 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qv8EMo1Y-1609824493733)(data:image/gif;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVQImWNgYGBgAAAABQABh6FO1AAAAABJRU5ErkJggg)] 发现出现了乱码问题我们需要设置一下他的编码格式为utf-8以及它返回的类型 通过RequestMaping的produces属性来实现修改下代码 //produces:指定响应体返回类型和编码 RequestMapping(value /json1,produces application/json;charsetutf-8)再次测试 http://localhost:8080/json1 乱码问题OK 【注意使用json记得处理乱码问题】 代码优化 乱码统一解决 上一种方法比较麻烦如果项目中有许多请求则每一个都要添加可以通过Spring配置统一指定这样就不用每次都去处理了 我们可以在springmvc的配置文件上添加一段消息StringHttpMessageConverter转换配置 mvc:annotation-drivenmvc:message-converters register-defaultstruebean classorg.springframework.http.converter.StringHttpMessageConverterconstructor-arg valueUTF-8//beanbean classorg.springframework.http.converter.json.MappingJackson2HttpMessageConverterproperty nameobjectMapperbean classorg.springframework.http.converter.json.Jackson2ObjectMapperFactoryBeanproperty namefailOnEmptyBeans valuefalse//bean/property/bean/mvc:message-converters/mvc:annotation-driven返回json字符串统一解决 在类上直接使用 RestController 这样子里面所有的方法都只会返回 json 字符串了不用再每一个都添加ResponseBody 我们在前后端分离开发中一般都使用 RestController 十分便捷 RestController public class UserController {//produces:指定响应体返回类型和编码RequestMapping(value /json1)public String json1() throws JsonProcessingException {//创建一个jackson的对象映射器用来解析数据ObjectMapper mapper new ObjectMapper();//创建一个对象User user new User(秦疆1号, 3, 男);//将我们的对象解析成为json格式String str mapper.writeValueAsString(user);//由于ResponseBody注解这里会将str转成json格式返回十分方便return str;}}启动tomcat测试结果都正常输出 测试集合输出 增加一个新的方法 RequestMapping(/json2) public String json2() throws JsonProcessingException {//创建一个jackson的对象映射器用来解析数据ObjectMapper mapper new ObjectMapper();//创建一个对象User user1 new User(秦疆1号, 3, 男);User user2 new User(秦疆2号, 3, 男);User user3 new User(秦疆3号, 3, 男);User user4 new User(秦疆4号, 3, 男);ListUser list new ArrayListUser();list.add(user1);list.add(user2);list.add(user3);list.add(user4);//将我们的对象解析成为json格式String str mapper.writeValueAsString(list);return str; }运行结果 : 十分完美没有任何问题 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-QayyJ48k-1609824493741)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104135200844.png)] 输出时间对象 增加一个新的方法 util包下面 RequestMapping(/json3) public String json3() throws JsonProcessingException {ObjectMapper mapper new ObjectMapper();//创建时间一个对象java.util.DateDate date new Date();//将我们的对象解析成为json格式String str mapper.writeValueAsString(date);return str; }运行结果 : 默认日期格式会变成一个数字是1970年1月1日到当前日期的毫秒数Jackson 默认是会把时间转成timestamps形式 解决方案取消timestamps形式 自定义时间格式 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WUBSCx53-1609824493744)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104135859499.png)] RequestMapping(/json4) public String json4() throws JsonProcessingException {ObjectMapper mapper new ObjectMapper();//不使用时间戳的方式mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);//自定义日期格式对象SimpleDateFormat sdf new SimpleDateFormat(yyyy-MM-dd HH:mm:ss);//指定日期格式mapper.setDateFormat(sdf);Date date new Date();String str mapper.writeValueAsString(date);return str; }运行结果 : 成功的输出了时间 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-A6XBrVOH-1609824493745)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104140013731.png)] 抽取为工具类 如果要经常使用的话这样是比较麻烦的我们可以将这些代码封装到一个工具类中我们去编写下 package com.kuang.utils;import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.SerializationFeature;import java.text.SimpleDateFormat;public class JsonUtils {public static String getJson(Object object) {return getJson(object,yyyy-MM-dd HH:mm:ss);}public static String getJson(Object object,String dateFormat) {ObjectMapper mapper new ObjectMapper();//不使用时间差的方式mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);//自定义日期格式对象SimpleDateFormat sdf new SimpleDateFormat(dateFormat);//指定日期格式mapper.setDateFormat(sdf);try {return mapper.writeValueAsString(object);} catch (JsonProcessingException e) {e.printStackTrace();}return null;} }我们使用工具类代码就更加简洁了 RequestMapping(/json5) public String json5() throws JsonProcessingException {Date date new Date();String json JsonUtils.getJson(date);return json; }大功告成完美 FastJson fastjson.jar是阿里开发的一款专门用于Java开发的包可以方便的实现json对象与JavaBean对象的转换实现JavaBean对象与json字符串的转换实现json对象与json字符串的转换。实现json的转换方法很多最后的实现结果都是一样的。 fastjson 的 pom依赖 dependencygroupIdcom.alibaba/groupIdartifactIdfastjson/artifactIdversion1.2.60/version /dependencyfastjson 三个主要的类 JSONObject 代表 json 对象 JSONObject实现了Map接口, 猜想 JSONObject底层操作是由Map实现的。JSONObject对应json对象通过各种形式的get()方法可以获取json对象中的数据也可利用诸如size()isEmpty()等方法获取键值对的个数和判断是否为空。其本质是通过实现Map接口并调用接口中的方法完成的。 JSONArray 代表 json 对象数组 内部是有List接口中的方法来完成操作的。 JSON代表 JSONObject和JSONArray的转化 JSON类源码分析与使用仔细观察这些方法主要是实现json对象json对象数组javabean对象json字符串之间的相互转化。 代码测试我们新建一个FastJsonDemo 类 package com.kuang.controller;import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSONObject; import com.kuang.pojo.User;import java.util.ArrayList; import java.util.List;public class FastJsonDemo {public static void main(String[] args) {//创建一个对象User user1 new User(秦疆1号, 3, 男);User user2 new User(秦疆2号, 3, 男);User user3 new User(秦疆3号, 3, 男);User user4 new User(秦疆4号, 3, 男);ListUser list new ArrayListUser();list.add(user1);list.add(user2);list.add(user3);list.add(user4);System.out.println(*******Java对象 转 JSON字符串*******);String str1 JSON.toJSONString(list);System.out.println(JSON.toJSONString(list)str1);String str2 JSON.toJSONString(user1);System.out.println(JSON.toJSONString(user1)str2);System.out.println(\n****** JSON字符串 转 Java对象*******);User jp_user1JSON.parseObject(str2,User.class);System.out.println(JSON.parseObject(str2,User.class)jp_user1);System.out.println(\n****** Java对象 转 JSON对象 ******);JSONObject jsonObject1 (JSONObject) JSON.toJSON(user2);System.out.println((JSONObject) JSON.toJSON(user2)jsonObject1.getString(name));System.out.println(\n****** JSON对象 转 Java对象 ******);User to_java_user JSON.toJavaObject(jsonObject1, User.class);System.out.println(JSON.toJavaObject(jsonObject1, User.class)to_java_user);} }这种工具类我们只需要掌握使用就好了在使用的时候在根据具体的业务去找对应的实现。和以前的commons-io那种工具包一样拿来用就好了 Json在我们数据传输中十分重要一定要学会使用 整合SSM [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-m0UFSMoE-1609824493746)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104155244053.png)] 环境要求 环境 IDEAMySQL 5.7.19Tomcat 9Maven 3.6 要求 需要熟练掌握MySQL数据库SpringJavaWeb及MyBatis知识简单的前端知识 数据库环境 创建一个存放书籍数据的数据库表 CREATE DATABASE ssmbuild;USE ssmbuild;DROP TABLE IF EXISTS books;CREATE TABLE books ( bookID INT(10) NOT NULL AUTO_INCREMENT COMMENT 书id, bookName VARCHAR(100) NOT NULL COMMENT 书名, bookCounts INT(11) NOT NULL COMMENT 数量, detail VARCHAR(200) NOT NULL COMMENT 描述, KEY bookID (bookID) ) ENGINEINNODB DEFAULT CHARSETutf8INSERT INTO books(bookID,bookName,bookCounts,detail)VALUES (1,Java,1,从入门到放弃), (2,MySQL,10,从删库到跑路), (3,Linux,5,从进门到进牢);基本环境搭建 1、新建一Maven项目ssmbuild 添加web的支持 2、导入相关的pom依赖 dependencies!--Junit--dependencygroupIdjunit/groupIdartifactIdjunit/artifactIdversion4.12/version/dependency!--数据库驱动--dependencygroupIdmysql/groupIdartifactIdmysql-connector-java/artifactIdversion5.1.47/version/dependency!-- 数据库连接池 --dependencygroupIdcom.mchange/groupIdartifactIdc3p0/artifactIdversion0.9.5.2/version/dependency!--Servlet - JSP --dependencygroupIdjavax.servlet/groupIdartifactIdservlet-api/artifactIdversion2.5/version/dependencydependencygroupIdjavax.servlet.jsp/groupIdartifactIdjsp-api/artifactIdversion2.2/version/dependencydependencygroupIdjavax.servlet/groupIdartifactIdjstl/artifactIdversion1.2/version/dependency!--Mybatis--dependencygroupIdorg.mybatis/groupIdartifactIdmybatis/artifactIdversion3.5.2/version/dependencydependencygroupIdorg.mybatis/groupIdartifactIdmybatis-spring/artifactIdversion2.0.2/version/dependency!--Spring--dependencygroupIdorg.springframework/groupIdartifactIdspring-webmvc/artifactIdversion5.1.9.RELEASE/version/dependencydependencygroupIdorg.springframework/groupIdartifactIdspring-jdbc/artifactIdversion5.1.9.RELEASE/version/dependency /dependencies3、Maven资源过滤设置 buildresourcesresourcedirectorysrc/main/java/directoryincludesinclude**/*.properties/includeinclude**/*.xml/include/includesfilteringfalse/filtering/resourceresourcedirectorysrc/main/resources/directoryincludesinclude**/*.properties/includeinclude**/*.xml/include/includesfilteringfalse/filtering/resource/resources /build4、建立基本结构和配置框架,连接数据库 com.kuang.pojo com.kuang.dao com.kuang.service com.kuang.controller mybatis-config.xml ?xml version1.0 encodingUTF-8 ? !DOCTYPE configurationPUBLIC -//mybatis.org//DTD Config 3.0//ENhttp://mybatis.org/dtd/mybatis-3-config.dtd configuration/configurationapplicationContext.xml ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd/beansMybatis层 1、数据库配置文件 database.properties jdbc.drivercom.mysql.jdbc.Driver jdbc.urljdbc:mysql://localhost:3306/ssmbuild?useSSLtrueuseUnicodetruecharacterEncodingutf8 jdbc.usernameroot jdbc.password1234562、IDEA关联数据库 3、编写MyBatis的核心配置文件 配置数据源交给spring去做 ?xml version1.0 encodingUTF-8 ? !DOCTYPE configurationPUBLIC -//mybatis.org//DTD Config 3.0//ENhttp://mybatis.org/dtd/mybatis-3-config.dtd configurationtypeAliasespackage namecom.kuang.pojo//typeAliasesmappersmapper resourcecom/kuang/dao/BookMapper.xml//mappers/configuration4、编写数据库对应的实体类 com.kuang.pojo.Books 使用lombok插件 dependencygroupIdorg.projectlombok/groupIdartifactIdlombok/artifactIdversion1.18.16/version /dependencypackage com.kuang.pojo;import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor;Data AllArgsConstructor NoArgsConstructor public class Books {private int bookID;private String bookName;private int bookCounts;private String detail;}5、编写Dao层的 Mapper接口 package com.kuang.dao;import com.kuang.pojo.Books; import java.util.List;public interface BookMapper {//增加一个Bookint addBook(Books book);//根据id删除一个Bookint deleteBookById(Param(bookTd) int id);//更新Bookint updateBook(Books books);//根据id查询,返回一个BookBooks queryBookById(int id);//查询全部Book,返回list集合ListBooks queryAllBook();}6、编写接口对应的 Mapper.xml 文件。需要导入MyBatis的包 ?xml version1.0 encodingUTF-8 ? !DOCTYPE mapperPUBLIC -//mybatis.org//DTD Mapper 3.0//ENhttp://mybatis.org/dtd/mybatis-3-mapper.dtdmapper namespacecom.kuang.dao.BookMapper!--增加一个Book--insert idaddBook parameterTypeBooksinsert into ssmbuild.books(bookName,bookCounts,detail)values (#{bookName}, #{bookCounts}, #{detail})/insert!--根据id删除一个Book--delete iddeleteBookById parameterTypeintdelete from ssmbuild.books where bookID#{bookId}/delete!--更新Book--update idupdateBook parameterTypeBooksupdate ssmbuild.booksset bookName #{bookName},bookCounts #{bookCounts},detail #{detail}where bookID #{bookID}/update!--根据id查询,返回一个Book--select idqueryBookById resultTypeBooksselect * from ssmbuild.bookswhere bookID #{bookID}/select!--查询全部Book--select idqueryAllBook resultTypeBooksSELECT * from ssmbuild.books/select/mappermappersmapper resourcecom/kuang/dao/BookMapper.xml/ /mappers7、编写Service层的接口和实现类 接口 package com.kuang.service;import com.kuang.pojo.Books;import java.util.List;//BookService:底下需要去实现,调用dao层 public interface BookService {//增加一个Bookint addBook(Books book);//根据id删除一个Bookint deleteBookById(int id);//更新Bookint updateBook(Books books);//根据id查询,返回一个BookBooks queryBookById(int id);//查询全部Book,返回list集合ListBooks queryAllBook(); }实现类 package com.kuang.service;import com.kuang.dao.BookMapper; import com.kuang.pojo.Books; import java.util.List;public class BookServiceImpl implements BookService {//调用dao层的操作设置一个set接口方便Spring管理private BookMapper bookMapper; //让spring进行托管public void setBookMapper(BookMapper bookMapper) {this.bookMapper bookMapper;}public int addBook(Books book) {return bookMapper.addBook(book);}public int deleteBookById(int id) {return bookMapper.deleteBookById(id);}public int updateBook(Books books) {return bookMapper.updateBook(books);}public Books queryBookById(int id) {return bookMapper.queryBookById(id);}public ListBooks queryAllBook() {return bookMapper.queryAllBook();} }OK到此底层需求操作编写完毕 Spring层 Spring整合MyBatis 1、配置Spring整合MyBatis我们这里数据源使用c3p0连接池 2、我们去编写Spring整合Mybatis的相关的配置文件spring-dao.xml ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttps://www.springframework.org/schema/context/spring-context.xsd!-- 配置整合mybatis --!-- 1.关联数据库文件 --context:property-placeholder locationclasspath:database.properties/!-- 2.数据库连接池 --!--数据库连接池dbcp 半自动化操作 不能自动连接c3p0 自动化操作自动的加载配置文件 并且设置到对象里面--bean iddataSource classcom.mchange.v2.c3p0.ComboPooledDataSource!-- 配置连接池属性 --property namedriverClass value${jdbc.driver}/property namejdbcUrl value${jdbc.url}/property nameuser value${jdbc.username}/property namepassword value${jdbc.password}/!-- c3p0连接池的私有属性 --property namemaxPoolSize value30/property nameminPoolSize value10/!-- 关闭连接后不自动commit --property nameautoCommitOnClose valuefalse/!-- 获取连接超时时间 --property namecheckoutTimeout value10000/!-- 当获取连接失败重试次数 --property nameacquireRetryAttempts value2//bean!-- 3.配置SqlSessionFactory对象 --bean idsqlSessionFactory classorg.mybatis.spring.SqlSessionFactoryBean!-- 注入数据库连接池 --property namedataSource refdataSource/!-- 配置MyBaties全局配置文件:mybatis-config.xml --property nameconfigLocation valueclasspath:mybatis-config.xml//bean!-- 4.配置扫描Dao接口包动态实现Dao接口注入到spring容器中 --!--解释 https://www.cnblogs.com/jpfss/p/7799806.html--bean classorg.mybatis.spring.mapper.MapperScannerConfigurer!-- 注入sqlSessionFactory --property namesqlSessionFactoryBeanName valuesqlSessionFactory/!-- 给出需要扫描Dao接口包 --property namebasePackage valuecom.kuang.dao//bean/beansSpring整合service层 ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd!-- 扫描service相关的bean --context:component-scan base-packagecom.kuang.service /!--BookServiceImpl注入到IOC容器中--bean idBookServiceImpl classcom.kuang.service.BookServiceImplproperty namebookMapper refbookMapper//bean!-- 配置事务管理器 --bean idtransactionManager classorg.springframework.jdbc.datasource.DataSourceTransactionManager!-- 注入数据库连接池 --property namedataSource refdataSource //bean/beansSpring层搞定再次理解一下Spring就是一个大杂烩一个容器对吧 SpringMVC层 1、web.xml ?xml version1.0 encodingUTF-8? web-app xmlnshttp://xmlns.jcp.org/xml/ns/javaeexmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsdversion4.0!--DispatcherServlet--servletservlet-nameDispatcherServlet/servlet-nameservlet-classorg.springframework.web.servlet.DispatcherServlet/servlet-classinit-paramparam-namecontextConfigLocation/param-name!--一定要注意:我们这里加载的是总的配置文件之前被这里坑了-- param-valueclasspath:applicationContext.xml/param-value/init-paramload-on-startup1/load-on-startup/servletservlet-mappingservlet-nameDispatcherServlet/servlet-nameurl-pattern//url-pattern/servlet-mapping!--encodingFilter--filterfilter-nameencodingFilter/filter-namefilter-classorg.springframework.web.filter.CharacterEncodingFilter/filter-classinit-paramparam-nameencoding/param-nameparam-valueutf-8/param-value/init-param/filterfilter-mappingfilter-nameencodingFilter/filter-nameurl-pattern/*/url-pattern/filter-mapping!--Session过期时间--session-configsession-timeout15/session-timeout/session-config/web-app2、spring-mvc.xml ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxmlns:mvchttp://www.springframework.org/schema/mvcxsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttps://www.springframework.org/schema/mvc/spring-mvc.xsd!-- 配置SpringMVC --!-- 1.开启SpringMVC注解驱动 --mvc:annotation-driven/!-- 2.静态资源默认servlet配置--mvc:default-servlet-handler/!-- 3.配置jsp 显示ViewResolver视图解析器 --bean classorg.springframework.web.servlet.view.InternalResourceViewResolverproperty nameviewClass valueorg.springframework.web.servlet.view.JstlView/property nameprefix value/WEB-INF/jsp//property namesuffix value.jsp//bean!-- 4.扫描web相关的bean --context:component-scan base-packagecom.kuang.controller//beans3、Spring配置整合文件applicationContext.xml ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdimport resourcespring-dao.xml/import resourcespring-service.xml/import resourcespring-mvc.xml//beans[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZmoryUOz-1609824493748)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104180419548.png)] [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-zAIOTRIj-1609824493749)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104180954684.png)] 配置文件暂时结束Controller 和 视图层编写 1、BookController 类编写 查询全部书籍 Controller RequestMapping(/book) public class BookController {AutowiredQualifier(BookServiceImpl)private BookService bookService;RequestMapping(/allBook)public String list(Model model) {ListBooks list bookService.queryAllBook();model.addAttribute(list, list);return allBook;} }2、编写首页 index.jsp % page languagejava contentTypetext/html; charsetUTF-8pageEncodingUTF-8 % !DOCTYPE HTML html headtitle首页/titlestyle typetext/cssa {text-decoration: none;color: black;font-size: 18px;}h3 {width: 180px;height: 38px;margin: 100px auto;text-align: center;line-height: 38px;background: deepskyblue;border-radius: 4px;}/style /head bodyh3a href${pageContext.request.contextPath}/book/allBook点击进入列表页/a /h3 /body /html3、书籍列表页面 allbook.jsp % taglib prefixc urihttp://java.sun.com/jsp/jstl/core % % page contentTypetext/html;charsetUTF-8 languagejava % html headtitle书籍列表/titlemeta nameviewport contentwidthdevice-width, initial-scale1.0!-- 引入 Bootstrap --link hrefhttps://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.cssrelstylesheet /head bodydiv classcontainerdiv classrow clearfixdiv classcol-md-12 columndiv classpage-headerh1small书籍列表 —— 显示所有书籍/small/h1/div/div/divdiv classrowdiv classcol-md-4 columna classbtn btn-primaryhref${pageContext.request.contextPath}/book/toAddBook新增/a/div/divdiv classrow clearfixdiv classcol-md-12 columntable classtable table-hover table-stripedtheadtrth书籍编号/thth书籍名字/thth书籍数量/thth书籍详情/thth操作/th/tr/theadtbodyc:forEach varbook items${requestScope.get(list)}trtd${book.getBookID()}/tdtd${book.getBookName()}/tdtd${book.getBookCounts()}/tdtd${book.getDetail()}/tdtdahref${pageContext.request.contextPath}/book/toUpdateBook?id${book.getBookID()}更改/a |ahref${pageContext.request.contextPath}/book/del/${book.getBookID()}删除/a/td/tr/c:forEach/tbody/table/div/div /div4、BookController 类编写 添加书籍 RequestMapping(/toAddBook) public String toAddPaper() {return addBook; }RequestMapping(/addBook) public String addPaper(Books books) {System.out.println(books);bookService.addBook(books);return redirect:/book/allBook; }5、添加书籍页面addBook.jsp % taglib prefixc urihttp://java.sun.com/jsp/jstl/core % % page contentTypetext/html;charsetUTF-8 languagejava %html headtitle新增书籍/titlemeta nameviewport contentwidthdevice-width, initial-scale1.0!-- 引入 Bootstrap --link hrefhttps://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.cssrelstylesheet /head body div classcontainerdiv classrow clearfixdiv classcol-md-12 columndiv classpage-headerh1small新增书籍/small/h1/div/div/divform action${pageContext.request.contextPath}/book/addBookmethodpost书籍名称input typetext namebookNamebrbrbr书籍数量input typetext namebookCountsbrbrbr书籍详情input typetext namedetailbrbrbrinput typesubmit value添加/form/div[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-j0007Vq7-1609824493751)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104184227092.png)] 6、BookController 类编写 修改书籍 RequestMapping(/toUpdateBook) public String toUpdateBook(Model model, int id) {Books books bookService.queryBookById(id);System.out.println(books);model.addAttribute(book,books );return updateBook; }RequestMapping(/updateBook) public String updateBook(Model model, Books book) {System.out.println(book);bookService.updateBook(book);Books books bookService.queryBookById(book.getBookID());model.addAttribute(books, books);return redirect:/book/allBook; }7、修改书籍页面 updateBook.jsp % taglib prefixc urihttp://java.sun.com/jsp/jstl/core % % page contentTypetext/html;charsetUTF-8 languagejava % html headtitle修改信息/titlemeta nameviewport contentwidthdevice-width, initial-scale1.0!-- 引入 Bootstrap --link hrefhttps://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.cssrelstylesheet /head body div classcontainerdiv classrow clearfixdiv classcol-md-12 columndiv classpage-headerh1small修改信息/small/h1/div/div/divform action${pageContext.request.contextPath}/book/updateBookmethodpostinput typehidden namebookID value${book.getBookID()}/书籍名称input typetext namebookNamevalue${book.getBookName()}/书籍数量input typetext namebookCountsvalue${book.getBookCounts()}/书籍详情input typetext namedetail value${book.getDetail() }/input typesubmit value提交//form/div前端隐藏域 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eU43WFt2-1609824493754)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104192804463.png)] [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-SunU7BqM-1609824493757)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104192944308.png)] 8、BookController 类编写 删除书籍 RequestMapping(/del/{bookId}) public String deleteBook(PathVariable(bookId) int id) {bookService.deleteBookById(id);return redirect:/book/allBook; }横切事务了解 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oWcLvMkx-1609824493758)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104191842691.png)] 导入jar包 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-apJQzPDM-1609824493760)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104192112127.png)] 配置Tomcat进行运行 到目前为止这个SSM项目整合已经完全的OK了可以直接运行进行测试这个练习十分的重要大家需要保证不看任何东西自己也可以完整的实现出来 项目结构图 小结及展望 这个是同学们的第一个SSM整合案例一定要烂熟于心 SSM框架的重要程度是不言而喻的学到这里大家已经可以进行基本网站的单独开发。但是这只是增删改查的基本操作。可以说学到这里大家才算是真正的步入了后台开发的门。也就是能找一个后台相关工作的底线。 或许很多人工作就做这些事情但是对于个人的提高来说还远远不够 Ajax 简介 AJAX Asynchronous JavaScript and XML异步的 JavaScript 和 XML。 AJAX 是一种在无需重新加载整个网页的情况下能够更新部分网页的技术。 Ajax 不是一种新的编程语言而是一种用于创建更好更快以及交互性更强的Web应用程序的技术。 在 2005 年Google 通过其 Google Suggest 使 AJAX 变得流行起来。Google Suggest能够自动帮你完成搜索单词。 Google Suggest 使用 AJAX 创造出动态性极强的 web 界面当您在谷歌的搜索框输入关键字时JavaScript 会把这些字符发送到服务器然后服务器会返回一个搜索建议的列表。 就和国内百度的搜索框一样! 传统的网页(即不用ajax技术的网页)想要更新内容或者提交一个表单都需要重新加载整个网页。 使用ajax技术的网页通过在后台服务器进行少量的数据交换就可以实现异步局部更新。 使用Ajax用户可以创建接近本地桌面应用的直接、高可用、更丰富、更动态的Web用户界面。 伪造Ajax 我们可以使用前端的一个标签来伪造一个ajax的样子。iframe标签 1、新建一个module sspringmvc-06-ajax 导入web支持 2、编写一个 ajax-frame.html 使用 iframe 测试感受下效果 !DOCTYPE html html head langenmeta charsetUTF-8titlekuangshen/title /head bodyscript typetext/javascriptwindow.onload function(){var myDate new Date();document.getElementById(currentTime).innerText myDate.getTime();};function LoadPage(){var targetUrl document.getElementById(url).value;console.log(targetUrl);document.getElementById(iframePosition).src targetUrl;}/scriptdivp请输入要加载的地址span idcurrentTime/span/ppinput idurl typetext valuehttps://www.baidu.com//input typebutton value提交 onclickLoadPage()/p /divdivh3加载页面位置/h3iframe idiframePosition stylewidth: 100%;height: 500px;/iframe /div/body /html3、使用IDEA开浏览器测试一下 利用AJAX可以做 注册时输入用户名自动检测用户是否已经存在。登陆时提示用户名密码错误删除数据行时将行ID发送到后台后台在数据库中删除数据库删除成功后在页面DOM中将数据行也删除。…等等 jQuery.ajax 纯JS原生实现Ajax我们不去讲解这里直接使用jquery提供的方便学习和使用避免重复造轮子有兴趣的可以去了解下JS原生XMLHttpRequest Ajax的核心是XMLHttpRequest对象(XHR)。XHR为向服务器发送请求和解析服务器响应提供了接口。能够以异步方式从服务器获取新数据。 jQuery 提供多个与 AJAX 有关的方法。 通过 jQuery AJAX 方法您能够使用 HTTP Get 和 HTTP Post 从远程服务器上请求文本、HTML、XML 或 JSON – 同时您能够把这些外部数据直接载入网页的被选元素中。 jQuery 不是生产者而是大自然搬运工。 jQuery Ajax本质就是 XMLHttpRequest对他进行了封装方便调用 jQuery.ajax(...)部分参数url请求地址type请求方式GET、POST1.9.0之后用methodheaders请求头data要发送的数据contentType即将发送信息至服务器的内容编码类型(默认: application/x-www-form-urlencoded; charsetUTF-8)async是否异步timeout设置请求超时时间毫秒beforeSend发送请求前执行的函数(全局)complete完成之后执行的回调函数(全局)success成功之后执行的回调函数(全局)error失败之后执行的回调函数(全局)accepts通过请求头发送给服务器告诉服务器当前客户端可接受的数据类型dataType将服务器端返回的数据转换成指定类型xml: 将服务器端返回的内容转换成xml格式text: 将服务器端返回的内容转换成普通文本格式html: 将服务器端返回的内容转换成普通文本格式在插入DOM中时如果包含JavaScript标签则会尝试去执行。script: 尝试将返回值当作JavaScript去执行然后再将服务器端返回的内容转换成普通文本格式json: 将服务器端返回的内容转换成相应的JavaScript对象jsonp: JSONP 格式使用 JSONP 形式调用函数时如 myurl?callback? jQuery 将自动替换 ? 为正确的函数名以执行回调函数我们来个简单的测试使用最原始的HttpServletResponse处理 , .最简单 , 最通用 1、配置web.xml 和 springmvc的配置文件复制上面案例的即可 【记得静态资源过滤和注解驱动配置上】 ?xml version1.0 encodingUTF-8? beans xmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexmlns:contexthttp://www.springframework.org/schema/contextxmlns:mvchttp://www.springframework.org/schema/mvcxsi:schemaLocationhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttps://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttps://www.springframework.org/schema/mvc/spring-mvc.xsd!-- 自动扫描指定的包下面所有注解类交给IOC容器管理 --context:component-scan base-packagecom.kuang.controller/mvc:default-servlet-handler /mvc:annotation-driven /!-- 视图解析器 --bean classorg.springframework.web.servlet.view.InternalResourceViewResolveridinternalResourceViewResolver!-- 前缀 --property nameprefix value/WEB-INF/jsp/ /!-- 后缀 --property namesuffix value.jsp //bean/beans2、编写一个AjaxController Controller public class AjaxController {RequestMapping(/a1)public void ajax1(String name , HttpServletResponse response) throws IOException{if (admin.equals(name)){response.getWriter().print(true);}else{response.getWriter().print(false);}}}3、导入jquery 可以使用在线的CDN 也可以下载导入 script srchttps://code.jquery.com/jquery-3.1.1.min.js/script script src${pageContext.request.contextPath}/statics/js/jquery-3.1.1.min.js/script4、编写index.jsp测试 % page contentTypetext/html;charsetUTF-8 languagejava % htmlheadtitle$Title$/title%--script srchttps://code.jquery.com/jquery-3.1.1.min.js/script--%script src${pageContext.request.contextPath}/statics/js/jquery-3.1.1.min.js/scriptscriptfunction a1(){$.post({url:${pageContext.request.contextPath}/a1,data:{name:$(#txtName).val()},success:function (data,status) {alert(data);alert(status);}});}/script/headbody%--onblur失去焦点触发事件--% 用户名:input typetext idtxtName onblura1()//body /html5、启动tomcat测试打开浏览器的控制台当我们鼠标离开输入框的时候可以看到发出了一个ajax的请求是后台返回给我们的结果测试成功 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-837r3Rwe-1609824493765)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210104234119790.png)] Springmvc实现 实体类user Data AllArgsConstructor NoArgsConstructor public class User {private String name;private int age;private String sex;}我们来获取一个集合对象展示到前端页面 RequestMapping(/a2) public ListUser ajax2(){ListUser list new ArrayListUser();list.add(new User(秦疆1号,3,男));list.add(new User(秦疆2号,3,男));list.add(new User(秦疆3号,3,男));return list; //由于RestController注解将list转成json格式返回 }前端页面 % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleTitle/title /head body input typebutton idbtn value获取数据/ table width80% aligncentertrtd姓名/tdtd年龄/tdtd性别/td/trtbody idcontent/tbody /tablescript src${pageContext.request.contextPath}/statics/js/jquery-3.1.1.min.js/script script$(function () {$(#btn).click(function () {$.post(${pageContext.request.contextPath}/a2,function (data) {console.log(data)var html;for (var i 0; i data.length ; i) {html tr td data[i].name /td td data[i].age /td td data[i].sex /td /tr}$(#content).html(html);});})}) /script /body /html成功实现了数据回显可以体会一下Ajax的好处 登录注册提示效果 我们再测试一个小Demo思考一下我们平时注册时候输入框后面的实时提示怎么做到的如何优化 我们写一个Controller RequestMapping(/a3) public String ajax3(String name,String pwd){String msg ;//模拟数据库中存在数据if (name!null){if (admin.equals(name)){msg OK;}else {msg 用户名输入错误;}}if (pwd!null){if (123456.equals(pwd)){msg OK;}else {msg 密码输入有误;}}return msg; //由于RestController注解将msg转成json格式返回 }前端页面 login.jsp % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleajax/titlescript src${pageContext.request.contextPath}/statics/js/jquery-3.1.1.min.js/scriptscriptfunction a1(){$.post({url:${pageContext.request.contextPath}/a3,data:{name:$(#name).val()},success:function (data) {if (data.toString()OK){$(#userInfo).css(color,green);}else {$(#userInfo).css(color,red);}$(#userInfo).html(data);}});}function a2(){$.post({url:${pageContext.request.contextPath}/a3,data:{pwd:$(#pwd).val()},success:function (data) {if (data.toString()OK){$(#pwdInfo).css(color,green);}else {$(#pwdInfo).css(color,red);}$(#pwdInfo).html(data);}});}/script /head body p用户名:input typetext idname onblura1()/span iduserInfo/span /p p密码:input typetext idpwd onblura2()/span idpwdInfo/span /p /body /html【记得处理json乱码问题】 mvc:annotation-drivenmvc:message-converters register-defaultstruebean classorg.springframework.http.converter.StringHttpMessageConverterconstructor-arg valueUTF-8//beanbean classorg.springframework.http.converter.json.MappingJackson2HttpMessageConverterproperty nameobjectMapperbean classorg.springframework.http.converter.json.Jackson2ObjectMapperFactoryBeanproperty namefailOnEmptyBeans valuefalse//bean/property/bean/mvc:message-converters /mvc:annotation-driven测试一下效果动态请求响应局部刷新就是如此 shift加F6 改文件名 拦截器 概述 SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。 **过滤器与拦截器的区别**拦截器是AOP思想的具体应用。 过滤器 servlet规范中的一部分任何java web工程都可以使用在url-pattern中配置了/*之后可以对所有要访问的资源进行拦截 拦截器 拦截器是SpringMVC框架自己的只有使用了SpringMVC框架的工程才能使用拦截器只会拦截访问的控制器方法 如果访问的是jsp/html/css/image/js是不会进行拦截的 自定义拦截器 那如何实现拦截器呢 想要自定义拦截器必须实现 HandlerInterceptor 接口。 1、新建一个Moudule springmvc-07-Interceptor 添加web支持 2、配置web.xml 和 springmvc-servlet.xml 文件 3、编写一个拦截器 package com.kuang.interceptor;import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;public class MyInterceptor implements HandlerInterceptor {//在请求处理的方法之前执行//如果返回true执行下一个拦截器//如果返回false就不执行下一个拦截器public boolean preHandle(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, Object o) throws Exception {System.out.println(------------处理前------------);return true;}//在请求处理方法执行之后执行public void postHandle(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView)throws Exception {System.out.println(------------处理后------------);}//在dispatcherServlet处理后执行,做清理工作.public void afterCompletion(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {System.out.println(------------清理------------);} }4、在springmvc的配置文件中配置拦截器 !--关于拦截器的配置-- mvc:interceptorsmvc:interceptor!--/** 包括路径及其子路径--!--/admin/* 拦截的是/admin/add等等这种 , /admin/add/user不会被拦截--!--/admin/** 拦截的是/admin/下的所有--mvc:mapping path/**/!--bean配置的就是拦截器--bean classcom.kuang.interceptor.MyInterceptor//mvc:interceptor /mvc:interceptors5、编写一个Controller接收请求 package com.kuang.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody;//测试拦截器的控制器 Controller public class InterceptorController {RequestMapping(/interceptor)ResponseBodypublic String testFunction() {System.out.println(控制器中的方法执行了);return hello;} }6、前端 index.jsp a href${pageContext.request.contextPath}/interceptor拦截器测试/a7、启动tomcat 测试一下 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OqZUcDzd-1609824493765)(data:image/gif;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVQImWNgYGBgAAAABQABh6FO1AAAAABJRU5ErkJggg)] 验证用户是否登录 (认证用户) 实现思路 1、有一个登陆页面需要写一个controller访问页面。 2、登陆页面有一提交表单的动作。需要在controller中处理。判断用户名密码是否正确。如果正确向session中写入用户信息。返回登陆成功。 3、拦截用户请求判断用户是否登陆。如果用户已经登陆。放行 如果用户未登陆跳转到登陆页面 测试 1、编写一个登陆页面 login.jsp % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleTitle/title /headh1登录页面/h1 hrbody form action${pageContext.request.contextPath}/user/login用户名input typetext nameusername br密码input typepassword namepwd brinput typesubmit value提交 /form /body /html2、编写一个Controller处理请求 package com.kuang.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping;import javax.servlet.http.HttpSession;Controller RequestMapping(/user) public class UserController {//跳转到登陆页面RequestMapping(/jumplogin)public String jumpLogin() throws Exception {return login;}//跳转到成功页面RequestMapping(/jumpSuccess)public String jumpSuccess() throws Exception {return success;}//登陆提交RequestMapping(/login)public String login(HttpSession session, String username, String pwd) throwsException {// 向session记录用户身份信息System.out.println(接收前端username);session.setAttribute(user, username);return success;}//退出登陆RequestMapping(logout)public String logout(HttpSession session) throws Exception {// session 过期session.invalidate();return login;} }3、编写一个登陆成功的页面 success.jsp % page contentTypetext/html;charsetUTF-8 languagejava % html headtitleTitle/title /head bodyh1登录成功页面/h1 hr${user} a href${pageContext.request.contextPath}/user/logout注销/a /body /html4、在 index 页面上测试跳转启动Tomcat 测试未登录也可以进入主页 % page contentTypetext/html;charsetUTF-8 languagejava % htmlheadtitle$Title$/title/headbodyh1首页/h1hr %--登录--%a href${pageContext.request.contextPath}/user/jumplogin登录/aa href${pageContext.request.contextPath}/user/jumpSuccess成功页面/a/body /html5、编写用户登录拦截器 package com.kuang.config;import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView;import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import java.io.IOException;public class LoginIntercepor implements HandlerInterceptor {public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {// 如果是登陆页面则放行System.out.println(uri: request.getRequestURI());if (request.getRequestURI().contains(login)) {return true;}HttpSession session request.getSession();// 如果用户已登陆也放行if(session.getAttribute(user) ! null) {return true;}// 用户没有登陆跳转到登陆页面request.getRequestDispatcher(/WEB-INF/jsp/login.jsp).forward(request,response);return false;}public void postHandle(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView)throws Exception {}public void afterCompletion(HttpServletRequest httpServletRequest,HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {} }6、在Springmvc的配置文件中注册拦截器 !--关于拦截器的配置-- mvc:interceptorsmvc:interceptormvc:mapping path/**/bean idloginInterceptor classcom.kuang.interceptor.LoginInterceptor//mvc:interceptor /mvc:interceptors7、再次重启Tomcat测试 OK测试登录拦截功能无误. 文件上传和下载 准备工作 文件上传是项目开发中最常见的功能之一 ,springMVC 可以很好的支持文件上传但是SpringMVC上下文中默认没有装配MultipartResolver因此默认情况下其不能处理文件上传工作。如果想使用Spring的文件上传功能则需要在上下文中配置MultipartResolver。 前端表单要求为了能上传文件必须将表单的method设置为POST并将enctype设置为multipart/form-data。只有在这样的情况下浏览器才会把用户选择的文件以二进制数据发送给服务器 对表单中的 enctype 属性做个详细的说明 application/x-wwwform-urlencoded默认方式只处理表单域中的 value 属性值采用这种编码方式的表单会将表单域中的值处理成 URL 编码方式。multipart/form-data这种编码方式会以二进制流的方式来处理表单数据这种编码方式会把文件域指定文件的内容也封装到请求参数中不会对字符编码。text/plain除了把空格转换为 “” 号外其他字符都不做编码处理这种方式适用直接通过表单发送邮件。 form action enctypemultipart/form-data methodpostinput typefile namefile/input typesubmit /form一旦设置了enctype为multipart/form-data浏览器即会采用二进制流的方式来处理表单数据而对于文件上传的处理则涉及在服务器端解析原始的HTTP响应。在2003年Apache Software Foundation发布了开源的Commons FileUpload组件其很快成为Servlet/JSP程序员上传文件的最佳选择。 Servlet3.0规范已经提供方法来处理文件上传但这种上传需要在Servlet中完成。而Spring MVC则提供了更简单的封装。Spring MVC为文件上传提供了直接的支持这种支持是用即插即用的MultipartResolver实现的。Spring MVC使用Apache Commons FileUpload技术实现了一个MultipartResolver实现类CommonsMultipartResolver。因此SpringMVC的文件上传还需要依赖Apache Commons FileUpload的组件。 文件上传 1、导入文件上传的jar包commons-fileupload Maven会自动帮我们导入他的依赖包 commons-io包 !--文件上传-- dependencygroupIdcommons-fileupload/groupIdartifactIdcommons-fileupload/artifactIdversion1.3.3/version /dependency !--servlet-api导入高版本的-- dependencygroupIdjavax.servlet/groupIdartifactIdjavax.servlet-api/artifactIdversion4.0.1/version /dependency2、配置beanmultipartResolver 【注意这个bena的id必须为multipartResolver 否则上传文件会报400的错误在这里栽过坑,教训】 !--文件上传配置-- bean idmultipartResolver classorg.springframework.web.multipart.commons.CommonsMultipartResolver!-- 请求的编码格式必须和jSP的pageEncoding属性一致以便正确读取表单的内容默认为ISO-8859-1 --property namedefaultEncoding valueutf-8/!-- 上传文件大小上限单位为字节1048576010M --property namemaxUploadSize value10485760/property namemaxInMemorySize value40960/ /beanCommonsMultipartFile 的 常用方法 String getOriginalFilename()获取上传文件的原名InputStream getInputStream()获取文件流void transferTo(File dest)将上传文件保存到一个目录文件中 我们去实际测试一下 3、编写前端页面 form action/upload enctypemultipart/form-data methodpostinput typefile namefile/input typesubmit valueupload /form4、Controller package com.kuang.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.multipart.commons.CommonsMultipartFile;import javax.servlet.http.HttpServletRequest; import java.io.*;Controller public class FileController {//RequestParam(file) 将namefile控件得到的文件封装成CommonsMultipartFile 对象//批量上传CommonsMultipartFile则为数组即可RequestMapping(/upload)public String fileUpload(RequestParam(file) CommonsMultipartFile file ,HttpServletRequest request) throws IOException {//获取文件名 : file.getOriginalFilename();String uploadFileName file.getOriginalFilename();//如果文件名为空直接回到首页if (.equals(uploadFileName)){return redirect:/index.jsp;}System.out.println(上传文件名 : uploadFileName);//上传路径保存设置String path request.getServletContext().getRealPath(/upload);//如果路径不存在创建一个File realPath new File(path);if (!realPath.exists()){realPath.mkdir();}System.out.println(上传文件保存地址realPath);InputStream is file.getInputStream(); //文件输入流OutputStream os new FileOutputStream(new File(realPath,uploadFileName));//文件输出流//读取写出int len0;byte[] buffer new byte[1024];while ((lenis.read(buffer))!-1){os.write(buffer,0,len);os.flush();}os.close();is.close();return redirect:/index.jsp;} }5、测试上传文件OK 采用file.Transto 来保存上传的文件 1、编写Controller /* * 采用file.Transto 来保存上传的文件 */ RequestMapping(/upload2) public String fileUpload2(RequestParam(file) CommonsMultipartFile file,HttpServletRequest request) throws IOException {//上传路径保存设置String path request.getServletContext().getRealPath(/upload);File realPath new File(path);if (!realPath.exists()){realPath.mkdir();}//上传文件地址System.out.println(上传文件保存地址realPath);//通过CommonsMultipartFile的方法直接写文件注意这个时候file.transferTo(new File(realPath / file.getOriginalFilename()));return redirect:/index.jsp; }2、前端表单提交地址修改 3、访问提交测试OK .getServletContext()爆红 将.getServletContext()更换为.getSession().getServletContext() 文件下载 文件下载步骤 1、设置 response 响应头 2、读取文件 – InputStream 3、写出文件 – OutputStream 4、执行操作 5、关闭流 先开后关 代码实现 RequestMapping(value/download) public String downloads(HttpServletResponse response ,HttpServletRequest request)throws Exception{//要下载的图片地址String path request.getServletContext().getRealPath(/upload);String fileName 基础语法.jpg;//1、设置response 响应头response.reset(); //设置页面不缓存,清空bufferresponse.setCharacterEncoding(UTF-8); //字符编码response.setContentType(multipart/form-data); //二进制传输数据//设置响应头response.setHeader(Content-Disposition,attachment;fileNameURLEncoder.encode(fileName, UTF-8));File file new File(path,fileName);//2、 读取文件--输入流InputStream inputnew FileInputStream(file);//3、 写出文件--输出流OutputStream out response.getOutputStream();byte[] buff new byte[1024];int index0;//4、执行 写出操作while((index input.read(buff))! -1){out.write(buff, 0, index);out.flush();}out.close();input.close();return null; }前端 a href/download点击下载/a测试文件下载OK大家可以和我们之前学习的JavaWeb原生的方式对比一下就可以知道这个便捷多了! java.net包下 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wMPdXivu-1609824493767)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210105124258999.png)] 总结 mabatis [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qD9LOz1R-1609824493769)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210105131601727.png)] spring [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YDo8lkVZ-1609824493772)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210105131644934.png)] springmvc [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bgl1q8ha-1609824493773)(C:\Users\王东梁\AppData\Roaming\Typora\typora-user-images\image-20210105131715639.png)]
http://www.yutouwan.com/news/307472/

相关文章:

  • 怎样模仿别人的网站云浮seo
  • 张家港城市建设规划局网站自己做彩票网站简单吗
  • 网站建设费在会计上怎么入账湖州市建设局网站6
  • 猪八戒网做动漫弹幕网站免费搭建手机自助网站
  • 珠海专业网站建设公司网站怎么能被百度收录
  • rp网站自动跳转图片怎么做宁波网络营销策划公司
  • 中国建设银行个人网站登录品牌建设管理办法
  • 站长之家论坛八大处网站建设
  • 好看的网站页面静态网站举例
  • 自建国际网站做电商做网站需要用什么语言开发
  • 网站素材包括哪些流量型网站
  • wordpress多站点文章调用万网可以花钱做网站
  • 一种子网站做的很好的视频广告图片在线设计网站
  • 湖北响应式网站建设设计se 网站优化
  • 北京市违法建设投诉网站首码项目推广网站
  • 网站建设公司怎么开拓业务如何做英文版网站
  • 什么学做网站来安县城乡规划建设局网站
  • 网站流量统计软件广州化妆品网站建设
  • 汉口网站制作企业简介ppt模板免费
  • 一个最简单的产品展示的asp网站应该如何做做宠物食品的网站
  • 温州建站模板搭建怎么样能够为一个网站做推广
  • 机关单位网站管理部门应建立宁波专业的网站建设团队
  • 上海专业网站制作公司黄岛网站建设哪家专业
  • wordpress外贸建站燕郊网站建设
  • 外贸流程全步骤 外贸篇北京网站推广优化公司
  • 标题正文型网站网站关闭了域名备案
  • 购物网站一般分几大模块长春代做网站
  • 上海网站建设 分类广告网站首页设计收费
  • 做阿里巴巴网站口碑wordpress 婚庆主题
  • 网站建设设计技术方案模板微信公众号怎么搞