搜索
bottom↓
回复: 4

小程序:基于NTP协议网络校时

[复制链接]

出0入4汤圆

发表于 2014-11-9 15:22:32 | 显示全部楼层 |阅读模式
本帖最后由 xiaoyigechaos 于 2014-11-9 15:26 编辑

先来个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.    
  31.     SOCKADDR_IN sin;
  32.     sin.sin_family = AF_INET;
  33.     sin.sin_port = htons(123);
  34.     sin.sin_addr.S_un.S_addr = inet_addr("198.123.30.132");
  35.     int len = sizeof(sin);
  36.    
  37.     char sendData[] = { //构造报文
  38.         0x1B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  39.         0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  40.         0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xD0, 0xAF, 0x5F, 0xF5, 0x23, 0xD7, 0x08, 0x00
  41.         };
  42.        
  43.         printf("Connect to NTP Server...\n");
  44.        
  45.         time_t T1=time(NULL);
  46.        
  47.         sendto(sclient, sendData, sizeof(sendData), 0, (SOCKADDR *)&sin, len);

  48.     char recvData[48]={0};
  49.     int ret = recvfrom(sclient, recvData, 48, 0, (SOCKADDR *)&sin, &len);
  50.     if(ret > 0)
  51.     {
  52.             time_t T4=time(NULL);
  53.            
  54.             printf("Connect success!\n\n");
  55.             printf("Now the local time is %s",ctime(&T4));
  56.            
  57.             NTP_MESSAGE *ntp_time=(NTP_MESSAGE *)recvData;
  58.            
  59.             /* 大小端变换 由于NTP以1900为始,需变换 */
  60.             ntp_time->T2.time=ntohl(ntp_time->T2.time)-JAN_1970;//T2
  61.             ntp_time->T3.time=ntohl(ntp_time->T3.time)-JAN_1970;//T3
  62.            
  63.             /* NTP报文的网络延时=(T4-T1)-(T3-T2) */
  64.             time_t net_delay=(T4-T1)-(ntp_time->T3.time-ntp_time->T2.time);
  65.             printf("Net delay is %ld second(s).\n",net_delay);
  66.            
  67.             /* 误差补偿秒数=((T2-T1)+(T3-T4))/2 */
  68.             time_t Deviation=(((time_t)ntp_time->T2.time-T1)+((time_t)ntp_time->T3.time-T4))/2;
  69.         printf("\nThe deviation is %d second(s).\n",Deviation);
  70.         
  71.         if(Deviation != 0)
  72.         {
  73.                 printf("Reset the local time...\n");
  74.                
  75.                 time_t now=time(NULL)+Deviation;
  76.                 struct tm *ptr;
  77.                 ptr=localtime(&now);
  78.                 char buffer[20]={0};
  79.                
  80.                 sprintf(buffer,"date %d-%d-%d",ptr->tm_year+1900,ptr->tm_mon+1,ptr->tm_mday);
  81.                 system(buffer);
  82.                
  83.                 sprintf(buffer,"time %d:%d:%d",ptr->tm_hour,ptr->tm_min,ptr->tm_sec);
  84.                 system(buffer);
  85.                 now=time(NULL);
  86.                 printf("Now the local time is %s",ctime(&now));
  87.         }
  88.         else //没有误差,无需校时
  89.         {
  90.                 printf("Here is no need to set local time.\n");
  91.         }
  92.     }

  93.     closesocket(sclient);
  94.     WSACleanup();
  95.     system("pause");
  96.     return 0;
  97. }
复制代码


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

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

下面给个截图


阿莫论坛20周年了!感谢大家的支持与爱护!!

知道什么是神吗?其实神本来也是人,只不过神做了人做不到的事情 所以才成了神。 (头文字D, 杜汶泽)

出0入0汤圆

发表于 2014-11-9 17:15:08 | 显示全部楼层
谢谢LZ分享,你这个是本地有NTP服务器么?那本地的NTP服务器也是同样的然后问外面的服务器对时?

出0入4汤圆

 楼主| 发表于 2014-11-9 17:22:44 | 显示全部楼层
bobqq 发表于 2014-11-9 17:15
谢谢LZ分享,你这个是本地有NTP服务器么?那本地的NTP服务器也是同样的然后问外面的服务器对时? ...

我没有NTP服务器

百度上可以找到一些可用的NTP服务器

自己搭建NTP服务器,其时钟来源是多样性的,比如GPS时钟

出0入0汤圆

发表于 2014-11-9 18:24:42 来自手机 | 显示全部楼层
顶下,谢谢楼主的贡献

出0入0汤圆

发表于 2015-3-26 16:04:54 | 显示全部楼层
这个不错,谢谢
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片。注意:要连续压缩2次才能满足要求!!】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-7-23 14:22

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表