Надежный метод получения MAC-адреса машины на C #

133

Мне нужен способ получить MAC-адрес машины независимо от ОС, в которой он работает, с использованием C #. Приложение должно работать на XP / Vista / Win7 32 и 64 бит, а также на этих ОС, но с иностранным языком по умолчанию. Многие команды C # и запросы ОС не работают в ОС. Любые идеи? Я очищал вывод «ipconfig / all», но это ужасно ненадежно, так как формат вывода отличается на каждой машине.

Спасибо


источник
7
Когда вы говорите «через ОС», вы имеете в виду разные ОС Microsoft?
Джон Велдон,

Ответы:

137

Более чистое решение

var macAddr = 
    (
        from nic in NetworkInterface.GetAllNetworkInterfaces()
        where nic.OperationalStatus == OperationalStatus.Up
        select nic.GetPhysicalAddress().ToString()
    ).FirstOrDefault();

Или:

String firstMacAddress = NetworkInterface
    .GetAllNetworkInterfaces()
    .Where( nic => nic.OperationalStatus == OperationalStatus.Up && nic.NetworkInterfaceType != NetworkInterfaceType.Loopback )
    .Select( nic => nic.GetPhysicalAddress().ToString() )
    .FirstOrDefault();
Мохаммед А. Фадил
источник
44
Или лямбда, если вам это нравится! return NetworkInterface.GetAllNetworkInterfaces().Where(nic => nic.OperationalStatus == OperationalStatus.Up).Select(nic => nic.GetPhysicalAddress().ToString()).FirstOrDefault();(Если это не ваша вещь, она должна быть вашей.)
GONeale
7
Краткий способ стать самым быстрым: var networks = NetworkInterface.GetAllNetworkInterfaces(); var activeNetworks = networks.Where(ni => ni.OperationalStatus == OperationalStatus.Up && ni.NetworkInterfaceType != NetworkInterfaceType.Loopback); var sortedNetworks = activeNetworks.OrderByDescending(ni => ni.Speed); return sortedNetworks.First().GetPhysicalAddress().ToString();
Грэм Лэйт
1
Выбор первого - не всегда лучший вариант. Выбор наиболее часто используемого соединения: stackoverflow.com/a/51821927/3667
Рамунас,
Примечание по оптимизации: вы можете позвонить FirstOrDefaultдо финала Select. Таким образом, он получит только физический адрес и сериализует его по фактическому полученному NetworkInterfaceвами адресу . Не забудьте добавить нулевую отметку (?) После FirstOrDefault.
GregaMohorko
Вычислительный более быстрый способ получить это, вам не нужно оценивать все сети, которые соответствуют заданному условию, вам нужна только первая из них: NetworkInterface .GetAllNetworkInterfaces() .FirstOrDefault(nic => nic.OperationalStatus == OperationalStatus.Up && nic.NetworkInterfaceType != NetworkInterfaceType.Loopback)? .GetPhysicalAddress().ToString();
Алессандро Муцци,
80

Вот код C #, который возвращает MAC-адрес первого рабочего сетевого интерфейса. Предполагая, что NetworkInterfaceсборка реализована в среде выполнения (например, Mono), используемой в других операционных системах, это будет работать в других операционных системах.

Новая версия: возвращает сетевой адаптер с максимальной скоростью, который также имеет действующий MAC-адрес.

/// <summary>
/// Finds the MAC address of the NIC with maximum speed.
/// </summary>
/// <returns>The MAC address.</returns>
private string GetMacAddress()
{
    const int MIN_MAC_ADDR_LENGTH = 12;
    string macAddress = string.Empty;
    long maxSpeed = -1;

    foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
    {
        log.Debug(
            "Found MAC Address: " + nic.GetPhysicalAddress() +
            " Type: " + nic.NetworkInterfaceType);

        string tempMac = nic.GetPhysicalAddress().ToString();
        if (nic.Speed > maxSpeed &&
            !string.IsNullOrEmpty(tempMac) &&
            tempMac.Length >= MIN_MAC_ADDR_LENGTH)
        {
            log.Debug("New Max Speed = " + nic.Speed + ", MAC: " + tempMac);
            maxSpeed = nic.Speed;
            macAddress = tempMac;
        }
    }

    return macAddress;
}

