Python語(yǔ)言編寫電腦時(shí)間自動(dòng)同步小工具
話不多說(shuō),小工具需求如下:
功能需求 -- 電腦開機(jī)后自動(dòng)執(zhí)行時(shí)間同步
非功能需求 -- 安裝執(zhí)行簡(jiǎn)單,無(wú)需安裝額外環(huán)境
一、代碼實(shí)現(xiàn)
基于以上需求,思路如下:訪問網(wǎng)絡(luò)獲取北京時(shí)間,然后調(diào)用命令行來(lái)設(shè)置系統(tǒng)時(shí)間。程序?qū)懗蒞indows Service,并設(shè)置為開機(jī)自動(dòng)運(yùn)行。正好前段時(shí)間在學(xué)習(xí)Python,所以打算用Python來(lái)寫這個(gè)工具。具體代碼如下:
獲取網(wǎng)絡(luò)時(shí)間
def getBeijinTime():
"""
獲取北京時(shí)間
"""
try:
conn = httplib.HTTPConnection("www.beijing-time.org")
conn.request("GET", "/time.asp")
response = conn.getresponse()
print response.status, response.reason
if response.status == 200:
#解析響應(yīng)的消息
result = response.read()
logging.debug(result)
data = result.split("\r\n")
year = data[1][len("nyear")+1 : len(data[1])-1]
month = data[2][len("nmonth")+1 : len(data[2])-1]
day = data[3][len("nday")+1 : len(data[3])-1]
#wday = data[4][len("nwday")+1 : len(data[4])-1]
hrs = data[5][len("nhrs")+1 : len(data[5])-1]
minute = data[6][len("nmin")+1 : len(data[6])-1]
sec = data[7][len("nsec")+1 : len(data[7])-1]
beijinTimeStr = "%s/%s/%s %s:%s:%s" % (year, month, day, hrs, minute, sec)
beijinTime = time.strptime(beijinTimeStr, "%Y/%m/%d %X")
return beijinTime
except:
logging.exception("getBeijinTime except")
return None
同步本地系統(tǒng)時(shí)間
def syncLocalTime():
"""
同步本地時(shí)間
"""
logging.info("current local time is: %d-%d-%d %d:%d:%d" % time.localtime()[:6])
beijinTime = getBeijinTime()
if beijinTime is None:
logging.info("get beijinTime is None, will try again in 30 seconds...")
timer = threading.Timer(30.0, syncLocalTime)
timer.start();
else:
logging.info("get beijinTime is: %d-%d-%d %d:%d:%d" % beijinTime[:6])
tm_year, tm_mon, tm_mday, tm_hour, tm_min, tm_sec = beijinTime[:6]
import os
os.system("date %d-%d-%d" % (tm_year, tm_mon, tm_mday)) #設(shè)置日期
os.system("time %d:%d:%d.0" % (tm_hour, tm_min, tm_sec)) #設(shè)置時(shí)間
logging.info("syncLocalTime complete, current local time: %d-%d-%d %d:%d:%d \n" % time.localtime()[:6])
二、部署安裝
為了讓Python程序能以Windows服務(wù)的方式運(yùn)行,需要用到py2exe(用來(lái)把Python程序編譯成exe)和Python Win32 Extensions 。(py2exe把Python代碼編譯成Winodws服務(wù)時(shí)依賴此組件)下載并安裝這兩個(gè)組件。安裝完畢后,在Python的安裝目錄下找到py2exe的Windows Service示例({PythonRoot}\Lib\site-packages\py2exe\samples\advanced\MyService.py)。然后仿照這個(gè)示例將上面的代碼完善一下。
Windows服務(wù)示例
import win32serviceutil
import win32service
import win32event
import win32evtlogutil
class SynctimeService(win32serviceutil.ServiceFramework):
_svc_name_ = "Synctime"
_svc_display_name_ = "Synctime"
_svc_description_ = "Synchronize local system time with beijin time"
_svc_deps_ = ["EventLog"]
def __init__(self, args):
win32serviceutil.ServiceFramework.__init__(self, args)
self.hWaitStop = win32event.CreateEvent(None, 0, 0, None)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
win32event.SetEvent(self.hWaitStop)
def SvcDoRun(self):
import servicemanager
# Write a 'started' event to the event log...
win32evtlogutil.ReportEvent(self._svc_name_,
servicemanager.PYS_SERVICE_STARTED,
0, # category
servicemanager.EVENTLOG_INFORMATION_TYPE,
(self._svc_name_, ''))
# wait for beeing stopped...
win32event.WaitForSingleObject(self.hWaitStop, win32event.INFINITE)
# and write a 'stopped' event to the event log.
win32evtlogutil.ReportEvent(self._svc_name_,
servicemanager.PYS_SERVICE_STOPPED,
0, # category
servicemanager.EVENTLOG_INFORMATION_TYPE,
(self._svc_name_, ''))
if __name__ == '__main__':
# Note that this code will not be run in the 'frozen' exe-file!!!
win32serviceutil.HandleCommandLine(SynctimeService)
之后,再編寫一個(gè)steup.py文件用來(lái)生成安裝文件。
Setup.py
from distutils.core import setup
import py2exe
setup(
# The first three parameters are not required, if at least a
# 'version' is given, then a versioninfo resource is built from
# them and added to the executables.
version = "0.0.1",
description = "Synchroniz local system time with beijin time",
name = "sysctime",
# targets to build
# console = ["synctime.py"],
service=["synctime"]
)
編譯生成windows程序,如下圖:

