客户端与服务器端时间保持一致

2024-05-14 22:58

本文主要是介绍客户端与服务器端时间保持一致,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

客户端实现准确的服务器时间同步 ,这篇博文是转过来的,原博文地址为:http://blog.csdn.net/chelen_jak/article/details/35335785,运气的缘故在我最需要秒杀倒计时时间同步问题时看到这篇博文,我首先说下我之前的思路,我使用setinterval来进行倒计时的时间,当然从服务器传过来一个时间戳,然后前台利用这个时间戳进行倒计时,当时傻傻的以为这就是与服务器同步,但是深入的想了下,这个想法非常的错误,这样少了一个至关重要的一点,传输时所消耗的时间是个重要的问题,而且并不能实时的与服务器交互,又不能完全使用本地时间,因为还是会出现差异。这样给服务器的压力太大,于是上网仔细查看了下,发现使用setinterval这个的话非常的不靠谱,转载的这篇博文也有说具体是为什么,所以说网上的很多所谓的倒计时代码都是并不能用。但是换句话,由于我做的是秒杀系统,对时间的要求非常高,所以要这样做,如果说你要做的没这么精确,所以并不需要这么的去做。
一、问题描述

需要解决的问题很简单,就是如何在页面上比较准确的显示服务器时间。目前比较常用的方法就是根据基准时间使用setTimeout或 setInterval来计算最新的时间,这样的问题在于setTimeout与setInterval的时间精度比较低,经测试一分钟大概能相差几秒 (与电脑性能以及运行的任务也相关),这样的精度在某些需求下是无法满足的。除此之外,如果要获得比较准确的时间可以定期与服务器进行校准,只是这样实现 的成本大一些。

 

本文尝试了一种改良的客户端实现时间同步的方式,具有以下的特点:

  • 1. 根据基准时间进行纯客户端计算,无需服务器校准
  • 2. 时间精度与客户端系统时间保持一致
  • 3. 不受客户端时间与服务器时间不同步造成的影响
  • 4. 不受客户端系统时间发生修改造成的影响
  • 5. 不受页面前进后退造成的影响
二、具体实现

1. 为了解决原方案中的时间精度问题,这里不再使用setTimeout和setInterval来直接计算时间,而是直接使用客户端时间(CT)。不过客户 端时间很可能与服务器时间(ST)不同步,这需要在页面加载的时候计算出客户端与服务器的时间差值(ΔT),这样只需在客户端时间上做一下修正即可得到准 确的服务器时间(ST’ = CT - ΔT)。

2. 由于客户端时间很可能被用户修改,因此直接按照步骤1中的方式计算,一旦用户修改了时间,计算出来的服务器时间也将随之发生变化。这就需要检测出客户端时 间的变化并消除这个变化。检测的方法很简单,即在每个计算周期(T)都将当时的客户端时间(CT2)与上一个周期的客户端时间(CT1)做比较,一旦两个 周期的差值(ΔT’ = CT2 - CT1 - T)大于某个预设值(S)时就将差值(ΔT’)加入到ΔT中,即此时的ΔT = ΔT + ΔT’。之所以需要设置一个预设值,是因为每个周期的时间本身不是固定的(依赖于setTimeout),因此ΔT’并不会等于0,如果每次都将 setTimeout造成的误差作为CT与ST之间的误差将会造成计算不准确。经过以上的计算,用户修改时间后将不会对计算结果产生影响。

3. 经JK提醒,完成以上两步还有一个问题,当用户离开当前页面之后后退回页面时,时间计算不准确。问题在于基准时间是服务器给的,在第一次进入页面的时候确 定,当用户后退回当前页面时,基准时间并没有变,这样会导致重新从过期的基准时间开始计算,导致不准确。需要解决这个问题就是需要解决跨页面的数据存储问 题,这在之前的《Ajax应用中浏览器历史的兼容性解决方案》一 文中已经说明,即通过表单元素来记忆。具体的实现方案是,页面第一次加载时创建两个input,一个用于存储最近一次的客户端时间,一个用于存储最近一次 的基准时间。如果发现已经存在input(前进、后退、非强制刷新)则比较上一次的客户端时间与当前客户端时间,如果其差值大于某个预设值则像步骤2中一 样进行校准,只不过使用的将是最新的基准值。

