วิธีปิดคอมพิวเตอร์จาก C #


138

วิธีที่ดีที่สุดในการปิดคอมพิวเตอร์จากโปรแกรม C # คืออะไร

ฉันพบวิธีการบางอย่างที่ใช้งานได้แล้ว - ฉันจะโพสต์ไว้ด้านล่าง - แต่ไม่มีวิธีใดที่สวยงามมาก ฉันกำลังมองหาบางสิ่งที่ง่ายขึ้นและเป็น. net

คำตอบ:


171

ทำงานเริ่มต้นด้วย windows XP, ไม่สามารถใช้ได้ใน win 2000 หรือต่ำกว่า:

นี่เป็นวิธีที่เร็วที่สุดที่จะทำ:

Process.Start("shutdown","/s /t 0");

มิฉะนั้นให้ใช้ P / Invoke หรือ WMI เหมือนที่คนอื่นพูด

แก้ไข: วิธีหลีกเลี่ยงการสร้างหน้าต่าง

var psi = new ProcessStartInfo("shutdown","/s /t 0");
psi.CreateNoWindow = true;
psi.UseShellExecute = false;
Process.Start(psi);

2
สิ่งนี้ดูเหมือนว่าจะทำงานได้จากบริการด้วย (อย่างน้อยในสถานการณ์ที่ฉันเกี่ยวข้อง) ฉันไม่เคยได้รับ WMI หรือ ExitWindowsEx วิธีการทำงานจากบริการ
James

1
@ James เป็นเพราะบริการมักไม่มีสิทธิ์ใช้งาน
AK_

สถานะการใช้พลังงานของเครื่องแตกต่างกันหลังจากใช้สิ่งนี้มากกว่าหลังจากใช้หน้าต่างข้อความปิดเครื่องแบบดั้งเดิม การกดปุ่มเพาเวอร์เพื่อบูตเครื่องจะดึงประมาณ 80-85 milliamps แทนที่จะเป็นมาตรฐาน 300 + ish จะโพสต์กลับมาที่นี่ถ้าฉันหาสาเหตุ สิ่งนี้ไม่ควรส่งผลกระทบต่อผู้ใช้ส่วนใหญ่
samuelesque

วิธีนี้ใช้งานได้ดียกเว้นความจริงที่ว่าถ้าคุณอยู่ใน WPF สิ่งนี้จะวางไข่หน้าต่างคอนโซลสำหรับเสี้ยววินาทีไม่ดูเป็นมืออาชีพ
ดัสตินเซ่น

80

นำมาจาก: โพสต์ Geekpedia

วิธีนี้ใช้WMIเพื่อปิดหน้าต่าง

คุณจะต้องเพิ่มการอ้างอิงถึง System.Management ไปยังโครงการของคุณเพื่อใช้สิ่งนี้

using System.Management;

void Shutdown()
{
    ManagementBaseObject mboShutdown = null;
    ManagementClass mcWin32 = new ManagementClass("Win32_OperatingSystem");
    mcWin32.Get();

    // You can't shutdown without security privileges
    mcWin32.Scope.Options.EnablePrivileges = true;
    ManagementBaseObject mboShutdownParams =
             mcWin32.GetMethodParameters("Win32Shutdown");

     // Flag 1 means we want to shut down the system. Use "2" to reboot.
    mboShutdownParams["Flags"] = "1";
    mboShutdownParams["Reserved"] = "0";
    foreach (ManagementObject manObj in mcWin32.GetInstances())
    {
        mboShutdown = manObj.InvokeMethod("Win32Shutdown", 
                                       mboShutdownParams, null);
    }
}

3
การใช้ WMI ทำให้การติดตามข้อผิดพลาดง่ายขึ้น จะเกิดอะไรขึ้นหากคำสั่งปิดเครื่องไม่ทำงานด้วยเหตุผลบางอย่าง
Rob Walker

2
ฉันใช้วิธีนี้เพื่อปิดหน้าต่างและสองในสามครั้งมันจะบอกฉันว่าฉันไม่มีสิทธิ์ แต่ครั้งที่สามมันคือ "ยอมแพ้" และรีสตาร์ทคอมพิวเตอร์แล้ว เกิดอะไรขึ้นกับสิ่งนั้น
DTI-Matt

