C# winform程序?qū)崿F(xiàn)開(kāi)機(jī)自啟動(dòng)并且識(shí)別是開(kāi)機(jī)啟動(dòng)還是雙擊啟動(dòng)
開(kāi)機(jī)啟動(dòng)程序,在很多場(chǎng)合都會(huì)用到,尤其是那種在后臺(tái)運(yùn)行的程序。
效果圖:
以上兩幅圖都用到了命令行啟動(dòng)程序,為了模擬開(kāi)機(jī)啟動(dòng)或者其他程序調(diào)用此程序。
第一幅圖:程序啟動(dòng)可以根據(jù)不同參數(shù),執(zhí)行不同的操作。如果是雙擊啟動(dòng),就自動(dòng)運(yùn)行邏輯代碼,如果是帶特定參數(shù)啟動(dòng),就自動(dòng)運(yùn)行邏輯代碼。
第二幅圖:winform 程序設(shè)置開(kāi)機(jī)啟動(dòng),模擬雙擊啟動(dòng)和開(kāi)機(jī)啟動(dòng)的不同效果。
開(kāi)機(jī)啟動(dòng)并自動(dòng)運(yùn)行方法:其實(shí)思路很簡(jiǎn)單,就是將程序添加到注冊(cè)表中,這設(shè)置值的時(shí)候,加一個(gè)參數(shù)就可以了。然后程序在入口函數(shù)處判斷啟動(dòng)參數(shù),如果有啟動(dòng)參數(shù),就走自動(dòng)運(yùn)行邏輯代碼,如果沒(méi)有,就只是程序啟動(dòng),并不運(yùn)行邏輯代碼。
【Main參數(shù)】
在C/C++中,我們很明確的知道 main(int argc, char *argv[]/*, char *envp[]*/) 函數(shù)有兩(三)個(gè)參數(shù),第三個(gè)參數(shù)一般用得少,我是還沒(méi)用到過(guò)。所以常見(jiàn)的都是用兩個(gè)參數(shù)。第一個(gè)是參數(shù)個(gè)數(shù),非負(fù)數(shù)。第二個(gè)是表示從執(zhí)行環(huán)境傳遞給程序的各個(gè)實(shí)參。也就是說(shuō),我們要用程序入口參數(shù),只需要判斷argc的值,然后使用數(shù)組取argv的值就行。
那么到了C#就變的更簡(jiǎn)單了。直接變成 string[] 了。那么你只要遍歷這個(gè)字符串?dāng)?shù)組就可以了。
我們用VS創(chuàng)建命令行工程的時(shí)候,如果是命令行程序那么VS會(huì)默認(rèn)使用帶參數(shù)的main函數(shù):void Main(string [] args);如果是winform工程,VS是默認(rèn)使用void Main()。其實(shí)無(wú)論默認(rèn)使用哪個(gè)都無(wú)所謂,主要是自己要清楚main函數(shù)的格式,以及代表的含義。main函數(shù)不光有參數(shù),還可以有返回值。其實(shí)我們平時(shí)寫的C#工程中main函數(shù)看似沒(méi)有返回值,其實(shí)是可以帶有int類型的返回值的。如果你不清楚這一塊,傳送門。
那如果我們用的是 void Main() 這種形式我們?cè)趺传@取程序入庫(kù)參數(shù)?這里其實(shí)是只是一個(gè)表面現(xiàn)象。別以為你不帶參數(shù),我就獲取不到了 ^_^ 。微軟為我們提供了一個(gè)類:Environment。這個(gè)類比較強(qiáng)大。如果你還不清楚怎么用,那去 MSDN 搜一下就會(huì)了。獲取命令行參數(shù)也就一個(gè)函數(shù)而已: string[] Environment.GetCommandLineArgs();使用這個(gè)方法需要注意就是返回值是數(shù)組類型,第一個(gè)元素包含正在執(zhí)行的程序的文件名,從第二個(gè)參數(shù)開(kāi)始,才是命令行參數(shù)。其實(shí)這個(gè)辦法就剛好解決了 winform 程序中獲取命令行參數(shù)的問(wèn)題。
【注冊(cè)表操作】
將程序啟動(dòng)寫入注冊(cè)表實(shí)現(xiàn)開(kāi)機(jī)啟動(dòng),這個(gè)感覺(jué)沒(méi)什么好說(shuō)的。使用固定方法操作就行。不過(guò)用有一點(diǎn)需要注意就是在訪問(wèn)注冊(cè)表的時(shí)候可能會(huì)提示沒(méi)有權(quán)限,你這個(gè)網(wǎng)上百度有好多方法。但是MSDN中也給出了方法。就是在工程的中添加應(yīng)用程序文件清單中修改一句就可以了。
核心代碼:
using System; using Microsoft.Win32; namespace AutoStartRun { public sealed class SystemHelper { private SystemHelper() { } /// <summary> /// 設(shè)置程序開(kāi)機(jī)啟動(dòng) /// </summary> /// <param name="strAppPath">應(yīng)用程序exe所在文件夾</param> /// <param name="strAppName">應(yīng)用程序exe名稱</param> /// <param name="bIsAutoRun">自動(dòng)運(yùn)行狀態(tài)</param> public static void SetAutoRun(string strAppPath, string strAppName, bool bIsAutoRun) { try { if (string.IsNullOrWhiteSpace(strAppPath) || string.IsNullOrWhiteSpace(strAppName)) { throw new Exception("應(yīng)用程序路徑或名稱為空!"); } RegistryKey reg = Registry.LocalMachine; RegistryKey run = reg.CreateSubKey(@"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run\"); if (bIsAutoRun) { run.SetValue(strAppName, strAppPath); } else { if (null != run.GetValue(strAppName)) { run.DeleteValue(strAppName); } } run.Close(); reg.Close(); } catch (Exception ex) { throw new Exception(ex.Message, ex); } } /// <summary> /// 判斷是否開(kāi)機(jī)啟動(dòng) /// </summary> /// <param name="strAppPath">應(yīng)用程序路徑</param> /// <param name="strAppName">應(yīng)用程序名稱</param> /// <returns></returns> public static bool IsAutoRun(string strAppPath, string strAppName) { try { RegistryKey reg = Registry.LocalMachine; RegistryKey software = reg.OpenSubKey(@"SOFTWARE"); RegistryKey run = reg.OpenSubKey(@"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run\"); object key = run.GetValue(strAppName); software.Close(); run.Close(); if (null == key || !strAppPath.Equals(key.ToString())) { return false; } return true; } catch (Exception ex) { throw new Exception(ex.Message, ex); } } } }
調(diào)用方法:
/// <summary> /// 設(shè)置程序開(kāi)機(jī)自啟動(dòng) /// </summary> private void SetAutoRun() { string strFilePath = Application.ExecutablePath; string strFileName = System.IO.Path.GetFileName(strFilePath); try { SystemHelper.SetAutoRun(strFilePath + " -autostart", strFileName, !menuAutoRun.Checked); menuAutoRun.Checked = !menuAutoRun.Checked; } catch (Exception ex) { MessageBox.Show(this, ex.Message, "錯(cuò)誤提示", MessageBoxButtons.OK, MessageBoxIcon.Error); } }
設(shè)置開(kāi)機(jī)啟動(dòng)就是如此簡(jiǎn)單。
【開(kāi)機(jī)啟動(dòng)并運(yùn)行】
那這個(gè)就不用說(shuō)了,將命令行參數(shù)和開(kāi)機(jī)注冊(cè)表操作結(jié)合起來(lái)就可以了。
示例代碼:
/// <summary> /// 檢查是否開(kāi)機(jī)啟動(dòng),并設(shè)置控件狀態(tài) /// </summary> private void CheckAutoRun() { string strFilePath = Application.ExecutablePath; string strFileName = System.IO.Path.GetFileName(strFilePath); if (SystemHelper.IsAutoRun(strFilePath + " -autostart", strFileName)) { menuAutoRun.Checked = true; } else { menuAutoRun.Checked = false; } } private void AutoRun() { if (menuAutoRun.Checked) { string[] strArgs = Environment.GetCommandLineArgs(); if (strArgs.Length >= 2 && strArgs[1].Equals("-autorun")) { labText.Text = "我是開(kāi)機(jī)自啟動(dòng)運(yùn)行..."; } } }
總結(jié)
以上所述是小編給大家介紹的C# winform程序?qū)崿F(xiàn)開(kāi)機(jī)自啟動(dòng)并且識(shí)別是開(kāi)機(jī)啟動(dòng)還是雙擊啟動(dòng),希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
C#線程委托BeginInvoke與EndInvoke的用法
這篇文章介紹了C#線程委托BeginInvoke與EndInvoke的用法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07Unity實(shí)現(xiàn)見(jiàn)縫插針小游戲
這篇文章主要為大家詳細(xì)介紹了Unity實(shí)現(xiàn)見(jiàn)縫插針小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04c#數(shù)據(jù)綁定之?dāng)?shù)據(jù)轉(zhuǎn)化為信息的示例
這篇文章主要介紹了c#數(shù)據(jù)綁定中的數(shù)據(jù)轉(zhuǎn)化為信息的示例,需要的朋友可以參考下2014-04-04c# DateTime常用操作實(shí)例(datetime計(jì)算時(shí)間差)
字符串操作DateTime操作,datetime計(jì)算時(shí)間差,取當(dāng)前時(shí)間,更多方法看下面代碼2013-12-12WPF實(shí)現(xiàn)動(dòng)畫效果(二)之From/To/By動(dòng)畫
這篇文章介紹了WPF實(shí)現(xiàn)動(dòng)畫效果之From/To/By動(dòng)畫,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06