综合项目 旅游网【2. 优化servlet】没有指定的js文件读不到文件 错误

news2025/1/10 16:58:37

优化servlet

目的

减少Servlet的数量,现在是一个功能一个Servlet,将其优化为一个模块一个Servlet,相当于在数据库中一张表对应一个Servlet,在Servlet中提供不同的方法,完成用户的请求。

如何解决测试时控制台中文乱码:

设置→maven →Runner→VM Options→-Dfile.encoding=gb2312

UserServlet的编写

将之前的Servlet实现的功能,抽取到UserServlet中的不同方法中实现,并且将UserService创建抽取到成员变量位置

package cn.itcast.travel.web.servlet;

import cn.itcast.travel.domain.ResultInfo;
import cn.itcast.travel.domain.User;
import cn.itcast.travel.service.UserService;
import cn.itcast.travel.service.impl.UserServiceImpl;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.commons.beanutils.BeanUtils;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.util.Map;

/**
 * @author 乱码酱
 * @date :2023-01-04 08:39
 * @program: hello_maven
 * @create:
 */@WebServlet("/user/*")// /user/registpublicclassUserServletextendsHttpServlet {
    //声明UserService业务对象privateUserServiceservice=newUserServiceImpl();

    /**
     * 注册功能
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */publicvoidregist(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {

        //验证校验Stringcheck= request.getParameter("check");
        //从sesion中获取验证码HttpSessionsession= request.getSession();
        Stringcheckcode_server= (String) session.getAttribute("CHECKCODE_SERVER");
        session.removeAttribute("CHECKCODE_SERVER");//为了保证验证码只能使用一次//比较if(checkcode_server == null || !checkcode_server.equalsIgnoreCase(check)){
            //验证码错误ResultInfoinfo=newResultInfo();
            //注册失败
            info.setFlag(false);
            info.setErrorMsg("验证码错误");
            //将info对象序列化为jsonObjectMappermapper=newObjectMapper();
            Stringjson= mapper.writeValueAsString(info);
            response.setContentType("application/json;charset=utf-8");
            response.getWriter().write(json);
            return;
        }

        //1.获取数据
        Map<String, String[]> map = request.getParameterMap();

        //2.封装对象Useruser=newUser();
        try {
            BeanUtils.populate(user,map);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        //3.调用service完成注册//UserService service = new UserServiceImpl();booleanflag= service.regist(user);
        ResultInfoinfo=newResultInfo();
        //4.响应结果if(flag){
            //注册成功
            info.setFlag(true);
        }else{
            //注册失败
            info.setFlag(false);
            info.setErrorMsg("注册失败!");
        }

        //将info对象序列化为jsonObjectMappermapper=newObjectMapper();
        Stringjson= mapper.writeValueAsString(info);

        //将json数据写回客户端//设置content-type
        response.setContentType("application/json;charset=utf-8");
        response.getWriter().write(json);

    }

    /**
     * 登录功能
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */publicvoidlogin(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        //1.获取用户名和密码数据
        Map<String, String[]> map = request.getParameterMap();
        //2.封装User对象Useruser=newUser();
        try {
            BeanUtils.populate(user,map);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        //3.调用Service查询// UserService service = new UserServiceImpl();Useru= service.login(user);

        ResultInfoinfo=newResultInfo();

        //4.判断用户对象是否为nullif(u == null){
            //用户名密码或错误
            info.setFlag(false);
            info.setErrorMsg("用户名密码或错误");
        }
        //5.判断用户是否激活if(u != null && !"Y".equals(u.getStatus())){
            //用户尚未激活
            info.setFlag(false);
            info.setErrorMsg("您尚未激活,请激活");
        }
        //6.判断登录成功if(u != null && "Y".equals(u.getStatus())){
            request.getSession().setAttribute("user",u);//登录成功标记//登录成功
            info.setFlag(true);
        }

        //响应数据ObjectMappermapper=newObjectMapper();

        response.setContentType("application/json;charset=utf-8");
        mapper.writeValue(response.getOutputStream(),info);
    }

    /**
     * 查询单个对象
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */publicvoidfindOne(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
//        //从session中获取登录用户//        Object user = request.getSession().getAttribute("user");//        //将user写回客户端       /* ObjectMapper mapper = new ObjectMapper();//        response.setContentType("application/json;charset=utf-8");//        mapper.writeValue(response.getOutputStream(),user);*///        writeValue(user,response);//从session中获取登录用户Objectuser= request.getSession().getAttribute("user");
//        将user写回客户端ObjectMappermapper=newObjectMapper();
        response.setContentType("application/json;charset=utf-8");
        mapper.writeValue(response.getOutputStream(),user);
    }

    /**
     * 退出功能
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */publicvoidexit(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        //1.销毁session
        request.getSession().invalidate();

        //2.跳转登录页面
        response.sendRedirect(request.getContextPath()+"/login.html");
    }

    /**
     * 激活功能
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */publicvoidactive(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {
        //1.获取激活码Stringcode= request.getParameter("code");
        if(code != null){
            //2.调用service完成激活//UserService service = new UserServiceImpl();booleanflag= service.active(code);

            //3.判断标记Stringmsg=null;
            if(flag){
                //激活成功
                msg = "激活成功,请<a href='login.html'>登录</a>";
            }else{
                //激活失败
                msg = "激活失败,请联系管理员!";
            }
            response.setContentType("text/html;charset=utf-8");
            response.getWriter().write(msg);
        }
    }
}

