gpt4 book ai didi

java - 使用同一浏览器的 2 个以上选项卡时,我的 Web 应用程序中 Servlet 中 Session 属性的范围和生命周期让我感到困惑

转载 作者:塔克拉玛干 更新时间:2023-11-01 22:16:36 26 4
gpt4 key购买 nike

我在做一个简单的网络项目(你可以看下面的代码)。据我所知, session 属性与一个 session 有关。当我打开同一浏览器的两个选项卡并运行键入 URL 时,只创建了一个 session ID,但同一 session 属性的两个不同对象正在运行(即我不想同时运行两个测验。但是,当我更改其中一个选项卡中的问题时,它不会影响另一个选项卡的 session 属性)。你能解释一下为什么会这样吗?如何更改我的代码以使 session 变量共享,以便当我更改其中一个选项卡中的 session 属性之一时,我希望其他选项卡的 session 变量受到影响?

/*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package com.quizServlet;

import QuizApp.Quiz;
import java.io.IOException;
import java.io.PrintWriter;
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;

/**
*
* @author Mati
*/
@WebServlet(name = "QuizServlet", urlPatterns = {"/Quiz"})
public class QuizServlet extends HttpServlet {

protected void processRequest(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
try {
} catch (Exception ex) {
out.write("<font style='color:red'><b>" + ex.getMessage() + "</b></font>");
} finally {
out.close();
}
}

@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {

response.setContentType("text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
try {
if (request.getSession().getAttribute("QuizzObject") == null) {
Quiz quiz = new Quiz();
quiz.addQuestion(new int[]{1, 2, 3, 4});
quiz.addQuestion(new int[]{1, 1, 2, 3, 5, 8});
quiz.addQuestion(new int[]{0, 5, 10, 15, 20, 25});
request.getSession().setAttribute("QuizzObject", quiz);
}
if (request.getSession().getAttribute("questionsLeft") == null) {
request.getSession().setAttribute("questionsLeft", true);
}
Quiz qq = (Quiz) request.getSession().getAttribute("QuizzObject");
qq.reset();

StringBuilder SB = new StringBuilder();

SB.append("<form name='myform' method='post'>");
SB.append("<h3>Have fun with NumberQuiz!</h3>");
SB.append("<p><input type='submit' name='btnNext' value='Start quiz' /></p>");
SB.append("</form>");
out.print(SB.toString());

} catch (Exception ex) {
out.write("<font style='color:red'><b>" + ex.getMessage() + "</b></font>");
} finally {
out.close();
}
}

@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
PrintWriter out = response.getWriter();

try {

StringBuilder SB = new StringBuilder();
String msg="";
SB.append("<html><head></head><body>");
Quiz qq = (Quiz) request.getSession().getAttribute("QuizzObject");
SB.append(request.getSession().getId());
boolean questionsLeft = (Boolean) request.getSession().getAttribute("questionsLeft");
if (questionsLeft) {
qq.addAttempts();
if (request.getParameter("txtAnswer") != null) {
if (qq.isCorrect(Integer.parseInt(request.getParameter("txtAnswer")))) {
qq.scoreAnswer();
} else {
msg="<p><font style='color:red'>Wrong Answer .. Try Again</font></p>";
}
}
if (qq.getCurrentQuestion() == null) {
request.getSession().setAttribute("questionsLeft", false);
SB.append("Congratulations, you have completed the quiz!");
SB.append("<br>Your final score is:" + qq.getScore());
SB.append("<br>Total attempts:" + qq.getAttempt());
qq.reset();
request.getSession().setAttribute("questionsLeft",null);
} else {
SB.append("<form name='myform' method='post'>");
//SB.append("<h3>Have fun with NumberQuiz!</h3>");
SB.append("<p>Your current score is " + qq.getScore() + ".</p>");
SB.append("<p>Guess the next number in the sequence!</p>");
SB.append("<p>" + qq.getCurrentQuestion().toString().replaceAll("\\?", "<font style='color:red'><b>?</b></font>") + "</p>");
SB.append("<p>Your answer:<input type='text' id='txtAnswer' name='txtAnswer' value='' /></p>");
SB.append("<p><input type='submit' name='btnNext' value='Next' onclick='return validate()' />");
SB.append("<input type='Reset' name='btnStart' value='Restart!' onclick=\"document.location.href='/QuizzWeb/Quiz';return false;\" /></p>");
SB.append(msg);
SB.append("</form>");
SB.append("<script type='text/javascript'>function validate(){if(document.getElementById('txtAnswer').value==''){alert('You should write an answer');return false;}return true;}</script>");
}
SB.append("</body></html>");
out.print(SB.toString());
}

} catch (Exception ex) {
out.print("<font style='color:red'><b>" + ex.getMessage() + "</b></font>");
} finally {
out.close();
}

}

@Override
public String getServletInfo() {
return "Short description";
}
}

