Chinaunix

标题: 分享一个自己用Perl写的GUI桌面应用程序 [打印本页]

作者: iamlimeng    时间: 2013-05-08 15:57
标题: 分享一个自己用Perl写的GUI桌面应用程序
本帖最后由 iamlimeng 于 2013-05-08 15:59 编辑

最近写个小程序,需要获得准确NTP时间,不依赖系统时间。我把其中的部分提出来,做成了一个可以用来同步本机时间的小桌面应用。因为Windows本身可以设置自动与授时服务器同步时间,从功能上看这个程序是多余的,但本版Perl写的桌面应用很少,尤其是GUI程序,更少,在此共享一下,希望跟大家多多交流,共同进步。

代码写得不好,大家将就参考啊。另外,对授时服务器的工作原理了解很少,如有更好的方式实现,希望前辈们多多指点。
  1. #!/usr/bin/perl

  2. use strict;
  3. #use warnings;
  4. use Win32::GUI qw( MB_ICONINFORMATION MB_OK );
  5. use POSIX qw/ _exit /;
  6. use threads;
  7. use threads::shared;
  8. use Thread::Queue;
  9. use Net::Time qw(inet_time);

  10. my $my_name = "NTP Time Synchronizer V2.0";
  11. my $about= <<EOF;
  12. /`\\ /`\\
  13. (/\\ V /\\)        $my_name
  14.   /6 6\\          -- Written by Limeng o_*, May 8, 2013
  15. >{= Y =}<
  16. /'-^-'\\         All Rights Reserved 2010-3000 ChangSha
  17. (_)""-(_)        Homepage: http://limeng.class22.net
  18.                  E-mail: iamlimeng\@163.com
  19. BlueIdea!

  20. 本程序用于从国际授时服务器获得时间,并校准系统时间!
  21. EOF

  22. my @time_sever = (
  23. 'time-a.nist.gov',
  24. 'time-b.nist.gov',
  25. 'utcnist.colorado.edu',
  26. 'time-nw.nist.gov',
  27. 'nist1-dc.glassey.com',
  28. 'nist1-ny.glassey.com',
  29. 'nist1-sj.glassey.com',
  30. 'time.nist.gov',
  31. 'salmon.maths.tcd.ie',
  32. 'nist1.symmetricom.com',
  33. '210.72.145.44',
  34. '133.100.11.8',
  35. 'time-a.timefreq.bldrdoc.gov',
  36. 'time-b.timefreq.bldrdoc.gov',
  37. 'time-c.timefreq.bldrdoc.gov',
  38. 'nist1.aol-ca.truetime.com',
  39. 'nist1.aol-va.truetime.com',
  40. );

  41. my $data_queue = new Thread::Queue;
  42. my $result_queue = new Thread::Queue;
  43. my $MAX_THREADS = 5;
  44. my $processing_count :shared = 0;
  45. my %checked :shared = ();
  46. my $net_time :shared = '';
  47. my $thread;

  48. my $w_m = 360;
  49. my $h_m = 240;
  50. my $icon = new Win32::GUI::Icon("icon.ico");
  51. my $wm_class = new Win32::GUI::Class(
  52.         -name => 'Limeng',
  53.         -icon => $icon,
  54. );

  55. my $font1 = Win32::GUI::Font->new(
  56.          -name => "Arial",
  57.          -size => 14,
  58. );

  59. my $font2 = Win32::GUI::Font->new(
  60.          -name => "宋体",
  61.          -size => 11,
  62. );

  63. my $main = Win32::GUI::Window->new(
  64.         -title => " $my_name",
  65.         -class => $wm_class,
  66.         -size  => [$w_m,$h_m],
  67.         -maximizebox        => 0,
  68.         -dialogui        => 1,
  69.         -noflicker        => 1,
  70.         -resizable        => 0,
  71. );

  72. $main->AddLabel(
  73.         -name => 'Local',
  74.         -font => $font1,
  75.         -pos  => [$w_m-315,50],
  76.         -size => [$w_m,20],
  77.         -text => "",
  78. );

  79. $main->AddLabel(
  80.         -name => 'Net',
  81.         -font => $font1,
  82.         -pos  => [$w_m-315,100],
  83.         -size => [$w_m,20],
  84.         -text => "",
  85. );

  86. $main->AddButton(
  87.         -name => 'B1',
  88.         -pos  => [$w_m-300,$h_m-70],
  89.         -font => $font2,
  90.         -text => " 校准时间 ",
  91.         -default => 1,
  92.         -ok      => 1,
  93.         -visible => 1,
  94.         -onClick => \&doAdjust,
  95. );

  96. $main->AddButton(
  97.         -name => 'B2',
  98.         -pos  => [$w_m-150,$h_m-70],
  99.         -text => " 关于程序 ",
  100.         -font => $font2,
  101.         -default => 1,
  102.         -ok      => 1,
  103.         -visible => 1,
  104.         -onClick => \&aboutMe,
  105. );
  106. my $timer1 = $main->AddTimer('T1',1000);
  107. $main->B1->Disable();
  108. $main->Center();
  109. $main->Show();
  110. &ObtainNetTime;
  111. Win32::GUI::Dialog();
  112. $main->Hide();
  113. undef $main;
  114. POSIX::_exit(0);

  115. sub Window_Terminate {
  116.         return -1;
  117. }

  118. sub doAdjust {
  119.         $thread = threads->new(\&Adjust);
  120.         $thread->detach();
  121.         return 0;
  122. }

  123. sub aboutMe {
  124.         my $self = shift;
  125.         $self->MessageBox("$about","关于本程序...",MB_ICONINFORMATION | MB_OK,);
  126.         return 0;
  127. }

  128. sub T1_Timer {
  129.         my $now = gettime();
  130.         $main->Local->Change(-text => "本机时间: $now");
  131. }

  132. sub T2_Timer {
  133.         $net_time++;
  134.         my $now = time_from_utc($net_time);
  135.         $main->Net->Change(-text => "网络时间: $now");
  136. }

  137. sub Adjust {
  138.         my ($date,$time) = split(/\s+/,time_from_utc($net_time));
  139.         my $result1 = system("date $date");
  140.         my $result2 = system("time $time");
  141.         if ($result1 or $result2) { $main->MessageBox("本机时间校准失败, 请以管理员身份运行本程序并重试!","本机时间校准失败!",MB_ICONINFORMATION | MB_OK,); }
  142.         else { $main->MessageBox("本机时间校准成功!","本机时间校准成功!",MB_ICONINFORMATION | MB_OK,); }
  143. }

  144. sub gettime {
  145.         my ($sec,$min,$hour,$day,$mon,$year,$weekday,$yeardate,$savinglightday) = (localtime(time));
  146.         $sec = ($sec < 10)? "0$sec":$sec;
  147.         $min = ($min < 10)? "0$min":$min;
  148.         $hour = ($hour < 10)? "0$hour":$hour;
  149.         $day = ($day < 10)? "0$day":$day;
  150.         $year += 1900;
  151.         $mon = ($mon < 9)? "0".($mon+1):($mon+1);
  152.         return("$year/$mon/$day $hour:$min:$sec");
  153. }

  154. sub ObtainNetTime {
  155.         $main->Net->Change(-text => "连接国际授时服务器...");
  156.         my $total = @time_sever;
  157.         my $sucess = 0;
  158.         for (my $n = 0; $n < $MAX_THREADS; $n++) {
  159.                 threads->create(\&thread_io);
  160.         }

  161.         foreach my $time_sever(@time_sever) {
  162.                  if ($data_queue ->pending() > $MAX_THREADS * 20) {
  163.                           select(undef, undef, undef, 0.02);
  164.                           redo;
  165.                  }
  166.                  $data_queue->enqueue("$time_sever");

  167.                  if ($result_queue->pending() > 0) {
  168.                           while (my $result = $result_queue->dequeue_nb()) {
  169.                                    if ($result) { $sucess++; }
  170.                           }
  171.                  }
  172.         }

  173.         my $times = 1;
  174.         while ($processing_count > 0 or $data_queue->pending() > 0 or $result_queue->pending() > 0) {
  175.                  select(undef, undef, undef, 0.02);
  176.                  while (my $result = $result_queue->dequeue_nb()) {
  177.                                    if ($result) { $sucess++; }
  178.                  }
  179.                  last if ($times > 300 or $sucess >= $total or $sucess >= 3);
  180.                  $times++;
  181.         }

  182.         foreach my $thread (threads->list(threads::all)) {
  183.                 $thread->detach();
  184.         }

  185.         # Check completion and recheck
  186.         if ($sucess < 3) {
  187.                  my @recheck;
  188.                  foreach (@time_sever) {
  189.                           if (!$checked{$_}) { push(@recheck,"$_"); }
  190.                  }
  191.                  if (@recheck) {
  192.                           foreach my $time_sever (@recheck) {
  193.                                    next if ($checked{$time_sever});
  194.                                    my $result = obtain_time($time_sever);
  195.                                    if ($result) { $sucess++; }
  196.                           }
  197.                  }
  198.         }

  199.         if ($sucess >= 3) {
  200.                  my @utc = sort values %checked;
  201.                  for (0..$#utc) {
  202.                          if (abs($utc[$_+1] - $utc[$_]) <= 10 or abs($utc[$_-1] - $utc[$_]) <= 10) {
  203.                                  $net_time = $utc[$_];
  204.                          }
  205.                  }
  206.         }
  207.          if ($net_time) {
  208.                  $main->B1->Enable();
  209.                 my $timer2 = $main->AddTimer('T2',1000);
  210.          }
  211.          else {
  212.                  $main->MessageBox("连接国际授时服务器失败, 请检查网络并重试!","连接国际授时服务器失败!",MB_ICONINFORMATION | MB_OK,);
  213.                  $main->Hide();
  214.                 undef $main;
  215.                 POSIX::_exit(0);
  216.          }
  217.          return(0);
  218. }

  219. sub thread_io()
  220. {
  221.         while (my $data = $data_queue->dequeue())
  222.         {
  223.                 {
  224.                         lock $processing_count;
  225.                         ++$processing_count;
  226.                 }

  227.                 my $result = obtain_time($data);
  228.                 $result_queue->enqueue($result);
  229.                 {
  230.                         lock $processing_count;
  231.                         --$processing_count;
  232.                 }
  233.         }
  234. }

  235. sub obtain_time {
  236.         my $time_sever = shift;
  237.         my $time = inet_time($time_sever,'udp',3);
  238.          if ($time) {
  239.                  $checked{$time_sever} = $time;
  240.                  return(1);
  241.          }
  242.          else { return(0); }
  243. }

  244. sub time_from_utc {
  245.         my $utc = shift;
  246.         my ($sec,$min,$hour,$day,$mon,$year,$weekday,$yeardate,$savinglightday) = (localtime($utc));
  247.         $sec = ($sec < 10)? "0$sec":$sec;
  248.         $min = ($min < 10)? "0$min":$min;
  249.         $hour = ($hour < 10)? "0$hour":$hour;
  250.         $day = ($day < 10)? "0$day":$day;
  251.          $mon = ($mon < 9)? "0".($mon+1):($mon+1);
  252.         $year += 1900;
  253.         return("$year/$mon/$day $hour:$min:$sec");
  254. }