Исходная версия: просто возвращает первую.

/// <summary>
/// Finds the MAC address of the first operation NIC found.
/// </summary>
/// <returns>The MAC address.</returns>
private string GetMacAddress()
{
    string macAddresses = string.Empty;

    foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
    {
        if (nic.OperationalStatus == OperationalStatus.Up)
        {
            macAddresses += nic.GetPhysicalAddress().ToString();
            break;
        }
    }

    return macAddresses;
}

Единственное, что мне не нравится в этом подходе, это то, что если у вас есть вроде Nortel Packet Miniport или какой-то тип VPN-соединения, он может быть выбран. Насколько я могу судить, нет способа отличить MAC реального физического устройства от виртуального сетевого интерфейса какого-либо типа.

blak3r
источник
6
Не выбирайте только первый рабочий интерфейс. Это может вернуть интерфейсы обратной петли, иногда подключаемые карты 3G и т. Д., Что, вероятно, не то, что вам нужно. NetworkInterfaceType ( msdn.microsoft.com/en-us/library/… ) предоставит вам дополнительную информацию о соединении NetworkInterface, чтобы вы могли сделать более осознанный выбор. Также имейте в виду, что на машине может быть много активных соединений, и их порядок может быть непредсказуемым.
Дэйв Р.
@DaveR. Я посмотрел на NetworkInterfaceType, он в основном почти всегда возвращает Ethernet, даже если это был виртуальный адаптер, по моему опыту, поэтому я нашел его довольно бесполезным.
blak3r
1
Вам следует выбрать интерфейс с самым низким значением GatewayMetric. Это должно быть соединение с «самым быстрым, надежным или наименее ресурсоемким маршрутом». В основном это даст вам интерфейс, который предпочитает использовать Windows. Однако я думаю, что для этого вам понадобится WMI. Я посмотрю, смогу ли я заставить это работать ...
AVee
6
Для полноты, доступ к классу NetworkInterface осуществляется с помощьюusing System.Net.NetworkInformation;
iancoleman
1
FWIW, если у вас установлен гигабитный сетевой адаптер и Hyper-V, у вас также будет 10-гигабитный виртуальный сетевой адаптер. :) Сложно решить ...
Кристофер Пейнтер
10

MACADDRESS свойство класса Win32_NetworkAdapterConfiguration WMI может предоставить вам MAC - адрес закрепительного в. (Пространство имен System.Management)

MACAddress

    Data type: string
    Access type: Read-only

    Media Access Control (MAC) address of the network adapter. A MAC address is assigned by the manufacturer to uniquely identify the network adapter.

    Example: "00:80:C7:8F:6C:96"

Если вы не знакомы с WMI API (инструментарий управления Windows), здесь есть хороший обзор приложений .NET.

WMI доступен во всех версиях окон со средой выполнения .Net.

Вот пример кода:

System.Management.ManagementClass mc = default(System.Management.ManagementClass);
ManagementObject mo = default(ManagementObject);
mc = new ManagementClass("Win32_NetworkAdapterConfiguration");

ManagementObjectCollection moc = mc.GetInstances();
    foreach (var mo in moc) {
        if (mo.Item("IPEnabled") == true) {
              Adapter.Items.Add("MAC " + mo.Item("MacAddress").ToString());
         }
     }
Баярд Рандель
источник
9

WMI - лучшее решение, если машина, к которой вы подключаетесь, является машиной Windows, но если вы смотрите на сетевой адаптер linux, mac или другой тип, вам нужно будет использовать что-то еще. Вот несколько вариантов:

  1. Используйте команду DOS nbtstat -a. Создайте процесс, вызовите эту команду, проанализируйте вывод.
  2. Сначала проверьте IP-адрес, чтобы убедиться, что ваш сетевой адаптер кэширует команду в своей таблице ARP, затем используйте команду DOS arp -a. Анализируйте вывод процесса, как в варианте 1.
  3. Используйте ужасный неуправляемый вызов sendarp в iphlpapi.dll

