6

我想使用唯一标识符来确定我的应用程序是否移动到另一台计算机。MAC 地址似乎适合此目的。我使用的代码是这样的:

Procedure TForm4.GetMacAddress;
var item: TListItem;
    objWMIService : OLEVariant;
    colItems      : OLEVariant;
    colItem       : OLEVariant;
    oEnum         : IEnumvariant;
    iValue        : LongWord;
    wmiHost, root, wmiClass: string;
    i: Int32;

  function GetWMIObject(const objectName: String): IDispatch;
  var
    chEaten: Integer;
    BindCtx: IBindCtx;//for access to a bind context
    Moniker: IMoniker;//Enables you to use a moniker object
  begin
    OleCheck(CreateBindCtx(0, bindCtx));
    OleCheck(MkParseDisplayName(BindCtx, StringToOleStr(objectName), chEaten, Moniker));//Converts a string into a moniker that identifies the object named by the string
    OleCheck(Moniker.BindToObject(BindCtx, nil, IDispatch, Result));//Binds to the specified object
  end;

begin
   wmiHost       := '.';
   root          := 'root\CIMV2';
   wmiClass      := 'Win32_NetworkAdapterConfiguration';
   objWMIService := GetWMIObject(Format('winmgmts:\\%s\%s',[wmiHost,root]));
   colItems      := objWMIService.ExecQuery(Format('SELECT * FROM %s',[wmiClass]),'WQL',0);
   oEnum         := IUnknown(colItems._NewEnum) as IEnumVariant;
   i := 0;
   while oEnum.Next(1, colItem, iValue) = 0 do
   begin
      Item := View.Items.Add;
      item.Caption := Copy (colItem.Caption, 2, 8);

      Item.SubItems.Add (colItem.Description);
      Item.SubItems.Add (colItem.ServiceName);
      Item.SubItems.Add (VarToStrNil (colItem.MACAddress));
      if (VarToStrNil(colItem.MACAddress) <> '')
         then Item.SubItems.Add ('yes')
         else Item.SubItems.Add ('no');
      if colItem.IPEnabled
         then Item.SubItems.Add ('yes')
         else Item.SubItems.Add ('no');
     Item.SubItems.Add (VarToStrNil (colItem.SettingID));
     Item.SubItems.Add (IntToStr (colItem.InterfaceIndex));
   end; // if
end; // GetMacAddress //

我的机器有一个网络端口,但这段代码找到了 18 个与网络相关的端口/事物/任何东西。其中有四个MAC地址。我假设一个网络端口应该启用 IP,以便留下两个(在图像中标记为 MAC)。假设在如此过滤的端口中,索引最低的端口是硬件端口是否正确?

在此处输入图像描述

在上面的快照中编辑Realtek 适配器是机器中唯一的物理适配器。另一个适配器是 VirtualBox 虚拟适配器。TLama 的答案识别了这两个适配器,但是有没有办法找到唯一的物理(Realtek)适配器的地址?

Update 1 EJP 指出可以更改 MAC 地址。这在某种程度上破坏了我的目的,但是当我正在寻找适合大多数情况的解决方案时,我决定接受它。

TLama 和 TOndrej 指出了几种解决方案。两者最终都会出现毫无疑问无法找到物理适配器的情况。

Update 2 TLama 的优秀阅读列表显示,可能没有确定物理适配器的方法。第一个项目符号中提到的文章展示了如何基于一些简单的假设来缩减适配器的数量。第三个项目中的文章展示了如何选择连接到 PCI 总线的适配器,这实际上正是我想知道的。文章中提到了一些奇怪的例外,但我认为这将在大多数情况下提供答案。

感谢大家的贡献!

4

2 回答 2

7

改用Win32_NetworkAdapter类。它有PhysicalAdapter成员。以下示例应列出物理适配器的 MAC 地址:

program Program1;

{$APPTYPE CONSOLE}

uses
  SysUtils, ActiveX, ComObj, Variants;

procedure GetWin32_NetworkAdapterInfo;
const
  WbemUser = '';
  WbemPassword = '';
  WbemComputer = 'localhost';
  wbemFlagForwardOnly = $00000020;
var
  ElementCount: LongWord;
  FWMIService: OleVariant;
  FWbemObject: OleVariant;
  EnumVariant: IEnumVARIANT;
  FSWbemLocator: OleVariant;
  FWbemObjectSet: OleVariant;
begin;
  FSWbemLocator := CreateOleObject('WbemScripting.SWbemLocator');
  FWMIService := FSWbemLocator.ConnectServer(WbemComputer, 'root\CIMV2', WbemUser, WbemPassword);
  FWbemObjectSet := FWMIService.ExecQuery('SELECT * FROM Win32_NetworkAdapter WHERE PhysicalAdapter = 1', 'WQL', wbemFlagForwardOnly);
  EnumVariant := IUnknown(FWbemObjectSet._NewEnum) as IEnumVariant;
  while EnumVariant.Next(1, FWbemObject, ElementCount) = 0 do
  begin
    Writeln(Format('MACAddress %s', [VarToStr(FWbemObject.MACAddress)]));
    FWbemObject := Unassigned;
  end;
end;

begin
  try
    CoInitialize(nil);
    try
      GetWin32_NetworkAdapterInfo;
    finally
      CoUninitialize;
    end;
  except
    on E:EOleException do
      Writeln(Format('EOleException %s %x', [E.Message,E.ErrorCode]));
    on E:Exception do
      Writeln(E.Classname, ':', E.Message);
  end;
  Writeln('Press Enter to exit');
  Readln;
end.

基于生成的代码WMI Delphi Code Creator

更新:

您实际上是在尝试过滤掉属于虚拟机的适配器,这并不容易,因为它们被模拟为物理适配器(您甚至可以在设备管理器中将它们视为物理适配器),因此您无法区分它们例如:

  • 由该类的DHCPEnabled成员Win32_NetworkAdapter,因为甚至可能配置虚拟机,以便它们从 DHCP 服务器获取 IP 地址
  • 由类的AdapterTypeId成员Win32_NetworkAdapter,因为虚拟适配器没有特殊类型
  • 由班级PhysicalAdapter成员Win32_NetworkAdapter,因为他们被模拟成物理的

补充阅读:

于 2012-05-15T08:53:14.953 回答
4

您还可以使用IP Helper库中GetAdaptersAddresses的 API 。对于 Delphi 翻译,Magenta Systems IP Helper Component乍一看还不错。

于 2012-05-15T08:56:35.593 回答