最佳答案

我想你可能对一些概念有点混淆,希望这个解释能理解并帮助你理清头绪。

session 存在于您的应用程序服务器上。创建后,它会通过使用 cookie(通常名为 JSESSIONID)与您的浏览器进行通信。当您的浏览器将该 cookie 作为请求的一部分提供给网络服务器时,服务器可以检索已经附加到该 session 的 session 和关联对象(应该是可序列化的,请参阅其他 SO 问题)(前提是该 session 尚未过期).

由于这些 session 变量仅存在于服务器上,因此服务器使用它们来构建给客户端的响应。但是为了得到回应,您的客户需要提出请求。您提出了一个请求并更改了第一个选项卡的状态,但是因为第二个选项卡没有发出自己的请求,所以它的状态没有更新。 (因为这些选项卡在同一个浏览器中,它们共享一个 session cookie,并检索同一个 session 来满足它们的请求)。随着更多的构建,您可以利用一些客户端技术(例如 AJAX)定期发出有关 session 状态的小请求并刷新浏览器窗口的显示。 (您可以通过让它们调用不同的资源或请求的不同接受类型来区分此类请求)。

现在您的代码设计...我没有深入研究它,但您可能想要更多地完成您的流程。似乎 GET 总是会重置您的测验并且帖子会继续吗? (这对我来说感觉有点奇怪,但我无法解释为什么......我建议阅读 REST 以及由此驱动的设计。JAX-RS 和 Jersey 非常棒:))。

编辑:这是一个更简单的 servlet,您可以用来玩弄它。将其投入 war ,并打开 2 个选项卡,一个仅针对 servlet 本身,另一个附加查询字符串 ?checkOnly=true。尝试独立刷新每个选项卡,看看计数会发生什么。

package test.servlets;

import java.io.IOException;
import java.io.PrintWriter;
import java.util.concurrent.atomic.AtomicInteger;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;

/**
* Counting servlet counts the number of requests to it.
* @author Charlie Huggard-Lee
*/
@SuppressWarnings("nls")
public class CountingServlet extends HttpServlet {

/**
* The serialVersionUID.
*/
private static final long serialVersionUID = 4279853716717632192L;

/**
* {@inheritDoc}
*/
@Override
protected void doGet(final HttpServletRequest req, final HttpServletResponse resp) throws IOException {
final HttpSession session = req.getSession();
AtomicInteger counter = (AtomicInteger) session.getAttribute("Count");
if (counter == null) {
counter = new AtomicInteger();
session.setAttribute("Count", counter);
}

final boolean checkOnly = Boolean.parseBoolean(req.getParameter("checkOnly"));

final int thisCount;
if (checkOnly) {
thisCount = counter.get();
} else {
thisCount = counter.getAndIncrement() + 1;
}

resp.setStatus(200);
resp.setHeader("Content-Type", "text/plain"); //$NON-NLS-1$ //$NON-NLS-2$
resp.setCharacterEncoding("UTF-8"); //$NON-NLS-1$
final PrintWriter writer = resp.getWriter();

if (session.isNew()) {
writer.append("Hey new user!\n");
} else {
writer.append("Welcome Back!\n");
}
writer.append("Session ID: ");
writer.append(session.getId());
writer.append("\n");
if (checkOnly) {
writer.append("(checking) ");
}
writer.append("Count: ");
writer.append(Integer.toString(thisCount));
}
}

关于java - 使用同一浏览器的 2 个以上选项卡时,我的 Web 应用程序中 Servlet 中 Session 属性的范围和生命周期让我感到困惑,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6003148/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com