Delphi实现通用的定时自动关机程序

news/2025/1/9 12:31:56/
一、问题的提出:运行某任务的计算机,尤其是服务器,如果能实现在无人职守的情况下,到达指定时间时自动关机,那么将极大地减轻系统管理员的负担,也会给我们的日常工作带来很大方便。 

  笔者用Delphi开发的这个定时自动关机程序,适用于目前两类的Windows系列操作系统:从Windows 95/98/Me到Windows NT/2000/XP。 

  二、程序的功能有: 

  1.用户自己设定关机时间,通过自定义函数IsValidTime()判断用户输入的时间是否有效。 

  2.定时强制自动关机:对于windows 95/98/Me,直接调用API函数ExitWindowsEx()关机。对于NT/2000/XP,需要取得计算机名,获得关机特权后,才能关机:首先调用OpenProcessToken()函数得到存取令牌的句柄,然后调用AdjustTokenPrivileges()函数来使能该特权。Win32API定义了一组字符串常量来标识不同的特权,如关机特权是 ’SeShutdownPrivilege’。 

  3.到达设定的关机时间时,延时30秒,以便用户保存文件,或取消关机。两类操作系统都显示倒记时,对于windows 95/98/Me,只通过程序界面显示;对于NT/2000/XP,将调用系统的倒记时界面显示。 

  4.为了不占用任务栏的空间,程序显示在托盘中。右键单击托盘中的图标,将显示快捷菜单。 

  5.如果未到设定的关机时间,系统要关闭,该程序能截获关机消息,由用户选择是否关机。原理是:当用户关闭Windows时,系统会发送给各应用程序一个消息wm_queryendsession,告诉各应用程序要关机了,如果反馈回来的消息值为0,就不能关机。因此,截获wm_queryendsession,并反馈回0,就大功告成了。 

  6.在内存中只运行本程序的一个实例。原理是:利用Windows 的全局原子表信息来实现此功能。Windows 的全局原子表可以被当前所有应用程序访问,它一共可包含37 项内容。程序运行时,首先检查在表中有无本程序的信息,如有,则提示后退出。如没有,则在表中增加该程序的信息。程序最后退出时要从表中移走信息以便程序能再运行。   

  四、源程序: 
unit AutoShut1; 
interface 

uses 
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, 
  Dialogs, StdCtrls, ExtCtrls, Menus,AppEvnts,shellapi; 
type 
  TForm1 = class(TForm) 
  Timer1: TTimer; 
  Timer2: TTimer; 
  ApplicationEvents1: TApplicationEvents; 
  PopupMenu1: TPopupMenu; 
  Edit1: TEdit; 
  Edit2: TEdit; 
  Label1: TLabel; 
  Label2: TLabel; 
  Label3: TLabel; 
  Btn_OK: TButton; 
  Btn_Abort: TButton; 
  procedure Timer1Timer(Sender: TObject); 
  procedure TrayMenu(Var Msg:TMessage); message WM_USER; 
  procedure TimeSetClick(Sender: TObject); 
  procedure ExitClick(Sender: TObject); 
  procedure Btn_OKClick(Sender: TObject); 
  procedure Btn_AbortClick(Sender: TObject); 
  procedure Timer2Timer(Sender: TObject); 
  procedure Edit2KeyPress(Sender: TObject; var Key: Char); 
  procedure WMQueryEndSession (var Msg : TWMQueryEndSession); 
  message WM_QueryEndSession; 
  procedure FormCreate(Sender: TObject); 
  procedure FormDestroy(Sender: TObject); 
  procedure FormCloseQuery(Sender: TObject; var CanClose: Boolean); 
private 
 { Private declarations } 
 Tray:NOTIFYICONDATA; 
 procedure ShowInTray(); 
public 
 { Public declarations } 
end; 

var 
  Form1: TForm1; 
  P,Ti1:Pchar; 
  Flags:Longint; 
  i:integer; 
  {关机延迟时间} 
  TimeDelay:integer; 
  atom:integer; 
  implementation 
 {$R *.dfm} 

{未到自动关机时间,系统要关闭时,截获关机消息 

wm_queryendsession,让用户决定是否关机} 
procedure TForm1.WMQueryEndSession (var Msg : TWMQueryEndSession); 
begin 
 if MessageDlg(’真的要关闭Windows吗?’,mtConfirmation,[mbYes,mbNo], 0) = mrNo then 
  Msg.Result := 0 
 else 
  Msg.Result := 1; 
 end; 