页面路径的更改

register.html中将"registUserServlet"改成"user/regist"

UserServlet中将所有方法写在一起,方便调用,对应的.html中的路径也要发生改变,login.html中"loginServlet"改成"user/login"

header.html中"findUserServlet"改成"user/findOne"

激活邮件,发送邮件正文则是在USerServiceImpl,需要将路径

"<a href='http://localhost/travel/activeUserServlet?code="中"activeUserServlet"部分替换成"user/active"

退出则是header.html中的"exitServlet"改成"user/exit"

BaseServlet编写

package cn.itcast.travel.web.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * @author 乱码酱
 * @date :2023-01-03 17:58
 * @program: hello_maven
 * @create:BaseServlet
 */@WebServlet("/baseServlet")publicclassBaseServletextendsHttpServlet {
    @Overrideprotectedvoidservice(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
        //System.out.println("baseServlet的service方法被执行了...");//完成方法分发//1.获取请求路径Stringuri= req.getRequestURI(); //   /travel/user/add
        System.out.println("请求uri:"+uri);//  /travel/user/add//2.获取方法名称                         从后往前数,所以要+1StringmethodName= uri.substring(uri.lastIndexOf('/') + 1);
        System.out.println("方法名称:"+methodName);
        //3.获取方法对象Method//谁调用我?我代表谁
        System.out.println(this);//UserServlet的对象cn.itcast.travel.web.servlet.UserServlet@4903d97etry {
            //获取方法Methodmethod=this.getClass().getMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
            //4.执行方法//暴力反射 可用于私密方法//method.setAccessible(true);
            method.invoke(this,req,resp);
            //invoke就是调用类中的方法,最简单的用法是可以把方法参数化invoke(class, method)
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
        
    }

}

【错误】

在查看代码时发现两个错误,一个是没有指定的js文件,一个是读不到文件

问题一:GET http://localhost/travel/js/angular.min.js net::ERR_ABORTED 404 (Not Found)

问题二:Uncaught TypeError: Cannot read properties of null (reading 'name')

at Object.eval [as success] (eval at <anonymous> (jquery-1.11.0.min.js:2:2616), <anonymous>:5:36)

at j (jquery-1.11.0.min.js:2:27136)

at Object.fireWith [as resolveWith] (jquery-1.11.0.min.js:2:27949)

at x (jquery-1.11.0.min.js:4:22244)

at XMLHttpRequest.b (jquery-1.11.0.min.js:4:26298)

【问题一解决】没有指定的js文件:

去官网https://code.angularjs.org找了一个

下载过来

【问题二解决】没获取到值

出现原因:这是由于还没登录,data.name 还没获取到值,所以为null

方法1:可以不管,它不会影响其他功能,当你登录后,就不会报这个错了

方法2:给它一个if判断是否为null

分类数据展示

效果

分析

代码实现

后台代码

在BaseServlet中添加序列化方法