Вот образец предмета №3. Это кажется лучшим вариантом, если WMI не является жизнеспособным решением:

using System.Runtime.InteropServices;
...
[DllImport("iphlpapi.dll", ExactSpelling = true)]
        public static extern int SendARP(int DestIP, int SrcIP, byte[] pMacAddr, ref uint PhyAddrLen);
...
private string GetMacUsingARP(string IPAddr)
{
    IPAddress IP = IPAddress.Parse(IPAddr);
    byte[] macAddr = new byte[6];
    uint macAddrLen = (uint)macAddr.Length;

    if (SendARP((int)IP.Address, 0, macAddr, ref macAddrLen) != 0)
        throw new Exception("ARP command failed");

    string[] str = new string[(int)macAddrLen];
    for (int i = 0; i < macAddrLen; i++)
        str[i] = macAddr[i].ToString("x2");

    return string.Join(":", str);
}

Чтобы отдать должное, где это необходимо, это основа для этого кода: http://www.pinvoke.net/default.aspx/iphlpapi.sendarp#

Брайан Дункан
источник
Я искал то же самое, что и OP, и это именно то, что мне нужно!
QueueHammer
В вариантах 1 и 2 вы имеете в виду команды DOS, если вы работаете на компьютере с Windows, и эквивалентную команду в Linux или Mac, верно?
Raikol Amaro
8

Мы используем WMI, чтобы получить MAC-адрес интерфейса с наименьшей метрикой, например, окна интерфейса предпочтут использовать, например:

public static string GetMACAddress()
{
    ManagementObjectSearcher searcher = new ManagementObjectSearcher("SELECT * FROM Win32_NetworkAdapterConfiguration where IPEnabled=true");
    IEnumerable<ManagementObject> objects = searcher.Get().Cast<ManagementObject>();
    string mac = (from o in objects orderby o["IPConnectionMetric"] select o["MACAddress"].ToString()).FirstOrDefault();
    return mac;
}

Или в Silverlight (требуется повышенное доверие):

public static string GetMACAddress()
{
    string mac = null;
    if ((Application.Current.IsRunningOutOfBrowser) && (Application.Current.HasElevatedPermissions) && (AutomationFactory.IsAvailable))
    {
        dynamic sWbemLocator = AutomationFactory.CreateObject("WbemScripting.SWBemLocator");
        dynamic sWbemServices = sWbemLocator.ConnectServer(".");
        sWbemServices.Security_.ImpersonationLevel = 3; //impersonate

        string query = "SELECT * FROM Win32_NetworkAdapterConfiguration where IPEnabled=true";
        dynamic results = sWbemServices.ExecQuery(query);

        int mtu = int.MaxValue;
        foreach (dynamic result in results)
        {
            if (result.IPConnectionMetric < mtu)
            {
                mtu = result.IPConnectionMetric;
                mac = result.MACAddress;
            }
        }
    }
    return mac;
}
AVEE
источник
7
public static PhysicalAddress GetMacAddress()
{
    var myInterfaceAddress = NetworkInterface.GetAllNetworkInterfaces()
        .Where(n => n.OperationalStatus == OperationalStatus.Up && n.NetworkInterfaceType != NetworkInterfaceType.Loopback)
        .OrderByDescending(n => n.NetworkInterfaceType == NetworkInterfaceType.Ethernet)
        .Select(n => n.GetPhysicalAddress())
        .FirstOrDefault();

    return myInterfaceAddress;
}
Тони
источник
Если я запущу этот код, получит ли он адрес человека, запускающего приложение? Это означает, что он не получит IP-адрес сервера, на котором он размещен, верно?
Nate Pet
Он получает MAC-адрес хост-машины, сервера.
Тони
6

ИМХО, возвращение первого MAC-адреса не является хорошей идеей, особенно когда размещены виртуальные машины. Поэтому я проверяю сумму отправленных / полученных байтов и выбираю наиболее часто используемое соединение, которое не идеально, но должно быть правильным 9/10 раз.

