现在的位置: 首页 > 综合 > 正文

设置系统时间

2018年07月10日 ⁄ 综合 ⁄ 共 4813字 ⁄ 字号 评论关闭

标准C库中只有获取系统时间的API,好像还没有设置系统时间的API,本文将谈谈如何在linux和windows平台设置系统时间,最后给出一个与平台无关的设置系统时间的封闭函数。

<!--[if !supportLists]-->一、<!--[endif]-->Linux下设置系统时间:

1.Linux下设置系统时间的函数有好几个,先来看看最常用的stime()函数,这个函数只能精确到秒。

#define _SVID_SOURCE /*如果你使用的是glib2的话,必须先定义这个宏才能使用*/

#include <time.h>

int stime(time_t *t);

参数说明:

t是以秒为单位的时间值,从GMT1970年1月1日0时0分0秒开始计算。

返回值:

成功返回0,错误返回-1,errno错误码,EFAULT表示传递的参数错误,如时间值是无效的值,EPERM表示权限不够,注意只有root用户才有修改系统时间的权限。如果要让普通程序修改系统时间,可以先切换到root用户操作,修改完成后,再切换到普通用户,或者用命令chmod +s给执行文件加上root用户的权限。

2.linux是如何管理时间的?

在系统启动时,Linux操作系统将时间从CMOS中读到系统时间变量中,以后修改时间通过修改系统时间实现。为了保持系统时间与CMOS时间的一致性,Linux每隔11分钟会将系统时间写入CMOS,同步时间。从这可以看出,获取系统时间有两个途径,一种是从CMOS中读,一种是从系统中读,但修改时间却只有一种,即修改linux系统中的时间,而修改CMOS中的时间是无效的,因为CMOS中的时间会被定时重写掉。另外还有一点要注意,修改了系统时间并不是马上生效的,假如你修改了系统时间并马上关机,再开机的时候,时间还是原来的,因为修改的时间还没有来得及写入CMOS中。

3.通过settimeofday()函数来设置系统时间,这个函数设置的精度可以精确到微秒。

#include <sys/time.h>

int settimeofday(const struct timeval *tv , const struct timezone *tz);

struct timeval {

    time_t      tv_sec;     /* seconds */

    suseconds_t tv_usec;    /* microseconds */

};

struct timezone {

    int tz_minuteswest;     /* minutes west of Greenwich */

    int tz_dsttime;         /* type of DST correction */

};

tz参数为时区,时区结构中tz_dsttime在linux中不支持,应该置为0,通常将参数tz设置为NULL,表示使用当前系统的时区。该函数是glib中的,但在mingw中没有实现。

该函数返回值与stime()一样,同样也需要root权限。

4.设置CMOS时间,其实它是通过RTC(Real-time clock)设备驱动来完成的,你可以用ioctl()函数来设置时间,当然也可以通过操作/dev/rtc设备文件,在此就不详细说明了。

二、windows下设置系统时间

1.设置当前时区的时间

#include <winbase.h>

BOOL SetLocalTime(const SYSTEMTIME* lpSystemTime);

typedef struct _SYSTEMTIME {  // st 

    WORD wYear;

    WORD wMonth; //月份从1开始

    WORD wDayOfWeek; //SetLocalTime()不使用这个参数

    WORD wDay;

    WORD wHour;

    WORD wMinute;

    WORD wSecond;

    WORD wMilliseconds;

} SYSTEMTIME;

函数成功返回非零,失败返回零。注意要求调用进程必需有SE_SYSTEMTIME_NAME权限。

2.另外还有一个函数SetSystemTime(),它的参数与SetLocalTime一样,只不过以UTC时区为基准的。

BOOL SetSystemTime(const SYSTEMTIME* lpSystemTime);

<!--[if !supportLists]-->二、<!--[endif]-->一个封装的设置系统时间的函数

//设置成功返回true,否则返回false

       bool set_local_time(struct tm& t)

