在專案中有一個需求是需要在區域網內跨PC遠端呼叫一個程式,並且要求有介面顯示,調查了一些資料,能實現遠端呼叫的.Net技術大概有PsExec、WMI、Schedule Task。
這三種方式都做了一個嘗試,結果發現PsExec、WMI都只能在程式列表中看到程式執行,卻無法顯示介面,無法執行程式中的管理員許可權操作,就連簡單的在C盤建立一個txt文字都做不到。
也許是我使用方法不對,無法達到我的需求。直到後來使用了Schedule Task的方式之後,才成功實現了我的需求。
第三種技術的主要思路是先通過CMD視窗呼叫schtasks命令在遠端PC上建立一個單次執行的計劃任務,在計劃任務中呼叫外部程式,然後立即執行,可以顯示介面,並且可以執行管理員許可權的操作。
用到的schtasks命令:
string queryTaskArg = string.Format(@" /query /s {0} -u domainname\{1} -p {2} /tn install", ip, username, password); string creatTaskArg = string.Format(@" /create /s {0} -u domainname\{1} -p {2} /sc ONCE /st 10:00 /tn installSelector /tr {3} /rl HIGHEST /ru Local /IT", ip, username, password, installPath); string runTaskArg = string.Format(@" /run /s {0} -u domainname\{1} -p {2} /tn install", ip, username, password); ; string deleteTaskArg = string.Format(@" /delete /s {0} -u domainname\{1} -p {2} /tn install /F", ip, username, password);
schtasks /create 建立計劃任務
schtasks /query 查詢計劃任務
schtasks /run z執行計劃任務
schtasks /delete 刪除計劃任務
ip:遠端PC的IP地址
username:遠端PC的登陸使用者名稱
password:遠端PC的登陸密碼
/tn 計劃任務的名字 /tr呼叫程式的路徑 /sc設定執行頻率 /rl 設定執行許可權
需要注意的是使用這個方法遠端呼叫程式會有相對路徑方面的問題,不建議在執行程式中使用相對路徑訪問其他檔案。
完整程式碼:
string creatTaskArg = string.Format(@" /create /s {0} -u domainname\{1} -p {2} /sc ONCE /st 10:00 /tn installSelector /tr {3} /rl HIGHEST /ru Local /IT", ip, username, password, installSelectorPath); string runTaskArg = string.Format(@" /run /s {0} -u domainname\{1} -p {2} /tn installSelector", ip, username, password); ; string deleteTaskArg = string.Format(@" /delete /s {0} -u domainname\{1} -p {2} /tn installSelector /F", ip, username, password); System.Diagnostics.Process p1 = new System.Diagnostics.Process(); p1.StartInfo.FileName = @"schtasks.exe"; p1.StartInfo.Arguments = string.Format(@" /query /s {0} -u domainname\{1} -p {2} /tn installSelector", ip, username, password); p1.StartInfo.UseShellExecute = false; p1.StartInfo.RedirectStandardError = true; p1.StartInfo.RedirectStandardOutput = true; p1.StartInfo.CreateNoWindow = true; p1.Start(); p1.WaitForExit(); string err = p1.StandardError.ReadToEnd(); string sop = p1.StandardOutput.ReadToEnd(); if (!string.IsNullOrEmpty(err) && string.IsNullOrEmpty(sop)) { p1.StartInfo.Arguments = creatTaskArg; p1.Start(); p1.WaitForExit(); err = p1.StandardError.ReadToEnd(); sop = p1.StandardOutput.ReadToEnd(); if (!sop.ToLower().Contains("success")) { throw new Exception(string.Format("Create schedule task failed on {0}", ip)); } } else { _logger.Error(err); } p1.StartInfo.Arguments = runTaskArg; p1.Start(); p1.WaitForExit(); err = p1.StandardError.ReadToEnd(); sop = p1.StandardOutput.ReadToEnd(); if (!string.IsNullOrEmpty(err) || !sop.ToLower().Contains("success")) { throw new Exception(string.Format("Run schedule task failed on {0}", ip)); } p1.StartInfo.Arguments = deleteTaskArg; p1.Start(); p1.WaitForExit(); err = p1.StandardError.ReadToEnd(); sop = p1.StandardOutput.ReadToEnd(); if (!string.IsNullOrEmpty(err) || !sop.ToLower().Contains("success")) { throw new Exception(string.Format("Delete schedule task failed on {0}", ip)); } p1.Close();