然后在控制臺(tái)中運(yùn)行:setup.py py2exe ,一切順利的話會(huì)在當(dāng)前目錄下生成build和dist目錄。

控制臺(tái)目錄切換到dist目錄,找到synctime.exe,在命令行中運(yùn)行:
synctime.exe –install (-remove) 安裝或移除時(shí)間同步服務(wù)。
現(xiàn)在可以運(yùn)行services.msc查看服務(wù)運(yùn)行情況
可以看到服務(wù)并沒有啟動(dòng),而且啟動(dòng)方式為手動(dòng)。在這里可以右擊服務(wù)選擇屬性手動(dòng)把服務(wù)啟動(dòng)起來(lái),并且設(shè)置為服務(wù)自動(dòng)啟動(dòng)。
好吧,我承認(rèn)。這樣操作跟上面的需求有點(diǎn)出入了,略顯麻煩。為了解決這個(gè)問題,自然想到的是用批處理來(lái)做。在dist目錄下分別建兩個(gè)批處理文件:
installservice.bat
@echo off
:: 安裝windows服務(wù)
echo 正在安裝服務(wù),請(qǐng)稍候...
synctime.exe -install
:: 設(shè)置服務(wù)自動(dòng)啟動(dòng)
echo 正在啟動(dòng)服務(wù)...
sc config Synctime start= AUTO
:: 啟動(dòng)服務(wù)
sc start Synctime
echo 服務(wù)啟動(dòng)成功, 按任意鍵繼續(xù)...
pause
removeserivce.bat
@echo off
:: 停止服務(wù)
echo 正在停止服務(wù),請(qǐng)稍候...
sc stop Synctime
echo 正在卸載服務(wù)...
:: 刪除windows服務(wù)
synctime.exe -remove
echo 服務(wù)卸載完成,請(qǐng)按任意鍵繼續(xù)剩余卸載...
pause
三、最終安裝效果圖


四、結(jié)尾
遺留的問題:
1、從上面的截圖可以看到,安裝程序在調(diào)用批處理時(shí)會(huì)顯示出控制臺(tái)窗口。這個(gè)問題我在網(wǎng)上查找資料,NSIS有相關(guān)的插件可以隱藏控制臺(tái)窗口調(diào)用bat文件。
2、我源代碼中有寫日志文件的操作,但是以Windows服務(wù)的方式運(yùn)行后,日志文件不能寫了,不知道有沒有好的解決辦法。
3、360 ...真是要人命啊....Orz..
時(shí)間同步工具及源代碼://www.dhdzp.com/softs/74865.html
編譯方法:
第一步: 安裝Python環(huán)境(什么?還沒有Python環(huán)境?... - -!)
第二步: 安裝依賴組件
1、py2exe (目前只支持到python2.7)
2、Python Win32 Extensions
第三步(可選):安裝Nsis環(huán)境,用來(lái)編譯腳本
第四步:將synctime.py編譯成windows程序
1、在當(dāng)前目錄下運(yùn)行"setup.py py2exe",順利的話會(huì)在當(dāng)前目錄下生成dist和build目錄
第五步: 運(yùn)行,有兩種運(yùn)行方式:
1、將installservice.bat 和 removeservice.bat 拷貝到dist中運(yùn)行即可
2(依賴第三步)、使用Nsis編譯Synctime.nsi腳本,生成安裝包, 安裝后運(yùn)行(推薦)
相關(guān)文章
Python 發(fā)送SMTP郵件的簡(jiǎn)單教程
SMTP(Simple Mail Transfer Protocol)簡(jiǎn)單郵件傳輸協(xié)議,Python內(nèi)置對(duì)SMTP的支持,可以發(fā)送純文本文件,HTML郵件以及附帶文件。本文講解如何使用python發(fā)送郵件2021-06-06
基于Python socket的端口掃描程序?qū)嵗a
這篇文章主要介紹了基于Python socket的端口掃描程序?qū)嵗a,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02
windows10安裝python依賴報(bào)錯(cuò)can‘t?create?or?remove?files?in?i
這篇文章主要介紹了windows10安裝python依賴報(bào)錯(cuò)can‘t?create?or?remove?files?in?install?directory問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2023-09-09
django數(shù)據(jù)庫(kù)migrate失敗的解決方法解析
這篇文章主要介紹了django數(shù)據(jù)庫(kù)migrate失敗的解決方法解析,分享了相關(guān)代碼示例,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-02-02
如何獲取Python簡(jiǎn)單for循環(huán)索引
這篇文章主要介紹了如何獲取Python簡(jiǎn)單for循環(huán)索引,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
解決Pycharm中import時(shí)無(wú)法識(shí)別自己寫的程序方法
今天小編就為大家分享一篇解決Pycharm中import時(shí)無(wú)法識(shí)別自己寫的程序方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2018-05-05
Python爬蟲圖片懶加載技術(shù) selenium和PhantomJS解析
這篇文章主要介紹了Python爬蟲圖片懶加載技術(shù) selenium和PhantomJS解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09
基于python實(shí)現(xiàn)Pycharm斷點(diǎn)調(diào)試
這篇文章主要介紹了基于python實(shí)現(xiàn)Pycharm斷點(diǎn)調(diào)試,在我們寫程序的時(shí)候,很容易遇到各種各樣的bug,然后編譯器提示程序出錯(cuò)的地方。很多時(shí)候可以通過提示的信息修改程序,但是有時(shí)我們想得到更多的信息,這個(gè)時(shí)候就需要進(jìn)行斷點(diǎn)調(diào)試,下面我們就一起來(lái)學(xué)習(xí)ycharm斷點(diǎn)調(diào)試2022-02-02