/**
     * 直接将传入的对象序列化为json,并且写回客户端
     * @param obj
     */publicvoidwriteValue(Object obj,HttpServletResponse response)throws IOException {
        ObjectMappermapper=newObjectMapper();
        response.setContentType("application/json;charset=utf-8");
        mapper.writeValue(response.getOutputStream(),obj);
    }

    /**
     * 将传入的对象序列化为json,返回调用者
     * @param obj
     * @return
     */public String writeValueAsString(Object obj)throws JsonProcessingException {
        ObjectMappermapper=newObjectMapper();
        return mapper.writeValueAsString(obj);
    }

我们在读取数据时是从数据库读取而不是前台给定,所以将之前的前台代码注释

查询分类数据代码

heder.html

//    查询分类数据
        $.get("category/findAll", {}, function (data) {
            //data(一个数组): [{cid:1,came:国内游},{},{}]varlis='<li class="nav-active"><a href="index.html">首页</a></li>';
            //遍历数组,拼接字符串<li>for (vari=0; i < data.length; i++) {
                varli='<li><a href="route_list.html">' + data[i].cname + '</a></li>';
                lis += li;
            }
        //拼接收藏排行榜的li,<li><a href="favoriterank.html">收藏排行榜</a></li>
            lis += '<li><a href="favoriterank.html">收藏排行榜</a></li>';
            //将lis字符串,设置到ul的html内容中
            $("#category").html(lis);

        });

对分类数据进行缓存优化

分析发现,分类的数据在每一次页面加载后都会重新请求数据库来加载,对数据库的压力比较大,而且分类的数据不会经常产生变化,所以可以使用redis来缓存这个数据。

分析:

优化代码实现

【出错】获取种类有异常数据

测试时发现获取种类有异常数据,回到数据库一看发现是数据库出错,于是修改数据库,但是修改后重启还是原来的异常数据,怀疑是浏览器自带的缓存问题,清空浏览器缓存还是无效,初步怀疑是redis缓存的问题

但是页面刷新还是原来的错误数据

【解决方法】

光清除浏览器缓存可不够,还要清除redis缓存

flushdb ——> 清空当前数据库中的所有 key

flushall ——> 清空整个 Redis 服务器的数据(删除所有数据库的所有 key )

到这里就清除完毕,不需要重启,不过大家要注意,你们服务器的缓存数据是否重要,是否可以清除,一定要确定清楚,否则不要轻易清除。

重新刷新记录正确,redis也查询得到数据

而且第一次显示是从数据库查询得到

删除控制台数据刷新网页,数据将从缓存中读取

CategoryServiceImpl 代码

package cn.itcast.travel.service.impl;

import cn.itcast.travel.dao.CategoryDao;
import cn.itcast.travel.dao.impl.CategoryDaoImpl;
import cn.itcast.travel.domain.Category;
import cn.itcast.travel.service.CategoryService;
import cn.itcast.travel.util.JedisUtil;
import redis.clients.jedis.Jedis;

import java.util.ArrayList;
import java.util.List;
import java.util.Set;

/**
 * @author 乱码酱
 * @date :2023-01-04 10:58
 * @program: hello_maven
 * @create:
 */publicclassCategoryServiceImplimplementsCategoryService {

    privateCategoryDaocategoryDao=newCategoryDaoImpl();

    @Override//查询所有方法public List<Category> findAll() {
        //1.从redis中查询//1.1获取jedis客户端Jedisjedis= JedisUtil.getJedis();
        //1.2可使用sortedset排序查询(有序且唯一)
        Set<String> categorys = jedis.zrange("category", 0, -1);
        List<Category> cs = null;
        //2.判断查询的集合是否为空if (categorys == null || categorys.size() == 0) {
            System.out.println("从数据库查询....");
            //3.如果为空,需要从数据库查询,在将数据存入redis//3.1 从数据库查询
            cs = categoryDao.findAll();
            //3.2 将集合数据存储到redis中名为category的keyfor (inti=0; i < cs.size(); i++) {
                jedis.zadd("category", cs.get(i).getCid(), cs.get(i).getCname());
            }
        } else {
            System.out.println("从redis中查询.....");

            //4.如果不为空,将set的数据存入list
            cs = newArrayList<Category>();
            for (String name : categorys) {
                Categorycategory=newCategory();
                category.setCname(name);
                cs.add(category);

            }
        }

        return cs;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/149958.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

