基于NTP协议网络校时

た 入场券 2022-08-13 17:57 288阅读 0赞

先来个NTP协议介绍
http://blog.163.com/yzc_5001/blog/static/2061963420121283050787/

关于时间戳的换算工具
http://tool.chinaz.com/Tools/unixtime.aspx#

下面是代码,C-Free + MinGW 编译,须在工程连接参数里添加 -lwsock32

  1. #include <stdio.h>
  2. #include <time.h>
  3. #include <winsock2.h>
  4. #define JAN_1970 0x83AA7E80 /* 2208988800 1970-1900 in seconds*/
  5. typedef union
  6. {
  7. int time;
  8. char time_array[4];
  9. } Timestamp;
  10. typedef struct
  11. {
  12. char message_hdr[16];//报文头部
  13. char RefTimestamp[8];
  14. Timestamp T1;
  15. char T1_decimal[4]; //T1的小数部分
  16. Timestamp T2;
  17. char T2_decimal[4]; //T2的小数部分
  18. Timestamp T3;
  19. char T3_decimal[4]; //T3的小数部分
  20. } NTP_MESSAGE;
  21. int main()
  22. {
  23. WORD socketVersion = MAKEWORD(2,2);
  24. WSADATA wsaData;
  25. if(WSAStartup(socketVersion, &wsaData) != 0)
  26. {
  27. return 0;
  28. }
  29. SOCKET sclient = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
  30. SOCKADDR_IN sin;
  31. sin.sin_family = AF_INET;
  32. sin.sin_port = htons(123);
  33. sin.sin_addr.S_un.S_addr = inet_addr("198.123.30.132");
  34. int len = sizeof(sin);
  35. char sendData[] = { //构造报文
  36. 0x1B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  37. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  38. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xD0, 0xAF, 0x5F, 0xF5, 0x23, 0xD7, 0x08, 0x00
  39. };
  40. printf("Connect to NTP Server...\n");
  41. time_t T1=time(NULL);
  42. sendto(sclient, sendData, sizeof(sendData), 0, (SOCKADDR *)&sin, len);
  43. char recvData[48]={0};
  44. int ret = recvfrom(sclient, recvData, 48, 0, (SOCKADDR *)&sin, &len);
  45. if(ret > 0)
  46. {
  47. time_t T4=time(NULL);
  48. printf("Connect success!\n\n");
  49. printf("Now the local time is %s",ctime(&T4));
  50. NTP_MESSAGE *ntp_time=(NTP_MESSAGE *)recvData;
  51. /* 大小端变换 由于NTP以1900为始,需变换 */
  52. ntp_time->T2.time=ntohl(ntp_time->T2.time)-JAN_1970;//T2
  53. ntp_time->T3.time=ntohl(ntp_time->T3.time)-JAN_1970;//T3
  54. /* NTP报文的网络延时=(T4-T1)-(T3-T2) */
  55. time_t net_delay=(T4-T1)-(ntp_time->T3.time-ntp_time->T2.time);
  56. printf("Net delay is %ld second(s).\n",net_delay);
  57. /* 误差补偿秒数=((T2-T1)+(T3-T4))/2 */
  58. time_t Deviation=(((time_t)ntp_time->T2.time-T1)+((time_t)ntp_time->T3.time-T4))/2;
  59. printf("\nThe deviation is %d second(s).\n",Deviation);
  60. if(Deviation != 0)
  61. {
  62. printf("Reset the local time...\n");
  63. time_t now=time(NULL)+Deviation;
  64. struct tm *ptr;
  65. ptr=localtime(&now);
  66. char buffer[20]={0};
  67. sprintf(buffer,"date %d-%d-%d",ptr->tm_year+1900,ptr->tm_mon+1,ptr->tm_mday);
  68. system(buffer);
  69. sprintf(buffer,"time %d:%d:%d",ptr->tm_hour,ptr->tm_min,ptr->tm_sec);
  70. system(buffer);
  71. now=time(NULL);
  72. printf("Now the local time is %s",ctime(&now));
  73. }
  74. else //没有误差,无需校时
  75. {
  76. printf("Here is no need to set local time.\n");
  77. }
  78. }
  79. closesocket(sclient);
  80. WSACleanup();
  81. system("pause");
  82. return 0;
  83. }

由于使用API 设置时间在 win7 下需要特殊的权限,故采用系统命令 date 和 time 设置,算是 不是办法的办法。
关于NTP协议,最开始给出的链接有具体的说明,在程序中,只需48个字节的必须内容即可,其它内容属可选项。

时间的小数部分抛弃不用,对于一般校时而言足够了。

下面给个截图

SouthEast

发表评论

表情:
评论列表 (有 0 条评论,288人围观)

还没有评论,来说两句吧...

相关阅读

    相关 NTP协议

    NTP(Network Time Protocol)是一种用于在计算机网络中同步各个节点时间的协议。它通过将一台计算机的时钟同步到另一台计算机的时钟来实现时间同步。NTP协议是