PHP无法直接获取视频观看时长,必须由前端JavaScript通过video事件定期上报currentTime等数据,PHP仅负责接收校验并存储。

PHP 无法直接获取用户观看视频时长
PHP 是服务端脚本语言,运行在服务器上,不接触浏览器播放器的运行状态。它**完全不知道**用户是否在看视频、看了多久、有没有暂停或拖拽。所谓“PHP 获取观看时长”,实际是前端主动上报、PHP 接收并存储的过程——PHP 只负责「接收」和「存」,不负责「获取」。
必须由前端 JavaScript 主动上报播放进度
真实可行的做法是:在页面中用 video 元素 + timeupdate 事件(或 requestAnimationFrame 定时采样),定期把当前 currentTime 和视频总时长 duration 发给 PHP 接口。关键点:
-
timeupdate触发频繁但不稳定(浏览器节流),建议用setInterval每 5–10 秒主动抓一次更可靠 - 需监听
ended、pause、seeking等事件,避免上报无效数据(比如用户拖到结尾又暂停) - 上报至少携带:
video_id、user_id(或 session/token)、current_time、timestamp(客户端时间戳,防服务端时钟偏差) - 避免高频请求:不要每秒都
fetch,可用防抖或合并上报(如 10 秒内只发最后一次)
PHP 后端接收与存储示例(简单版)
假设前端用 POST 发送 JSON:
POST /api/report_watch.php Content-Type: application/json{ "video_id": "vid_abc123", "user_id": "u789", "current_time": 42.6, "timestamp": 1717023456 }
对应 PHP 接收逻辑(注意校验和防重复):
立即学习“PHP免费学习笔记(深入)”;
'Method not allowed']);
exit;
}
$data = json_decode(file_get_contents('php://input'), true);
if (!$data || !isset($data['video_id'], $data['user_id'], $data['current_time'])) {
http_response_code(400);
echo json_encode(['error' => 'Missing required fields']);
exit;
}
// 简单去重:同一用户对同一视频,10 秒内只存最新一条
$cache_key = 'watch:' . $data['user_id'] . ':' . $data['video_id'];
if (apcu_exists($cache_key)) {
$last_ts = apcu_fetch($cache_key);
if (time() - $last_ts < 10) {
echo json_encode(['ok' => 'ignored']);
exit;
}
}
apcu_store($cache_key, time());
// 存数据库(示例结构:watch_records 表)
$stmt = $pdo->prepare("INSERT INTO watch_records (video_id, user_id, current_time, reported_at) VALUES (?, ?, ?, ?)");
$stmt->execute([
$data['video_id'],
$data['user_id'],
(float)$data['current_time'],
date('Y-m-d H:i:s')
]);
echo json_encode(['ok' => true]);
?>
容易被忽略的关键细节
很多项目卡在这几处:
- 没处理用户关闭页面/刷新前的最后进度——要用
beforeunload或visibilitychange补发一次 - 没区分「有效观看」和「挂机」:单纯记录
currentTime不够,要结合播放状态(video.paused === false)判断是否真在播 - 移动端 Safari 对
timeupdate支持差,且后台标签页会暂停 JS,必须降级用visibilitychange+ 手动计时 - PHP 接口没做频率限制,被恶意刷导致数据库写爆——加 IP + user_id 组合限流(如 1 分钟最多 12 次)
真正的难点不在 PHP 怎么写,而在于前端如何鲁棒地采集、抗干扰、保精度。PHP 这边只是个安静的接收方。
