cristian算法是一种时钟同步算法, 用于通过客户端进程与时间服务器同步时间。此算法适用于低延迟网络, 其中往返时间与准确性相比, 它是短的, 而易于冗余的分布式系统/应用程序与该算法不能并存。这里的往返时间是指请求开始到相应响应结束之间的持续时间。
以下是模仿cristian算法工作的示意图:
data:image/s3,"s3://crabby-images/5e7b1/5e7b19f0c300872f0b66a1d2d0de0e59d5a703b8" alt="cristian的事前插图"
算法:
1)客户端计算机上的进程在以下时间将获取时钟时间(服务器时间)的请求发送到时钟服务器
data:image/s3,"s3://crabby-images/d86ee/d86ee6933141a11f5d8153bc2f282cad917fe43e" alt="T_0"
.
2)Clock Server侦听客户端进程发出的请求, 并以Clock server time的形式返回响应。
3)客户端进程会及时从Clock Server获取响应
data:image/s3,"s3://crabby-images/3db9d/3db9d178c69725a5455e213ad2ee094eede11c7a" alt="T_1"
并使用以下公式计算同步的客户端时钟时间。
![\ [T_ {CLIENT} = T_ {SERVER} +(T_1-T_0)/ 2 \]](https://www.lsbin.com/wp-content/uploads/2021/05/quicklatex.com-18b40d7e02831dd6a81f8643483c787f_l3.png)
其中
data:image/s3,"s3://crabby-images/41a8a/41a8af91e8d5c5b6d775cc45af7bac4f325ae2dc" alt="T_ {CLIENT}"
指同步时钟时间,
data:image/s3,"s3://crabby-images/bd982/bd9828c3c452d0adae5ab2c2c7f718c8581dd83e" alt="T_ {SERVER}"
指服务器返回的时钟时间,
data:image/s3,"s3://crabby-images/64d7a/64d7a025c15229599f00b1c4b37c379dccc20bd9" alt="T_0"
指的是客户端进程发送请求的时间,
data:image/s3,"s3://crabby-images/d3ba5/d3ba5ca767eb2e7c07a1ecfcfdbe081e8ccdd68b" alt="T_1"
指的是客户端进程收到响应的时间
以上公式的工作/可靠性:
data:image/s3,"s3://crabby-images/f24c8/f24c8cf7ec67c21b2ede2df8e63b2e08b86eca0b" alt="T_1-T_0"
指网络和服务器将请求传输到服务器, 处理请求并将响应返回给客户端进程所花费的总时间, 假设网络延迟
data:image/s3,"s3://crabby-images/e3263/e3263418e0f9b32c13c1b61e84bd290354119cdb" alt="T_0"
和
data:image/s3,"s3://crabby-images/b7c88/b7c88eb7c037828bf4ee3105d7d01ca165d7ab8c" alt="T_1"
大致相等。
客户端的时间最多与实际时间不同
data:image/s3,"s3://crabby-images/5d699/5d6997239b724e3a353b5b2a18f306a1793f9bd4" alt="(T_1-T_0)/ 2"
秒。使用上面的陈述, 我们可以得出结论, 同步错误最多为
data:image/s3,"s3://crabby-images/5d052/5d0527665f0af82d351795db98de6a5ac897d2f8" alt="(T_1-T_0)/ 2"
秒。
因此,
![\ [错误\,\ epsilon \,[-(T_1-T_0)/ 2,\,(T_1-T_0)/ 2] \]](https://www.lsbin.com/wp-content/uploads/2021/05/quicklatex.com-476cf18f368563acdaf63b14dae0981d_l3.png)
以下Python代码说明了Cristian算法的工作原理:
以下代码用于在本地计算机上启动时钟服务器的原型:
# Python3 program imitating a clock server
import socket
import datetime
# function used to initiate the Clock Server
def initiateClockServer():
s = socket.socket()
print ( "Socket successfully created" )
# Server port
port = 8000
s.bind(('', port))
# Start listening to requests
s.listen( 5 )
print ( "Socket is listening..." )
# Clock Server Running forever
while True :
# Establish connection with client
connection, address = s.accept()
print ( 'Server connected to' , address)
# Respond the client with server clock time
connection.send( str (
datetime.datetime.now()).encode())
# Close the connection with the client process
connection.close()
# Driver function
if __name__ = = '__main__' :
# Trigger the Clock Server
initiateClockServer()
输出如下:
Socket successfully created
Socket is listening...
以下代码用于在本地计算机上启动客户端进程的原型:
# Python3 program imitating a client process
import socket
import datetime
from dateutil import parser
from timeit import default_timer as timer
# function used to Synchronize client process time
def synchronizeTime():
s = socket.socket()
# Server port
port = 8000
# connect to the clock server on local computer
s.connect(( '127.0.0.1' , port))
request_time = timer()
# receive data from the server
server_time = parser.parse(s.recv( 1024 ).decode())
response_time = timer()
actual_time = datetime.datetime.now()
print ( "Time returned by server: " + str (server_time))
process_delay_latency = response_time - request_time
print ( "Process Delay latency: " \
+ str (process_delay_latency) \
+ " seconds" )
print ( "Actual clock time at client side: " \
+ str (actual_time))
# synchronize process client clock time
client_time = server_time \
+ datetime.timedelta(seconds = \
(process_delay_latency) /2 )
print ( "Synchronized process client time: " \
+ str (client_time))
# calculate synchronization error
error = actual_time - client_time
print ( "Synchronization error : "
+ str (error.total_seconds()) + " seconds" )
s.close()
# Driver function
if __name__ = = '__main__' :
# synchronize time using clock server
synchronizeTime()
输出如下:
Time returned by server: 2018-11-07 17:56:43.302379
Process Delay latency: 0.0005150819997652434 seconds
Actual clock time at client side: 2018-11-07 17:56:43.302756
Synchronized process client time: 2018-11-07 17:56:43.302637
Synchronization error : 0.000119 seconds
时钟同步的即兴性:
使用网络上的迭代测试, 我们可以定义一个最小传输时间, 以此可以制定一个改进的同步时钟时间(减少同步误差)。
在这里, 通过定义一个最小的传输时间(具有较高的可信度), 我们可以说服务器时间将
总是在之后产生
data:image/s3,"s3://crabby-images/189cc/189ccd22e776d1764d8ff955b65523b9fd98bd0c" alt="T_0 + T_ {min}"
和
data:image/s3,"s3://crabby-images/18578/18578f4a984776ba510227a49015e863c8ec966f" alt="T_ {SERVER}"
将始终在之前生成
data:image/s3,"s3://crabby-images/13603/1360369b0806fad39f1f573a7e024aeed3281283" alt="T_1-T_ {min}"
, 在哪里
data:image/s3,"s3://crabby-images/6563f/6563f993a617866a9abf5f04ccfdab006682bfac" alt="T_ {min}"
是最小传输时间, 它是
data:image/s3,"s3://crabby-images/c1f6a/c1f6a5bf6d594da3d3ebcd700f0d7b678f98061e" alt="T_ {REQUEST}"
和
data:image/s3,"s3://crabby-images/aa232/aa232ea4f2c422d1e3aacd3133781fd465094e9a" alt="T_ {RESPONSE}"
在几次迭代测试中。同步错误可以表述为:
![\ [错误\,\ epsilon \,[-((T_1-T_0)/ 2-T_ {min}),\,((T_1-T_0)/ 2-T_ {min})] \]](https://www.lsbin.com/wp-content/uploads/2021/05/quicklatex.com-df72c8775f172461209e3fc0e769de27_l3.png)
同样, 如果
data:image/s3,"s3://crabby-images/8c8a0/8c8a0477d28db1e6125fce2c5c869b7416b9344d" alt="T_ {REQUEST}"
和
data:image/s3,"s3://crabby-images/bcbbd/bcbbd6af7deb0b1bacf7e3bef0d9ee399591ee21" alt="T_ {RESPONSE}"
相差相当长的时间, 我们可以替代
data:image/s3,"s3://crabby-images/2fb39/2fb39471621bfc98419e32438eb3a1f81920dd3f" alt="T_ {min}"
by
data:image/s3,"s3://crabby-images/7f2fc/7f2fce653efe02364ca55cead70bd330e45459f4" alt="T_ {min1}"
和
data:image/s3,"s3://crabby-images/e6186/e6186bc4f36543f9989301f7175ba1d05c55b67c" alt="T_ {min2}"
, 在哪里
data:image/s3,"s3://crabby-images/4370d/4370d14ed0ad7499ec41d8927bdb2ab34e8a3b0c" alt="T_ {min1}"
是观察到的最短请求时间, 并且
data:image/s3,"s3://crabby-images/09c5c/09c5cff4c96059da30fa1e5500369df543f468ba" alt="T_ {min2}"
是指在网络上观察到的最小响应时间。
在这种情况下, 同步时钟时间可以计算为:
![\ [T_ {CLIENT} = T_ {SERVER} +(T_1-T_0)/ 2 +(T_ {min2}-T_ {min1})/ 2 \]](https://www.lsbin.com/wp-content/uploads/2021/05/quicklatex.com-949cdd9ee2b55ab66bf57b8f5c2fb9be_l3.png)
因此, 通过仅将响应时间和请求时间引入为单独的时间延迟, 我们可以改善时钟时间的同步性, 从而减少总体同步误差。要运行的迭代测试的数量取决于整体时钟漂移观测到的。
参考文献:
1)https://en.wikipedia.org/wiki/Cristian%27s_algorithm
2)https://en.wikipedia.org/wiki/Round-trip_delay_time
3)https://www.lsbin.org/socket-programming-python
4)https://en.wikipedia.org/wiki/Clock_drift
首先, 你的面试准备可通过以下方式增强你的数据结构概念:Python DS课程。