标签:powershell trace-command pipe
今天在powershell.com 看见了一个很有趣的问题。问题如下,有一个csv文件,大概内容是这样的
然后请问为什么这个命令不工作?
import-csv name.csv | get-service
报错:
理论上管道应该可以通过byvalue或者bypropertyname传递,get-service有-computername这个选项,前面导入的csv文件也有comutername这个属性,理论上应该自动匹配才对啊?那么问题出在哪里呢?
管道的基本实现方式参见 http://beanxyz.blog.51cto.com/5570417/1678609
我们可以用trace-command 命令来跟踪具体发生了什么
trace-command -PSHost -name ParameterBinding -Expression{ import-csv name.csv | get-service }
首先是import-csv的命令,没问题
然后接下来管道要把他获取的内容传递给 get-service
注意看我以为会自动匹配的computername并没有自动匹配进去,反而是name这个参数匹配进去了!尽管他第一次匹配因为类型不同忽略了,第二次他自动进行了一个类型转换,强行和name匹配成功
原因何在呢? 查看一下get-service的帮助文档。小技巧:我一般是使用-showwindows 或者-online,这样我可以新开一个窗口进行搜索,不过win10和ps5有个bug,-show的内容不完善所以我使用在线查看。
help get-service -online
搜索byvalue,我们可以看见有两个参数接受,分别是
搜索bypropertyname, 也有两个参数接受这个方式:
然后再看看position的参数, 发现-Name是设定为1的,也就是说默认情况下如果我们没有指定参数,他会认为自动用-name来匹配对应的字符串!
当管道尝试匹配的时候,因为我们没有指定参数,他会自动用-Name参数来进行匹配,尽管他的类型对不上,他会自动把PSCustomObject转换成字符串进行尝试,这样一来类型一样了,他就自动匹配成了-Name而不是-ComputerName,因为后者根本没有机会来匹配。
怎么解决呢?很简单我们需要明确告诉他哪些参数和属性要配对。
比如
import-csv name.csv | get-service -name * -ComputerName {$_.computername}
或者
get-service -computername ((import-csv name.csv ).computername)
这样就行啦。
这个命令也不会工作,尽管管道前面和后面都有一样的名字“computerName”
get-adcomputer -filter * | select @{n=‘computername‘;e={$_.name}} |invoke-command -ScriptBlock { dir }
为什么?因为
Invoke-command 的computername根本就不支持管道
他的-inputobject 接受任何类型!!也就是说任何尝试传递给invoke-command的管道参数都会被接管,其他的管道参数设置是毫无用处的。
怎么解决?和上面一样直接跑吧 别用管道了
invoke-command -ScriptBlock { dir } -computername (get-adcomputer -filter * | select -expand name)
本文出自 “麻婆豆腐” 博客,请务必保留此出处http://beanxyz.blog.51cto.com/5570417/1751700
标签:powershell trace-command pipe
原文地址:http://beanxyz.blog.51cto.com/5570417/1751700