管理 ifaddrs 以在 Swift 中返回 MAC 地址
Posted
技术标签:
【中文标题】管理 ifaddrs 以在 Swift 中返回 MAC 地址【英文标题】:Manage ifaddrs to return MAC addresses as well in Swift 【发布时间】:2016-03-04 18:07:27 【问题描述】:我有以下代码可以成功检索连接到我的路由器的所有 IP。但我需要获取每个 IP 的 MAC 地址。
所以地址不是作为数组返回[ips]
,而是作为字典返回[ips:0000000, mac: 000000]
是否可以通过更改以下代码来实现(来自How to get Ip address in swift)?
func getIFAddresses() -> [String]
print("GET IF ADDRESSSE")
var addresses = [String]()
// Get list of all interfaces on the local machine:
var ifaddr : UnsafeMutablePointer<ifaddrs> = nil
if getifaddrs(&ifaddr) == 0
print("getifaddrs\(getifaddrs)")
// For each interface ...
for (var ptr = ifaddr; ptr != nil; ptr = ptr.memory.ifa_next)
let flags = Int32(ptr.memory.ifa_flags)
var addr = ptr.memory.ifa_addr.memory
print("flags\(flags)")
print("addr\(addr)")
// Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING)
if addr.sa_family == UInt8(AF_INET) || addr.sa_family == UInt8(AF_INET6)
print("addr.sa_family\(addr.sa_family)")
// Convert interface address to a human readable string:
var hostname = [CChar](count: Int(NI_MAXHOST), repeatedValue: 0)
print("hostname\(hostname)")
if (getnameinfo(
&addr, socklen_t(addr.sa_len),
&hostname,
socklen_t(hostname.count),
nil,
socklen_t(0), NI_NUMERICHOST) == 0)
if let address = String.fromCString(hostname)
addresses.append(address)
freeifaddrs(ifaddr)
print("freeifaddrs\(freeifaddrs)")
print("ADDRESSES \(addresses)")
return addresses
【问题讨论】:
你试过了吗?代码有效吗?如果不是什么问题? @DanBeaulieu 你是对的,正在删除 cmets。我应该知道最好不要尝试解释为什么通常不建议投反对票。是的,我的答案已被复制并用作另一个问题的答案。 【参考方案1】:(备注/澄清:这是对“在 Swift 中管理 ifaddrs 以返回 MAC 地址” 和 “是否有可能修改 How to get Ip address in swift 的代码以返回 MAC 地址”。 这不是“检索连接到我的路由器的所有 IP”的解决方案 问题正文中也提到了这一点。)
这里是引用代码的扩展,它返回
本地(启动并运行)接口为
(接口名称、IP 地址、MAC 地址)的数组 元组。
MAC 地址是从AF_LINK
类型的接口检索的
它们在接口列表中存储为sockaddr_dl
结构。
这是一个可变长度结构,Swift 的严格类型系统使得一些指针杂耍变得必要。
重要提示:此代码旨在在 Mac 计算机上运行。 在 ios 设备上获取 MAC 地址不起作用。 iOS 有意返回“02:00:00:00:00:00”作为硬件地址 对于出于隐私原因的所有接口,请参见例如 Trouble with MAC address in iOS 7.0.2.)
func getInterfaces() -> [(name : String, addr: String, mac : String)]
var addresses = [(name : String, addr: String, mac : String)]()
var nameToMac = [ String : String ]()
// Get list of all interfaces on the local machine:
var ifaddr : UnsafeMutablePointer<ifaddrs> = nil
if getifaddrs(&ifaddr) == 0
// For each interface ...
var ptr = ifaddr
while ptr != nil
defer ptr = ptr.memory.ifa_next
let flags = Int32(ptr.memory.ifa_flags)
let addr = ptr.memory.ifa_addr
if let name = String.fromCString(ptr.memory.ifa_name)
// Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING)
if addr.memory.sa_family == UInt8(AF_LINK)
// Get MAC address from sockaddr_dl structure and store in nameToMac dictionary:
let dl = UnsafePointer<sockaddr_dl>(ptr.memory.ifa_addr)
let lladdr = UnsafeBufferPointer(start: UnsafePointer<Int8>(dl) + 8 + Int(dl.memory.sdl_nlen),
count: Int(dl.memory.sdl_alen))
if lladdr.count == 6
nameToMac[name] = lladdr.map String(format:"%02hhx", $0).joinWithSeparator(":")
if addr.memory.sa_family == UInt8(AF_INET) || addr.memory.sa_family == UInt8(AF_INET6)
// Convert interface address to a human readable string:
var hostname = [CChar](count: Int(NI_MAXHOST), repeatedValue: 0)
if (getnameinfo(addr, socklen_t(addr.memory.sa_len), &hostname, socklen_t(hostname.count),
nil, socklen_t(0), NI_NUMERICHOST) == 0)
if let address = String.fromCString(hostname)
addresses.append( (name: name, addr: address, mac : "") )
freeifaddrs(ifaddr)
// Now add the mac address to the tuples:
for (i, addr) in addresses.enumerate()
if let mac = nameToMac[addr.name]
addresses[i] = (name: addr.name, addr: addr.addr, mac : mac)
return addresses
你必须添加
#include <ifaddrs.h>
#include <net/if_dl.h>
到桥接头文件进行编译。
示例用法:
for addr in getInterfaces()
print(addr)
// ("en0", "fe80::1234:7fff:fe2e:8765%en0", "a9:55:6f:2e:57:78")
// ("en0", "192.168.2.108", "a9:55:6f:2e:57:78")
// ...
Swift 3 (Xcode 8) 更新:
func getInterfaces() -> [(name : String, addr: String, mac : String)]
var addresses = [(name : String, addr: String, mac : String)]()
var nameToMac = [ String: String ]()
// Get list of all interfaces on the local machine:
var ifaddr : UnsafeMutablePointer<ifaddrs>?
guard getifaddrs(&ifaddr) == 0 else return []
guard let firstAddr = ifaddr else return []
// For each interface ...
for ptr in sequence(first: firstAddr, next: $0.pointee.ifa_next )
let flags = Int32(ptr.pointee.ifa_flags)
if let addr = ptr.pointee.ifa_addr
let name = String(cString: ptr.pointee.ifa_name)
// Check for running IPv4, IPv6 interfaces. Skip the loopback interface.
if (flags & (IFF_UP|IFF_RUNNING|IFF_LOOPBACK)) == (IFF_UP|IFF_RUNNING)
switch Int32(addr.pointee.sa_family)
case AF_LINK:
// Get MAC address from sockaddr_dl structure and store in nameToMac dictionary:
addr.withMemoryRebound(to: sockaddr_dl.self, capacity: 1) dl in
dl.withMemoryRebound(to: Int8.self, capacity: 8 + Int(dl.pointee.sdl_nlen + dl.pointee.sdl_alen))
let lladdr = UnsafeBufferPointer(start: $0 + 8 + Int(dl.pointee.sdl_nlen),
count: Int(dl.pointee.sdl_alen))
if lladdr.count == 6
nameToMac[name] = lladdr.map String(format:"%02hhx", $0).joined(separator: ":")
case AF_INET, AF_INET6:
// Convert interface address to a human readable string:
var hostname = [CChar](repeating: 0, count: Int(NI_MAXHOST))
if (getnameinfo(addr, socklen_t(addr.pointee.sa_len),
&hostname, socklen_t(hostname.count),
nil, socklen_t(0), NI_NUMERICHOST) == 0)
let address = String(cString: hostname)
addresses.append( (name: name, addr: address, mac : "") )
default:
break
freeifaddrs(ifaddr)
// Now add the mac address to the tuples:
for (i, addr) in addresses.enumerated()
if let mac = nameToMac[addr.name]
addresses[i] = (name: addr.name, addr: addr.addr, mac : mac)
return addresses
【讨论】:
感谢您花时间研究它。我很感激。很抱歉在使用您的代码时没有将您的积分添加到我的问题中。 此代码旨在让所有设备连接到 wifi 网络或我的本地机器信息? @GuiSoySauce:这适用于本地机器上的接口(参考代码***.com/questions/25626117/…和您上一个问题中的代码chrisandtennille.com/code/IPAddress.c也是如此)。 @Martin R 如何在 swift 框架的模块映射中添加 if_dl.h 出现此错误:“必须从模块 'Darwin.POSIX.sys.types' 之前导入'u_char'声明是必需的“ @Martin R 我解决了这个问题,我在这个答案链接中添加了:***.com/questions/39036255/…【参考方案2】:如果你将在框架中使用它,你必须在其中添加一个带有此配置的 .modelmap 文件
module ifaddrs [system] [extern_c]
header
"/Applications/Xcode_7.3.1.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS.sdk/usr/include/ifaddrs.h"
export *
module net [system] [extern_c]
module types
header "/usr/include/sys/types.h"
export *
module if_dl
header
"/usr/include/net/if_dl.h"
export *
然后在你的 .swift 文件中
import ifaddrs
import net.if_dl
【讨论】:
以上是关于管理 ifaddrs 以在 Swift 中返回 MAC 地址的主要内容,如果未能解决你的问题,请参考以下文章