public string GetDefaultMacAddress()
{
    Dictionary<string, long> macAddresses = new Dictionary<string, long>();
    foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
    {
        if (nic.OperationalStatus == OperationalStatus.Up)
            macAddresses[nic.GetPhysicalAddress().ToString()] = nic.GetIPStatistics().BytesSent + nic.GetIPStatistics().BytesReceived;
    }
    long maxValue = 0;
    string mac = "";
    foreach(KeyValuePair<string, long> pair in macAddresses)
    {
        if (pair.Value > maxValue)
        {
            mac = pair.Key;
            maxValue = pair.Value;
        }
    }
    return mac;
}
Ramunas
источник
6

Этот метод определит MAC-адрес сетевого интерфейса, используемого для подключения к указанному URL-адресу и порту.

Все ответы здесь не способны достичь этой цели.

Я написал этот ответ много лет назад (в 2014 году). Поэтому я решил немного "подтянуть лицо". Пожалуйста, посмотрите раздел обновлений

    /// <summary>
    /// Get the MAC of the Netowrk Interface used to connect to the specified url.
    /// </summary>
    /// <param name="allowedURL">URL to connect to.</param>
    /// <param name="port">The port to use. Default is 80.</param>
    /// <returns></returns>
    private static PhysicalAddress GetCurrentMAC(string allowedURL, int port = 80)
    {
        //create tcp client
        var client = new TcpClient();

        //start connection
        client.Client.Connect(new IPEndPoint(Dns.GetHostAddresses(allowedURL)[0], port));

        //wai while connection is established
        while(!client.Connected)
        {
            Thread.Sleep(500);
        }

        //get the ip address from the connected endpoint
        var ipAddress = ((IPEndPoint)client.Client.LocalEndPoint).Address;

        //if the ip is ipv4 mapped to ipv6 then convert to ipv4
        if(ipAddress.IsIPv4MappedToIPv6)
            ipAddress = ipAddress.MapToIPv4();        

        Debug.WriteLine(ipAddress);

        //disconnect the client and free the socket
        client.Client.Disconnect(false);
        
        //this will dispose the client and close the connection if needed
        client.Close();

        var allNetworkInterfaces = NetworkInterface.GetAllNetworkInterfaces();

        //return early if no network interfaces found
        if(!(allNetworkInterfaces?.Length > 0))
            return null;

        foreach(var networkInterface in allNetworkInterfaces)
        {
            //get the unicast address of the network interface
            var unicastAddresses = networkInterface.GetIPProperties().UnicastAddresses;
           
            //skip if no unicast address found
            if(!(unicastAddresses?.Count > 0))
                continue;

            //compare the unicast addresses to see 
            //if any match the ip address used to connect over the network
            for(var i = 0; i < unicastAddresses.Count; i++)
            {
                var unicastAddress = unicastAddresses[i];

                //this is unlikely but if it is null just skip
                if(unicastAddress.Address == null)
                    continue;
                
                var ipAddressToCompare = unicastAddress.Address;

                Debug.WriteLine(ipAddressToCompare);

                //if the ip is ipv4 mapped to ipv6 then convert to ipv4
                if(ipAddressToCompare.IsIPv4MappedToIPv6)
                    ipAddressToCompare = ipAddressToCompare.MapToIPv4();

                Debug.WriteLine(ipAddressToCompare);

                //skip if the ip does not match
                if(!ipAddressToCompare.Equals(ipAddress))
                    continue;

                //return the mac address if the ip matches
                return networkInterface.GetPhysicalAddress();
            }
              
        }

        //not found so return null
        return null;
    }

Чтобы вызвать его, вам нужно передать URL-адрес для подключения следующим образом:

var mac = GetCurrentMAC("www.google.com");

Вы также можете указать номер порта. Если не указано иное, значение по умолчанию - 80.

ОБНОВЛЕНИЕ:

2020

  • Добавлены комментарии для объяснения кода.
  • Исправлено для использования с новыми операционными системами, которые используют IPV4, сопоставленный с IPV6 (например, Windows 10).
  • Уменьшенная вложенность.
  • Модернизированный код использует "var".