复制代码
工作原理:用多线程从多个授时服务器获取时间,成功后,可以用授时服务器时间来同步本机时间。

打包后的程序如下:
NetTimeSynchronizer.rar (1.31 MB, 下载次数: 612)

同时,也希望大家多分享代码,以资共同提高。



作者: nixiaoweihunter    时间: 2013-05-08 16:37
楼主是做什么工作的,经常写这种程序?
作者: iakuf    时间: 2013-05-09 10:06
多谢分享,没怎么写过 GUI 但看起来很有意思
作者: iamlimeng    时间: 2013-05-09 10:09
回复 2# nixiaoweihunter


  Perl是业余爱好,没事写几行,解决一些日常小问题,顺便防止老年痴呆。   
作者: wxlfh    时间: 2013-05-09 10:56
回复 4# iamlimeng


    防止老年痴呆的这个主意不错。
作者: hkkkyy    时间: 2013-05-09 11:48
运行不了啊,一运行,Perl就停止工作了,window7 activePerl
作者: iamlimeng    时间: 2013-05-09 11:52
回复 6# hkkkyy


    跟我的运行环境一样啊,win7+activeperl 5.10.1008,请检查一下,代码所需的的模块都已经安装。
作者: hkkkyy    时间: 2013-05-09 11:55
我是5.16,包都装上了,难道因为是x64.回复 7# iamlimeng


   
作者: iamlimeng    时间: 2013-05-09 12:01
回复 8# hkkkyy


   我是x86,代码本身没问题,请调试一下,看问题在哪里。
