linux

使用dmidecode

CPU ID
// 代码:
sudo dmidecode -t 4 | grep ID
// 主板序列号
代码:
sudo dmidecode -t 2 | grep Serial
// MAC地址
代码:
sudo lshw -c network | grep serial | head -n 1

这个需要使用root权限,同时对于jetson 板子,dmidecode没用

14、获取CPU序列号或者主板序列号sudodmidecode –t 4 | grep ID15、查看序列号:sudodmidecode | grep Serial | more16、查看cpusudodmidecode –t 4 | more17、查看BIOS信息sudodmidecode –t 0 | more18、查看主板信息sudodmidecode –t 2 | more19、查看OEMSudodmidecode –t 11 | more20、显示当前内存大小free –m| grep “Mem” | awk ‘{print $2}’21:显示系统运行时间uptime22:查看内核限制23:查看Ubuntu版本信息cat /etc/issue或 cat/etc/lsb-release或 lsb_release–dsc24:查看系统是32位还是64位#查看long的位数,返回32或64getconf LONG_BIT#查看文件信息,包含32-bit就是32位,包含64-bit就是64位file /sbin/init

linux使用dmidecode代码获取主板序列号

/* ************************************************************************
*       Filename:  system_test.c
*    Description:
*        Version:  1.0
*        Created:  2015年04月12日 10时41分49秒
*       Revision:  none
*       Compiler:  gcc
*         Author:  YOUR NAME (),
*        Company:
* ************************************************************************/#include <stdio.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>
int main( int argc, char *argv[] )
{pid_t  pid;int ret = 0;int    fd[2]   = { 0 };/* 创建管道 */ret = pipe( fd );if ( ret == -1 ){perror( "pipe" );_exit( 1 );}/* 创建子进程,目的  1exec 2复制管道文件描述符 */pid = vfork();if ( pid < 0 ){perror( "vfork" );}else if ( pid == 0 ){dup2( fd[1], 1 );                       /* 标准输出重定向到管道的写端 */char str[50] = "dmidecode -s system-serial-number";execlp( "/bin/sh", "sh", "-c", str, NULL );}else  {char result[100] = "";read( fd[0], result, sizeof(result) );  /* 从管道的读端读取数据 */char msg[100] = "";sprintf( msg, "%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c:%c%c",result[7], result[8], result[10], result[11], result[13], result[14], result[16], result[17],result[19], result[20], result[22], result[23], result[25],result[26], result[28], result[29], result[31], result[32],result[34], result[35], result[37], result[38], result[40],result[41], result[43], result[44], result[46], result[47],result[49], result[50], result[52], result[53] );printf( "---->%s\n", msg );}return(0);
}

获取CPUID:

CPUID并不是所有的Intel CPU都支持的。如果支持,汇编调用为:eax置0000_0003,调用cpuid。
以下为实现代码(在我的CPU上,并没有得到):

#define cpuid(in,a,b,c,d)  asm("cpuid": "=a" (a), "=b" (b), "=c" (c), "=d" (d) : "a" (in));static int getcpuid (char *id, size_t max)
{int i;unsigned long li, maxi, maxei, ebx, ecx, edx, unused;cpuid (0, maxi, unused, unused, unused);maxi &= 0xffff;if (maxi < 3){return -1;}cpuid (3, eax, ebx, ecx, edx);snprintf (id, max, "%08lx %08lx %08lx %08lx", eax, ebx, ecx, edx);fprintf (stdout, "get cpu id: %s/n", id);return 0;
}

或者

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <arpa/inet.h>
#include <string>
#include <fstream>static bool get_cpu_id_by_asm(std::string & cpu_id)
{cpu_id.clear();unsigned int s1 = 0;unsigned int s2 = 0;asm volatile("movl $0x01, %%eax; \n\t""xorl %%edx, %%edx; \n\t""cpuid; \n\t""movl %%edx, %0; \n\t""movl %%eax, %1; \n\t": "=m"(s1), "=m"(s2));if (0 == s1 && 0 == s2){return(false);}char cpu[32] = { 0 };snprintf(cpu, sizeof(cpu), "%08X%08X", htonl(s2), htonl(s1));std::string(cpu).swap(cpu_id);return(true);
}static void parse_cpu_id(const char * file_name, const char * match_words, std::string & cpu_id)
{cpu_id.c_str();std::ifstream ifs(file_name, std::ios::binary);if (!ifs.is_open()){return;}char line[4096] = { 0 };while (!ifs.eof()){ifs.getline(line, sizeof(line));if (!ifs.good()){break;}const char * cpu = strstr(line, match_words);if (NULL == cpu){continue;}cpu += strlen(match_words);while ('\0' != cpu[0]){if (' ' != cpu[0]){cpu_id.push_back(cpu[0]);}++cpu;}if (!cpu_id.empty()){break;}}ifs.close();
}static bool get_cpu_id_by_system(std::string & cpu_id)
{cpu_id.c_str();const char * dmidecode_result = ".dmidecode_result.txt";char command[512] = { 0 };snprintf(command, sizeof(command), "dmidecode -t 4 | grep ID > %s", dmidecode_result);if (0 == system(command)){parse_cpu_id(dmidecode_result, "ID:", cpu_id);}unlink(dmidecode_result);return(!cpu_id.empty());
}static bool get_cpu_id(std::string & cpu_id)
{if (get_cpu_id_by_asm(cpu_id)){return(true);}if (0 == getuid()){if (get_cpu_id_by_system(cpu_id)){return(true);}}return(false);
}static void test_1()
{std::string cpu_id;if (get_cpu_id(cpu_id)){printf("cpu_id: [%s]\n", cpu_id.c_str());}else{printf("can not get cpu id\n");}
}static void test_2()
{{std::string cpu_id;if (get_cpu_id_by_asm(cpu_id)){printf("cpu_id_by_asm: [%s]\n", cpu_id.c_str());}else{printf("can not get cpu id\n");}}{std::string cpu_id;if (get_cpu_id_by_system(cpu_id)){printf("cpu_id_by_sys: [%s]\n", cpu_id.c_str());}else{printf("can not get cpu id\n");}}
}int main(int argc, char* argv[])
{test_1();test_2();return(0);
}