1
วิธีนี้ไม่ได้ผลสำหรับฉัน ฉันได้รับข้อยกเว้น "สิทธิพิเศษที่ไม่ได้จัด" แม้ว่าฉันจะเรียกใช้โปรแกรมภายใต้ผู้ใช้ของผู้ดูแลระบบ
Fanda

@roomaroo วิธีนี้ใช้ไม่ได้ มันมีข้อยกเว้น: ข้อยกเว้นการจัดการไม่มีสิทธิ์พิเศษ
somethingSomething

หากคุณต้องการปิดการบังคับอย่างแรงคุณควรใช้ mboShutdownParams ["Flags"] = "5"; ค่า 5 หมายถึงการปิดระบบแบบบังคับ
SaneDeveloper

32

เธรดนี้แสดงรหัสที่จำเป็น: http://bytes.com/forum/thread251367.html

แต่นี่คือรหัสที่เกี่ยวข้อง:

using System.Runtime.InteropServices;

[StructLayout(LayoutKind.Sequential, Pack=1)]
internal struct TokPriv1Luid
{
    public int Count;
    public long Luid;
    public int Attr;
}

[DllImport("kernel32.dll", ExactSpelling=true) ]
internal static extern IntPtr GetCurrentProcess();

[DllImport("advapi32.dll", ExactSpelling=true, SetLastError=true) ]
internal static extern bool OpenProcessToken( IntPtr h, int acc, ref IntPtr
phtok );

[DllImport("advapi32.dll", SetLastError=true) ]
internal static extern bool LookupPrivilegeValue( string host, string name,
ref long pluid );

[DllImport("advapi32.dll", ExactSpelling=true, SetLastError=true) ]
internal static extern bool AdjustTokenPrivileges( IntPtr htok, bool disall,
ref TokPriv1Luid newst, int len, IntPtr prev, IntPtr relen );

[DllImport("user32.dll", ExactSpelling=true, SetLastError=true) ]
internal static extern bool ExitWindowsEx( int flg, int rea );

internal const int SE_PRIVILEGE_ENABLED = 0x00000002;
internal const int TOKEN_QUERY = 0x00000008;
internal const int TOKEN_ADJUST_PRIVILEGES = 0x00000020;
internal const string SE_SHUTDOWN_NAME = "SeShutdownPrivilege";
internal const int EWX_LOGOFF = 0x00000000;
internal const int EWX_SHUTDOWN = 0x00000001;
internal const int EWX_REBOOT = 0x00000002;
internal const int EWX_FORCE = 0x00000004;
internal const int EWX_POWEROFF = 0x00000008;
internal const int EWX_FORCEIFHUNG = 0x00000010;

private void DoExitWin( int flg )
{
    bool ok;
    TokPriv1Luid tp;
    IntPtr hproc = GetCurrentProcess();
    IntPtr htok = IntPtr.Zero;
    ok = OpenProcessToken( hproc, TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, ref htok );
    tp.Count = 1;
    tp.Luid = 0;
    tp.Attr = SE_PRIVILEGE_ENABLED;
    ok = LookupPrivilegeValue( null, SE_SHUTDOWN_NAME, ref tp.Luid );
    ok = AdjustTokenPrivileges( htok, false, ref tp, 0, IntPtr.Zero, IntPtr.Zero );
    ok = ExitWindowsEx( flg, 0 );
    }

การใช้งาน:

DoExitWin( EWX_SHUTDOWN );

หรือ

DoExitWin( EWX_REBOOT );

คุณสามารถอ่านเกี่ยวกับค่าคงที่ EWX_ อื่น ๆ ได้ที่นี่: msdn.microsoft.com/en-us/library/windows/desktop/?hl=th
TripleAntigen

1
เมื่อย้ายค่าคงที่ตัวเลขไปที่ C # แนวปฏิบัติที่เหมาะสมที่สุดคือใช้ enum นั่นคือสิ่งที่ enum ถูกออกแบบมาให้ทำ มันให้การพิมพ์ที่แข็งแกร่งรอบค่าคงที่ตัวเลขสนับสนุนทางเลือกธง / bitmasks และปลดเปลื้องไปมาได้อย่างง่ายดายไปยังประเภทตัวเลขพื้นฐาน
Andrew Rondeau