作者: raoweijian    时间: 2013-05-09 16:15
可以运行嘿嘿
作者: rubyish    时间: 2013-05-10 02:34
多谢分享,没Win
但看起来很有意思
谢谢
作者: capfsxl    时间: 2013-09-28 17:15
回复 1# iamlimeng

能否消息于我你的微信或手机号,以便酬谢!
   
作者: dsadjkasd    时间: 2013-09-28 22:32
写的不错,但提个小建议,就是获取本机时间那个补0的写的太烦,用sprintf就可以很简单处理,简单的修改了一下gettime的sub。
  1. sub gettime {
  2.         my ($sec,$min,$hour,$day,$mon,$year,$weekday,$yeardate,$savinglightday) = (localtime(time));
  3.         $year += 1900;
  4.         $mon++;
  5.         return sprintf("%04s/%02s/%02s %02s:%02s:%02s",$year,$mon,$day,$hour,$min,$sec);
  6. }
复制代码

作者: iamlimeng    时间: 2013-09-28 22:40
回复 13# dsadjkasd

这是好多年前写的代码,运行没问题,没想过去改进。太好了,非常感谢你的建议!
作者: dsadjkasd    时间: 2013-09-28 22:46
回复 14# iamlimeng
感觉你一直对perl的论坛很有贡献,win32gui的文档很少,只能拿写范例来学习,我也写过一些供大家参考。你是做什么工作的?我平时工作一直写些perl,处理大量的数据,只不过感觉PERL今非昔比了。

   
作者: iamlimeng    时间: 2013-09-28 22:54
回复 15# dsadjkasd


    你高抬我了。工作跟计算机无关,纯业余爱好。偶尔要分析点数据,用Perl也是分分钟搞定,Perl是一匹好骆驼,会一直爱下去,希望多交流。
作者: sinian126    时间: 2013-09-30 09:00
提示: 作者被禁止或删除 内容自动屏蔽




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2