获取MAC地址:(可以考虑加入ifconfig -a的解析,因为lshw实在太慢了)

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <unistd.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string>
#include <fstream>bool get_mac_address_by_ioctl(std::string & mac_address)
{mac_address.clear();int sock = socket(AF_INET, SOCK_STREAM, 0);if (sock < 0){return(false);}struct ifreq ifr = { 0 };strncpy(ifr.ifr_name, "eth0", sizeof(ifr.ifr_name) - 1);bool ret = (ioctl(sock, SIOCGIFHWADDR, &ifr) >= 0);close(sock);const char hex[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };char mac[16] = { 0 };for (int index = 0; index < 6; ++index){size_t value = ifr.ifr_hwaddr.sa_data[index] & 0xFF;mac[2 * index + 0] = hex[value / 16];mac[2 * index + 1] = hex[value % 16];}std::string(mac).swap(mac_address);return(ret);
}static void parse_mac_address(const char * file_name, const char * match_words, std::string & mac_address)
{mac_address.c_str();std::ifstream ifs(file_name, std::ios::binary);if (!ifs.is_open()){return;}char line[4096] = { 0 };while (!ifs.eof()){ifs.getline(line, sizeof(line));if (!ifs.good()){break;}const char * mac = strstr(line, match_words);if (NULL == mac){continue;}mac += strlen(match_words);while ('\0' != mac[0]){if (' ' != mac[0] && ':' != mac[0]){mac_address.push_back(mac[0]);}++mac;}if (!mac_address.empty()){break;}}ifs.close();
}static bool get_mac_address_by_system(std::string & mac_address)
{mac_address.c_str();const char * lshw_result = ".lshw_result.txt";char command[512] = { 0 };snprintf(command, sizeof(command), "lshw -c network | grep serial | head -n 1 > %s", lshw_result);if (0 == system(command)){parse_mac_address(lshw_result, "serial:", mac_address);}unlink(lshw_result);return(!mac_address.empty());
}static bool get_mac_address(std::string & mac_address)
{if (get_mac_address_by_ioctl(mac_address)){return(true);}if (get_mac_address_by_system(mac_address)){return(true);}return(false);
}static void test_1()
{std::string mac_address;if (get_mac_address(mac_address)){printf("mac_address: [%s]\n", mac_address.c_str());}else{printf("can not get mac address\n");}
}static void test_2()
{{std::string mac_address;if (get_mac_address_by_ioctl(mac_address)){printf("mac_address: [%s]\n", mac_address.c_str());}else{printf("can not get mac address\n");}}{std::string mac_address;if (get_mac_address_by_system(mac_address)){printf("mac_address: [%s]\n", mac_address.c_str());}else{printf("can not get mac address\n");}}
}int main(int argc, char * argv[])
{test_1();test_2();return(0);
}

获取硬盘序列号:

#include <cctype>
#include <cstdlib>
#include <cstring>
#include <fcntl.h>
#include <unistd.h>
#include <scsi/sg.h>
#include <sys/ioctl.h>
#include <linux/hdreg.h>
#include <string>
#include <fstream>static bool get_disk_name(std::string & disk_name)
{disk_name.c_str();std::ifstream ifs("/etc/mtab", std::ios::binary);if (!ifs.is_open()){return(false);}char line[4096] = { 0 };while (!ifs.eof()){ifs.getline(line, sizeof(line));if (!ifs.good()){break;}const char * disk = line;while (isspace(disk[0])){++disk;}const char * space = strchr(disk, ' ');if (NULL == space){continue;}const char * mount = space + 1;while (isspace(mount[0])){++mount;}if ('/' != mount[0] || ' ' != mount[1]){continue;}while (space > disk && isdigit(space[-1])){--space;}if (space > disk){std::string(disk, space).swap(disk_name);break;}}ifs.close();return(!disk_name.empty());
}static void trim_serial(const void * serial, size_t serial_len, std::string & serial_no)
{const char * serial_s = static_cast<const char *>(serial);const char * serial_e = serial_s + serial_len;while (serial_s < serial_e){if (isspace(serial_s[0])){++serial_s;}else if ('\0' == serial_e[-1] || isspace(serial_e[-1])){--serial_e;}else{break;}}if (serial_s < serial_e){std::string(serial_s, serial_e).swap(serial_no);}
}static bool get_disk_serial_by_way_1(const std::string & disk_name, std::string & serial_no)
{serial_no.clear();int fd = open(disk_name.c_str(), O_RDONLY);if (-1 == fd){return(false);}struct hd_driveid drive = { 0 };if (0 == ioctl(fd, HDIO_GET_IDENTITY, &drive)){trim_serial(drive.serial_no, sizeof(drive.serial_no), serial_no);}close(fd);return(!serial_no.empty());
}static bool scsi_io(int fd, unsigned char * cdb, unsigned char cdb_size, int xfer_dir, unsigned char * data, unsigned int data_size, unsigned char * sense, unsigned int sense_len)
{sg_io_hdr_t io_hdr = { 0 };io_hdr.interface_id = 'S';io_hdr.cmdp = cdb;io_hdr.cmd_len = cdb_size;io_hdr.sbp = sense;io_hdr.mx_sb_len = sense_len;io_hdr.dxfer_direction = xfer_dir;io_hdr.dxferp = data;io_hdr.dxfer_len = data_size;io_hdr.timeout = 5000;if (ioctl(fd, SG_IO, &io_hdr) < 0){return(false);}if (SG_INFO_OK != (io_hdr.info & SG_INFO_OK_MASK) && io_hdr.sb_len_wr > 0){return(false);}if (io_hdr.masked_status || io_hdr.host_status || io_hdr.driver_status){return(false);}return(true);
}static bool get_disk_serial_by_way_2(const std::string & disk_name, std::string & serial_no)
{serial_no.clear();int fd = open(disk_name.c_str(), O_RDONLY);if (-1 == fd){return(false);}int version = 0;if (ioctl(fd, SG_GET_VERSION_NUM, &version) < 0 || version < 30000){close(fd);return(false);}const unsigned int data_size = 0x00ff;unsigned char data[data_size] = { 0 };const unsigned int sense_len = 32;unsigned char sense[sense_len] = { 0 };unsigned char cdb[] = { 0x12, 0x01, 0x80, 0x00, 0x00, 0x00 };cdb[3] = (data_size >> 8) & 0xff;cdb[4] = (data_size & 0xff);if (scsi_io(fd, cdb, sizeof(cdb), SG_DXFER_FROM_DEV, data, data_size, sense, sense_len)){int page_len = data[3];trim_serial(data + 4, page_len, serial_no);}close(fd);return(!serial_no.empty());
}static bool parse_serial(const char * line, int line_size, const char * match_words, std::string & serial_no)
{const char * serial_s = strstr(line, match_words);if (NULL == serial_s){return(false);}serial_s += strlen(match_words);while (isspace(serial_s[0])){++serial_s;}const char * serial_e = line + line_size;const char * comma = strchr(serial_s, ',');if (NULL != comma){serial_e = comma;}while (serial_e > serial_s && isspace(serial_e[-1])){--serial_e;}if (serial_e <= serial_s){return(false);}std::string(serial_s, serial_e).swap(serial_no);return(true);
}static void get_serial(const char * file_name, const char * match_words, std::string & serial_no)
{serial_no.c_str();std::ifstream ifs(file_name, std::ios::binary);if (!ifs.is_open()){return;}char line[4096] = { 0 };while (!ifs.eof()){ifs.getline(line, sizeof(line));if (!ifs.good()){break;}if (0 == ifs.gcount()){continue;}if (parse_serial(line, ifs.gcount() - 1, match_words, serial_no)){break;}}ifs.close();
}static bool get_disk_serial_by_way_3(const std::string & disk_name, std::string & serial_no)
{serial_no.c_str();const char * hdparm_result = ".hdparm_result.txt";char command[512] = { 0 };snprintf(command, sizeof(command), "hdparm -i %s | grep SerialNo > %s", disk_name.c_str(), hdparm_result);if (0 == system(command)){get_serial(hdparm_result, "SerialNo=", serial_no);}unlink(hdparm_result);return(!serial_no.empty());
}static bool get_disk_serial_by_way_4(std::string & serial_no)
{serial_no.c_str();const char * lshw_result = ".lshw_result.txt";char command[512] = { 0 };snprintf(command, sizeof(command), "lshw -class disk | grep serial > %s", lshw_result);if (0 == system(command)){get_serial(lshw_result, "serial:", serial_no);}unlink(lshw_result);return(!serial_no.empty());
}static bool get_disk_serial_number(std::string & serial_no)
{if (0 != getuid()){return(false);}std::string disk_name;if (get_disk_name(disk_name)){if (get_disk_serial_by_way_1(disk_name, serial_no)){return(true);}if (get_disk_serial_by_way_2(disk_name, serial_no)){return(true);}if (get_disk_serial_by_way_3(disk_name, serial_no)){return(true);}}if (get_disk_serial_by_way_4(serial_no)){return(true);}return(false);
}static void test_1()
{std::string serial_no;if (get_disk_serial_number(serial_no)){printf("serial_number: [%s]\n", serial_no.c_str());}else{printf("get serial number failed\n");}
}static void test_2()
{std::string disk_name;if (get_disk_name(disk_name)){printf("disk_name:[%s]\n", disk_name.c_str());{std::string serial_no;get_disk_serial_by_way_1(disk_name, serial_no);printf("get_serial_by_way_1:[%s]\n", serial_no.c_str());}{std::string serial_no;get_disk_serial_by_way_2(disk_name, serial_no);printf("get_serial_by_way_2:[%s]\n", serial_no.c_str());}{std::string serial_no;get_disk_serial_by_way_3(disk_name, serial_no);printf("get_serial_by_way_3:[%s]\n", serial_no.c_str());}}{std::string serial_no;get_disk_serial_by_way_4(serial_no);printf("get_serial_by_way_4:[%s]\n", serial_no.c_str());}
}int main(int argc, char * argv[])
{printf("---------------\n");test_1();printf("---------------\n");test_2();printf("---------------\n");return(0);
}

简化:

#include <stdio.h>
#include <sys/ioctl.h>
#include <linux/hdreg.h>
#include <sys/stat.h>
#include <fcntl.h> static int getdiskid (char *hardc)
{int fd;struct hd_driveid hid;fd = open ("/dev/sda", O_RDONLY);if (fd < 0){return -1;}if (ioctl (fd, HDIO_GET_IDENTITY, &hid) < 0){return -1;}close (fd);sprintf(hardc,"%s", hid.serial_no);return 0;
}int main(void)
{char hardseri[50];getdiskid(hardseri);printf("%s",hardseri);return 0;
}

