ps1aniuge
2018-11-08 17:19:05 +08:00
问:powershell 经过 ssh,远程运行命令,比 bash 强在哪?
答:
=============linux 远程命令 & ps 远程命令 对比例子============
linux:
ssh aaa@1.1.1.1 "以用户 aaa 权限执行的,命令 xxx"
ps:
$a = 1
[scriptblock]$备份命令 =
{
Get-Date
$b = $using:a + 2 #引用客户机变量,需要用$using:
}
$连接 1 = New-PSSession -HostName 1.1.1.1 -UserName root #手动输入密码或用-KeyFilePath 选项
invoke-command -ScriptBlock { $备份命令 } -Session $连接 1
============================================================
bash 的远程命令,简单直接。就好像我左手这盘蛋炒饭,简单解饿,但是不够强。更适用于 简单远程命令场合。
你再看看我右手这盘盖饭好在哪?答:生菜垫底,萝卜雕花围边。
bash 远程传递的是 [字符串] ,powershell 传递的是 [代码块] 。特色是 [对象垫底,大花括号围边] 。
字符串传递到远程时,经常需要要转义。代码块不用。
代码块,支持多行,格式化,使代码美观。
变量名,函数名支持中文。
代码块中,支持引用客户端变量,一律加上 [$using:] ,即客户机上的 [$a] ,在服务器上叫 [$using:a]
代码块中,支持引用服务器端变量,即服务器上的 [$a] ,在服务器还上叫 [$a]
即使变量重名,两个$a 也绝不会弄混。
ps 用大花括号包围代码,不用单双引号,代码嵌套很容易。
而代码嵌套容易,使的 ps 的 ssh 远程,从 server1 (跳板机,堡垒机)经 ssh 进入 server2,再 ssh 进入 server3,进入 33 层 ssh server 执行命令很容易。而 shell 难。
远程代码天生不老稳定的,有时没反应,或卡住,或中途断了。遇到此情形,每行 ps 代码都可以在外面套上 try-cacth,比 shell 更稳。
批量 ssh,ps 采用 [多线程] ,比 bash 用 [多进程] 快,时间准,省内存。
问:本地脚本和远程脚本有何区别?如何编写远程脚本?
答:
由于下述限制颇多,顾应尽量少用 [直接] 远程调用。多用 [进入远程机子后] 的本地调用。
1 模块化。要把远程任务,分成 n 个独立的单元。
我们编写的本地脚本,有时候没分的这么清楚。但远程脚本,就要分成 n 个脚本,或 n 个函数。
2 每个单元让它返回$true,或 false。
我们要改造本地脚本,人为让每个单元能返回$true,或 false。
好让我们远程判断每个单元是否执行成功。
3 对于偶尔会失败的任务,根据上述返回$true,或 false,把脚本改造成,有一定循环次数。
令脚本的每个单元,都让它形成闭环。
本地脚本中的本地命令,大都是不报错即成功。
对于 bash 这就尴尬了,癌症无解。powershell 有 try+catch 解决这问题。
4 每个单元让它,在远程机子上,写入 log。log 中带上日期时间。
5 每个单元,甚至!每个命令,必须!设定超时时间。
郭德纲说过,远程脚本,天生不老稳定的;)
本地脚本中的本地命令,大都不具备超时选项,远程运行时卡住就完了。
解决这个问题的最佳实践,是用多线程,和计时器。
把单元放入后台线程,然后主线程 sleep n 秒。超时就 kill 掉。
这是笨办法,好在简单,bash 也能实现。但 bash 没有线程,只有进程。
powershell 中,无需使用此笨办法。powershell 有计时器。计时器是多线程的。
new 一个计时器对象,并绑定代码后,你就可以-开始计时器-停止计时器-调整计时周期。
现在我们假定一个任务,前 3 次必然卡住超时,第 4 次必然成功。
那么我们只需在任务代码中,加上停止计时器的代码。并启动计时器即可。
则前 3 次运行到卡住代码,超时后被计时器重启。最后 1 次通过了卡住代码,运行了计时器停止代码。
这涉及到了两个线程之间传值,和控制。
而 bash 多进程,进程之间传值控制,不如线程之间方便。
所以说 [多线程,计时器,是给命令加上超时选项的最佳实践] 。
这里面还有一个细节。若你用 ps 的远程线程,来运行 [代码单元] 。可以起一个线程名,作为超时 kill 的标识符。
但是对于 bash,就不能用进程名了。你需要一个标明进程是唯一的方法。
这种方法是有,但是又要加不少代码。在这里就不做展开详谈了。