{

#ifdef _WIN32

       SYSTEMTIME st;

       memset(&st, 0, sizeof(SYSTEMTIME));

       st.wYear = t.tm_year + 1970; //注意struct tm结构中的年是从1970年开始的计数

       st.wMonth = t.tm_mon + 1; //注意struct tm结构中的月份是从0开始的

       st.wDay = t.tm_mday;

       st.wHour = t.tm_hour;

       st.wMinute = t.tm_min;

       st.wSecond = t.tm_sec;

       if(!SetLocalTime(&st))

              return true;

              else

                     return false;

       #else

              //将struct tm结构时间转换成GMT时间time_t

              struct time_t st;

              st = mktime(&t);

              if(st==-1)

                     return false;

              if(!stime(st))

                     return true;

              else

                     return false;

#endif

}

 

 

 

头文件:time.h  

函数原型:time_t time(time_t * timer)  

功 能: 获取当前的系统时间,返回的结果是一个time_t类型(即int64类型),其实就是一个大整数,其值表示从CUT(Coordinated Universal Time)时间1970年1月1日00:00:00(称为UNIX系统的Epoch时间)到当前时刻的秒数。

可以通过调用localtime将time_t所表示的CUT时间转换为本地时间(我们是+8区,比CUT多8个小时)并转成struct tm类型,该类型的各数据成员分别表示年月日时分秒。

struct tm的结构为:

struct tm {
        int tm_sec;     /* seconds after the minute - [0,61] */
        int tm_min;     /* minutes after the hour - [0,59] */
        int tm_hour;    /* hours since midnight - [0,23] */
        int tm_mday;    /* day of the month - [1,31] */
        int tm_mon;     /* months since January - [0,11] */
        int tm_year;    /* years since 1900 */
        int tm_wday;    /* days since Sunday - [0,6] */
        int tm_yday;    /* days since January 1 - [0,365] */
        int tm_isdst;   /* daylight savings time flag */
        };

在Win32中有SYSTEMTIME类数据结构。  

SYSTEMTIME结构定义如下:  

SYSTEMTIME STRUCT{  

WORD wYear ;  // 年  

WORD wMonth ;  // 月  

WORD wDayOfWeek ;  //  星期,0=星期日,1=星期一...  

WORD wDay ; // 日  

WORD wHour ; // 时  

WORD wMinute ; // 分  

WORD wSecond ; // 秒  

WORD wMilliseconds ; // 毫秒  

};

/*
** SYSTEMTIME转time_t
*/

time_t systime_to_timet(const SYSTEMTIME& st)
{
    struct tm gm = {st.wSecond, st.wMinute, st.wHour, st.wDay, st.wMonth-1, st.wYear-1900, st.wDayOfWeek, 0, 0};
    return mktime(&gm);
}

由上可以看出struct tm结构和struct SYSTEMTIME结构的年和月的取值范围是不一样的:

tm.tm_mon = systemtime.wMonth - 1

tm.tm_year = systemtime.wYear - 1900

/*

**time_t转SYSTEMTIME

*/

SYSTEMTIME Time_tToSystemTime(time_t t)

{

    tm temptm = *localtime(&t);

    SYSTEMTIME st = {1900 + temptm.tm_year, 

                                   1 + temptm.tm_mon, 

                                   temptm.tm_wday, 

                                   temptm.tm_mday, 

                                   temptm.tm_hour, 

                                   temptm.tm_min, 

                                   temptm.tm_sec, 

                                   0};

    return st;

}

还有一种是通过struct FILETIME作为中间量来转换time_t和systemtime

/*

**time_t转SYSTEMTIME

*/

SYSTEMTIME TimetToSystemTime(time_t t)

{

    FILETIME ft;

    SYSTEMTIME pst;

    LONGLONG nLL = Int32x32To64(t, 10000000) + 116444736000000000;

    ft.dwLowDateTime = (DWORD)nLL;

    ft.dwHighDateTime = (DWORD)(nLL >> 32);

    FileTimeToSystemTime(&ft, &pst);

    return pst;

}

/*

**SYSTEMTIME转time_t

*/

time_t SystemTimeToTimet(SYSTEMTIME st)

{

    FILETIME ft;

    SystemTimeToFileTime( &st, &ft );

    LONGLONG nLL;

    ULARGE_INTEGER ui;

    ui.LowPart = ft.dwLowDateTime;

    ui.HighPart = ft.dwHighDateTime;

    nLL = (ft.dwHighDateTime << 32) + ft.dwLowDateTime;

    time_t pt = (long)((LONGLONG)(ui.QuadPart - 116444736000000000) / 10000000);

    return pt;

}

抱歉!评论已关闭.