java如何从servlet向所有登录用户发送数据
我的要求是管理员应该向所有登录的用户发送数据。我有一个servlet第一个用户发送登录HTTP req并从servlet获得登录成功响应,现在该用户的会话已启动。类似地,其他用户也会登录。现在,admin也是一个用户,可以登录并看到两个用户user1和user2登录。管理员应向所有登录用户发送数据。Servlet应该将数据推送到该用户,而无需用户HTTP请求。请注意,在会话期间,用户不会发送任何其他HTTP请求
我没有使用comet,而是尝试存储每个登录用户的响应obj,后来使用该obj尝试将数据发回给用户,但没有成功
请检查以下代码:LoginServlet。java
public class LoginServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
ServletContext sc;
protected ArrayList<HttpServletResponse> connections = new ArrayList<HttpServletResponse>();
public LoginServlet() {
super();
}
public void init() throws ServletException {
super.init();
sc = getServletContext();
}
protected void doGet(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
}
protected void doPost(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
synchronized (connections) {
connections.add(response);
}
ServletContext sc = getServletContext();
Integer x = (Integer) sc.getAttribute("hit");
if (x == null)
x = 1;
else
x++;
sc.setAttribute("hit", x);
String username = request.getParameter("username");
String password = request.getParameter("password");
HttpSession session = request.getSession(true);
session.setAttribute("username", username);
session.setAttribute("password", password);
response.setContentType("text/html");
ArrayList<LoggedInUser> loggedInUsers = (ArrayList<LoggedInUser>) sc
.getAttribute("users");
if (loggedInUsers == null) {
System.out.println("loggedInUsers creates");
loggedInUsers = new ArrayList<LoggedInUser>();
}
loggedInUsers.add(new LoggedInUser(username, password));
sc.setAttribute("users", loggedInUsers);
PrintWriter out = response.getWriter();
System.out.println("loggedInUsers ==>" + loggedInUsers.size());
if (username != null && username.equals("admin")) {
out.println("Admin Login Success");
ServletContext scx = getServletContext();
Integer xx = (Integer) scx.getAttribute("hit");
out.println("<H1>Users visited " + xx + "</h1>");
loggedInUsers = (ArrayList<LoggedInUser>) sc.getAttribute("users");
for (int i = 0; i <= loggedInUsers.size() - 1; i++) {
LoggedInUser LoggedInUser = loggedInUsers.get(i);
out.println(LoggedInUser.getmUsername()+ "<br>");
}
for (int i = 0; i < connections.size(); i++) {
try {
LoggedInUser LoggedInUser = loggedInUsers.get(i);
PrintWriter writer = connections.get(i).getWriter();
System.out.println("inside loope " + i + "\n" + writer);
writer.println("MESSAGE FROM ADMIN HERE" + "<br>");
writer.flush();
writer.close();
} catch (IOException e) {
log("IOExeption sending message", e);
}
}
} else {
out.println("Welcome " + username + " Login Success");
}
}
}
用户登录arraylist中存储的resp obj后
synchronized (connections) {
connections.add(response);
}
后来使用这个obj得到了打印作者obj
PrintWriter writer = connections.get(i).getWriter();
问题:
- 如何向所有登录用户发送一条特定消息李>
- 没有彗星,我们能做到这一点吗李>
如果我做错了什么,请纠正代码,并提供其他建议。谢谢
# 1 楼答案
这在技术上是不可能的。HTTP协议根本不允许这样做。浏览器只有在发送请求时才能从服务器获得响应。一旦完成读取响应,它将不再尝试从其套接字读取任何内容。。。直到它发出下一个请求
你可以做的是在所有页面中加入一些javascript,这些页面会定期向服务器发出AJAX请求,询问是否有要显示的管理消息
我想你也可以用COMET来做这件事
当前保留最后一个响应并尝试向其写入更多数据的方法失败,因为它违反了HTTP协议和servlet状态机。当登录servlet完成原始请求时,响应的输出流被刷新到套接字并关闭。它无法重新打开,即使可以,浏览器也不知道如何处理您编写的数据
更新:这方面有一个变化可能会起作用。它涉及WebSocket。。。其中,客户端在请求中发送一些特殊的头,服务器发送101响应,并切换协议;见https://en.wikipedia.org/wiki/WebSocket