我有一个使用Wildfly 10作为服务器的JSF 2.2应用程序。
用户已登录并添加到在线用户列表中:
HttpSession session = request.getSession();
funcionario.setSessao(session);
funcionario.setIp(request.getRemoteAddr());
session.setAttribute("usuarioLogado", funcionario);
我从在线用户列表中删除了退出的用户。
public class ActiveUserListener implements HttpSessionAttributeListene {
@Inject
Fixo fixo;
@Override
public void attributeAdded(HttpSessionBindingEvent event) {
if (event.getValue() instanceof Funcionario) {
fixo.getLogins().add((Funcionario) event.getValue());
}
}
@Override
public void attributeRemoved(HttpSessionBindingEvent event) {
if (event.getValue() instanceof Funcionario) {
if (event.getValue() instanceof Funcionario) {
fixo.getLogins().remove((Funcionario) event.getValue());
}
}
}
@Override
public void attributeReplaced(HttpSessionBindingEvent event) {
if (event.getValue() instanceof Funcionario) {
fixo.getLogins().add((Funcionario) event.getValue());
}
} }
public class SessionCounter implements HttpSessionListener {
@Inject
Fixo fixo;
public void sessionDestroyed(HttpSessionEvent se) {
if (se.getSession().getAttribute("usuarioLogado") != null) {
Funcionario f = fixo.getLogins().stream().filter(o -> o.getSessaoId().equals(se.getSession().getId()))
.findAny().get();
fixo.getLogins().remove(f);
}
}
@Override
public void sessionCreated(HttpSessionEvent arg0) {
// TODO Auto-generated method stub
}}´
当我尝试打印一些用户/会话详细信息时:
public void usuConsole() {
System.out.println("nº " + fixo.getLogins().size());
for (Funcionario f : fixo.getLogins()) {
System.out.println(f.getMatricula());
if (f.getSessao() != null) {
System.out.println(f.getSessao().getId());
System.out.println(f.getSessao().getCreationTime());
System.out.println(f.getSessao().getLastAccessedTime());
}
}
}
在某些时候打印一些行:
会话无效R5EB6hKAzanm50PSRYqxcv361UMD6nGjZWxJVc5P
如果很少有用户登录就可以,但是在很多用户登录和注销后,出现错误
答案 0 :(得分:1)
选项1:
尝试在Fixo类中为您的登录集合使用java.util.concurrent.CopyOnWriteArrayList实现。
选项2
手动同步登录集合:
a)使收集结束(使用对象进行同步时的良好做法。
b)在事件监听器方法中同步登录集合插入和删除:
synchronized(fixo.getLogins()){
fixo.getLogins().remove((Funcionario) event.getValue());
}
synchronized(fixo.getLogins()){
fixo.getLogins().add((Funcionario) event.getValue());
}
c)同步每个循环:
synchronized(fixo.getLogins()){
for (Funcionario f : fixo.getLogins()) {
System.out.println(f.getMatricula());
if (f.getSessao() != null) {
System.out.println(f.getSessao().getId());
System.out.println(f.getSessao().getCreationTime());
System.out.println(f.getSessao().getLastAccessedTime());
}
}
}
现在,在任何这些同步块中,任何对登录集合的公共方法调用都将被阻止,直到应用程序离开块。 这意味着当for循环运行时,事件侦听器方法中的集合的任何修改都将保持,直到循环结束。 根据监听器规范,在sessionDestroyed方法完成处理之前,会话不会失效。
至少是理论,尝试一下,让我们看看它是怎么回事。