26

วิธีการที่แตกต่างกัน:

A. System.Diagnostics.Process.Start("Shutdown", "-s -t 10");

B. เครื่องมือจัดการ Windows (WMI)

C. System.Runtime.InteropServices Pinvoke

D. การจัดการระบบ

หลังจากที่ฉันส่งฉันได้เห็นคนอื่น ๆ อีกมากมายที่โพสต์ ...


2
B และ D เป็นวิธีการเดียวกัน (WMI)
Lucas

อี Powershell รันสคริปต์จากรหัสblogs.msdn.microsoft.com/kebab/2014/04/28/...
user1785960

14

วิธีน่าเกลียดโรงเรียนเก่า ใช้ExitWindowsExฟังก์ชันจาก Win32 API

using System.Runtime.InteropServices;

void Shutdown2()
{
    const string SE_SHUTDOWN_NAME = "SeShutdownPrivilege";
    const short SE_PRIVILEGE_ENABLED = 2;
    const uint EWX_SHUTDOWN = 1;
    const short TOKEN_ADJUST_PRIVILEGES = 32;
    const short TOKEN_QUERY = 8;
    IntPtr hToken;
    TOKEN_PRIVILEGES tkp;

    // Get shutdown privileges...
    OpenProcessToken(Process.GetCurrentProcess().Handle, 
          TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, out hToken);
    tkp.PrivilegeCount = 1;
    tkp.Privileges.Attributes = SE_PRIVILEGE_ENABLED;
    LookupPrivilegeValue("", SE_SHUTDOWN_NAME, out tkp.Privileges.pLuid);
    AdjustTokenPrivileges(hToken, false, ref tkp, 0U, IntPtr.Zero, 
          IntPtr.Zero);

    // Now we have the privileges, shutdown Windows
    ExitWindowsEx(EWX_SHUTDOWN, 0);
}

// Structures needed for the API calls
private struct LUID
{
    public int LowPart;
    public int HighPart;
}
private struct LUID_AND_ATTRIBUTES
{
    public LUID pLuid;
    public int Attributes;
}
private struct TOKEN_PRIVILEGES
{
    public int PrivilegeCount;
    public LUID_AND_ATTRIBUTES Privileges;
}

[DllImport("advapi32.dll")]
static extern int OpenProcessToken(IntPtr ProcessHandle, 
                     int DesiredAccess, out IntPtr TokenHandle);

