2

私は PSList が好きで、強制終了する必要があるプロセスを特定するために CPU と経過時間を使用します。次のようなことができるように、パイプライン値を返す powershell 関数でラップしたいと思います。

get-remoteprocess server1 | where  {$_.CPUMinutes -gt 4}

関数から値を返すことに行き詰まります-オブジェクトの配列である必要があることを理解しています。

私の最初のバージョンは次のようになります。

function get-remoteproc {
param ($hostname)
$results = pslist "\\$hostname" 

for ($i= 3; $i -le $results.length; $i++) {
    $strline = $results[$i]
    $StrWithPrefix = " "+$results[$i]
    $trimmedline = $StrWithPrefix -replace '\s+', " ";
    $Splitline = $trimmedline.Split(" ")
    $ProcessName = $Splitline[1]
    .
    .
    $ProcessCPUTime = ":"+[string]$Splitline[7]
    $SplitCpuTime = $ProcessCPUTime.Split(":")

    $CpuHours = $SplitCpuTime[1]
    $CpuMinutes = $SplitCpuTime[2]
    $CpuSeconds = $SplitCpuTime[3]
    .
    .
    .

    $obj = New-Object PSObject
       $obj | Add-Member Noteproperty -Name "Name" -Value $Name
       $obj | Add-Member Noteproperty -Name "PPid" -Value $Ppid
       $obj | Add-Member Noteproperty -Name "Pri" -Value $Pri

}

Doug Finke の提案を取り入れて、これは非常に簡潔ですが、パイプラインの値でうまく機能する、私のわずかに適応したバージョンを次に示します。

function get-remoteproc {
    param ($hostname=$env:COMPUTERNAME)

    $results = PsList "\\$hostname"

    foreach($record in $results[3..($results.Count)]) {
        # Remove spaces
        while ($record.IndexOf("  ") -ge 0) {
            $record = $record -replace "  "," "
        }

        $Name,$Processid,$Pri,$Thd,$Hnd,$Priv,$CPUTime,$ElapsedTime = $record.Split(" ")

        $properties = @{
            Name = $Name
            Pid = $Processid
            Pri = $Pri
            Thd = $Thd
            Hnd = $Hnd
            Priv = $Priv
            CPUTime = $CPUTime
            ElapsedTime = $ElapsedTime
        }

        New-Object PSObject -Property $properties |
            Add-Member -PassThru ScriptProperty CPUH   {[int]$this.CPUTime.Split(":")[0]} |
            Add-Member -PassThru ScriptProperty CPUM {[int]$this.CPUTime.Split(":")[1]} |
            Add-Member -PassThru ScriptProperty CPUS {[int]$this.CPUTime.Split(":")[2]} |
            Add-Member -PassThru ScriptProperty ElaH {[int]$this.ElapsedTime.Split(":")[0]} |
            Add-Member -PassThru ScriptProperty ElaM {[int]$this.ElapsedTime.Split(":")[1]} |
            Add-Member -PassThru ScriptProperty ElaS {[int]$this.ElapsedTime.Split(":")[2]} 
    }
}

そして、オブジェクトがパイプラインによって消費されるように正しく展開されていることを示す関数の呼び出し:

get-remoteproc "Server1" | where {(($_.CPUM * $_.CPUS) -gt 60) -and ($_.name -notmatch "Idle" )}|
ft name, pid, pri, thd, hnd, priv, CPUH, cpuM, cpuS, ElaH, ElaM, ElaS   -auto

みんな、ありがとう!

4

2 に答える 2

1

empo が指摘しているように、$obj をパイプラインに戻す必要があります。plist テキストを PowerShell オブジェクトに変換する別の方法を次に示します。

function get-remoteproc {
    param ($hostname=$env:COMPUTERNAME)

    $results = PsList "\\$hostname"

    foreach($record in $results[3..($results.Count)]) {
        # Remove spaces
        while ($record.IndexOf("  ") -ge 0) {
            $record = $record -replace "  "," "
        }

        $Name,$Processid,$Pri,$Thd,$Hnd,$Priv,$CPUTime,$ElapsedTime = $record.Split(" ")

        $properties = @{
            Name = $Name
            Pid = $Processid
            Pri = $Pri
            Thd = $Thd
            Hnd = $Hnd
            Priv = $Priv
            CPUTime = $CPUTime
            ElapsedTime = $ElapsedTime
        }

        New-Object PSObject -Property $properties |
            Add-Member -PassThru ScriptProperty CPUHours   {$this.CPUTime.Split(":")[0]} |
            Add-Member -PassThru ScriptProperty CPUMinutes {$this.CPUTime.Split(":")[1]} |
            Add-Member -PassThru ScriptProperty CPUSeconds {$this.CPUTime.Split(":")[2]} 

    }
}
于 2011-05-08T12:32:09.227 に答える