VS2019+Opencv3.4+Win10配置详解

一.下载opencv 官网&#xff1a;Releases - OpenCV 不同版本vs对应不同版本的opencv,其中高版本vs可以配置低版本vc&#xff0c;低版本不能配置高版本vc。 windows系统直接下载Windows版本就可以&#xff08;下载的文件是一个exe文件&#xff0c;运行相当于解压缩&#xff0…

1143汉诺塔

题目描述汉诺塔问题是这样的&#xff1a;有3根柱子A,B,C&#xff0c;其中A柱上有64个盘子&#xff0c;盘子大小不等&#xff0c;大的在下&#xff0c;小的在上。要求把这64个盘子从A柱移到C柱上&#xff0c;在移动过程中可以借助B柱&#xff0c;每次只允许移动一个盘子&#xf…

什么是 Java 泛型?怎样使用 Java 泛型?

目录 1、为什么使用泛型&#xff1f; 2、什么是泛型类&#xff1f;如何定义一个泛型类&#xff1f; 泛型的命名约定 3、什么是泛型方法&#xff1f;如何定义一个泛型方法&#xff1f; 4、什么是有界类型参数&#xff1f;如何定义有界类型参数&#xff1f; &#xff08;1&…

Maven高级-私服

分模块合作开发 9.2)Nexus Nexus是Sonatype公司的一款maven私服产品 下载地址&#xff1a;https://help.sonatype.com/repomanager3/download Nexus*安装、启动与配置** 启动服务器&#xff08;命令行启动&#xff09; nexus.exe /run nexus访问服务器&#xff08;默认端口…

linux安装部署vsftpd

yum直接安装yum -y install vsftpd ftp 创建新用户&#xff1a;ftpd更新ftpd密码&#xff1a;echo "123456" |passwd --stdin ftpd创建ftp目录&#xff1a;mkdir -p /home/ftpd/test授权&#xff1a;chown -R ftpd:ftpd /home/ftpd/testchmod 777 -R /home/ftpd/test…

某程序员哀叹:最近阳的人越来越多,面对员工们纷纷倒下,公司领导公然宣称“发烧请病假不等于在家睡大觉,再不回复工作就滚蛋”...

最近阳的人越来越多&#xff0c;面对员工们纷纷倒下&#xff0c;有的公司通情达理&#xff0c;有的公司却开始“不当人”了。一位网友曝光公司领导在群里所有人&#xff0c;称“发烧请病假不意味着在家睡大觉&#xff0c;啥也不管&#xff0c;联系不上&#xff0c;安排不予响应…

【SAP Hana】SAP HANA SQL 基础教程

SAP HANA SQL 基础教程1、SQL 标准简介2、HANA STUDIO 的安装3、HANA STUDIO 的设置4、HANA SQL 基础教程&#xff08;1&#xff09;查看表数据&#xff08;2&#xff09;查看表结构&#xff08;3&#xff09;SELECT&#xff08;4&#xff09;WHERE&#xff08;5&#xff09;WH…

B站直播带货,带货直播数据如何查看?

随着时代发展&#xff0c;直播电商带货也是越来越火&#xff0c;在这个直播带货火热期&#xff0c;B站也是当仁不让的加入到直播带货行业中&#xff0c;在今年双11中&#xff0c;B站第一次参加双十一直播电商混战&#xff0c;但是并未像其他电商平台一般&#xff0c;趁双十一流…

【自学Python】Python浮点型(float)

Python浮点型(float) Python浮点型(float)教程 Python 浮点型数值用于保存带小数点的数值。Python 的浮点数有两种表示形式&#xff0c;即十进制形式和科学计数法形式。 Python浮点型(float)详解 十进制形式 Python 最常见的浮点型数就是十进制形式的浮点型数。Python 中的…

Java-类加载

静态加载和动态加载 4种加载时机&#xff0c;只有反射是动态加载 静态加载举个例子 Cat父类Animal mao是Cat类独有方法 Animal anew Cat(); a.mao();//编译看左边 //左边类型为Animal&#xff08;会加载Animal类&#xff0c;编译时进行加载叫静态加载&#xff09; //然后加载…

