基本的NTP客户端代码示例

用于向NTP服务器请求时间。向NTP服务器发送一个48字节大小的请求报文,再使用recvfrom()函数接收服务器的响应。最后,解析得到的时间戳并转换为Unix时间进行显示。


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <time.h>

#define NTP_SERVER_IP "ntp1.aliyun.com"
#define NTP_SERVER_PORT 123
#define NTP_PACKET_SIZE 48

// NTP报文结构体
struct ntp_packet {
    uint8_t flags;// 标志位,使用0x1B表示客户端请求时间信息
    uint8_t stratum;// stratum层级,一般使用1表示客户端直接与服务器通信  
    uint8_t poll;// 投票计数器,表示网络状况,一般使用0或10表示正常情况  
    uint8_t precision;// 时间精度,一般使用0或10表示正常情况  
    uint32_t rootDelay;// 服务器到客户端的延迟,单位是秒  
    uint32_t rootDispersion;// 服务器到客户端的散度,单位是秒
    uint32_t referenceIdentifier;// 参考标识符,通常是服务器的IP地址或域名 
    uint32_t referenceTimestamp[2];// 参考时间戳,一般是服务器的本地时间
    uint32_t originateTimestamp[2];// 起始时间戳,一般是客户端发送请求的时间
    uint32_t receiveTimestamp[2];// 接收时间戳,一般是服务器收到请求的时间  
    uint32_t transmitTimestamp[2];// 传输时间戳,一般是服务器发送响应的时间
};


int main() {
    int sockfd;
    struct sockaddr_in servaddr;
  
    // 创建套接字
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&servaddr, 0, sizeof(servaddr));
  
    // 设置服务器地址和端口
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(NTP_SERVER_PORT);
    servaddr.sin_addr.s_addr = inet_addr("120.25.115.20");
/*
    if(inet_pton(AF_INET, NTP_SERVER_IP, &(servaddr.sin_addr)) <= 0) {
        perror("inet_pton failed");
        exit(EXIT_FAILURE);
    }
*/
    // 构造请求报文
    struct ntp_packet packet;
    memset(&packet, 0, sizeof(packet));
    packet.flags = 0x1B;

    // 发送请求报文到服务器
    if(sendto(sockfd, &packet, sizeof(packet), 0, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) { 
        perror("sendto failed");
        exit(EXIT_FAILURE);
    }
  
    // 接收响应报文
    memset(&packet, 0, sizeof(packet));
    if(recv(sockfd, &packet, sizeof(packet), 0) < 0) {
        perror("recv failed");
        exit(EXIT_FAILURE);
    }

    // 解析响应报文中的时间戳
    time_t transmitTime = ntohl(packet.transmitTimestamp[0]) - 2208988800U;
    printf("NTP server time: %s", ctime(&transmitTime));

    close(sockfd);
  
    return 0;
}

2208988800是一个特定的偏移量,用于将NTP(网络时间协议)时间戳转换为UNIX时间戳。NTP时间戳是从1900年1月1日00:00:00开始计算的,而UNIX时间戳是从1970年1月1日00:00:00开始计算的。因此,需要进行一个转换来将NTP时间戳转换为UNIX时间戳。

这个偏移量是1970年与1900年之间的差值,也就是2208988800秒。在进行转换时,将NTP时间戳减去这个偏移量,就可以得到对应的UNIX时间戳。这样做的目的是为了在不同系统之间进行时间同步和比较。

ntp服务端

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <time.h>

#define NTP_PORT 123      // NTP使用的默认端口号
#define NTP_PACKET_SIZE 48  // NTP报文的大小

void fillNTPPacket(char *ntpPacket) {
    memset(ntpPacket, 0, NTP_PACKET_SIZE);

    // 设置NTP报文的首字节
    ntpPacket[0] = 0x1B;  // 设置LI、Version和Mode字段

    // 设置时间戳字段为当前时间
    time_t currentTime = time(NULL);
    uint32_t ntpTime = htonl(currentTime + 2208988800U);
    memcpy(&ntpPacket[40], &ntpTime, sizeof(ntpTime));
}

