以编程方式更改系统日期
Posted
技术标签:
【中文标题】以编程方式更改系统日期【英文标题】:Change system date programmatically 【发布时间】:2010-10-13 15:28:58 【问题描述】:如何使用 C# 以编程方式更改本地系统的日期和时间?
【问题讨论】:
有趣的是,像这样的一些问题是如何得到回答的,而其他问题却是“你试过什么?” ...奇怪... 【参考方案1】:Here is where I found the answer.;我已在此处重新发布以提高清晰度。
定义这个结构:
[StructLayout(LayoutKind.Sequential)]
public struct SYSTEMTIME
public short wYear;
public short wMonth;
public short wDayOfWeek;
public short wDay;
public short wHour;
public short wMinute;
public short wSecond;
public short wMilliseconds;
将以下extern
方法添加到您的类中:
[DllImport("kernel32.dll", SetLastError = true)]
public static extern bool SetSystemTime(ref SYSTEMTIME st);
然后用你的结构实例调用方法,如下所示:
SYSTEMTIME st = new SYSTEMTIME();
st.wYear = 2009; // must be short
st.wMonth = 1;
st.wDay = 1;
st.wHour = 0;
st.wMinute = 0;
st.wSecond = 0;
SetSystemTime(ref st); // invoke this method.
【讨论】:
编写自定义 C++/CLI 包装器并引入另一个程序集比编写 ~9 行结构更容易?? 只是不要让 Marc Gravell 看到你的结构! ;-) 由于某种原因,我对答案的编辑被拒绝了,但至少对于 Win 7,我发现我需要以管理员身份运行程序才能使其正常工作。见:***.com/questions/2818179/… 可以说这个方法设置了UTC时间。因此,如果您通过 Datetime.Now 获取本地时间,那么它将设置错误的时间。遇到这个,搞了半天不明白是什么问题…… 值得一提的是,该程序需要管理员权限才能运行...【参考方案2】:这里已经有很多很棒的观点和方法,但是这里有一些目前被遗漏的规范,我觉得它们可能会让一些人感到困惑。
-
在 Windows Vista、7、8 操作系统上,这将需要一个 UAC 提示,以获得成功执行
SetSystemTime
函数所需的管理权限。原因是调用进程需要SE_SYSTEMTIME_NAME权限。
SetSystemTime
函数在协调世界时 (UTC) 中期待 SYSTEMTIME
结构。否则它将无法按预期工作。
根据您获取DateTime
值的位置/方式,在设置SYSTEMTIME
结构中的相应值之前,最好谨慎行事并使用ToUniversalTime()
。
代码示例:
DateTime tempDateTime = GetDateTimeFromSomeService();
DateTime dateTime = tempDateTime.ToUniversalTime();
SYSTEMTIME st = new SYSTEMTIME();
// All of these must be short
st.wYear = (short)dateTime.Year;
st.wMonth = (short)dateTime.Month;
st.wDay = (short)dateTime.Day;
st.wHour = (short)dateTime.Hour;
st.wMinute = (short)dateTime.Minute;
st.wSecond = (short)dateTime.Second;
// invoke the SetSystemTime method now
SetSystemTime(ref st);
【讨论】:
我不能使用这个直接改变系统时间 我已经在多个项目中成功使用了这段代码。您是否以管理员身份运行可执行文件?否则这段代码肯定行不通。 哇,这个解决了我的问题。问题是您当地时间的时区妨碍了获取正确时间,因此“DateTime dateTime = tempDateTime.ToUniversalTime();”行解决了一切。【参考方案3】:您可以使用对 DOS 命令的调用,但在 windows dll 中调用该函数是一种更好的方法。
public struct SystemTime
public ushort Year;
public ushort Month;
public ushort DayOfWeek;
public ushort Day;
public ushort Hour;
public ushort Minute;
public ushort Second;
public ushort Millisecond;
;
[DllImport("kernel32.dll", EntryPoint = "GetSystemTime", SetLastError = true)]
public extern static void Win32GetSystemTime(ref SystemTime sysTime);
[DllImport("kernel32.dll", EntryPoint = "SetSystemTime", SetLastError = true)]
public extern static bool Win32SetSystemTime(ref SystemTime sysTime);
private void button1_Click(object sender, EventArgs e)
// Set system date and time
SystemTime updatedTime = new SystemTime();
updatedTime.Year = (ushort)2009;
updatedTime.Month = (ushort)3;
updatedTime.Day = (ushort)16;
updatedTime.Hour = (ushort)10;
updatedTime.Minute = (ushort)0;
updatedTime.Second = (ushort)0;
// Call the unmanaged function that sets the new date and time instantly
Win32SetSystemTime(ref updatedTime);
【讨论】:
【参考方案4】:使用此功能更改系统时间(在window 8中测试)
void setDate(string dateInYourSystemFormat)
var proc = new System.Diagnostics.ProcessStartInfo();
proc.UseShellExecute = true;
proc.WorkingDirectory = @"C:\Windows\System32";
proc.CreateNoWindow = true;
proc.FileName = @"C:\Windows\System32\cmd.exe";
proc.Verb = "runas";
proc.Arguments = "/C date " + dateInYourSystemFormat;
try
System.Diagnostics.Process.Start(proc);
catch
MessageBox.Show("Error to change time of your system");
Application.ExitThread();
void setTime(string timeInYourSystemFormat)
var proc = new System.Diagnostics.ProcessStartInfo();
proc.UseShellExecute = true;
proc.WorkingDirectory = @"C:\Windows\System32";
proc.CreateNoWindow = true;
proc.FileName = @"C:\Windows\System32\cmd.exe";
proc.Verb = "runas";
proc.Arguments = "/C time " + timeInYourSystemFormat;
try
System.Diagnostics.Process.Start(proc);
catch
MessageBox.Show("Error to change time of your system");
Application.ExitThread();
示例: 调用表单的加载方法 setDate("5-6-92"); setTime("2:4:5 AM");
【讨论】:
【参考方案5】:-
PInvoke 调用 Win32 API SetSystemTime,(example)
System.Management 类与 WMI 类 Win32_OperatingSystem 并在该类上调用 SetDateTime。
两者都要求调用者已被授予 SeSystemTimePrivilege 并启用此权限。
【讨论】:
【参考方案6】:由于我在评论中提到过,这里是一个 C++/CLI 包装器:
#include <windows.h>
namespace JDanielSmith
public ref class Utilities abstract sealed /* abstract sealed = static */
public:
CA_SUPPRESS_MESSAGE("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands")
static void SetSystemTime(System::DateTime dateTime)
LARGE_INTEGER largeInteger;
largeInteger.QuadPart = dateTime.ToFileTimeUtc(); // "If your compiler has built-in support for 64-bit integers, use the QuadPart member to store the 64-bit integer."
FILETIME fileTime; // "...copy the LowPart and HighPart members [of LARGE_INTEGER] into the FILETIME structure."
fileTime.dwHighDateTime = largeInteger.HighPart;
fileTime.dwLowDateTime = largeInteger.LowPart;
SYSTEMTIME systemTime;
if (FileTimeToSystemTime(&fileTime, &systemTime))
if (::SetSystemTime(&systemTime))
return;
HRESULT hr = HRESULT_FROM_WIN32(GetLastError());
throw System::Runtime::InteropServices::Marshal::GetExceptionForHR(hr);
;
C# 客户端代码现在非常简单:
JDanielSmith.Utilities.SetSystemTime(DateTime.Now);
【讨论】:
我试用了您的代码,但它似乎不起作用。 gist.github.com/jtara1/07cfd5ebffab8296564f86000c50510e 无论如何,我找到了我想要的解决方案并对其进行了测试github.com/jtara1/MiscScripts/blob/…【参考方案7】:为正在寻找的其他人提供的复制/粘贴课程
using System;
using System.ComponentModel;
using System.Runtime.InteropServices;
public static class SystemDateTime
[DllImport("kernel32.dll", EntryPoint = "SetSystemTime", SetLastError = true)]
private static extern bool Win32SetSystemTime(ref SystemTime sysTime);
[StructLayout(LayoutKind.Sequential)]
public struct SystemTime
public ushort Year;
public ushort Month;
public ushort DayOfWeek;
public ushort Day;
public ushort Hour;
public ushort Minute;
public ushort Second;
public ushort Millisecond;
;
public static void SetSystemDateTime(int year, int month, int day, int hour,
int minute, int second, int millisecond)
SystemTime updatedTime = new SystemTime
Year = (ushort) year,
Month = (ushort) month,
Day = (ushort) day,
Hour = (ushort) hour,
Minute = (ushort) minute,
Second = (ushort) second,
Millisecond = (ushort) millisecond
;
// If this returns false, then the problem is most likely that you don't have the
// admin privileges required to set the system clock
if (!Win32SetSystemTime(ref updatedTime))
throw new Win32Exception(Marshal.GetLastWin32Error());
public static void SetSystemDateTime(DateTime dateTime)
SetSystemDateTime(dateTime.Year, dateTime.Month, dateTime.Day, dateTime.Hour, dateTime.Minute,
dateTime.Second, dateTime.Millisecond);
【讨论】:
效果很好【参考方案8】:小心! 如果您从结构中删除未使用的属性,则会设置错误的时间。我因此失去了 1 天。我认为结构的顺序很重要。
这是正确的结构:
public struct SystemTime
public ushort Year;
public ushort Month;
public ushort DayOfWeek;
public ushort Day;
public ushort Hour;
public ushort Minute;
public ushort Second;
public ushort Millisecond;
;
如果您运行 SetSystemTime(),它会按预期工作。 对于测试,我将时间设置如下;
SystemTime st = new SystemTime();
st.Year = 2019;
st.Month = 10;
st.Day = 15;
st.Hour = 10;
st.Minute = 20;
st.Second = 30;
SetSystemTime(ref st);
时间设置:15.10.2019 10:20,没关系。
但我删除了未使用的 DayOfWeek 属性;
public struct SystemTime
public ushort Year;
public ushort Month;
public ushort Day;
public ushort Hour;
public ushort Minute;
public ushort Second;
public ushort Millisecond;
;
SystemTime st = new SystemTime();
st.Year = 2019;
st.Month = 10;
st.Day = 15;
st.Hour = 10;
st.Minute = 20;
st.Second = 30;
SetSystemTime(ref st);
运行相同的代码,但时间设置为:10.10.2019 20:30
请注意 SystemTime 结构的顺序和所有字段。 优素福
【讨论】:
SetSystemTime
期望具有这些确切字段的结构。如果您传递具有不同字段的结构(更少的字段,不同的顺序,不同的类型,......),它将无法工作。如果你删除了DayOfWeek
,SetSystemTime
仍然期望它,所以你从Day
开始的字段都被转移了,最后一个字段丢失了。【参考方案9】:
proc.Arguments = "/C Date:" + dateInYourSystemFormat;
这是工作函数:
void setDate(string dateInYourSystemFormat)
var proc = new System.Diagnostics.ProcessStartInfo();
proc.UseShellExecute = true;
proc.WorkingDirectory = @"C:\Windows\System32";
proc.CreateNoWindow = true;
proc.FileName = @"C:\Windows\System32\cmd.exe";
proc.Verb = "runas";
proc.Arguments = "/C Date:" + dateInYourSystemFormat;
try
System.Diagnostics.Process.Start(proc);
catch
MessageBox.Show("Error to change time of your system");
Application.ExitThread();
【讨论】:
以上是关于以编程方式更改系统日期的主要内容,如果未能解决你的问题,请参考以下文章