获取主板序列号:(没有找到纯代码的实现方法)

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <string>
#include <fstream>static void parse_board_serial(const char * file_name, const char * match_words, std::string & board_serial)
{board_serial.c_str();std::ifstream ifs(file_name, std::ios::binary);if (!ifs.is_open()){return;}char line[4096] = { 0 };while (!ifs.eof()){ifs.getline(line, sizeof(line));if (!ifs.good()){break;}const char * board = strstr(line, match_words);if (NULL == board){continue;}board += strlen(match_words);while ('\0' != board[0]){if (' ' != board[0]){board_serial.push_back(board[0]);}++board;}if ("None" == board_serial){board_serial.clear();continue;}if (!board_serial.empty()){break;}}ifs.close();
}static bool get_board_serial_by_system(std::string & board_serial)
{board_serial.c_str();const char * dmidecode_result = ".dmidecode_result.txt";char command[512] = { 0 };snprintf(command, sizeof(command), "dmidecode -t 2 | grep Serial > %s", dmidecode_result);if (0 == system(command)){parse_board_serial(dmidecode_result, "Serial Number:", board_serial);}unlink(dmidecode_result);return(!board_serial.empty());
}static bool get_board_serial_number(std::string & board_serial)
{if (0 == getuid()){if (get_board_serial_by_system(board_serial)){return(true);}}return(false);
}static void test()
{std::string board_serial;if (get_board_serial_number(board_serial)){printf("board_serial: [%s]\n", board_serial.c_str());}else{printf("can not get board id\n");}
}int main(int argc, char* argv[])
{test();return(0);
}

方便测试的Makefile:

build :g++ -o get_cpu_id get_cpu_id.cppg++ -o get_mac_address get_mac_address.cppg++ -o get_disk_serial_number get_disk_serial_number.cppg++ -o get_board_serial_number get_board_serial_number.cpprun   :@echo "--------------------"@- ./get_cpu_id@echo "--------------------"@- ./get_mac_address@echo "--------------------"@- ./get_disk_serial_number@echo "--------------------"@- ./get_board_serial_number@echo "--------------------"clean : -rm get_cpu_id-rm get_mac_address-rm get_disk_serial_number-rm get_board_serial_numberrebuild : clean build

编译:make 或者 make build

运行:make run 或者 sudo make run (上面大多数信息都需要超级用户权限才能获取到结果)

清理:make clean (这个写得太死了,本来是想删除非cpp文件的,shell写不出来)

重编:make rebuild

windows:

命令行:

查看主机序列号:

C:\Users\Administrator>wmic bios get serialnumber
SerialNumber
WB12345678-------->>>>这是我笔记本的主机序列号,在不换主板的情况下,该序列号应该与机器机身上贴的序列号一致C:\Users\Administrator>

查看磁盘序列号:

一、查看ECS实例中块存储的设备名
lsblk

二、使用以下命令获取块存储设备的序列号

udevadm info --query=all --name=磁盘设备名 | grep ID_SERIAL

例如

udevadm info --query=all --name=/dev/vda | grep ID_SERIAL


##获取CPU序列号

C:\Users\jesson>wmic CPU get ProcessorID
ProcessorId
BFEBFBFF000206A7

获取mac地址和ip信息

一台机器上可能不只有一个网卡,但每一个网卡只有一个MAC地址,而每一个网卡可能配置有多个IP地址;如平常的笔记本电脑中,就会有无线网卡和有线网卡(网线接口)两种;因此,如果要获得本机所有网卡的IP和MAC地址信息,则必须顺序获得每个网卡,再依次获取其信息等;在windows sdk中,用IP_ADAPTER_INFO结构体存储网卡信息,包括网卡名、网卡描述、网卡MAC地址、网卡IP等,该结构体的主要描述如下所示:

