ntp_test.lua 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. --[[
  2. @module ntp_test
  3. @summary ntp_test测试功能模块
  4. @version 1.0
  5. @date 2025.10.24
  6. @author 马亚丹
  7. @usage
  8. 本demo演示的功能为:使用Air780EPM开发板,演示用ntp网络对时后,获取本地时间和UTC时间的功能
  9. 核心逻辑:
  10. 1.判断是否联网,
  11. 2.网络就绪后开始时间同步
  12. 3.时间同步成功,获取本地时间和UTC时间,按默认间隔时间循环打印获取的时间信息
  13. 4.时间同步失败,打印提醒
  14. ]]
  15. -- 自定义NTP服务器列表,可选配置,默认使用内置的ntp服务器地址ntp.aliyun.com
  16. local ntp_servers = {
  17. "ntp.aliyun.com",
  18. "ntp.air32.cn",
  19. "time1.cloud.tencent.com"
  20. }
  21. --本地时间是指:当前时区的时间,默认是东八区北京时间,可以通过rtc.timezone接口查询或者设置时区
  22. --UTC时间是指:0时区的时间
  23. --东八区的时间是在UTC时间的基础上增加8个小时
  24. -- 打印时间信息的工具函数
  25. local function print_time_details()
  26. --设置为东3区时间
  27. --rtc.timezone(12)
  28. --获取东3区的时间字符串
  29. --东3区时间字符串 Fri Oct 24 12:01:20 2025
  30. --log.info("sntp", "东3区时间字符串", os.date())
  31. -- 获取本地时间字符串
  32. -- 本地时间字符串 Fri Oct 24 17:01:15 2025
  33. log.info("sntp", "本地时间字符串", os.date())
  34. -- 获取UTC时间字符串
  35. -- UTC时间字符串 Fri Oct 24 09:01:15 2025
  36. log.info("sntp", "UTC时间字符串", os.date("!%c"))
  37. -- 格式化本地时间字符串
  38. -- 本地时间字符串 2025-10-24 17:01:15
  39. log.info("sntp", "格式化本地时间字符串", os.date("%Y-%m-%d %H:%M:%S"))
  40. -- 格式化UTC时间字符串
  41. -- UTC时间字符串 2025-10-24 09:01:15
  42. log.info("sntp", "格式化UTC时间字符串", os.date("!%Y-%m-%d %H:%M:%S"))
  43. -- RTC时钟原始数据(UTC时间)
  44. local rtc_time = rtc.get()
  45. -- RTC时钟(UTC) {"year":2025,"min":1,"hour":9,"mon":10,"sec":15,"day":24}
  46. log.info("sntp", "RTC时钟(UTC)", json.encode(rtc_time))
  47. -- 本地时间戳(秒级)
  48. --本地时间戳 1761296475
  49. log.info("sntp", "本地时间戳", os.time())
  50. -- 获取本地时间的table
  51. -- 本地时间结构 {"wday":6,"min":1,"yday":297,"hour":17,"isdst":false,"year":2025,"month":10,"sec":15,"day":24}
  52. local local_struct_time = os.date("*t")
  53. log.info("sntp", "本地时间结构", json.encode(local_struct_time))
  54. --结构时间转时间戳 1761325275
  55. log.info("sntp", "结构时间转时间戳", os.time(local_struct_time))
  56. end
  57. -- 打印高精度时间戳
  58. local function print_high_precision_time()
  59. local ntp_time = socket.ntptm()
  60. if ntp_time and ntp_time.tsec then
  61. --tm数据 {"sms":89,"tms":320,"vaild":true,"tsec":1761296475,"lms":231,"ndeley":28,"lsec":18,"ssec":1761296457}
  62. log.info("tm数据", json.encode(ntp_time))
  63. -- 格式化:秒.毫秒
  64. --高精度时间戳 1761296475.320
  65. log.info("sntp", "高精度时间戳", string.format("%u.%03d", ntp_time.tsec, ntp_time.tms))
  66. else
  67. log.warn("sntp", "高精度时间戳获取失败")
  68. end
  69. end
  70. -- SNTP同步主逻辑
  71. local function sntp_sync_loop()
  72. --查看网卡适配器的联网状态是否IP_READY,true表示已经准备好可以联网了,false暂时不可以联网
  73. while not socket.adapter(socket.dft()) do
  74. log.warn("sntp", "wait IP_READY", socket.dft())
  75. -- 在此处阻塞等待默认网卡连接成功的消息"IP_READY"
  76. -- 或者等待1秒超时退出阻塞等待状态;
  77. -- 注意:此处的1000毫秒超时不要修改的更长;
  78. -- 因为当使用exnetif.set_priority_order配置多个网卡连接外网的优先级时,会隐式的修改默认使用的网卡
  79. -- 当exnetif.set_priority_order的调用时序和此处的socket.adapter(socket.dft())判断时序有可能不匹配
  80. -- 此处的1秒,能够保证,即使时序不匹配,也能1秒钟退出阻塞状态,再去判断socket.adapter(socket.dft())
  81. sys.waitUntil("IP_READY", 1000)
  82. end
  83. -- 检测到了IP_READY消息,设置默认网络适配器编号
  84. log.info("sntp", "recv IP_READY", socket.dft())
  85. while true do
  86. log.info("sntp", "开始同步:")
  87. -- 方式1:使用自定义服务器列表
  88. -- log.info("sntp", "开始同步,服务器列表:", json.encode(ntp_servers))
  89. -- 选择自己要用的的服务器
  90. --socket.sntp(ntp_servers[2])
  91. -- 方式2: 使用默认的ntp服务器地址:ntp.aliyun.com
  92. socket.sntp()
  93. -- 等待同步结果(成功:NTP_UPDATE,失败:超时)
  94. local sync_success = sys.waitUntil("NTP_UPDATE", 5000)
  95. if sync_success then
  96. --ntp同步成功之后,已经自动将系统时间设置为本地时间
  97. --在此处的脚本中,不需要再调用任何时间设置接口去手动设置时间
  98. --本地时间:默认为东八区北京时间,如果使用rtc.timezone(zone)设置过本地时间,则本地时间为自己设置过的时区对应的时间
  99. -- 同步成功:打印时间信息
  100. log.info("sntp", "时间同步成功")
  101. print_time_details()
  102. print_high_precision_time()
  103. --时间同步成功
  104. --再等待1小时再次发起下次时间同步,这里的等待时长可以按自己的需求修改。
  105. sys.wait(3600*1000)
  106. else
  107. log.warn("sntp", "时间同步失败")
  108. --时间同步失败
  109. --再等待10秒重新发起时间同步,这里的等待时长可以按自己的需求修改。
  110. sys.wait(10*1000)
  111. end
  112. end
  113. end
  114. -- 订阅NTP错误消息
  115. sys.subscribe("NTP_ERROR", function(err_info)
  116. log.error("sntp", "同步过程发生错误", err_info or "未知错误")
  117. end)
  118. -- 启动SNTP同步任务
  119. sys.taskInit(sntp_sync_loop)