39

我试图检查端口是否打开,如下所示使用powershell。

(new-object Net.Sockets.TcpClient).Connect("10.45.23.109", 443)

此方法有效,但输出对用户不友好。这意味着如果没有错误,则它可以访问。有没有办法检查成功并显示一些消息,如“端口 443 正在运行”?

4

10 回答 10

67

如果您运行的是 Windows 8/Windows Server 2012 或更高版本,则可以在 PowerShell中使用Test-NetConnection命令。

前任:

Test-NetConnection -Port 53 -ComputerName LON-DC1
于 2014-09-18T23:33:45.567 回答
44

我以多种方式改进了 Salselvaprabu 的答案:

  1. 它现在是一项功能 - 您可以放入您的 powershell 配置文件并在需要时随时使用
  2. 它可以接受主机作为主机名或 IP 地址
  3. 如果主机或端口不可用,则不再有例外 - 只是文本

像这样称呼它:

Test-Port example.com 999
Test-Port 192.168.0.1 80

function Test-Port($hostname, $port)
{
    # This works no matter in which form we get $host - hostname or ip address
    try {
        $ip = [System.Net.Dns]::GetHostAddresses($hostname) | 
            select-object IPAddressToString -expandproperty  IPAddressToString
        if($ip.GetType().Name -eq "Object[]")
        {
            #If we have several ip's for that address, let's take first one
            $ip = $ip[0]
        }
    } catch {
        Write-Host "Possibly $hostname is wrong hostname or IP"
        return
    }
    $t = New-Object Net.Sockets.TcpClient
    # We use Try\Catch to remove exception info from console if we can't connect
    try
    {
        $t.Connect($ip,$port)
    } catch {}

    if($t.Connected)
    {
        $t.Close()
        $msg = "Port $port is operational"
    }
    else
    {
        $msg = "Port $port on $ip is closed, "
        $msg += "You may need to contact your IT team to open it. "                                 
    }
    Write-Host $msg
}
于 2014-03-21T08:02:49.773 回答
16

实际上,Shay levy 的回答几乎是正确的,但正如我在他的评论栏中提到的那样,我遇到了一个奇怪的问题。所以我将命令分成两行,它工作正常。

$Ipaddress= Read-Host "Enter the IP address:"
$Port= Read-host "Enter the port number to access:"

$t = New-Object Net.Sockets.TcpClient
$t.Connect($Ipaddress,$Port)
    if($t.Connected)
    {
        "Port $Port is operational"
    }
    else
    {
        "Port $Port is closed, You may need to contact your IT team to open it. "
    }
于 2012-03-08T07:06:40.573 回答
14

您可以检查 Connected 属性是否设置为 $true 并显示友好消息:

    $t = New-Object Net.Sockets.TcpClient "10.45.23.109", 443 

    if($t.Connected)
    {
        "Port 443 is operational"
    }
    else
    {
        "..."
    }
于 2012-03-05T12:54:24.550 回答
8

在最新版本的 PowerShell 中,有一个新的 cmdlet,Test-NetConnection。

实际上,此 cmdlet 允许您 ping 端口,如下所示:

Test-NetConnection -ComputerName <remote server> -Port nnnn

我知道这是一个老问题,但如果你点击这个页面(就像我一样)寻找这个信息,这个添加可能会有所帮助!

于 2017-07-18T19:41:17.787 回答
4

我试图改进 mshutov 的建议。我添加了将输出用作对象的选项。

 function Test-Port($hostname, $port)
    {
    # This works no matter in which form we get $host - hostname or ip address
    try {
        $ip = [System.Net.Dns]::GetHostAddresses($hostname) | 
            select-object IPAddressToString -expandproperty  IPAddressToString
        if($ip.GetType().Name -eq "Object[]")
        {
            #If we have several ip's for that address, let's take first one
            $ip = $ip[0]
        }
    } catch {
        Write-Host "Possibly $hostname is wrong hostname or IP"
        return
    }
    $t = New-Object Net.Sockets.TcpClient
    # We use Try\Catch to remove exception info from console if we can't connect
    try
    {
        $t.Connect($ip,$port)
    } catch {}

    if($t.Connected)
    {
        $t.Close()
        $object = [pscustomobject] @{
                        Hostname = $hostname
                        IP = $IP
                        TCPPort = $port
                        GetResponse = $True }
        Write-Output $object
    }
    else
    {
        $object = [pscustomobject] @{
                        Computername = $IP
                        TCPPort = $port
                        GetResponse = $False }
        Write-Output $object

    }
    Write-Host $msg
}
于 2014-08-04T09:56:35.093 回答
4

