范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

Java统计用户在线人数,这样做才优雅

  统计用户在线人数
  在统计用户在人数的时候,我们用到了监听器,监听器大致分为以下三种: ServletRequestListener :用于监听请求的监听接口HttpSessionListener :用于监听会话的监听接口ServletContextListener :用于监听应用的回话接口错误的统计办法监听Request域
  这种统计办法是错误的认为每次刷新页面后进行进行一次的 count++ 运算import javax.servlet.*; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionAttributeListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import javax.servlet.http.HttpSessionBindingEvent;  @WebListener() public class MyRequestListener implements ServletRequestListener{     private ServletContext sc;     private Integer count;     @Override     //请求被初始化 Request     public void requestInitialized(ServletRequestEvent sre) {         //获取全局域         sc = sre.getServletContext();         //将count从全局域中获取出来         count = (Integer) sc.getAttribute("count");         System.out.println(count);         count++;         System.out.println(count);         sc.setAttribute("count",count);     } } import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionAttributeListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import javax.servlet.http.HttpSessionBindingEvent;  @WebListener() public class MyServletContextListener implements ServletContextListener{     private ServletContext sc;     @Override     //Application被初始化的时候创建     public void contextInitialized(ServletContextEvent sce) {         Integer count = 0;         //获取全局域         sc = sce.getServletContext();         //将count放入到全局域中         sc.setAttribute("count",count);     } } <%@ page contentType="text/html;charset=UTF-8" language="java" %>         $Titlelt;/title>   </head>   <body>   <center><h1>You are the ${applicationScope.count} customer to visit. </h1></center>   </body> </html>
  这种错误地做法导致的是每刷新一次页面 就会导致count进行累加操作,最终产生错误的在线人数,所以此时想到不应该监听Request域,而应该监听Session域。 监听Session域
  在第二次监听Session域之后,发现每次刷新页面后不改变count但是在启动不同的浏览器后 count++ 会实现,但是,这样做并不是我们要统计的在线人数,所以此种做法错误。由于代码只是将原来写在Request监听器中的代码转移到Session监听器中,所以其他没变的代码将不重复。import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionAttributeListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import javax.servlet.http.HttpSessionBindingEvent;  @WebListener() public class MySessionListener implements HttpSessionListener{      private ServletContext sc;     private Integer count;      @Override     //当对话产生时激活此方法     public void sessionCreated(HttpSessionEvent se) {         sc = se.getSession().getServletContext();         count = (Integer) sc.getAttribute("count");         count++;         sc.setAttribute("count",count);     } }
  这时我们发现对于在线人数的统计,不是网页访问的次数,也不是浏览器打开的个数,对需求的理解的错误理解。所以正确的做法是统计其IP的数量,这样的话,不管你在一台电脑上开启多少客户端,都会只有一个。 正确的统计方法
  统计其IP的数量,将IP的数量作为当前的在线人数,那么如何统计IP的数量呢?这样将会导出以下问题: 如何获取用户的IP? IP将如何存储? 如何判断IP之前已经存在?
  现在来解决这些问题: 只能从请求中获取 通过2、3问题,我们想到了集合(List),因为集合不仅可以存储任何字符串,还可以通过遍历来判断之前是否有重复的IP出现。
  到了这里又冒出来一个问题集合(List)放到哪个域里呢?
  ServletContext域 import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.HttpSessionAttributeListener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import javax.servlet.http.HttpSessionBindingEvent; import java.util.ArrayList; import java.util.List;  @WebListener() public class MyServletContextListener implements ServletContextListener{     private ServletContext sc;      @Override     //Application被初始化的时候创建     public void contextInitialized(ServletContextEvent sce) {         //创建一个链表来存储IP         List<String> ips = new ArrayList<>();         sc = sce.getServletContext();         //将创建好的链表对象,放到Application域中         sc.setAttribute("ips",ips);     } }
  由于IP只能在Request域中获取,所以遍历判断在Request域中进行。 import javax.servlet.*; import javax.servlet.annotation.WebListener; import javax.servlet.http.*; import java.util.List;  @WebListener() public class MyRequestListener implements ServletRequestListener{      private HttpServletRequest sr;     private String clientIp;     private ServletContext sc;     private List<String> ips;     private HttpSession session;      @Override     //请求被初始化 Request     public void requestInitialized(ServletRequestEvent sre) {         //从请求域中获取IP         sr = (HttpServletRequest) sre.getServletRequest();         clientIp = sr.getRemoteAddr();         session = sr.getSession();         session.setAttribute("clientIp",clientIp);          //测试         // System.out.println("clientIp = "+ clientIp);         //获取Application域中的List         sc = sre.getServletContext();         ips = (List<String>) sc.getAttribute("ips");         //遍历ips         for (String ip :                 ips) {             if (clientIp.equals(ip))                 return;         }         ips.add(clientIp);         sc.setAttribute("ips",ips);     } }
  因为要统计在线人数,所以要设置退出按钮,点击退出按钮之后,因为要从List域中移除,所以使用Session域监听器来判断session回话的关闭 import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.*; import java.util.List;  @WebListener() public class MySessionListener implements HttpSessionListener{      private ServletContext sc;     private List<String> ips;     private HttpSession session;     private Object clientIp;      @Override     public void sessionDestroyed(HttpSessionEvent se) {         sc = se.getSession().getServletContext();         ips = (List<String>) sc.getAttribute("ips");         session = se.getSession();         clientIp = session.getAttribute("clientIp");         //删除ip,如何获取IP,但是不可以从session获取到IP         //因为Session获取不到Request         //一个Session包含多个Request         //一个Request只对应一个Session 所以获取不到,这时只能先从Request域中获取到的ips,放置到Session域         //然后从Session 域中读取         ips.remove(clientIp);         // session一失效就马上将此IP从链表中移除是错误的         //应该看此IP是否有另外的回话存在,如果有的话不能删除     } }
  此处代码是页面点击关闭后,激活的退出方法 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;  @WebServlet(name = "LogoutServlet",urlPatterns = "/logoutServlet") public class LogoutServlet extends HttpServlet {      private HttpSession session;      protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {         //从域中获取一个session,设置为false 如果域中存在一个session,则直接获取,如果不存在,则返回一个空的session         session = request.getSession(false);         if (session != null){             //使session失效             session.invalidate();             //失效后,需要进行的操作,List链表中需要减去,用到了Session域监听器         }     }      protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {         doPost(request,response);     } }
  在jsp页面进行读取的时候,因为ips是以List链表的形式存在的,所以要想判断当前在线人数,所以必须要判断链表的长度,所以是 applicationScope.ips.size() <%@ page contentType="text/html;charset=UTF-8" language="java" %> <html>   <head>     <title>$Titlelt;/title>   </head>   <body>   <center><h1>You are the ${applicationScope.ips.size()} customer to visit. </h1>
  <h3>       安全退出     </h3>   </center>   </body> </html>
  好了?,这时候,程序写完了,如何判断呢?
  此时,我们的程序是部署在本地的Tomcat上的,对于本台电脑,只有一个IP,如何实现多个IP呢?其实啊我们的电脑可以有三个IP,在访问服务器的时候,服务器的IP多写几个,相当于本机的IP多出来几个。是哪三个IP呢?
  1、默认clientIp : 0:0:0:0:0:0:0:1
  2、127.0.0.1
  这时大家可能会问127.0.0.1和localhost有什么区别呢,其实在这里要区分三个概念:
  localhost、127.0.0.1 和 本机IP之间的区别: localhost等于127.0.0.1,不过localhost是域名,127.0.0.1是IP地址。 localhost和127.0.0.1不需要联网,都是本机访问。 本机IP需要联网,本机IP是本机或外部访问, 本机 IP 就是本机对外放开访问的IP地址,这个网址就 是与物理网卡绑定的IP地址。
  3、IPv4地址:192.168.1.110
  这样就很完美的实现了本地三个IP的测试。
  写到这里,似乎已经可以简单的测试当前在线人数,也许仔细的人会发现在Session域被销毁的方法中的注释中发现一些猫腻。大家可以仔细想想,如果客户端用不同的浏览器,相同的IP去访问呢?点击退出后,会不会出现错误情况呢?答案是会的。演示结果如下图
  最完美的代码
  所以在点击退出登录的按钮之后,不可以直接将IP移除,要判断有没有另外的回话存在,如果有另外的回话存在,此IP是不可以删掉的,问题由此变的复杂了,因为还要统计此IP所发出的会话有多少。
  整体思路:
  在全局域中,将不是直接将iP存放在List的链表中,而是以一个Map的形式存在,Map的键为String类型,Key为List类型,List中存放的是当前IP所激发的会话对象,这样就可以统计,一个IP触发的sessions有多少个。
  通过调用Map的get方法,将当前IP最为参数,将可以获取到他所激发的会话集合。但是,此集合可能为空,因为有可能当前IP一次也没有访问此页面,所以在List为空的时候好要创建一个ArrayList来存放sessions,然后将变化后的List重新写回到Map,再将变化后的Map写回到全局域中 。这样创建过程基本完成。
  然后考虑销毁过程,IP还需方法放到Session域中,当session被销毁的时候,应该把当前Session从List 中删除,但是Map中此sessions对应的IP可是不能直接删,要判断List中的sessions的个数(Entry对象),个数为1的时候才可以删除,不然就不可以删除。
  所以,要将当前IP通过Request域存放到当前Session域中,
  然后,要考虑的问题是,每次刷新页面后sessions的个数会增加,这是错误的,原因是什么?
  答案是,因为在存放sessions的时候,创建数组直接进行的添加,这样的话,每次一刷新页面,就会导致sessions的添加,所以在此之前应该判断,sessions中是否有此session,有的话直接跳出。
  这样添加就没问题了 Servlet域中添加Map
  在Map中,需要使用键值对的方式,Key为IP,Value为List,那么List中存放什么呢?存放的是此IP发出的所有回话的HttpSession的对象,所以List的泛型是HttpSession。
  请求,在请求中,因为将当前Session 对象存放到List中, List在Map中,Map在全局域中,所以首先得从全局域获取到Map,然后,从Map中获取由当前IP所发出的所有Session的组成的List,判断当前的List是否为NULL,若为NULL,则创建List,否则,将当前SessioncurrentSession放入List中。 import javax.servlet.*; import javax.servlet.annotation.WebListener; import javax.servlet.http.*; import java.util.ArrayList; import java.util.List; import java.util.Map;  @WebListener() public class MyRequestListener implements ServletRequestListener{      private HttpServletRequest sr;     private String clientIp;     private ServletContext sc;     private List<String> ips;     private HttpSession currentSession;     private Map<String,List<HttpSession>> map;     private List<HttpSession> sessions;       @Override     //请求被初始化 Request     public void requestInitialized(ServletRequestEvent sre) {         //从请求域中获取IP         sr = (HttpServletRequest) sre.getServletRequest();         clientIp = sr.getRemoteAddr();         currentSession  = sr.getSession();         //将当前Session 对象存放到List中, List在Map中,Map在全局域中,         sc = sre.getServletContext();         map = (Map<String, List<HttpSession>>) sc.getAttribute("map");         //从Map中获取由当前IP所发出的所有Session的组成的List         sessions = map.get(clientIp);         //判断当前的List是否为NULL,若为NULL,则创建List,否则,将当前Session放入List         if (sessions == null){             sessions = new ArrayList<>();         } //        遍历List的session 对象,若有则不添加,若没有则添加         for (HttpSession session :                 sessions) {             if (session == currentSession)                 return;         }         sessions.add(currentSession);           //将变化过的List重新写回到Map         map.put(clientIp,sessions);         //再将变化的Map写回到全局域中         sc.setAttribute("map",map);          //将当前IP放入到当前Session         currentSession.setAttribute("clientIp",clientIp);     }  } ServletContext
  这里将不使用ips了,所以将其删除 import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.*; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;  @WebListener() public class MyServletContextListener implements ServletContextListener{     private ServletContext sc;     @Override     //Application被初始化的时候创建     public void contextInitialized(ServletContextEvent sce) {         //创建一个Map,key为IP,value为该IP上所发出的会话的对象         Map<String,List<HttpSession>> map = new HashMap<>();         sc = sce.getServletContext();         //将map放到全局域中         sc.setAttribute("map",map);     } } Session监听器
  接下来剖析Session的删除工作,获取当前Session对象,这里有之前传递过来的IP,在进行删除操作的时候,要注意此处,删除的是List中的sessions,删除之后,还要判断其IP的是否要删除,如果List中没有该元素,则说明当前IP所发出的会话全部关闭,就可以从map中将当前IP对应的Entry对象删除,否则,当前IP所发出的会话任存在,那么使用put方法将变化过的List写回到map。 import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.annotation.WebListener; import javax.servlet.http.*; import java.util.List; import java.util.Map;  @WebListener() public class MySessionListener implements HttpSessionListener{      private ServletContext sc;     private List<String> ips;     private HttpSession currentSession;     private String clientIp;     private Map<String,List<HttpSession>> map;     private List<HttpSession> sessions;      @Override     public void sessionDestroyed(HttpSessionEvent se) {         sc = se.getSession().getServletContext();          currentSession = se.getSession();         clientIp = (String) currentSession.getAttribute("clientIp");         map = (Map<String, List<HttpSession>>) sc.getAttribute("map");         //从Map中获取List         sessions = map.get(clientIp);         //从List中删除当前Session对象         sessions.remove(currentSession);         //如果List中没有该元素,则说明当前IP所发出的会话全部关闭,就可以从map中         //将当前IP对应的Entry对象删除         //若List中仍有元素,当前IP所发出的会话任存在,那么将变化过的List写回到map          if (sessions.size() == 0){              map.remove(clientIp);          }else {              map.put(clientIp,sessions);          }          sc.setAttribute("map",map);     } }
  因为处理的退出的页面 /logoutServlet 不需要做任何不同的处理,所以这里将不再重复。
  因为在jsp用到了JSP标准库,所以到导两个包。 <%@ page contentType="text/html;charset=UTF-8" language="java" %> <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %> <html>   <head>     <title>$Titlelt;/title>   </head>   <body>   <center><h1>You are the ${applicationScope.map.size()} customer to visit. </h1>
  <h3>       安全退出
  </h3>     <h2>       <c:forEach items="${map}" var="entry">         ${entry.key }=${entry.value.size()}
  </c:forEach>     </h2>   </center>   </body> </html>
  最后 测试成功,这就是一个完美的统计当前用户的在线人数。
  来源:https://mp.weixin.qq.com/s/gM-lyh_9FeFKRRAJlPY83g</pre><hr>
<div class="middle_list"><a href="at_330028.html">吃他汀的时候,还能不能吃柚子?听听药师怎么说</a>有一位服用阿托伐他汀的朋友问华子,他听说在服用他汀类药物的时候,就不能吃柚子了,这是为什么呢?华子说,柚子主要指的是葡萄柚(也称西柚),因为其中含有较多的柚苷和呋喃香豆素,有可能干<a href="at_330029.html">白开水放凉后等于慢性毒药?医生告诉你真相,不懂得赶紧来看看</a>白开水放凉后等于慢性毒药?医生告诉你真相,不懂得赶紧来看看水是生命之源,每个人每天都需要喝水,但是喝水也需要讲究方法方式,甚至现在网络上出现这样一种说法,开水放凉以后会变成慢性毒药<a href="at_330030.html">俗语说人过六十,四不想,是哪四不想?过了60岁的人不妨看看</a>由于以前人们的生活条件艰苦,无法解决温饱问题,所以人均寿命非常的低,但现在随着生活条件的改善,如果能活到60岁是一件特别常见的事情。生活当中有一些人60岁可能已经退休,对于退休后的<a href="at_330031.html">冬至前每天喝一碗正气汤,打通上中下三焦,温补正气,强体质</a>大家好,我是米医生霜降之后,寒风骤然登临,昼夜温差增大,冷得愈发彻底了。回想夏天的极端高温,当时我就说过今年妥妥是一个寒冬。壬寅年风邪当道,立冬后更是太阳寒水厥阴风木,风寒湿邪肆虐<a href="at_330032.html">秋初食梨正当时</a>养生之道现代医学认为,梨对高血压心脏病便秘头昏失眠多梦等病症,有良好的辅助治疗作用。同时,对肝炎肝硬化患者来说,更是良好的保健食品。夏末秋初多吃梨适合于气管炎患者,也适合于患有干燥<a href="at_330033.html">长期喝白开水与长期喝茶的人相比,哪个更健康一些?医生说出答案</a>水是生命的源泉,我们的生命活动离不开水的支持,在人体当中60以上都是水分,如果身体当中缺少水分,不仅会影响各器官细胞的正常运转,同时也会对我们的生命带来一定的安全隐患。随着人们的物<a href="at_330034.html">LV最值得买的4款包包(小个子)</a>不管你的身高是高是低,也不管你的肤色是白是黑,更不管你的风格是随性还是小香风,你都能在LV找到一款属于你的包包。我自己身高155,100斤,平时穿衣服比较休闲,偶尔会有点小心思,买<a href="at_330035.html">AddisonRae现身街头,穿黑色运动装健硕有型,手端着咖啡轻松惬意</a>近日,AddisonRae现身洛杉矶的街头,当天,她身穿黑色运动装健硕有型,手端着咖啡轻松惬意。她是一个很开朗的女孩,总是面带着微笑,看起来很有亲和力。艾迪生雷(AddisonRa<a href="at_330036.html">丰台的秋景大片,等您来定格!</a>秋天,无论在什么地方的秋天,总是好的。郁达夫故都的秋(节选)老舍说秋天一定要住北平。天堂是什么样子,我不晓得,但是从我的生活经验去判断,北平之秋便是天堂。名家笔下的秋天总是美得让人<a href="at_330037.html">陕西有个湖泊,被誉为高原明珠,您知道吗?</a>现在国家很多地方都在发展着自己的旅游业,但是在各个地方,他们的发展速度也是各不相同,特别是那些原本发展良好的城市,他们的发展速度更快,但是他们的发展速度却让人有些惊讶,比如有些湖可<a href="at_330038.html">临夏诗影水杉长廊</a>水杉长廊文驮夫摄影丁仲忻不知不觉又来到你的身旁季节的更替给了你独有的韵律霜花入怀更显旖旎长长的走廊里一颦,一笑,一回眸放慢脚步独享那份惬意婉约,优雅,洒脱是你高大挺拔,身在云天里心</div>
<div class="middle_page"><a href="/udt/ls_30888.html"><<<</a><a href="/udt/ls_30887.html"><<</a><a href="/udt/ls_30886.html"><</a><span>-</span><a href="/udt/ls_30884.html">></a><a href="/udt/ls_30883.html">>></a><a href="/udt/ls_30882.html">>>></a></div></div>
<div class="middle_right"><a href="https://www.vkhz.com/think.html"><img src="/static/vkhz.png" width="100%"></a><a href="at_106024.html">是情感慰藉还是混乱麻烦?世界杯决赛前,梅西的家人们都在做什么</a>世界杯决赛前梅西在社交媒体曾经发布了一张家族合影,他为这张合影配字Familia,这张图将几位极少曝光的家庭成员现于台前,也再一次引起了人们对梅西家庭的好奇心,对于梅西而言,他稳定<a href="at_106025.html">英雄无敌3二级魔法横评</a>爱生活爱游戏!欢迎大家一起探讨经典游戏!今天和大家探讨一下英雄无敌3中四系18种二级魔法的实用性。为了避免文章篇幅过长,所以各种魔法只是对其高级状态作简单说明,不同等级的魔耗作用效<a href="at_106026.html">2。母乳喂养的好处</a>1首选母乳喂养婴儿降生后越早吸食母乳越好,越早刺激乳房越好2母乳喂养是有菌喂养乳头上的需氧菌和乳头内的厌氧菌,进入婴儿肠道后组建了人体消化道正常菌群的最初基础3母乳前奶头禁用消毒纸<a href="at_106027.html">每个人都要补充的维D怎么选?怎么吃?</a>花爸说几乎所有妈妈们对宝宝缺钙都特别重视(担心长不高),其实过度焦虑了。今天的两篇文章,一篇说补钙一篇说补维D。VD比钙重要,所以放在前面。下一篇补钙的文章中说了,奶量够的婴幼儿吃<a href="at_106028.html">在日本研究虐儿的安徽女学者</a>日本华侨报总主笔蒋丰读报,是我多年的习惯。日本社会的大事小情,国际世界的新闻趋势,都能通过读报及时获得。不过,日本报纸读久了,一个疑问也在我心里越积越深都说物以稀为贵,可为什么在高<a href="at_106029.html">生完二胎,我每天都在尿裤子</a>原创孕事咳咳咳咳咳。。很多阳了的妈妈,虽然阳康了,但咳嗽还是不止,睡觉咳,白天咳,吃饭也咳。频繁的咳嗽,对有些产后妈妈来说,真的很不友好!前阵子孕事妈就刷到一则新闻,湖南株洲一位3<a href="at_106030.html">孕期得了妊娠糖尿病?别慌,做好这些事,母子平安不受伤</a>(声明本文仅用于科普用途,为了保护患者隐私,以下内容里的相关信息已进行处理)基本信息女,27岁疾病类型妊娠期糖尿病治疗医院重庆医科大学附属第一医院(三甲)治疗方案饮食运动定期监测血<a href="at_106031.html">我第一次看到宝宝大哭时汗毛竖起,她整个嘴角都向右边歪着</a>2020年的最后一个月,三妹妹在全家人的翘首期待中出生了。我躺在产房还没有缓过来,但做母亲的直觉让我总觉得哪里不对劲三妹妹在医院的小睡床里面一直哇哇大哭,旁边几个助产士却挨着脑袋窃<a href="at_106032.html">暴亏42亿!为人民造车的网红车企,为什么卖不动了?</a>2019年的暑期档,国产动画电影哪吒之魔童降世横空出世,夺得2019年的电影票房冠军。这场爆火不仅带动了一系列相关的国产ip产业链,也拯救了一个汽车品牌,那就是合众新能源汽车有限公<a href="at_106033.html">峡谷的野怪,有哪些不为人知的秘密</a>峡谷的野怪多种多样,可一大部分的人只会在意野怪所掉落的经济,不会在意背后的故事,那么,野怪背后都有什么故事呢首先,峡谷的地理位置在日落海和云中漠地的相接之处,就像下图这样其次,王者<a href="at_106034.html">你从没玩过的阴间游戏!Steam另类模拟游戏推荐</a>停尸间助手英文名TheMortuaryAssistantSteam售价80评价特别好评(1,831,90)标签恐怖灵异超现实推荐指数(没中文扣一星)就在我以为冲就完事模拟器可以霸占<div class="middle_right_youlian"></div></div></div>
</body></html>