具体的代码实现如下

 

 /*定义*/    var SyncTimer = (function(){    /*跨页面数据存储器*/    //存储最近一次的客户端时间,用于在页面前进、后退时进行时间矫正    var memoryElementID = 'sync_timer_memory_el';    //存储矫正后的最新基准时间,当页面前进、后退到当前页面时会以此值为新的基准时间    var memoryBaseTimeElementID = 'sync_timer_memory_base_time_el';    document.write('   
<input type="text" id="' + memoryElementID + '">');    document.write('   
<input type="text" id="' + memoryBaseTimeElementID + '">');    return{    /*  * @param { Integer } baseTime 基准时间  * @param { Function } updater 时间更新时的监听器  * @param { Integer } interval 校准计算周期时长,默认为200ms。  * @param { Integer } threshold 两个检查周期之间的时间误差(差值-周期时长)如果大于阈值则视为客户端时间有调整,默认为500ms。  */    run: function(baseTime,updater,interval,threshold){    interval = interval || 200;    threshold = threshold || 500;                       var memoryEl = document.getElementById(memoryElementID);    var baseTimeEl = document.getElementById(memoryBaseTimeElementID);    /*前进、后退或刷新,则矫正baseTime*/    if( memoryEl.value != '' ){    //计算当前客户端时间与上次存储的客户端时间之差,如果差值超过阈值则更新基准时间    var diff = +new Date - parseInt(memoryEl.value);    if( Math.abs( diff ) - interval > threshold ){    baseTime = parseInt(baseTimeEl.value);    baseTime += diff;    }    }    var ct = +new Date;    var diff = ct - baseTime;    var pt = ct,cct;    (function(){    cct = +new Date;    /*计算当前计算周期与上一个计算周期的时间差,如果差值大于设定的阈值则进行矫正(处理客户端时间调整的情况)*/    var secDiff = cct - pt;    if( Math.abs( secDiff ) - interval > threshold ){    diff += (secDiff - interval);    }    var fixedTime = cct - diff;    updater( fixedTime );    pt = memoryEl.value = cct;    baseTimeEl.value = fixedTime;    setTimeout(arguments.callee,interval);    })();    }    }    })();    /*使用*/    window.onload = function(){    var serverTime = parseInt($('dateWrapper').getAttribute('date'))*1000;    SyncTimer.run(serverTime,function(date){    var d = new Date(date);    $('dateWrapper').innerHTML = d.format('yyyy-MM-dd hh:mm:ss');    $('dateWrapper').setAttribute('date',parseInt(date/1000));    });    }    

 

三、总结
  • 总体实现还是比较麻烦,如果对时间精度要求不高可不必这么做。
  • 还有一种情况未解决:用户从当前页面进入别的页面后修改客户端时间,之后后退到当前页面,此时时间计算不正确,但是暂时未找到解决方案。
  • 此外发现两个有意思的东西:1. 在Firefox下如果将客户端时间改慢会导致setInterval停止运行,而setTimeout则不会;2. 在Chrome中,当用户修改了客户端时间后,setInterval中取到的Date的值并不会随用户的修改而修改。
  • 下面写上我修改后的代码,可以直接粘贴使用的倒计时代码,而不是系统时间的代码
  • <script>/*定义*/    var SyncTimer = (function(){    /*跨页面数据存储器*/    //存储最近一次的客户端时间,用于在页面前进、后退时进行时间矫正    var memoryElementID = 'sync_timer_memory_el';    //存储矫正后的最新基准时间,当页面前进、后退到当前页面时会以此值为新的基准时间    var memoryBaseTimeElementID = 'sync_timer_memory_base_time_el';    document.write('<input type="hidden" id="' + memoryElementID + '">');    document.write('<input type="hidden" id="' + memoryBaseTimeElementID + '">');    return{    /*  * @param { Integer } baseTime 基准时间  * @param { Function } updater 时间更新时的监听器  * @param { Integer } interval 校准计算周期时长,默认为200ms。  * @param { Integer } threshold 两个检查周期之间的时间误差(差值-周期时长)如果大于阈值则视为客户端时间有调整,默认为500ms。  */    run: function(baseTime,updater,interval,threshold){    interval = interval || 200;    threshold = threshold || 500;                       var memoryEl = document.getElementById(memoryElementID);    var baseTimeEl = document.getElementById(memoryBaseTimeElementID);    /*前进、后退或刷新,则矫正baseTime*/    if( memoryEl.value != '' ){    //计算当前客户端时间与上次存储的客户端时间之差,如果差值超过阈值则更新基准时间    var diff = +new Date - parseInt(memoryEl.value);    if( Math.abs( diff ) - interval > threshold ){    baseTime = parseInt(baseTimeEl.value);    baseTime += diff;    }    }    var ct = +new Date;    var diff = ct - baseTime;    var pt = ct,cct;    (function(){    cct = +new Date;    /*计算当前计算周期与上一个计算周期的时间差,如果差值大于设定的阈值则进行矫正(处理客户端时间调整的情况)*/    var secDiff = cct - pt;    if( Math.abs( secDiff ) - interval > threshold ){    diff += (secDiff - interval);    }    var fixedTime = cct - diff;    updater( fixedTime );    pt = memoryEl.value = cct;    baseTimeEl.value = fixedTime;    setTimeout(arguments.callee,interval);    })();    }    }    })();    /*使用*/    window.onload = function(){    var serverTime = parseInt({$time})*1000;    SyncTimer.run(serverTime,function(date){var intDiff = (1437364800+18000)*1000 - date;intDiff =  Math.floor(intDiff / 1000);var day=0,hour=0,minute=0,second=0;//时间默认值		if(intDiff > 0){day = Math.floor(intDiff / (60 * 60 * 24));hour = Math.floor(intDiff / (60 * 60)) - (day * 24);minute = Math.floor(intDiff / 60) - (day * 24 * 60) - (hour * 60);second = Math.floor(intDiff) - (day * 24 * 60 * 60) - (hour * 60 * 60) - (minute * 60);}if (minute <= 9) minute = '0' + minute;if (second <= 9) second = '0' + second;-$('#day_show').html(day+"天");$('#hour_show').html('<s id="h"></s>'+hour+'时');$('#minute_show').html('<s></s>'+minute+'分');$('#second_show').html('<s></s>'+second+'秒');if(minute == '00' && second == '00' && hour == '0' && day == '0'){temp = window.clearInterval(temp);$('#now').removeClass('btn-danger');$('#now').addClass('btn-success');}});    }    </script>


这篇关于客户端与服务器端时间保持一致的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/990095

相关文章

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

MySQL按时间维度对亿级数据表进行平滑分表

《MySQL按时间维度对亿级数据表进行平滑分表》本文将以一个真实的4亿数据表分表案例为基础,详细介绍如何在不影响线上业务的情况下,完成按时间维度分表的完整过程,感兴趣的小伙伴可以了解一下... 目录引言一、为什么我们需要分表1.1 单表数据量过大的问题1.2 分表方案选型二、分表前的准备工作2.1 数据评估

MySQL中DATE_FORMAT时间函数的使用小结

《MySQL中DATE_FORMAT时间函数的使用小结》本文主要介绍了MySQL中DATE_FORMAT时间函数的使用小结,用于格式化日期/时间字段,可提取年月、统计月份数据、精确到天,对大家的学习或... 目录前言DATE_FORMAT时间函数总结前言mysql可以使用DATE_FORMAT获取日期字段

Python标准库datetime模块日期和时间数据类型解读

《Python标准库datetime模块日期和时间数据类型解读》文章介绍Python中datetime模块的date、time、datetime类,用于处理日期、时间及日期时间结合体,通过属性获取时间... 目录Datetime常用类日期date类型使用时间 time 类型使用日期和时间的结合体–日期时间(

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Java获取当前时间String类型和Date类型方式

《Java获取当前时间String类型和Date类型方式》:本文主要介绍Java获取当前时间String类型和Date类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Java获取当前时间String和Date类型String类型和Date类型输出结果总结Java获取