如果您使用的是旧版本的 Powershell,其中 Test-NetConnection 不可用,这里是主机名“my.hostname”和端口“123”的单行:

$t = New-Object System.Net.Sockets.TcpClient 'my.hostname', 123; if($t.Connected) {"OK"}

返回 OK 或错误消息。

于 2017-10-19T21:54:14.077 回答
0

将其归结为单行将变量“$port389Open”设置为 True 或 false - 它可以快速且轻松地复制端口列表

try{$socket = New-Object Net.Sockets.TcpClient($ipAddress,389);if($socket -eq $null){$Port389Open = $false}else{Port389Open = $true;$socket.close()}}catch{Port389Open = $false}

如果您想真正发疯,可以返回整个数组-

Function StdPorts($ip){
    $rst = "" |  select IP,Port547Open,Port135Open,Port3389Open,Port389Open,Port53Open
    $rst.IP = $Ip
    try{$socket = New-Object Net.Sockets.TcpClient($ip,389);if($socket -eq $null){$rst.Port389Open = $false}else{$rst.Port389Open = $true;$socket.close();$ipscore++}}catch{$rst.Port389Open = $false}
    try{$socket = New-Object Net.Sockets.TcpClient($ip,53);if($socket -eq $null){$rst.Port53Open = $false}else{$rst.Port53Open = $true;$socket.close();$ipscore++}}catch{$rst.Port53Open = $false}
    try{$socket = New-Object Net.Sockets.TcpClient($ip,3389);if($socket -eq $null){$rst.Port3389Open = $false}else{$rst.Port3389Open = $true;$socket.close();$ipscore++}}catch{$rst.Port3389Open = $false}
    try{$socket = New-Object Net.Sockets.TcpClient($ip,547);if($socket -eq $null){$rst.Port547Open = $false}else{$rst.Port547Open = $true;$socket.close();$ipscore++}}catch{$rst.Port547Open = $false}
    try{$socket = New-Object Net.Sockets.TcpClient($ip,135);if($socket -eq $null){$rst.Port135Open = $false}else{$rst.Port135Open = $true;$socket.close();$SkipWMI = $False;$ipscore++}}catch{$rst.Port135Open = $false}
    Return $rst
}
于 2018-04-01T21:49:12.563 回答
0

mshutov 和 Salselvaprabu 给出了很好的答案。我需要一些更强大的东西,并且检查所有提供的 IPAddress,而不是只检查第一个。

我还想复制一些参数名称和功能,而不是 Test-Connection 功能。

这个新功能允许您设置重试次数的计数,以及每次尝试之间的延迟。享受!

function Test-Port {

    [CmdletBinding()]
    Param (
        [string] $ComputerName,
        [int] $Port,
        [int] $Delay = 1,
        [int] $Count = 3
    )

    function Test-TcpClient ($IPAddress, $Port) {

        $TcpClient = New-Object Net.Sockets.TcpClient
        Try { $TcpClient.Connect($IPAddress, $Port) } Catch {}

        If ($TcpClient.Connected) { $TcpClient.Close(); Return $True }
        Return $False

    }

    function Invoke-Test ($ComputerName, $Port) {

        Try   { [array]$IPAddress = [System.Net.Dns]::GetHostAddresses($ComputerName) | Select-Object -Expand IPAddressToString } 
        Catch { Return $False }

        [array]$Results = $IPAddress | % { Test-TcpClient -IPAddress $_ -Port $Port }
        If ($Results -contains $True) { Return $True } Else { Return $False }

    }

    for ($i = 1; ((Invoke-Test -ComputerName $ComputerName -Port $Port) -ne $True); $i++)
    {
        if ($i -ge $Count) {
            Write-Warning "Timed out while waiting for port $Port to be open on $ComputerName!"
            Return $false
        }

        Write-Warning "Port $Port not open, retrying..."
        Sleep $Delay
    }

    Return $true

}
于 2017-10-25T13:10:02.653 回答
-1

扫描关闭的端口时,它会长时间无响应。将 fqdn 解析为 ip 时似乎更快,例如:

[System.Net.Dns]::GetHostAddresses("www.msn.com").IPAddressToString
于 2015-04-17T18:05:45.267 回答