今天在powershell.com 看见了一个很有趣的问题。问题如下,有一个csv文件,大概内容是这样的

wKioL1bov12Cq2-NAAAJxMn4ZLE508.png


然后请问为什么这个命令不工作?


import-csv name.csv | get-service

报错:

Trace-Commnd 跟踪管道参数传递_第1张图片


理论上管道应该可以通过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的命令,没问题

Trace-Commnd 跟踪管道参数传递_第2张图片

然后接下来管道要把他获取的内容传递给 get-service


注意看我以为会自动匹配的computername并没有自动匹配进去,反而是name这个参数匹配进去了!尽管他第一次匹配因为类型不同忽略了,第二次他自动进行了一个类型转换,强行和name匹配成功

Trace-Commnd 跟踪管道参数传递_第3张图片


原因何在呢? 查看一下get-service的帮助文档。小技巧:我一般是使用-showwindows 或者-online,这样我可以新开一个窗口进行搜索,不过win10和ps5有个bug,-show的内容不完善所以我使用在线查看。

help get-service -online


搜索byvalue,我们可以看见有两个参数接受,分别是

-InputObject

-Name


搜索bypropertyname, 也有两个参数接受这个方式:

-ComputerName

-Name


然后再看看position的参数, 发现-Name是设定为1的,也就是说默认情况下如果我们没有指定参数,他会认为自动用-name来匹配对应的字符串!


Trace-Commnd 跟踪管道参数传递_第4张图片


当管道尝试匹配的时候,因为我们没有指定参数,他会自动用-Name参数来进行匹配,尽管他的类型对不上,他会自动把PSCustomObject转换成字符串进行尝试,这样一来类型一样了,他就自动匹配成了-Name而不是-ComputerName,因为后者根本没有机会来匹配。

Trace-Commnd 跟踪管道参数传递_第5张图片


怎么解决呢?很简单我们需要明确告诉他哪些参数和属性要配对。

比如

import-csv name.csv | get-service -name * -ComputerName {$_.computername}


或者

get-service -computername ((import-csv name.csv ).computername)

这样就行啦。


还有一个类似的例子 (http://powershell.com/cs/blogs/donjones/archive/2011/12/10/troubleshooting-pipeline-parameter-binding-by-peeking-inside.aspx)


这个命令也不会工作,尽管管道前面和后面都有一样的名字“computerName”

get-adcomputer -filter * | select @{n='computername';e={$_.name}} |invoke-command -ScriptBlock { dir }


为什么?因为

  1. Invoke-command 的computername根本就不支持管道

  2. 他的-inputobject 接受任何类型!!也就是说任何尝试传递给invoke-command的管道参数都会被接管,其他的管道参数设置是毫无用处的。


Trace-Commnd 跟踪管道参数传递_第6张图片

    Trace-Commnd 跟踪管道参数传递_第7张图片

怎么解决?和上面一样直接跑吧 别用管道了

invoke-command -ScriptBlock { dir } -computername (get-adcomputer -filter * | select -expand name)