Джонатан Альфаро
источник
1
Это очень интересно, я попробую, так как в моем случае я хотел бы, чтобы клиент обнаружил: а) адрес источника, используемый для связи с моим сервером (он НЕ обязательно будет через Интернет) и б) какой MAC-адрес принадлежит сетевому адаптеру, который предоставляет этот IP-адрес ...
Брайан Б.
5

Вы можете использовать NIC ID:

 foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces()) {
     if (nic.OperationalStatus == OperationalStatus.Up){
         if (nic.Id == "yay!")
     }
 }

Это не MAC-адрес, но это уникальный идентификатор, если это то, что вы ищете.

MMR
источник
2

Мне очень нравится решение AVee с самой низкой метрикой IP-соединения! Но если будет установлен второй nic с той же метрикой, сравнение MAC может не сработать ...

Лучше храни описание интерфейса с MAC. В более поздних сравнениях вы можете определить правильный ник по этой строке. Вот пример кода:

   public static string GetMacAndDescription()
    {
        ManagementObjectSearcher searcher = new ManagementObjectSearcher("SELECT * FROM Win32_NetworkAdapterConfiguration where IPEnabled=true");
        IEnumerable<ManagementObject> objects = searcher.Get().Cast<ManagementObject>();
        string mac = (from o in objects orderby o["IPConnectionMetric"] select o["MACAddress"].ToString()).FirstOrDefault();
        string description = (from o in objects orderby o["IPConnectionMetric"] select o["Description"].ToString()).FirstOrDefault();
        return mac + ";" + description;
    }

    public static string GetMacByDescription( string description)
    {
        ManagementObjectSearcher searcher = new ManagementObjectSearcher("SELECT * FROM Win32_NetworkAdapterConfiguration where IPEnabled=true");
        IEnumerable<ManagementObject> objects = searcher.Get().Cast<ManagementObject>();
        string mac = (from o in objects where o["Description"].ToString() == description select o["MACAddress"].ToString()).FirstOrDefault();
        return mac;
    }
Детлеф
источник
2

скажем, у меня есть TcpConnection, использующий мой локальный IP-адрес 192.168.0.182. Затем, если я захочу узнать MAC-адрес этого сетевого адаптера, я назову метод следующим образом:GetMacAddressUsedByIp("192.168.0.182")

public static string GetMacAddressUsedByIp(string ipAddress)
    {
        var ips = new List<string>();
        string output;

        try
        {
            // Start the child process.
            Process p = new Process();
            // Redirect the output stream of the child process.
            p.StartInfo.UseShellExecute = false;

            p.StartInfo.RedirectStandardOutput = true;
            p.StartInfo.UseShellExecute = false;
            p.StartInfo.CreateNoWindow = true;
            p.StartInfo.FileName = "ipconfig";
            p.StartInfo.Arguments = "/all";
            p.Start();
            // Do not wait for the child process to exit before
            // reading to the end of its redirected stream.
            // p.WaitForExit();
            // Read the output stream first and then wait.
            output = p.StandardOutput.ReadToEnd();
            p.WaitForExit();

        }
        catch
        {
            return null;
        }

        // pattern to get all connections
        var pattern = @"(?xis) 
(?<Header>
     (\r|\n) [^\r]+ :  \r\n\r\n
)
(?<content>
    .+? (?= ( (\r\n\r\n)|($)) )
)";

        List<Match> matches = new List<Match>();

        foreach (Match m in Regex.Matches(output, pattern))
            matches.Add(m);

        var connection = matches.Select(m => new
        {
            containsIp = m.Value.Contains(ipAddress),
            containsPhysicalAddress = Regex.Match(m.Value, @"(?ix)Physical \s Address").Success,
            content = m.Value
        }).Where(x => x.containsIp && x.containsPhysicalAddress)
        .Select(m => Regex.Match(m.content, @"(?ix)  Physical \s address [^:]+ : \s* (?<Mac>[^\s]+)").Groups["Mac"].Value).FirstOrDefault();

        return connection;
    }
Тоно Нам
источник
Это неэффективно ... Я бы не рекомендовал это делать.
Ivandro IG Jao
2