{判断时间S格式是否是有效} 

function IsValidTime(s:string):bool; 
begin 
 if  Length(s)<>5 then IsValidTime:=False 
 else 
 begin 
  if(s[1]<’0’)or(s[1]>’2’)or(s[2]<’0’)or 
       (s[2]>’9’) or (s[3] <> ’:’) or 
       (s[4]<’0’) or (s[4]>’5’) or 
       (s[5]<’0’) or (s[5]>’9’)then IsValidTime:=False 
  else 
   IsValidTime:=True; 
  end; 
end; 


{判断是哪类操作系统,以确定关机方式} 

function GetOperatingSystem: string; 
 var  osVerInfo: TOSVersionInfo; 
begin 
 Result :=’’; 
 osVerInfo.dwOSVersionInfoSize := SizeOf(TOSVersionInfo); 
 if GetVersionEx(osVerInfo) then 
  case osVerInfo.dwPlatformId of 
   VER_PLATFORM_WIN32_NT: 
   begin 
    Result := ’Windows NT/2000/XP’ 
  end; 
  VER_PLATFORM_WIN32_WINDOWS: 
  begin 
   Result := ’Windows 95/98/98SE/Me’; 
  end; 
 end; 
end; 


{获得计算机名} 

function GetComputerName: string; 
var 
 buffer: array[0..MAX_COMPUTERNAME_LENGTH + 1] of Char; 
 Size: Cardinal; 
begin 
  Size := MAX_COMPUTERNAME_LENGTH + 1; 
  Windows.GetComputerName(@buffer, Size); 
  Result := strpas(buffer); 
end; 

  
{定时关机函数 ,各参数的意义如下: 

Computer: 计算机名;Msg:显示的提示信息; 
Time:时间延迟; Force:是否强制关机; 
Reboot: 是否重启动} 
function TimedShutDown(Computer: string; Msg: string; 
 Time: Word; Force: Boolean; Reboot: Boolean): Boolean; 
var 
 rl: Cardinal; 
 hToken: Cardinal; 
 tkp: TOKEN_PRIVILEGES; 
begin 
  {获得用户关机特权,仅对Windows NT/2000/XP} 
  OpenProcessToken(GetCurrentProcess, TOKEN_ADJUST_PRIVILEGES or TOKEN_QUERY, hToken); 
  if LookupPrivilegeValue(nil, ’SeShutdownPrivilege’, tkp.Privileges[0].Luid) then 
  begin 
   tkp.Privileges[0].Attributes := SE_PRIVILEGE_ENABLED; 
   tkp.PrivilegeCount := 1; 
   AdjustTokenPrivileges(hToken, False, tkp, 0, nil, rl); 
  end; 
  Result := InitiateSystemShutdown(PChar(Computer), PChar(Msg), Time, Force, Reboot) 
end; 

{窗体最小化后,显示在托盘中} 

procedure tform1.ShowInTray; 
Begin 
 Tray.cbSize:=sizeof(Tray); 
 Tray.Wnd:=Self.Handle; 
 Tray.uFlags:=NIF_ICON+NIF_MESSAGE+NIF_TIP; 
 Tray.uCallbackMessage:=WM_USER; 
 Tray.hIcon:=application.Icon.Handle ; 
 Tray.szTip:=’定时关机’; 
 Shell_NotifyIcon(NIM_ADD,@Tray); 
End; 

{右键单击托盘中的图标,显示快捷菜单} 

procedure Tform1.TrayMenu(var Msg:TMessage); 
var 
 X,Y:Tpoint; 
 J,K:Integer; 
Begin 
 GetCursorPos(X); 
 GetCursorPos(Y); 
 J:=X.X; 
 K:=Y.Y; 
 if Msg.LParam=WM_RBUTTONDOWN then PopupMenu1.Popup(J,K); 
 End; 
  
