前言:
使用Tomcat和Sevlet并结合了JDBCtemplate,数据库等知识,做一个简单的登录界面。完成用户在html页面输入账户密码,点击提交按钮,用POST请求方式进行提交。然后Servlet接受到POST请求消息,执行代码并完成数据库操作校验功能。
问题A:
在写完所有代码后(包括HTML页面,LoginServlet,failServlet,successServlet,JDBC工具类,User封装类,UserDao操作数据库类),启动了Tomcat服务器,打开了HTML页面,输入完用户名和密码之后,跳转失败了,页面提示信息 404
尝试A :
首先怀疑的是HTML页面中写的表单提交有问题,
<form action="有问题的地方" method="post">
...... </form>
action属性的值原本写的是 /LoginServet,此时各文件路径是这样的
然后我开始改 action属性的值,从什么地方开始改呢?
首先,我为了确保我写的 LoginServlet 是可以被成功访问到的,就首先直接在 LoginServlet 文件下右键运行服务器并自动访问,结果表示是可以被成功访问到的,访问URL为
http://localhost:8080/TomcatAndServletDemo_LoginPractice/LoginServet 手动访问Servlet的URL
然后,根据上面的URL,来改 action 属性的值。
<form action=" 属性值" method="post">
属性值的改动尝试有:
/TomcatAndServletDemo_LoginPractice/Java Resource/src/Servlet/LoginServet
/TomcatAndServletDemo_LoginPractice/src/Servlet/LoginServet
/TomcatAndServletDemo_LoginPractice/Servlet/LoginServet
/TomcatAndServletDemo_LoginPractice/LoginServet (每次删掉一个粗体的内容)
最后,当试到最后一个,打开HTML页面填写表单,跳转到Servlet页面时,地址栏的URL和手动访问Servlet的URL终于一样了,这就说明,从HTML页面可以成功访问到Servlet页面
http://localhost:8080/TomcatAndServletDemo_LoginPractice/LoginServet 手动访问Servlet的URL
http://localhost:8080/TomcatAndServletDemo_LoginPractice/LoginServet 这个是跳转打开Servlet的URL
完全一样
小结A:
经过验证,如果 HTML页面 和 Servlet文件 的相对位置关系 如上图所示,那么在 HTML页面 写表单提交的时候,
action 属性値应该 写 1级路径(Web项目名)/ 3级路径 ( Servlet名 )
至于为什么中间的 2级路径 src那些不用写,猜测是 Tomcat 自己会默认忽视src目录和其目录下的包,直接找到指定的 Servlet
结果A:(问题B )
虽然已成功解决从HTML页面成功跳转访问到Servlet,但依旧没有完成我们想要的功能
经过多次尝试发现我在 failServlet 和 successServlet 中简化代码时
在 doPost() 方法中 错把 this.doGet( request, response )写作 this.doPost( request, response )
所以才一直调用 doPost()方法
解决B(问题C) :
改过来后解决了问题,但是发现,即使我输入正确的用户名和密码,也会跳转到 failServlet 页面,这又是为什么呢?
尝试C:
于是我检查了 failServlet 和 successServlet 的代码,并在关键地方都加入了 System.out.println( )
用来从控制台观察出错的地方
发现代码运行到 UserDao 类的时候出来异常
Incorrect result size: expected 1, actual 0异常
解决C:
1. 查CSDN后,https://blog.csdn.net/qq_40194399/article/details/84168598,
说是 使用queryForObject做数据库查询时出现如下异常
queryForObject有且只能查询一条数据,如果数据库中没有这条数据或者数据库中这条数据有相同的那 么会抛出此异常。
解决办法:
可以使用Spring中的query方法,或者使用try-catch捕获异常(在做登录时可以使用)
2. 经过对比,原本我的 UserDao类 就有 try catch 语句,所以CSDN提供的解决方案并不适合我,而且如果 使用query方法则不符合我要获取封装好的对象的原意。
但它倒是提醒了我,原来我的 sql 语句写错了,
"select * from user where username = ? and password = ? " 写成了
"select * from user where id = ? and password = ? "
3. 改完 sql 语句后,终于解决了!!!!
总结:
1,在HTML页面写表单提交的时候,应该注意 action 属性值的写法,路径应该怎么写,要非常注意
2,在写Servlet方法的时候,为了简化代码,代码复用,doGet()或者 doPost()方法中的
this.doPost(request , response )要写对,别写成一直调用自己了
3,解决Incorrect result size: expected 1, actual 0异常,应该注意 sql 语句的正确,各项值要对应好,
id 对 id ,username 对 username ,password 对 password