Java-根据IP统计访问次数
本文简要介绍如何实现一个IP访问计数器。
为了简单, 使用 JSP 来实现, 但读者需要明白, Java代码可以在到处运行。
示例Demo页面: http://www.cncounter.com/test/counter.jsp
返回JSON: http://www.cncounter.com/test/counter.jsp?format=json
清空本IP的计数: http://www.cncounter.com/test/counter.jsp?action=clear
下面列出实现代码, 文档和说明就在代码中:
解析客户端IP的工具类 IPUtils.java
:
package com.cncounter.util.net;
import javax.servlet.http.HttpServletRequest;
/**
* IP工具类
*/
public class IPUtils {
public static final String SEMICOLON = ";";
// Nginx代理传递的实际客户端 IP-header
public static final String[] HEADERS_TO_TRY = {
"X-Forwarded-For",
"X-REAL-IP",
"Proxy-Client-IP",
"WL-Proxy-Client-IP",
"HTTP_X_FORWARDED_FOR",
"HTTP_X_FORWARDED",
"HTTP_X_CLUSTER_CLIENT_IP",
"HTTP_CLIENT_IP",
"HTTP_FORWARDED_FOR",
"HTTP_FORWARDED",
"HTTP_VIA",
"REMOTE_ADDR",
"REMOTE-HOST"
};
/**
* 获取客户端的IP地址
*/
public static String getClientIp(HttpServletRequest request) {
String clientIp = _getClientIp(request);
if (null != clientIp && !clientIp.trim().isEmpty()) {
return clientIp;
}
return request.getRemoteAddr();
}
private static String _getClientIp(HttpServletRequest request) {
for (String header : HEADERS_TO_TRY) {
String ip = request.getHeader(header);
if (ip != null && ip.length() != 0 && !"unknown".equalsIgnoreCase(ip)) {
return ip;
}
}
return request.getRemoteAddr();
}
}
其中依赖了 FastJSON, FastJSON的信息请参考: https://github.com/alibaba/fastjson/wiki/Quick-Start-CN
改为其他JSON库也可以。但强烈推荐使用简单强大灵活的FastJSON。
JSP文件 counter.jsp
实现代码如下:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isErrorPage="true" %>
<%@ page trimDirectiveWhitespaces="true" %>
<%@ page import="java.util.concurrent.ConcurrentHashMap" %>
<%@ page import="java.util.concurrent.atomic.AtomicInteger" %>
<%@ page import="com.cncounter.util.net.IPUtils" %>
<%@ page import="com.alibaba.fastjson.JSON" %>
<%-- trimDirectiveWhitespaces 的作用是去除多余的空行 --%>
<%!
// 访问计数器Map<IP地址, 次数>
private static ConcurrentHashMap<String, AtomicInteger> visitCounterMap
= new ConcurrentHashMap<String, AtomicInteger>();
// 增加并获取最新的访问次数
private static int incrementCounter(String clientIp) {
//
AtomicInteger visitCounter = visitCounterMap.get(clientIp);
if (null == visitCounter) {
visitCounter = new AtomicInteger();
AtomicInteger oldValue = visitCounterMap.putIfAbsent(clientIp, visitCounter);
if (null != oldValue) {
// 使用 putIfAbsent 时注意: 判断是否有并发导致的原有值。
visitCounter = oldValue;
}
}
// 先增加, 再返回
int count = visitCounter.incrementAndGet();
return count;
}
// 清除某个IP的访问次数
private static int clearCounter(String clientIp) {
visitCounterMap.remove(clientIp);
return 0;
}
//
private static final String CONST_PARAM_NAME_ACTION = "action";
private static final String CONST_ACTION_VALUE_CLEAR = "clear";
//
private static final String CONST_PARAM_NAME_FORMAT = "format";
private static final String CONST_FORMAT_VALUE_JSON = "json";
//
private static final String CONST_ATTR_NAME_CLIENTIP = "clientIp";
private static final String CONST_ATTR_NAME_VISITCOUNT = "visitCount";
%>
<%
// 获取客户端IP地址
String clientIp = IPUtils.getClientIp(request);
Integer visitCount = 0;
if (null != clientIp) {
// 获取访问次数
visitCount = incrementCounter(clientIp);
}
// 如果需要清空数据
String action = request.getParameter(CONST_PARAM_NAME_ACTION);
if (CONST_ACTION_VALUE_CLEAR.equalsIgnoreCase(action)) {
visitCount = clearCounter(clientIp);
}
// 如果需要返回JSON格式的数据
String format = request.getParameter(CONST_PARAM_NAME_FORMAT);
if (CONST_FORMAT_VALUE_JSON.equalsIgnoreCase(format)) {
// 返回JSON
Map<String, Object> result = new HashMap<String, Object>();
result.put(CONST_ATTR_NAME_CLIENTIP, clientIp);
result.put(CONST_ATTR_NAME_VISITCOUNT, visitCount);
%>
<%=JSON.toJSONString(result)%>
<%
return; // 如果返回JSON数据, 则不往下执行
}
%>
<HTML>
<HEAD>
<TITLE>统计页面访问次数</TITLE>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<meta name="viewport"
content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no">
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<style>
body {
word-wrap: break-word;
word-break: break-all;
}
</style>
</HEAD>
<BODY>
<H1>统计页面访问次数</H1>
<table border="1" width="100%">
<thead>
<tr>
<td align="center">
<h3>IP地址</h3>
</td>
<td align="center">
<h3>访问次数</h3>
</td>
</tr>
</thead>
<tbody>
<%
Set<String> keySet = visitCounterMap.keySet();
// 排序?
// 根据值排序?
for (String key : keySet) {
%>
<tr>
<td>
<%=key%>
</td>
<td>
<%=visitCounterMap.getOrDefault(key, new AtomicInteger(0)).intValue()%>
</td>
</tr>
<%
}
%>
</tbody>
</table>
</BODY>
</HTML>
并未进行持久化, 如有需要, 可以定时保存到数据库。
只需要在保存之后清空本地的值即可。
一般情况下, 统计访问次数等功能, 不需要考虑极端情况, 只要不影响系统的性能即可。
然后访问相应的地址:
示例Demo页面: http://www.cncounter.com/test/counter.jsp
返回JSON: http://www.cncounter.com/test/counter.jsp?format=json
清空本IP的计数: http://www.cncounter.com/test/counter.jsp?action=clear
返回的JSON 如下所示:
{"clientIp":"61.50.103.90","visitCount":4}
时间: 2018年2月9日
作者: 铁锚

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
阿里云短信服务接口的c++实现
最近在调用阿里云短信服务接口发现并没有c++的资料,网上查询发现也是零星描述,由于自身项目基础累积了C++的太多,不想掺杂太多语言,就自行实现c++的阿里云短信接口,其难点就在于签名而已,希望能给大家提供参考。 1)我的网页调用接口采用acl_master实现的,acl_master是一个跨平台c/c++库,提供了网络通信库及服务器编程框架,同时提供更多的实用功能库及示例。(下载地址:Github: https://github.com/acl-dev/acl),读者可采用其他库实现。 2)参考阿里云的“HTTP协议及签名”(https://help.aliyun.com/document_detail/56189.html?spm=a2c4g.11186623.6.581.fy2faU)描述文档,先做一些必要的准备: *注册阿里云,有需要的阿里云产品通用代金券的可以点击下面链接进行领取: https://promotion.aliyun.com/ntms/yunparter/invite.html?userCode=pfb80n4a 开通短信服务后,在你账户头像下AccessKey...
-
下一篇
Swift里的protocol--协议
我感觉它和java或go里的接口,差不多一个意思吧。 protocol FullyNamed { var fullName: String { get } } struct Person: FullyNamed { var fullName: String } class Starship: FullyNamed { var prefix: String? var name: String init(name: String, prefix: String? = nil) { self.name = name self.prefix = prefix } var fullName: String { return (prefix != nil ? prefix! + " " : "") + name } } var ncc1701 = Starship(name: "Enterprise", prefix: "USS") print(ncc1701.fullName) let john = Person(fullName: "John Appleseed") print(john.fu...
相关文章
文章评论
共有0条评论来说两句吧...