[DllImport("advapi32.dll", SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool AdjustTokenPrivileges(IntPtr TokenHandle,
    [MarshalAs(UnmanagedType.Bool)]bool DisableAllPrivileges,
    ref TOKEN_PRIVILEGES NewState,
    UInt32 BufferLength,
    IntPtr PreviousState,
    IntPtr ReturnLength);

[DllImport("advapi32.dll")]
static extern int LookupPrivilegeValue(string lpSystemName, 
                       string lpName, out LUID lpLuid);

[DllImport("user32.dll", SetLastError = true)]
static extern int ExitWindowsEx(uint uFlags, uint dwReason);

ในรหัสการผลิตคุณควรตรวจสอบค่าที่ส่งคืนของการเรียก API แต่ฉันปล่อยมันไว้เพื่อให้ตัวอย่างชัดเจนขึ้น


12

สั้นและหวาน เรียกโปรแกรมภายนอก:

    using System.Diagnostics;

    void Shutdown()
    {
        Process.Start("shutdown.exe", "-s -t 00");
    }

หมายเหตุ: สิ่งนี้เรียกโปรแกรม Windows 'Shutdown.exe ดังนั้นมันจะทำงานเฉพาะเมื่อโปรแกรมนั้นพร้อมใช้งาน คุณอาจมีปัญหาบน Windows 2000 (ที่ Shutdown.exe จะใช้ได้เฉพาะในชุดทรัพยากร) หรือXP ฝัง


9
System.Diagnostics.Process.Start("shutdown", "/s /t 0")

ควรทำงาน.

สำหรับการรีสตาร์ทมัน / r

สิ่งนี้จะรีสตาร์ทกล่องพีซีโดยตรงและหมดจดโดยไม่มีกล่องโต้ตอบ


นี่คือคำตอบที่สมบูรณ์แบบสำหรับระบบที่ทันสมัย ​​(2015+)
Fattie

ขอบคุณคุณช่วยอธิบายว่า / s และ / t 0 ทำอะไรได้บ้าง
Vladimir verleg

1
@Peterverleg แน่นอน อาร์กิวเมนต์ "/ s" บอกให้คอมพิวเตอร์ปิดและ "/ t" แจ้งให้คอมพิวเตอร์รอ x วินาทีก่อนที่จะปิดเครื่อง ฉันรู้จากประสบการณ์ส่วนตัวว่าอาร์กิวเมนต์ "/ t" ไม่ได้ทำอะไรใน Windows 8.1 แต่มันใช้ได้ใน 7 อย่างแน่นอน คุณสามารถใช้ฟังก์ชันเหล่านี้ได้เช่นกันshutdown /s /t 0 //For shutdown shutdown /r /t 0 //For restart shutdown /h /t 0 //For hibernateลองพิมพ์ลงใน CMD เพื่อดูผลลัพธ์เดียวกัน
คา Vertal

6

คุณสามารถเริ่มกระบวนการปิดระบบได้:

  • shutdown -s -t 0 - ปิดตัวลง
  • shutdown -r -t 0 - เริ่มต้นใหม่


5

ฉันมีปัญหาในการพยายามใช้วิธีการ WMI ที่ได้รับการยอมรับข้างต้นเพราะฉันมักจะมีสิทธิ์ไม่ได้ถูกยกเว้นแม้จะใช้โปรแกรมเป็นผู้ดูแลระบบ

การแก้ปัญหาสำหรับกระบวนการในการขอสิทธิ์สำหรับตัวเอง ฉันพบคำตอบได้ที่http://www.dotnet247.com/247reference/msgs/58/292150.aspxเขียนโดยคนที่ชื่อ Richard Hill

ฉันได้วางการใช้งานพื้นฐานของโซลูชันไว้ด้านล่างในกรณีที่ลิงก์นั้นเก่า

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Management;
using System.Runtime.InteropServices;
using System.Security;
using System.Diagnostics;

namespace PowerControl
{
    public class PowerControl_Main
    {


        public void Shutdown()
        {
            ManagementBaseObject mboShutdown = null;
            ManagementClass mcWin32 = new ManagementClass("Win32_OperatingSystem");
            mcWin32.Get();

            if (!TokenAdjuster.EnablePrivilege("SeShutdownPrivilege", true))
            {
                Console.WriteLine("Could not enable SeShutdownPrivilege");
            }
            else
            {
                Console.WriteLine("Enabled SeShutdownPrivilege");
            }

            // You can't shutdown without security privileges
            mcWin32.Scope.Options.EnablePrivileges = true;
            ManagementBaseObject mboShutdownParams = mcWin32.GetMethodParameters("Win32Shutdown");

            // Flag 1 means we want to shut down the system
            mboShutdownParams["Flags"] = "1";
            mboShutdownParams["Reserved"] = "0";

            foreach (ManagementObject manObj in mcWin32.GetInstances())
            {
                try
                {
                    mboShutdown = manObj.InvokeMethod("Win32Shutdown",
                                                   mboShutdownParams, null);
                }
                catch (ManagementException mex)
                {
                    Console.WriteLine(mex.ToString());
                    Console.ReadKey();
                }
            }
        }


    }


    public sealed class TokenAdjuster
    {
        // PInvoke stuff required to set/enable security privileges
        [DllImport("advapi32", SetLastError = true),
        SuppressUnmanagedCodeSecurityAttribute]
        static extern int OpenProcessToken(
        System.IntPtr ProcessHandle, // handle to process
        int DesiredAccess, // desired access to process
        ref IntPtr TokenHandle // handle to open access token
        );

        [DllImport("kernel32", SetLastError = true),
        SuppressUnmanagedCodeSecurityAttribute]
        static extern bool CloseHandle(IntPtr handle);

        [DllImport("advapi32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        static extern int AdjustTokenPrivileges(
        IntPtr TokenHandle,
        int DisableAllPrivileges,
        IntPtr NewState,
        int BufferLength,
        IntPtr PreviousState,
        ref int ReturnLength);

        [DllImport("advapi32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        static extern bool LookupPrivilegeValue(
        string lpSystemName,
        string lpName,
        ref LUID lpLuid);

        [StructLayout(LayoutKind.Sequential)]
        internal struct LUID
        {
            internal int LowPart;
            internal int HighPart;
        }

        [StructLayout(LayoutKind.Sequential)]
        struct LUID_AND_ATTRIBUTES
        {
            LUID Luid;
            int Attributes;
        }

        [StructLayout(LayoutKind.Sequential)]
        struct _PRIVILEGE_SET
        {
            int PrivilegeCount;
            int Control;
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 1)] // ANYSIZE_ARRAY = 1
            LUID_AND_ATTRIBUTES[] Privileges;
        }

        [StructLayout(LayoutKind.Sequential)]
        internal struct TOKEN_PRIVILEGES
        {
            internal int PrivilegeCount;
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)]
            internal int[] Privileges;
        }
        const int SE_PRIVILEGE_ENABLED = 0x00000002;
        const int TOKEN_ADJUST_PRIVILEGES = 0X00000020;
        const int TOKEN_QUERY = 0X00000008;
        const int TOKEN_ALL_ACCESS = 0X001f01ff;
        const int PROCESS_QUERY_INFORMATION = 0X00000400;

        public static bool EnablePrivilege(string lpszPrivilege, bool
        bEnablePrivilege)
        {
            bool retval = false;
            int ltkpOld = 0;
            IntPtr hToken = IntPtr.Zero;
            TOKEN_PRIVILEGES tkp = new TOKEN_PRIVILEGES();
            tkp.Privileges = new int[3];
            TOKEN_PRIVILEGES tkpOld = new TOKEN_PRIVILEGES();
            tkpOld.Privileges = new int[3];
            LUID tLUID = new LUID();
            tkp.PrivilegeCount = 1;
            if (bEnablePrivilege)
                tkp.Privileges[2] = SE_PRIVILEGE_ENABLED;
            else
                tkp.Privileges[2] = 0;
            if (LookupPrivilegeValue(null, lpszPrivilege, ref tLUID))
            {
                Process proc = Process.GetCurrentProcess();
                if (proc.Handle != IntPtr.Zero)
                {
                    if (OpenProcessToken(proc.Handle, TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
                    ref hToken) != 0)
                    {
                        tkp.PrivilegeCount = 1;
                        tkp.Privileges[2] = SE_PRIVILEGE_ENABLED;
                        tkp.Privileges[1] = tLUID.HighPart;
                        tkp.Privileges[0] = tLUID.LowPart;
                        const int bufLength = 256;
                        IntPtr tu = Marshal.AllocHGlobal(bufLength);
                        Marshal.StructureToPtr(tkp, tu, true);
                        if (AdjustTokenPrivileges(hToken, 0, tu, bufLength, IntPtr.Zero, ref ltkpOld) != 0)
                        {
                            // successful AdjustTokenPrivileges doesn't mean privilege could be changed
                            if (Marshal.GetLastWin32Error() == 0)
                            {
                                retval = true; // Token changed
                            }
                        }
                        TOKEN_PRIVILEGES tokp = (TOKEN_PRIVILEGES)Marshal.PtrToStructure(tu,
                        typeof(TOKEN_PRIVILEGES));
                        Marshal.FreeHGlobal(tu);
                    }
                }
            }
            if (hToken != IntPtr.Zero)
            {
                CloseHandle(hToken);
            }
            return retval;
        }

    }
}