OpenShift 容器平台企业版 OCP 4.11.9 部署(基于KVM,CentOS)

参考&#xff1a; 阿里云上Openshift-4.10.5搭建 OpenShift4.8在oVirt下的自动化安装 红帽OpenShift安装部署-阿里云帮助中心 安装配置操作节点&#xff08;Operator&#xff09;&#xff0c;并获取OCP离线安装文件 OCP安装定制文件准备_frank0521的博客-CSDN博客 第 23 章…

【Java数据结构与算法】第二十一章 元组

【Java数据结构与算法】第二十一章 元组 文章目录【Java数据结构与算法】第二十一章 元组1.概念2.自定义元组3.第三方Jar包1.概念 元组&#xff08;Tuple&#xff09;是一种数据结构&#xff0c;可以存放多个元素&#xff0c;每个元素的数据类型可以不同。用List与Tuple类比&a…

深入了解Netty,这一篇就够了

一、Netty简介 Netty是由JBOSS提供的一个java开源框架&#xff0c;现为 Github上的独立项目。Netty提供异步的、事件驱动的网络应用程序框架和工具&#xff0c;用以快速开发高性能、高可靠性的网络服务器和客户端程序。 也就是说&#xff0c;Netty 是一个基于NIO的客户、服务器…

微分方程(人口预测+传染病模型)

一、定义 微分方程&#xff1a;含导数或微分的方程 微分方程的阶数&#xff1a;所含导数或微分的最高阶数&#xff0c;如y’’’2y’’-2x0是三阶微分方程 微分方程的解&#xff1a;使得微分方程成立的函数 例如y’-2x0的解可以为x或者x1 微分方程的通解和特解&#xff1a;特…

【pat】分而治之【图】

分而治之&#xff0c;各个击破是兵家常用的策略之一。在战争中&#xff0c;我们希望首先攻下敌方的部分城市&#xff0c;使其剩余的城市变成孤立无援&#xff0c;然后再分头各个击破。为此参谋部提供了若干打击方案。本题就请你编写程序&#xff0c;判断每个方案的可行性。输入…

MySQL触发器相关知识

1、什么是触发器 触发器&#xff08;trigger&#xff09;是mysql的数据库对象之一&#xff0c;是一种与表操作有关的数据库对象&#xff0c;当触发器所在表上出现指定事件时&#xff08;这些事件包括insert、update、delete三种&#xff09;&#xff0c;将调用该对象&#xff0…

2023年安装Flutter开发环境_在C盘空间占用空间

2023年安装Flutter开发环境&#xff0c;C盘空间还剩多少&#xff1f; 1&#xff1a;Flutter开发对磁盘空间的要求 2&#xff1a;其余日常辅助软件安装D盘&#xff08;占用8GB&#xff09; 3&#xff1a;消耗时间&#xff08;3天–网络有时会中断&#xff09;–【劝退提示】 安…

Hudi(12):Hudi集成Flink之sql-client方式

目录 0. 相关文章链接 1. 启动sql-client 1.1. 修改flink-conf.yaml配置 1.2. local模式 1.3. yarn-session模式 2. 插入数据 3. 查询数据 4. 更新数据 5. 流式插入 5.1. 创建测试表 5.2. 执行插入 5.3. 查看job 5.4. 查看job 5.5. 查看HDFS目录 5.6. 查询结果 …

行为型模式 - 解释器模式Interpreter

学习而来&#xff0c;代码是自己敲的。也有些自己的理解在里边&#xff0c;有问题希望大家指出。 模式的定义与特点 解释器模式&#xff08;Interperter Pattern&#xff09;&#xff0c;给定一个语言&#xff0c;定义它的文法表示&#xff0c;并定义一个解释器&#xff0c;这个…

智引未来,深兰科技机器人家族首次亮相TechG

12月31日&#xff0c;首届上海国际消费电子技术展(简称TechG)在南京国际博览中心圆满落下帷幕。作为全球消费电子技术领域的顶级行业盛会&#xff0c;本届展会共吸引了来自全球的300余家企业出席&#xff0c;共计逾2万名专业人士到场参观。阿里巴巴、蚂蚁科技、海尔、科大讯飞、…