页面停留时长、日志上报,你们都是怎么玩的?

01 引言

在业务中,我们很多时候需要记录用户的行为。但是并不是所有的行为都会和服务端交互,服务端就显的心余而力不足。

尤其在一些类似支付、拍卖等过程中引起的纠纷。用户或没有什么都没有操作被扣款了,但是因为和服务端没有交互,并不到任何的线索。这时候就需要前端采集用户的行为日志,然后按需上报,以记录用户的页面行为。

有需求,就有市场。类似的框架应运而生如,美团的Logan、神策网络的神策等。为了分析用户的行为喜好,页面的停留时长就变的尤为紧要。

我们就以后端的视角,窥探一下前端技术。

02 难点

用户的行为日志记录相对比较简单,但是页面的停留时长似乎有点难。难点如下:

  • 用户打开监控页面,又打开新的页面算不算离开?
  • 用户将页面最小化,算不算离开?
  • 用户将页面最大化,停留时长要不要重新计算?
  • 用户关闭新开页后,重新回到监控页面,停留时长要不要算?

其实,关于算不算,要不要算是产品问题。从技术角度来看,我们通过什么样的方式或者函数能够监控这些变化呢?

有两种方式可供参考:

  • pagehide | pageshow
  • visibilitychange

03 pagehide | pageshow

当浏览器从展示会话历史中的另一个页面过程中隐藏当前页面时,会向 Window 发送 pagehide 事件。例如,当用户点击浏览器的后退按钮时,在显示前一个页面之前,当前页面会接收到一个 pagehide 事件。

当一条会话历史记录被执行的时候将会触发页面显示 (pageshow) 事件。(这包括了后退/前进按钮操作,同时也会在 onload 事件触发后初始化页面时触发)

3.1 pagehide

pagehide是基于window的事件,简单来说,就是页面被覆盖或者关闭时,触发该事件。

js 复制代码
window.addEventListener('pagehide', function(event) {
    console.info("页面被关闭了");
});

// 另外一种写法
window.onpagehide = (event) =>{
    console.info("页面被关闭了");
}

文档地址:

developer.mozilla.org/zh-CN/docs/...

3.2 pageshow

pageshow 同样是基于window的事件,简单来说,就是页面只要被重新加载,就会触发该事件。

js 复制代码
window.addEventListener('pageshow', function(event) {
    console.info("页面被关闭了");
});

// 另外一种写法
window.onpageshow = (event) =>{
    console.info("页面被关闭了");
}

文档地址:

developer.mozilla.org/zh-CN/docs/...

3.3 实战

页面的开启和关闭,我们都想服务端发送请求,其中发送请求的方法:

navigator.sendBeacon(url, data)

文档地址:

developer.mozilla.org/zh-CN/docs/...

案例:

js 复制代码
$(function (){
    window.addEventListener('pagehide', function(event) {
        const formData = new FormData();
        formData.append('data', '窗口关闭了!');
        navigator.sendBeacon("/file/windowClose", formData);
    });

    window.onpageshow = (event) =>{
        const formData = new FormData();
        formData.append('data', '窗口打开了!');
        navigator.sendBeacon("/file/windowOpen", formData);
    }
});
java 复制代码
@RequestMapping("windowClose")
public void windowClose(String data) {
    System.out.println("window is closed......" + data);
}

@RequestMapping("windowOpen")
public void windowOpen(String data) {
    System.out.println("window is opened......" + data);
}

结果:

3.5 浏览器兼容

04 visibilitychange

当其选项卡的内容变得可见或被隐藏时,会在 document 上触发 visibilitychange 事件。

文档地址:

developer.mozilla.org/zh-CN/docs/...

4.1 案例

JS代码

js 复制代码
document.addEventListener("visibilitychange", function logData() {
    if (document.visibilityState === "hidden") {
        const formData = new FormData();
        formData.append('data', '窗口关闭了!');
        navigator.sendBeacon("/file/windowClose", formData);
    }
});

document.onvisibilitychange = (event) => {
    if (document.visibilityState === "visible") {
        const formData = new FormData();
        formData.append('data', '窗口打开了!');
        navigator.sendBeacon("/file/windowOpen", formData);
    }
}

也可以使用Jquery绑定事件:

js 复制代码
$(document).on('visibilitychange', function() {
    if (document.visibilityState === 'hidden') {
        // 关闭
    }else if (document.visibilityState === 'visible'){
        // 显示
    }
});

结果:

4.2 浏览器兼容性

05 小结

两种方式都存在兼容性问题,移动端适用不适用,怎么也不清楚,总之应用在PC端,案例并没有问题。这两种方式就能监控窗口的显示和关闭,用来统计用户的浏览时长,同时也可以用户类似Websocket等客户端的关闭。收藏以后备用。

相关推荐
不会敲代码131 分钟前
TCP/IP 与前端性能:从数据包到首次渲染的底层逻辑
前端·tcp/ip
kyriewen43 分钟前
奥特曼借GPT-5.5干杯,而你的Copilot正按Token收钱
前端·github·openai
callJJ43 分钟前
Spring Data Redis 两种编程模型详解:同步 vs 响应式
java·spring boot·redis·python·spring
AC赳赳老秦1 小时前
投标合规提效:用 OpenClaw 实现标书 / 合同自动审核、关键词校验、格式优化,降低废标风险
开发语言·前端·python·eclipse·emacs·deepseek·openclaw
kyriewen1 小时前
代码写成一锅粥?3个设计模式让你的项目“起死回生”
前端·javascript·设计模式
不会敲代码11 小时前
从零搭建 AI 日记助手:用 Milvus 向量数据库实现语义搜索
javascript·openai
千寻girling1 小时前
《 Git 详细教程 》
前端·后端·面试
wbs_scy1 小时前
Linux线程同步与互斥(三):线程同步深度解析之POSIX 信号量与环形队列生产者消费者模型,从原理到源码彻底吃透
java·开发语言
threelab2 小时前
Three.js UV 图像变换效果 | 三维可视化 / AI 提示词
javascript·人工智能·uv
之歆3 小时前
DAY08_CSS浮动与行内块布局实战指南(下)
前端·css