procedure TForm1.Timer1Timer(Sender: TObject); 
begin 
 Edit1.Text:=FormatDateTime(’hh:mm’, Now); 
 {两个时间相等,计算机将在TimeDelay秒内强制关机} 
 if edit1.text=edit2.Text then 
 Begin 
  TimeDelay:=30; 
  timer1.Enabled:=False; 
 if GetOperatingSystem=’Windows NT/2000/XP’ then 
  begin 
   {调用系统的关机提示窗口,只限于Windows NT/2000/XP。} 
   TimedShutDown(getcomputername, ’系统将要关机!’, 
   TimeDelay, true, false); 
   btn_abort.Enabled :=true; 
   timer2.Enabled :=true; 
  end; 
 if  GetOperatingSystem=’Windows 95/98/98SE/Me’ then 
  begin 
    timer2.Enabled :=true; 
    {在顶层显示本程序的窗口,显示时间倒记时} 
    Application.Restore; 
    SetWindowPos(Handle,HWND_TOPMOST,Left,Top,Width,Height, 
               SWP_NOACTIVATE); 
  end; 
 end; 
end; 

procedure TForm1.Timer2Timer(Sender: TObject); 
begin 
  btn_abort.Enabled :=true; 
  label3.Caption :=’离关机时间还有’+inttostr(timedelay)+’秒。’; 
  if timedelay>0 then timedelay:=timedelay-1 
  else 
   begin 
    timer2.Enabled :=false; 
    {强制Windows 95/98/98SE/Me关机} 
    ExitWindowsEx(EWX_SHUTDOWN+EWX_FORCE,0); 
    end; 
  end; 

{通过控件PopupMenu1定义的快捷菜单,包括"设置关机时间"和"退出"。 

PopupMenu1的AutoPopup为False,下面是"设置关机时间"的代码} 
procedure TForm1.TimeSetClick(Sender: TObject); 
begin 
  {设置本程序窗口位于最顶层} 
  SetWindowPos(Handle,HWND_TOPMOST,Left,Top,Width,Height, 
               SWP_NOACTIVATE); 
  ShowWindow(Application.Handle,SW_NORMAL); 
  edit2.SetFocus ; 
  edit2.SelectAll ; 
end; 

{快捷菜单中"退出"的代码} 

procedure TForm1.ExitClick(Sender: TObject); 
begin 
  {如果已经开始倒记时,禁止退出,而是显示程序窗口} 
  if Timer2.Enabled=false then 
  begin 
    Application.Terminate; 
  end 
  else  ShowWindow(Application.Handle,SW_NORMAL); 
end; 

{确定按钮} 

procedure TForm1.Btn_OKClick(Sender: TObject); 
begin 
  btn_abort.Enabled :=false; 
  label3.Caption :=’提示:关机时间格式 HH:MM’; 
  if timer1.Enabled =false then timer1.Enabled :=true; 
  {关机时间设置有效,程序将显示在托盘中,无效则提示。} 
  if IsValidTime(edit2.Text) then 
    begin 
      ShowWindow(Application.Handle,sw_minimize); 
      ShowWindow(Application.Handle,sw_hide); 
      ShowInTray; 
    end 
  else 
    showmessage(’提示:时间格式错误,’+chr(13)+ 
    ’请输入正确的关机时间 HH:MM。’); 
end; 

{取消关机按钮} 

procedure TForm1.Btn_AbortClick(Sender: TObject); 
begin 
  if  GetOperatingSystem=’Windows NT/2000/XP’ then 
    {对于Windows NT/2000/XP,取消关机} 
    begin 
      AbortSystemShutdown(pchar(getcomputername)); 
    end; 
    {停止倒记时} 
  if timer2.Enabled =true then timer2.Enabled :=false; 
  btn_abort.Enabled :=false; 
end; 

{输入关机时间后,可直接按回车} 