Мне действительно неприятно раскапывать этот старый пост, но я считаю, что этот вопрос заслуживает другого ответа, относящегося к Windows 8-10.

Используя NetworkInformation из пространства имен Windows.Networking.Connectivity , вы можете получить идентификатор используемого окнами сетевого адаптера. Затем вы можете получить MAC-адрес интерфейса из ранее упомянутого GetAllNetworkInterfaces ().

Это не будет работать в приложениях Магазина Windows, поскольку NetworkInterface в System.Net.NetworkInformation не предоставляет GetAllNetworkInterfaces.

string GetMacAddress()
{
    var connectionProfile = NetworkInformation.GetInternetConnectionProfile();
    if (connectionProfile == null) return "";

    var inUseId = connectionProfile.NetworkAdapter.NetworkAdapterId.ToString("B").ToUpperInvariant();
    if(string.IsNullOrWhiteSpace(inUseId)) return "";

    var mac = NetworkInterface.GetAllNetworkInterfaces()
        .Where(n => inUseId == n.Id)
        .Select(n => n.GetPhysicalAddress().GetAddressBytes().Select(b=>b.ToString("X2")))
        .Select(macBytes => string.Join(" ", macBytes))
        .FirstOrDefault();

    return mac;
}
Грег Горман
источник
2
string mac = "";
foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
            {

                if (nic.OperationalStatus == OperationalStatus.Up && (!nic.Description.Contains("Virtual") && !nic.Description.Contains("Pseudo")))
                {
                    if (nic.GetPhysicalAddress().ToString() != "")
                    {
                        mac = nic.GetPhysicalAddress().ToString();
                    }
                }
            }
MessageBox.Show(mac);
хади-сафари
источник
2
Этот ответ можно улучшить, кратко объяснив, что делает код и как он решает проблему.
Greg the Incredulous
1

Немного изменил свой код blak3r. Если у вас есть два адаптера с одинаковой скоростью. Сортируйте по MAC, чтобы всегда получать одно и то же значение.

public string GetMacAddress()
{
    const int MIN_MAC_ADDR_LENGTH = 12;
    string macAddress = string.Empty;
    Dictionary<string, long> macPlusSpeed = new Dictionary<string, long>();
    try
    {
        foreach(NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
        {
            System.Diagnostics.Debug.WriteLine("Found MAC Address: " + nic.GetPhysicalAddress() + " Type: " + nic.NetworkInterfaceType);

            string tempMac = nic.GetPhysicalAddress().ToString();

            if(!string.IsNullOrEmpty(tempMac) && tempMac.Length >= MIN_MAC_ADDR_LENGTH)
                macPlusSpeed.Add(tempMac, nic.Speed);
        }

        macAddress = macPlusSpeed.OrderByDescending(row => row.Value).ThenBy(row => row.Key).FirstOrDefault().Key;
    }
    catch{}

    System.Diagnostics.Debug.WriteLine("Fastest MAC address: " + macAddress);

    return macAddress;
}
user369122
источник
1
foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
{
     if (nic.OperationalStatus == OperationalStatus.Up)
     {
            PhysicalAddress Mac = nic.GetPhysicalAddress();
     }
}
Мохамад Джавади
источник
0

ipconfig.exeреализован с использованием различных DLL, включая iphlpapi.dll... Поиск в Google iphlpapiобнаруживает соответствующий API Win32, задокументированный в MSDN.

ChrisW
источник
0

Попробуй это:

    /// <summary>
    /// returns the first MAC address from where is executed 
    /// </summary>
    /// <param name="flagUpOnly">if sets returns only the nic on Up status</param>
    /// <returns></returns>
    public static string[] getOperationalMacAddresses(Boolean flagUpOnly)
    {
        string[] macAddresses = new string[NetworkInterface.GetAllNetworkInterfaces().Count()];

        int i = 0;
        foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
        {
            if (nic.OperationalStatus == OperationalStatus.Up || !flagUpOnly)
            {
                macAddresses[i] += ByteToHex(nic.GetPhysicalAddress().GetAddressBytes());
                //break;
                i++;
            }
        }
        return macAddresses;
    }
Ричард
источник