Swift 2检查端口是否忙碌

时间:2015-10-12 16:53:12

标签: network-programming port swift2

如果Port忙,有人可以告诉我如何在Swift2中找到它吗?

因为我编写了一个具有自编写Tcp服务器的mac应用程序,但有时它不会启动,因为它“无法绑定到端口”。那么如何检查端口是否未被使用,以阻止Tcp服务器的启动按钮,直到端口空闲,再次?

我不想要新的框架。

由于

2 个答案:

答案 0 :(得分:4)

主要来自Swifter的代码:https://github.com/glock45/swifter

func checkTcpPortForListen(port: in_port_t) -> (Bool, descr: String){

    let socketFileDescriptor = socket(AF_INET, SOCK_STREAM, 0)
    if socketFileDescriptor == -1 {
        return (false, "SocketCreationFailed, \(descriptionOfLastError())")
    }

    var addr = sockaddr_in()
      addr.sin_len = __uint8_t(sizeof(sockaddr_in))
      addr.sin_family = sa_family_t(AF_INET)
      addr.sin_port = Int(OSHostByteOrder()) == OSLittleEndian ? _OSSwapInt16(port) : port
      addr.sin_addr = in_addr(s_addr: inet_addr("0.0.0.0"))
      addr.sin_zero = (0, 0, 0, 0, 0, 0, 0, 0)
    var bind_addr = sockaddr()
    memcpy(&bind_addr, &addr, Int(sizeof(sockaddr_in)))

    if bind(socketFileDescriptor, &bind_addr, socklen_t(sizeof(sockaddr_in))) == -1 {
        let details = descriptionOfLastError()
        release(socketFileDescriptor)
        return (false, "\(port), BindFailed, \(details)")
    }
    if listen(socketFileDescriptor, SOMAXCONN ) == -1 {
        let details = descriptionOfLastError()
        release(socketFileDescriptor)
        return (false, "\(port), ListenFailed, \(details)")
    }
    release(socketFileDescriptor)
    return (true, "\(port) is free for use") 
}

func release(socket: Int32) {
    Darwin.shutdown(socket, SHUT_RDWR)
    close(socket)
}
func descriptionOfLastError() -> String {
    return String.fromCString(UnsafePointer(strerror(errno))) ?? "Error: \(errno)"
}

答案 1 :(得分:3)

更新Swift 4的正确答案:

<label data-bind="text: $parent.Value.SpecificValue"></label>

编辑: 调用此函数的示例:

func checkTcpPortForListen(port: in_port_t) -> (Bool, descr: String) {

    let socketFileDescriptor = socket(AF_INET, SOCK_STREAM, 0)
    if socketFileDescriptor == -1 {
        return (false, "SocketCreationFailed, \(descriptionOfLastError())")
    }

    var addr = sockaddr_in()
    let sizeOfSockkAddr = MemoryLayout<sockaddr_in>.size
    addr.sin_len = __uint8_t(sizeOfSockkAddr)
    addr.sin_family = sa_family_t(AF_INET)
    addr.sin_port = Int(OSHostByteOrder()) == OSLittleEndian ? _OSSwapInt16(port) : port
    addr.sin_addr = in_addr(s_addr: inet_addr("0.0.0.0"))
    addr.sin_zero = (0, 0, 0, 0, 0, 0, 0, 0)
    var bind_addr = sockaddr()
    memcpy(&bind_addr, &addr, Int(sizeOfSockkAddr))

    if Darwin.bind(socketFileDescriptor, &bind_addr, socklen_t(sizeOfSockkAddr)) == -1 {
        let details = descriptionOfLastError()
        release(socket: socketFileDescriptor)
        return (false, "\(port), BindFailed, \(details)")
    }
    if listen(socketFileDescriptor, SOMAXCONN ) == -1 {
        let details = descriptionOfLastError()
        release(socket: socketFileDescriptor)
        return (false, "\(port), ListenFailed, \(details)")
    }
    release(socket: socketFileDescriptor)
    return (true, "\(port) is free for use")
}

func release(socket: Int32) {
    Darwin.shutdown(socket, SHUT_RDWR)
    close(socket)
}

func descriptionOfLastError() -> String {
    return String.init(cString: (UnsafePointer(strerror(errno))))
}