procedure TForm1.Edit2KeyPress(Sender: TObject; var Key: Char); 
begin 
  if (key=#13)  then  Btn_OK.Click; 
end; 

{搜寻系统原子表看是否程序已运行} 

procedure TForm1.FormCreate(Sender: TObject); 
begin 
  {如果没运行则在表中增加信息 } 
  if GlobalFindAtom(’PROGRAM_RUNNING’) = 0 then 
    atom := GlobalAddAtom(’PROGRAM_RUNNING’) 
  else begin 
    {如果程序已运行则显示信息然后退出 } 
    MessageDlg(’程序已经在运行!’,mtWarning,[mbOK],0); 
    Halt; 
  end; 
end; 

procedure TForm1.FormDestroy(Sender: TObject); 
begin 
  {程序退出时,从原子表中移走信息} 
  GlobalDeleteAtom(atom); 
  {删除托盘中的图标} 
  Shell_NotifyIcon(NIM_DELETE,@Tray); 
end; 

procedure TForm1.FormCloseQuery(Sender: TObject; var CanClose: Boolean); 
begin 
  {如果已经开始倒记时,禁止关闭程序窗口} 
  if timer2.Enabled =true then canclose:=false; 
end; 
end. 


  五、说明:本程序在Windows XP下,用Delphi 6.0开发,在Windows 95/98/Me和Windows NT/2000/XP下运行成功。

http://www.ppmy.cn/news/566413.html

相关文章

戴尔服务器开机自动关机,戴尔台式电脑自动关机怎么办

戴尔电脑运行的好好的自动就关机或者自动重新启动了,那么为什么电脑会自动关机与自动重新启动呢?学习啦小编为大家分享了具体的操作方法&#xff0c;接下来大家就跟着学习啦小编一起去了解下吧。 戴尔台式电脑自动关机怎么办 电脑自动关机的原因&#xff1a;病毒 1、一般来说不…

小白的黑客技术(狗头),用C语言写一个自动关机程序,恶搞神器

学计算机这么久终于学到点有黑客那味的东西了&#xff0c;哈哈哈哈 (大牛禁止嘲笑小白&#xff0c;不许笑) 效果如下 准备工作 首先&#xff0c;这个程序需要用到sysem()函数&#xff0c;是执行系统命令的函数&#xff0c;shutdown -s -t 60 是60秒关机的指令&#xff1b;使…

计算机自动关机原理,电脑自动关机是什么原因怎样处理

我们常常把电脑自动关机或重启的问题归结于内存条的问题。电脑自动关机或重启这类问题,到底是什么原因导致的,又如何解决这一的问题呢?一起来看看电脑自动关机是什么原因怎样处理,欢迎查阅! 电脑自动关机可能原因 1、电脑散热不良导致电脑自动关机。 2、电源供电不足 3、内…

OpenText Documentum 平台 实现企业内容管理现代化

OpenText Documentum 平台 实现企业内容管理现代化 OpenText™ Documentum™ 平台组织、保存信息并使其易于访问&#xff0c;同时确保信息遵守所有隐私和安全协议。Documentum 为未来创建了一个架构&#xff0c;跨文件存储管理内容并与企业应用程序&#xff08;包括 SAP、Sale…

houdini批渲染结束自动关机程序

最近需要一个能在houdini批渲染结束后自动关机的小程序 所以就写了一个&#xff0c;也没用多长时间 使用的openframeworks 0.84 本来是想用PsSetCreateProcessNotifyRoutine的但是需要ddk windows 开发包&#xff0c;看起来就很麻烦 所以想着干脆刷进程吧 下面是代码 #in…

C语言实现电脑自动关机程序--可以用来恶搞舍友电脑

实现电脑关机主要用到shutdown命令 shutdown命令 shutdown -a 取消关机 shutdown -s 关机 shutdown -f  强行关闭应用程序 shutdown -m \\计算机名 控制远程计算机 shutdown -i 显示“远程关机”图形用户界面&#xff0c;但必须是Shutdown的第一个参数 shutdown -l 注…

打工人准时下班踩点利器——python写一个自动关机程序并打包成exe文件

嗨害大家好鸭&#xff0c;我是小熊猫&#x1f5a4; 前一阵子工具人表哥和我说很想和时间赛跑 关机的速度还不够快 要是自动关机就好了 到点就走人 这种事情…肯定是拿来当我整活的素材啦 ~ ~ ~ 有什么python相关报错解答自己不会的、或者源码资料/模块安装/女装大佬精通技…

Win11如何设置自动关机

1、首先建立一个【文本文档】&#xff0c;然后输入【shutdown -s -t 0】命令&#xff0c;并且命名为【shutdown】。 2、将刚才名称为 shutdown 的文本文档&#xff0c;修改成后缀为 .bat 的批处理文件&#xff1b; 3、打开任务计划程序&#xff0c;点击顶部【操作】&#xff0c…