背景
- 私信中一些时延相关的统计需要用时间戳来做差值但是用客户端本地的时间戳可能会由于时区的问题产生误差,因此想需要服务端的时间戳做个校准。
- 例如:在统计接收消息时延时,需要用接收方收到消息的时间戳减去发送方发送消息的时间戳。但是如果两个人的时区不同的话,得到的结果是错误的。因此需要统一按照server的时间来做运算
原理
由于可能受到物理因素[1]等干预。会导致本地时钟与服务端时钟存在误差。因此需要进行时间校准;
目前最简单且精准的时间校准方案可参考NTP协议实现方式:
[1] 物理因素:钟表和计算机内部有一个叫做晶体振荡器的东西,给它加上电压,它就会以固定的频率振动。但这个振动频率的稳定性,取决于它的制造工艺以及外界环境的影响。
出于成本考虑,钟表的制作工艺没那么高,所以它更容易有误差。而电脑制作工艺虽然比较高,但它内部的晶体振荡器会受到温度变化带来的影响,实际工作过程中也会产生误差。
向server请求服务端时钟时间,并在网络请求上添加时间戳,配合计算网络延迟。从而修改本地时间;
根据图示可以计算出网络传输延迟,以及本地时钟与服务端时钟的时延:
- 网络传输延迟 = (t4 - t1) - (t3 - t2)
- 本地与服务端时钟时延 = t2 - t1 - 网络传输延迟 / 2 = ((t2 - t1) + (t3 - t4)) / 2
计算过程假设网络上下回路对称,传输延迟一致
获取时钟时延后,便可以对本地时钟进行修改,保证与服务端时间的一致性;
方案
- server在所有请求的response中增加两个通用字段,request_arrived_time(请求到达中台的时间),server_execution_end_time(中台返回response的时间),单位都是毫秒
- 客户端在收到response时,计算与server时间的delta(误差),计算公式如下:
request_duration = client_request_end - client_request_start;
server_execution_duration = server_execution_end_time - request_arrived_time;
estimate_request_arrived_time = (request_duration - server_execution_duration) / 2 + client_request_start;
delta = request_arrived_time - estimate_request_arrived_time;
客户端在每次计算出的delta做累加得到total_delta,并记录计算次数times,avg_delta = total_delta / time,用avg_delta来作为当前客户端与server的时间误差,后续在做统计时统一用客户端的时间戳加上avg_delta
代码实现
请求服务端获取请求发送相关时间戳,并发送时钟对齐的消息
mainThreadHandler.sendMessageDelayed(Message.obtain().apply {
what = MESSAGE_TOKEN
obj = timestampModel
}, DELAY_TIME)
收到消息后进行计算
val mainThreadHandler = object : Handler(Looper.getMainLooper()) {
override fun handleMessage(msg: Message?) {
super.handleMessage(msg)
msg ?: return
when (msg.what) {
MESSAGE_TOKEN -> {
val timestampModel = msg.obj as? RequestTimestampModel ?: return
Task.delay(0)
.continueWith {
// 可能存在并发问题,所以需要考虑线程安全问题
timestampModel.apply {
val requestDuration = clientEndTime - clientStartTime
val serverExecuteDuration = serverExecutionEndTime - serverArrivedTime
val estimateRequestArrivedTime = (requestDuration - serverExecuteDuration) / 2 + clientStartTime
lastDeltaFromServerTime = serverArrivedTime - estimateRequestArrivedTime
serverSyncTimes++
totalDeltaFromServerTime += lastDeltaFromServerTime
avgDeltaFromServerTime = totalDeltaFromServerTime / serverSyncTimes
Depend.Log.i(TAG,
"sync Client timestamp $serverSyncTimes cmd:$cmd,($requestDuration,$serverExecuteDuration),singleDelta:$lastDeltaFromServerTime,avg:$avgDeltaFromServerTime")
}
}
}
}
}
}
存在问题
PTP的时间戳是在硬件上产生的,不受软件影响,效率更高。同步精度可以实现100ns。
网络同步时延,受网络流量影响也很大。网络波动越大,测得的路径时延就越不准确。可以通过重复对齐解决问题。