typedef struct _IP_ADAPTER_INFO {struct _IP_ADAPTER_INFO* Next;//指向链表中下一个适配器信息的指针DWORD ComboIndex;//预留值char AdapterName[MAX_ADAPTER_NAME_LENGTH + 4];//使用ANSI字符串表示的适配器名称char Description[MAX_ADAPTER_DESCRIPTION_LENGTH + 4];//使用ANSI字符串表示的适配器描述UINT AddressLength;//适配器硬件地址以字节计算的长度BYTE Address[MAX_ADAPTER_ADDRESS_LENGTH];//硬件地址以BYTE数组所表示DWORD Index;//适配器索引UINT Type;//适配器类型,主要有以下几种:/**   MIB_IF_TYPE_OTHER     1*   MIB_IF_TYPE_ETHERNET     6*   MIB_IF_TYPE_TOKENRING     9*   MIB_IF_TYPE_FDDI     15*   MIB_IF_TYPE_PPP     23*   MIB_IF_TYPE_LOOPBACK      24*   MIB_IF_TYPE_SLIP      28*/UINT DhcpEnabled;//指定这个适配器是否开启DHCPPIP_ADDR_STRING CurrentIpAddress;//预留值IP_ADDR_STRING IpAddressList;//该适配器的IPv4地址链表IP_ADDR_STRING GatewayList;//该适配器的网关IPv4地址链表IP_ADDR_STRING DhcpServer;//该适配器的DHCP服务器的IPv4 地址链表BOOL HaveWins;IP_ADDR_STRING PrimaryWinsServer;IP_ADDR_STRING SecondaryWinsServer;time_t LeaseObtained;time_t LeaseExpires;} IP_ADAPTER_INFO,*PIP_ADAPTER_INFO;

由于可能有多个网卡,因此struct _IP_ADAPTER_INFO* Next字段为一个链表结构指针,由于一个网卡可能有多个IP,因此IP_ADDR_STRING字段应该也是一个链表结构,其信息如下所示:

typedef struct _IP_ADDR_STRING
{struct _IP_ADDR_STRING* Next;  //指向同类型节点,即下一个IP(如果有多IP的话)IP_ADDRESS_STRING IpAddress;  //IP地址信息IP_MASK_STRING IpMask; //IP子网掩码DWORD Context;// 网络表入口。这个值对应着AddIPAddredd和DeleteIPAddress函数中的NTEContext参数
} IP_ADDR_STRING;

由于可能有多个网卡,因此struct _IP_ADAPTER_INFO* Next字段为一个链表结构指针,由于一个网卡可能有多个IP,因此IP_ADDR_STRING字段应该也是一个链表结构,其信息如下所示:

typedef struct _IP_ADDR_STRING
{struct _IP_ADDR_STRING* Next;  //指向同类型节点,即下一个IP(如果有多IP的话)IP_ADDRESS_STRING IpAddress;  //IP地址信息IP_MASK_STRING IpMask; //IP子网掩码DWORD Context;// 网络表入口。这个值对应着AddIPAddredd和DeleteIPAddress函数中的NTEContext参数
} IP_ADDR_STRING;

在基本了解以上信息后,就可以调用GetAdaptersInfo函数来获取相关网卡信息了,其通用的代码如下所示:

#include <WinSock2.h>
#include <Iphlpapi.h>
#include <iostream>
using namespace std;
#pragma comment(lib,"Iphlpapi.lib") //需要添加Iphlpapi.lib库int main(int argc, char* argv[])
{//PIP_ADAPTER_INFO结构体指针存储本机网卡信息PIP_ADAPTER_INFO pIpAdapterInfo = new IP_ADAPTER_INFO();//得到结构体大小,用于GetAdaptersInfo参数unsigned long stSize = sizeof(IP_ADAPTER_INFO);//调用GetAdaptersInfo函数,填充pIpAdapterInfo指针变量;其中stSize参数既是一个输入量也是一个输出量int nRel = GetAdaptersInfo(pIpAdapterInfo,&stSize);//记录网卡数量int netCardNum = 0;//记录每张网卡上的IP地址数量int IPnumPerNetCard = 0;if (ERROR_BUFFER_OVERFLOW == nRel){//如果函数返回的是ERROR_BUFFER_OVERFLOW//则说明GetAdaptersInfo参数传递的内存空间不够,同时其传出stSize,表示需要的空间大小//这也是说明为什么stSize既是一个输入量也是一个输出量//释放原来的内存空间delete pIpAdapterInfo;//重新申请内存空间用来存储所有网卡信息pIpAdapterInfo = (PIP_ADAPTER_INFO)new BYTE[stSize];//再次调用GetAdaptersInfo函数,填充pIpAdapterInfo指针变量nRel=GetAdaptersInfo(pIpAdapterInfo,&stSize);    }if (ERROR_SUCCESS == nRel){//输出网卡信息//可能有多网卡,因此通过循环去判断while (pIpAdapterInfo){cout<<"网卡数量:"<<++netCardNum<<endl;cout<<"网卡名称:"<<pIpAdapterInfo->AdapterName<<endl;cout<<"网卡描述:"<<pIpAdapterInfo->Description<<endl;switch(pIpAdapterInfo->Type){case MIB_IF_TYPE_OTHER:cout<<"网卡类型:"<<"OTHER"<<endl;break;case MIB_IF_TYPE_ETHERNET:cout<<"网卡类型:"<<"ETHERNET"<<endl;break;case MIB_IF_TYPE_TOKENRING:cout<<"网卡类型:"<<"TOKENRING"<<endl;break;case MIB_IF_TYPE_FDDI:cout<<"网卡类型:"<<"FDDI"<<endl;break;case MIB_IF_TYPE_PPP:printf("PP\n");cout<<"网卡类型:"<<"PPP"<<endl;break;case MIB_IF_TYPE_LOOPBACK:cout<<"网卡类型:"<<"LOOPBACK"<<endl;break;case MIB_IF_TYPE_SLIP:cout<<"网卡类型:"<<"SLIP"<<endl;break;default:break;}cout<<"网卡MAC地址:";for (DWORD i = 0; i < pIpAdapterInfo->AddressLength; i++)if (i < pIpAdapterInfo->AddressLength-1){printf("%02X-", pIpAdapterInfo->Address[i]);}else{printf("%02X\n", pIpAdapterInfo->Address[i]);}cout<<"网卡IP地址如下:"<<endl;//可能网卡有多IP,因此通过循环去判断IP_ADDR_STRING *pIpAddrString =&(pIpAdapterInfo->IpAddressList);do {cout<<"该网卡上的IP数量:"<<++IPnumPerNetCard<<endl;cout<<"IP 地址:"<<pIpAddrString->IpAddress.String<<endl;cout<<"子网地址:"<<pIpAddrString->IpMask.String<<endl;cout<<"网关地址:"<<pIpAdapterInfo->GatewayList.IpAddress.String<<endl;pIpAddrString=pIpAddrString->Next;} while (pIpAddrString);pIpAdapterInfo = pIpAdapterInfo->Next;cout<<"--------------------------------------------------------------------"<<endl;}}//释放内存空间if (pIpAdapterInfo){delete pIpAdapterInfo;}return 0;
}

执行结果:

通过WMI获取网卡MAC地址、硬盘序列号、主板序列号、CPU ID、BIOS序列号

头文件:WMI_DeviceQuery.h

/* ----------------------------------------------------------
文件名称:WMI_DeviceQuery.h作者:秦建辉 MSN:splashcn@msn.com 版本历史: V1.4    2010年05月17日 修正了硬盘序列号处理中的错误。现在和EVEREST Ultimate Edition 5.5一致。 V1.3    2010年05月11日 增加了对网卡原生MAC地址的查询。 V1.2    2010年05月05日 增加对硬盘序列号的进一步处理。 V1.1    2010年04月30日 修正微软MSDN例子错误,并增加对虚拟机网卡的判断。 V1.0    2010年04月27日 完成正式版本。 功能描述: 基于WMI获取设备属性: 0:网卡原生MAC地址 1:硬盘序列号 2:主板序列号 3:CPU ID 4:BIOS序列号 5:主板型号 6:网卡当前MAC地址 接口函数: WMI_DeviceQuery
------------------------------------------------------------ */
#pragma once  #include <windows.h>  #ifndef MACRO_T_DEVICE_PROPERTY  #define MACRO_T_DEVICE_PROPERTY  #define PROPERTY_MAX_LEN    128 // 属性字段最大长度  typedef struct _T_DEVICE_PROPERTY  {  TCHAR szProperty[PROPERTY_MAX_LEN];  } T_DEVICE_PROPERTY;
#endif  #define WMI_QUERY_TYPENUM   7   // WMI查询支持的类型数  #ifdef __cplusplus
extern "C"
{
#endif  /*
功能:通过WMI获取设备属性
参数说明: iQueryType:需要查询的设备属性 0:网卡原生MAC地址 1:硬盘序列号 2:主板序列号 3:CPU ID 4:BIOS序列号 5:主板型号 6:网卡当前MAC地址 properties:存储设备属性值 iSize:可存储的最大设备个数
返回值: -1:不支持的设备属性值 -2:WMI连接失败 -3:不正确的WQL查询语句 >=0:获取的设备个数
*/
INT WMI_DeviceQuery( INT iQueryType, T_DEVICE_PROPERTY *properties, INT iSize );  #ifdef __cplusplus
}
#endif

实现文件:WMI_DeviceQuery.cpp

#include "WMI_DeviceQuery.h"
#include <comutil.h>
#include <Wbemidl.h>
#include <tchar.h>
#include <strsafe.h>
#include <algorithm>
#include <atlconv.h>
#include <ntddndis.h>  #pragma comment (lib, "comsuppw.lib")
#pragma comment (lib, "wbemuuid.lib")  typedef struct _T_WQL_QUERY
{  CHAR*   szSelect;       // SELECT语句  WCHAR*  szProperty;     // 属性字段
} T_WQL_QUERY;  // WQL查询语句
const T_WQL_QUERY szWQLQuery[] = {  // 网卡原生MAC地址  "SELECT * FROM Win32_NetworkAdapter WHERE (MACAddress IS NOT NULL) AND (NOT (PNPDeviceID LIKE 'ROOT%'))",  L"PNPDeviceID",  // 硬盘序列号  "SELECT * FROM Win32_DiskDrive WHERE (SerialNumber IS NOT NULL) AND (MediaType LIKE 'Fixed hard disk%')",  L"SerialNumber",  // 主板序列号  "SELECT * FROM Win32_BaseBoard WHERE (SerialNumber IS NOT NULL)",  L"SerialNumber",      // 处理器ID  "SELECT * FROM Win32_Processor WHERE (ProcessorId IS NOT NULL)",  L"ProcessorId",  // BIOS序列号  "SELECT * FROM Win32_BIOS WHERE (SerialNumber IS NOT NULL)",  L"SerialNumber",  // 主板型号  "SELECT * FROM Win32_BaseBoard WHERE (Product IS NOT NULL)",  L"Product",  // 网卡当前MAC地址  "SELECT * FROM Win32_NetworkAdapter WHERE (MACAddress IS NOT NULL) AND (NOT (PNPDeviceID LIKE 'ROOT%'))",  L"MACAddress",
};  // 通过“PNPDeviceID”获取网卡原生MAC地址
static BOOL WMI_DoWithPNPDeviceID( const TCHAR *PNPDeviceID, TCHAR *MacAddress, UINT uSize )
{  TCHAR   DevicePath[MAX_PATH];  HANDLE  hDeviceFile;      BOOL    isOK = FALSE;  // 生成设备路径名  StringCchCopy( DevicePath, MAX_PATH, TEXT(".//") );  StringCchCat( DevicePath, MAX_PATH, PNPDeviceID );  StringCchCat( DevicePath, MAX_PATH, TEXT("#{ad498944-762f-11d0-8dcb-00c04fc3358c}") );  // 将“PNPDeviceID”中的“/”替换成“#”,以获得真正的设备路径名  std::replace( DevicePath + 4, DevicePath + 4 + _tcslen(PNPDeviceID), TEXT('//'), TEXT('#') );   // 获取设备句柄  hDeviceFile = CreateFile( DevicePath,  0,  FILE_SHARE_READ | FILE_SHARE_WRITE,  NULL,  OPEN_EXISTING,  0,  NULL);  if( hDeviceFile != INVALID_HANDLE_VALUE )  {     ULONG   dwID;  BYTE    ucData[8];  DWORD   dwByteRet;        // 获取网卡原生MAC地址  dwID = OID_802_3_PERMANENT_ADDRESS;  isOK = DeviceIoControl( hDeviceFile, IOCTL_NDIS_QUERY_GLOBAL_STATS, &dwID, sizeof(dwID), ucData, sizeof(ucData), &dwByteRet, NULL );  if( isOK )  {   // 将字节数组转换成16进制字符串  for( DWORD i = 0; i < dwByteRet; i++ )  {  StringCchPrintf( MacAddress + (i << 1), uSize - (i << 1), TEXT("%02X"), ucData[i] );  }  MacAddress[dwByteRet << 1] = TEXT('/0');  // 写入字符串结束标记  }  CloseHandle( hDeviceFile );  }  return isOK;
}  static BOOL WMI_DoWithHarddiskSerialNumber( TCHAR *SerialNumber, UINT uSize )
{  UINT    iLen;  UINT    i;  iLen = _tcslen( SerialNumber );  if( iLen == 40 )    // InterfaceType = "IDE"  {   // 需要将16进制编码串转换为字符串  TCHAR ch, szBuf[32];  BYTE b;       for( i = 0; i < 20; i++ )  {   // 将16进制字符转换为高4位  ch = SerialNumber[i * 2];  if( (ch >= '0') && (ch <= '9') )  {  b = ch - '0';  }  else if( (ch >= 'A') && (ch <= 'F') )  {  b = ch - 'A' + 10;  }  else if( (ch >= 'a') && (ch <= 'f') )  {  b = ch - 'a' + 10;  }  else  {   // 非法字符  break;  }  b <<= 4;  // 将16进制字符转换为低4位  ch = SerialNumber[i * 2 + 1];  if( (ch >= '0') && (ch <= '9') )  {  b += ch - '0';  }  else if( (ch >= 'A') && (ch <= 'F') )  {  b += ch - 'A' + 10;  }  else if( (ch >= 'a') && (ch <= 'f') )  {  b += ch - 'a' + 10;  }  else  {   // 非法字符  break;  }  szBuf[i] = b;  }  if( i == 20 )  {   // 转换成功  szBuf[i] = L'/0';  StringCchCopy( SerialNumber, uSize, szBuf );  iLen = _tcslen( SerialNumber );  }  }  // 每2个字符互换位置  for( i = 0; i < iLen; i += 2 )  {  std::swap( SerialNumber[i], SerialNumber[i+1] );  }  // 去掉空格  std::remove( SerialNumber, SerialNumber + _tcslen(SerialNumber) + 1, L' ' );  return TRUE;
}  static BOOL WMI_DoWithProperty( INT iQueryType, TCHAR *szProperty, UINT uSize )
{  BOOL isOK = TRUE;  switch( iQueryType )  {  case 0:     // 网卡原生MAC地址          isOK = WMI_DoWithPNPDeviceID( szProperty, szProperty, uSize );  break;  case 1:     // 硬盘序列号  isOK = WMI_DoWithHarddiskSerialNumber( szProperty, uSize );  break;  case 6:     // 网卡当前MAC地址  // 去掉冒号  std::remove( szProperty, szProperty + _tcslen(szProperty) + 1, L':' );  break;  default:  // 去掉空格  std::remove( szProperty, szProperty + _tcslen(szProperty) + 1, L' ' );  }  return isOK;
}  // 基于Windows Management Instrumentation(Windows管理规范)
INT WMI_DeviceQuery( INT iQueryType, T_DEVICE_PROPERTY *properties, INT iSize )
{  HRESULT hres;  INT iTotal = 0;  // 判断查询类型是否支持  if( (iQueryType < 0) || (iQueryType >= sizeof(szWQLQuery)/sizeof(T_WQL_QUERY)) )  {  return -1;  // 查询类型不支持  }  // 初始化COM  hres = CoInitializeEx( NULL, COINIT_MULTITHREADED );   if( FAILED(hres) )  {  return -2;  }  // 设置COM的安全认证级别  hres = CoInitializeSecurity(   NULL,   -1,   NULL,   NULL,   RPC_C_AUTHN_LEVEL_DEFAULT,   RPC_C_IMP_LEVEL_IMPERSONATE,  NULL,  EOAC_NONE,  NULL  );  if( FAILED(hres) )  {  CoUninitialize();  return -2;  }  // 获得WMI连接COM接口  IWbemLocator *pLoc = NULL;  hres = CoCreateInstance(   CLSID_WbemLocator,               NULL,   CLSCTX_INPROC_SERVER,   IID_IWbemLocator,  reinterpret_cast<LPVOID*>(&pLoc)  );   if( FAILED(hres) )  {  CoUninitialize();  return -2;  }  // 通过连接接口连接WMI的内核对象名"ROOT//CIMV2"  IWbemServices *pSvc = NULL;  hres = pLoc->ConnectServer(  _bstr_t( L"ROOT//CIMV2" ),  NULL,  NULL,  NULL,  0,  NULL,  NULL,  &pSvc  );      if( FAILED(hres) )  {  pLoc->Release();   CoUninitialize();  return -2;  }  // 设置请求代理的安全级别  hres = CoSetProxyBlanket(  pSvc,  RPC_C_AUTHN_WINNT,  RPC_C_AUTHZ_NONE,  NULL,  RPC_C_AUTHN_LEVEL_CALL,  RPC_C_IMP_LEVEL_IMPERSONATE,  NULL,  EOAC_NONE  );  if( FAILED(hres) )  {  pSvc->Release();  pLoc->Release();       CoUninitialize();  return -2;  }  // 通过请求代理来向WMI发送请求  IEnumWbemClassObject *pEnumerator = NULL;  hres = pSvc->ExecQuery(  bstr_t("WQL"),   bstr_t( szWQLQuery[iQueryType].szSelect ),  WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,   NULL,  &pEnumerator  );  if( FAILED(hres) )  {  pSvc->Release();  pLoc->Release();  CoUninitialize();  return -3;  }  // 循环枚举所有的结果对象    while( pEnumerator )  {  IWbemClassObject *pclsObj = NULL;  ULONG uReturn = 0;  if( (properties != NULL) && (iTotal >= iSize) )  {  break;  }  pEnumerator->Next(  WBEM_INFINITE,  1,   &pclsObj,  &uReturn  );  if( uReturn == 0 )  {  break;  }  if( properties != NULL )  {   // 获取属性值  VARIANT vtProperty;  VariantInit( &vtProperty );   pclsObj->Get( szWQLQuery[iQueryType].szProperty, 0, &vtProperty, NULL, NULL );  StringCchCopy( properties[iTotal].szProperty, PROPERTY_MAX_LEN, W2T(vtProperty.bstrVal) );  VariantClear( &vtProperty );  // 对属性值做进一步的处理  if( WMI_DoWithProperty( iQueryType, properties[iTotal].szProperty, PROPERTY_MAX_LEN ) )  {  iTotal++;  }  }  else  {  iTotal++;  }  pclsObj->Release();  } // End While  // 释放资源  pEnumerator->Release();  pSvc->Release();  pLoc->Release();      CoUninitialize();  return iTotal;
}

linux或者windows获取唯一硬件信息-命令行和c++代码相关推荐

  1. c# 获取电脑硬件信息通用查询类[测试通过]

    C#获取电脑硬件信息通用类[Computer]代码展示和分析,简介如下: 1.项目中添加System.Management引用. 2.添加类Computer,把下面代码全选,复制,粘贴. 3.使用方法 ...

  2. Linux下获取详细硬件信息的工具:Dmidecode命令详解

    Dmidecode 这款软件允许你在 Linux 系统下获取有关硬件方面的信息.Dmidecode 遵循 SMBIOS/DMI 标准,其输出的信息包括 BIOS.系统.主板.处理器.内存.缓存等等.偶 ...

  3. 汇总 Linux下获取详细硬件信息的工具:Dmidecode命令详解

    Dmidecode 这款软件允许你在 Linux 系统下获取有关硬件方面的信息.Dmidecode 遵循 SMBIOS/DMI 标准,其输出的信息包括 BIOS.系统.主板.处理器.内存.缓存等等.偶 ...

  4. linux查看显示器名称命令,linux 查看显示器信息Linux下查看硬件信息命令大全

    /proc 虚拟的目录,是系统内存的映射.可直接访问这个目录来获取系统信息.其中也包含下面的信息: 主机CPU信息:cpuinfo 主机DMA通道信息:dma 文件系统信息:filesystems 主 ...

  5. 怎么用Linux命令查看BIOS信息,LINUX下怎样获取主板的信息用到什么命令

    在Linux系统中,我们常常需要使用命令来实现各种操作,比如从系统中获取主板信息参数,很多用户都不太清楚,本文教给大家在LINUX下怎样获取主板的信息. 方法步骤 dmidecode取出的信息可能不是 ...

  6. 检测硬件的批处理命令,检测硬件bat,一键获取电脑硬件信息

    警告:运行BAT源码是一种危险的动作,如果你不熟悉,请不要尝试! 批处理语言: 简体中文 授权方式: 免费软件 运行环境: Windows平台 检测硬件批处理命令.一键获取.直接双击就可以查看 @ec ...

  7. 查看Linux硬件信息命令的使用

    为什么80%的码农都做不了架构师?>>>    查看Linux硬件信息命令的使用 http://www.xxlinux.com/newxxlinux/syste ... 5/315. ...

  8. linux内核4.14.10查看网卡型号,Linux中查看硬件信息命令

    Linux中查看硬件信息命令 1, 主板信息 .查看主板的序列号 -------------------------------------------------- #使用命令 dmidecode ...

  9. 查看服务器芯片组型号,linux查看硬件信息命令

    在使用Linux系统的时候,我们有时候需要查看服务器硬件相关信息,接下来吾爱编程为大家详细的介绍一下linux查看硬件信息命令,有需要的小伙伴可以参考一下: 1.查看服务器硬件信息 (1).查看服务器 ...

最新文章

  1. ExtJs_关于combobox的那些分页二三事
  2. ThinkPHP笔记——配置分组产生无法加载Index错误解决办法
  3. C4.5-Release8的代码架构图
  4. 参考文献_参考文献:
  5. java crud_Java 8流中的数据库CRUD操作
  6. 题解-ZeroJudge-c686 高斯符號
  7. Windows2008管理---第12章 终端服务器
  8. Android开发者应该深入学习的10个开源应用项目[转]
  9. sql server2008如何修改mac地址_QCC304x/QCC514x:修改蓝牙MAC地址及名称
  10. 2021年7月国产数据库大事记
  11. BM3D 算法原理详细解析 按过程步骤讲解(附C++实现代码)
  12. FILD FLD FSTP
  13. 计算机单位-字节到底是什么?(小白简单易懂)
  14. C程序设计语言逆波兰式计算器学习心得
  15. 陶哲轩实分析 习题 12.5.12
  16. Nova8pro自动调节亮度忽亮忽暗问题
  17. 我的jQuery学习之路_笔记(五)
  18. 固态硬盘(SSD)原理及相关介绍
  19. java实现“两数之和”
  20. 算法设计与分析(Java实现)—— 动态规划(入门)斐波那契数列

热门文章

  1. java一个岗位上千人竞争,『震惊』国家公务员考试这些岗位竞争比例低到令人怀疑...
  2. Oracle监听器服务名称,查看oracle监听器的状态及打开监听器服务
  3. 字体图标库icomoon和iconfont使用方法
  4. Android之锁屏后,防止服务被关闭
  5. Mysql日期格式化 查询 问题
  6. Linux下多线程编程---02(线程退出与线程资源的回收)
  7. 城市排水系统优化解决方案
  8. 软件构造—ADT的理解
  9. python超市买苹果
  10. Problem1:将x的平方赋值给y