2
สิ่งนี้ได้ผลแม้ว่าฉันจะไม่ชอบก็ตามไม่รู้ว่าทำไม ผมสงสัยว่าฉันควรจะได้ไปเพียงกับคำสั่ง "ปิด" ...
แดนปลัดอำเภอ

5

เพียงเพิ่มคำตอบของ Pop Catalin ต่อไปนี้เป็นสายการบินเดียวที่ปิดคอมพิวเตอร์โดยไม่แสดงหน้าต่างใด ๆ :

Process.Start(new ProcessStartInfo("shutdown", "/s /t 0") {
  CreateNoWindow = true, UseShellExecute = false
});

2

ฉันลองใช้วิธี WMI ของ roomarooเพื่อปิด Windows Server 2003 แต่จะไม่ทำงานจนกว่าฉันจะเพิ่ม `[สถานะของผู้อ่าน] '(เช่น" เธรดเดี่ยวอพาร์ตเมนต์แบบเธรด ") ในการประกาศหลัก ():

[STAThread]
public static void Main(string[] args) {
    Shutdown();
}

ฉันพยายามปิดระบบจากเธรดและเพื่อให้ทำงานได้ฉันต้องตั้งค่า "สถานะอพาร์ทเมนต์" ของเธรดเป็น STA เช่นกัน:

using System.Management;
using System.Threading;

public static class Program {

    [STAThread]
    public static void Main(string[] args) {
        Thread t = new Thread(new ThreadStart(Program.Shutdown));
        t.SetApartmentState(ApartmentState.STA);
        t.Start();
        ...
    }

    public static void Shutdown() {
        // roomaroo's code
    }
}

ฉันเป็น C # noob ดังนั้นฉันจึงไม่แน่ใจในความสำคัญของเธรด STA ในแง่ของการปิดระบบ (แม้หลังจากอ่านลิงก์ที่ฉันโพสต์ไว้ด้านบน) บางทีคนอื่นสามารถอธิบายรายละเอียด ...


ที่จริงแล้วเฉพาะเธรดที่เรียกใช้ WMI จำเป็นต้องเป็นเธรด STA หากที่ไม่ด้ายหลักไม่จำเป็นต้องMain() [STAThread]
SLAK

2

** คำตอบเพิ่มเติม ...

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
// Remember to add a reference to the System.Management assembly
using System.Management;
using System.Diagnostics;

namespace ShutDown
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void btnShutDown_Click(object sender, EventArgs e)
        {
            ManagementBaseObject mboShutdown = null;
            ManagementClass mcWin32 = new ManagementClass("Win32_OperatingSystem");
            mcWin32.Get();

            // You can't shutdown without security privileges
            mcWin32.Scope.Options.EnablePrivileges = true;
            ManagementBaseObject mboShutdownParams = mcWin32.GetMethodParameters("Win32Shutdown");

            // Flag 1 means we want to shut down the system
            mboShutdownParams["Flags"] = "1";
            mboShutdownParams["Reserved"] = "0";

            foreach (ManagementObject manObj in mcWin32.GetInstances())
            {
                mboShutdown = manObj.InvokeMethod("Win32Shutdown", mboShutdownParams, null);
            }
        }
    }
}

1

ใช้ shutdown.exe เพื่อหลีกเลี่ยงปัญหาเกี่ยวกับการส่งผ่านการดำเนินการที่ซับซ้อนการดำเนินการจาก WindowForms ใช้สคริปต์เรียกใช้ PowerShell:

using System.Management.Automation;
...
using (PowerShell PowerShellInstance = PowerShell.Create())
{
    PowerShellInstance.AddScript("shutdown -a; shutdown -r -t 100;");
    // invoke execution on the pipeline (collecting output)
    Collection<PSObject> PSOutput = PowerShellInstance.Invoke();
} 

System.Management.Automation.dll ควรติดตั้งบนระบบปฏิบัติการและมีอยู่ใน GAC

ขออภัยสำหรับภาษาอังกฤษของฉัน


0

ไม่มี. net พื้นเมืองวิธีสำหรับการปิดเครื่องคอมพิวเตอร์ คุณต้อง P / เรียกใช้การเรียก ExitWindows หรือ ExitWindowsEx API


0

หากคุณต้องการปิดคอมพิวเตอร์จากระยะไกลคุณสามารถใช้งานได้

Using System.Diagnostics;

เมื่อคลิกปุ่มใด ๆ

{
    Process.Start("Shutdown","-i");
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.