golang獲取網卡信息操作
我就廢話不多說瞭,大傢還是直接看代碼吧~
package main import ( "fmt" "os/exec" "strings" "github.com/safchain/ethtool" ) func main() { baseNicPath := "/sys/class/net/" cmd := exec.Command("ls", baseNicPath) buf, err := cmd.Output() if err != nil { //fmt.Println("Error:", err) return } output := string(buf) for _, device := range strings.Split(output, "\n") { if len(device) > 1 { fmt.Println(device) ethHandle, err := ethtool.NewEthtool() if err != nil { panic(err.Error()) } defer ethHandle.Close() stats, err := ethHandle.LinkState(device) if err != nil { panic(err.Error()) } fmt.Printf("LinkName: %s LinkState: %d\n", device, stats) } } } ©
補充:GO 語言取得 Ethernet 類型的網卡地址
在 C# 中取得 Ethernet 類型的網卡地址
在 C# 中,因為有 NetworkInterface .NetworkInterfaceType == NetworkInterfaceType.Ethernet 所以,很容易在代碼中進行判斷。
public static PhysicalAddress GetMacAddress() { foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces()) { // Only consider Ethernet network interfaces if (nic.NetworkInterfaceType == NetworkInterfaceType.Ethernet) // nic.OperationalStatus == OperationalStatus.Up { return nic.GetPhysicalAddress(); } } return null; }
在 GO 中的處理辦法
搜索百度百十回,沒有找到答案,無奈,搜索 net 包的源碼,找到蛛絲馬跡,在未公開的方法中,找到一個類型的判斷語句。net/interface_windows.go 中,有 case windows.IF_TYPE_ETHERNET_CSMACD 的類型判斷。
再查找 IF_TYPE_ETHERNET_CSMACD 的資料,正是這裡所需要的。
參考:IP_INTERFACE_NAME_INFO_W2KSP1 structure
於是,將需要的部分復制出來加以改造,得到 IsEthernet 方法,結合 net 包中已有的方法,參數使用 net.Interface 的 Index 來使用。
完整代碼如下:
package machine import ( "errors" "net" "os" "strings" "syscall" "unsafe" "golang.org/x/sys/windows" ) func GetMACAddress() (string, error) { netInterfaces, err := net.Interfaces() if err != nil { panic(err.Error()) } mac, macerr := "", errors.New("no valid mac address") for i := 0; i < len(netInterfaces); i++ { if (netInterfaces[i].Flags&net.FlagLoopback) == 0 && strings.Contains(netInterfaces[i].Flags.String(), "broadcast") { index := netInterfaces[i].Index if isEthernet(index) { mac = netInterfaces[i].HardwareAddr.String() return mac, nil } } } return mac, macerr } // 根據網卡接口 Index 判斷其是否為 Ethernet 網卡 func isEthernet(ifindex int) bool { aas, err := adapterAddresses() if err != nil { return false } result := false for _, aa := range aas { index := aa.IfIndex if ifindex == int(index) { switch aa.IfType { case windows.IF_TYPE_ETHERNET_CSMACD: result = true } if result { break } } } return result } // 從 net/interface_windows.go 中復制過來 func adapterAddresses() ([]*windows.IpAdapterAddresses, error) { var b []byte l := uint32(15000) // recommended initial size for { b = make([]byte, l) err := windows.GetAdaptersAddresses(syscall.AF_UNSPEC, windows.GAA_FLAG_INCLUDE_PREFIX, 0, (*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])), &l) if err == nil { if l == 0 { return nil, nil } break } if err.(syscall.Errno) != syscall.ERROR_BUFFER_OVERFLOW { return nil, os.NewSyscallError("getadaptersaddresses", err) } if l <= uint32(len(b)) { return nil, os.NewSyscallError("getadaptersaddresses", err) } } var aas []*windows.IpAdapterAddresses for aa := (*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])); aa != nil; aa = aa.Next { aas = append(aas, aa) } return aas, nil }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- GoLang讀取文件的10種方法實例
- Golang的strings.Split()踩坑記錄
- 基於go interface{}==nil 的幾種坑及原理分析
- Go語言學習筆記之錯誤和異常詳解
- Golang標準庫unsafe源碼解讀