int main() {
    int sockfd;
    struct sockaddr_in servaddr, cliaddr;
    char ntpPacket[NTP_PACKET_SIZE];

    // 创建UDP套接字
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&servaddr, 0, sizeof(servaddr));

    // 设置服务器地址和端口
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = INADDR_ANY;
    servaddr.sin_port = htons(NTP_PORT);

    // 绑定套接字到服务器地址
    if (bind(sockfd, (const struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    while (1) {
        int len, ntpSize;

        // 接收来自客户端的请求
        len = sizeof(cliaddr);
        ntpSize = recvfrom(sockfd, (char *)ntpPacket, NTP_PACKET_SIZE, 0, (struct sockaddr *)&cliaddr, &len);

        if (ntpSize < 0) {
            perror("recvfrom failed");
            exit(EXIT_FAILURE);
        }

        // 填充NTP报文
        fillNTPPacket(ntpPacket);

        // 发送NTP响应给客户端
        if (sendto(sockfd, (const char *)ntpPacket, NTP_PACKET_SIZE, 0, (const struct sockaddr *)&cliaddr, len) < 0) {
            perror("sendto failed");
            exit(EXIT_FAILURE);
        }
    }

    close(sockfd);

    return 0;
}

  • 6
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的 NTP 客户端示例代码,使用 C 语言编写: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <netdb.h> #include <netinet/in.h> #include <sys/socket.h> #include <sys/time.h> #include <time.h> #include <unistd.h> #define NTP_TIMESTAMP_DELTA 2208988800ull typedef struct { uint8_t li_vn_mode; // Leap indicator, version and mode uint8_t stratum; // Stratum level of the local clock uint8_t poll; // Maximum interval between successive messages uint8_t precision; // Precision of the local clock uint32_t rootDelay; // Total round trip delay time uint32_t rootDispersion; // Max error aloud from primary clock source uint32_t refId; // Reference clock identifier uint32_t refTm_s; // Reference time-stamp seconds uint32_t refTm_f; // Reference time-stamp fraction of a second uint32_t origTm_s; // Originate time-stamp seconds uint32_t origTm_f; // Originate time-stamp fraction of a second uint32_t rxTm_s; // Received time-stamp seconds uint32_t rxTm_f; // Received time-stamp fraction of a second uint32_t txTm_s; // Transmit time-stamp seconds uint32_t txTm_f; // Transmit time-stamp fraction of a second } ntp_packet; void error(const char *msg) { perror(msg); exit(1); } int main(int argc, char *argv[]) { int sockfd, n; struct sockaddr_in serv_addr; struct hostent *server; char buffer[256]; ntp_packet packet; if (argc < 2) { fprintf(stderr,"usage %s hostname\n", argv[0]); exit(0); } sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); if (sockfd < 0) { error("ERROR opening socket"); } server = gethostbyname(argv[1]); if (server == NULL) { fprintf(stderr,"ERROR, no such host\n"); exit(0); } bzero((char *) &serv_addr, sizeof(serv_addr)); serv_addr.sin_family = AF_INET; bcopy((char *)server->h_addr, (char *)&serv_addr.sin_addr.s_addr, server->h_length); serv_addr.sin_port = htons(123); memset(&packet, 0, sizeof(ntp_packet)); packet.li_vn_mode = (0x3 << 6) | (0x3 << 3) | 0x3; // NTPv4, client mode n = sendto(sockfd, (char *)&packet, sizeof(ntp_packet), 0, (struct sockaddr *)&serv_addr, sizeof(serv_addr)); if (n < 0) { error("ERROR in sendto"); } n = recvfrom(sockfd, (char *)&packet, sizeof(ntp_packet), 0, NULL, NULL); if (n < 0) { error("ERROR in recvfrom"); } packet.txTm_s = ntohl(packet.txTm_s); packet.txTm_f = ntohl(packet.txTm_f); time_t txTm = (time_t)(packet.txTm_s - NTP_TIMESTAMP_DELTA); printf("Time: %s", ctime((const time_t *)&txTm)); close(sockfd); return 0; } ``` 该代码中的 `ntp_packet` 结构体表示 NTP 协议中的数据包格式。在 `main` 函数中,它首先创建了一个 UDP 套接字,然后向指定的 NTP 服务器发送一个 NTP 数据包,并等待接收服务器返回的数据包。最后,从接收到的数据包中解析出服务器的时间戳,并将其转换为可读的时间格式,输出到标准输出中。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值