Windows系统&Linux系统(Docker内)获取硬件信息Java

    应公司要求,做一个需求,需要获取本机或者服务器的某些唯一信息,比如MAC地址,CPU序列号等硬件信息;方法都大同小异,主要的就是通过命令来获取信息,这里只列举方法!

    在Windows系统上运行代码基本没有什么难点和容易犯的错误,各种方法都可,容易出问题的地方是在linux上获取硬件信息;

// 主板序列号 windows
	private static String getMainBordId_windows() {
		String result = "";
		try {
			File file = File.createTempFile("realhowto", ".vbs");
			file.deleteOnExit();
			FileWriter fw = new java.io.FileWriter(file);
 
			String vbs = "Set objWMIService = GetObject(\"winmgmts:\\\\.\\root\\cimv2\")\n"
					+ "Set colItems = objWMIService.ExecQuery _ \n" + "   (\"Select * from Win32_BaseBoard\") \n"
					+ "For Each objItem in colItems \n" + "    Wscript.Echo objItem.SerialNumber \n"
					+ "    exit for  ' do the first cpu only! \n" + "Next \n";
 
			fw.write(vbs);
			fw.close();
			Process p = Runtime.getRuntime().exec("cscript //NoLogo " + file.getPath());
			BufferedReader input = new BufferedReader(new InputStreamReader(p.getInputStream()));
			String line;
			while ((line = input.readLine()) != null) {
				result += line;
			}
			input.close();
		} catch (Exception e) {
			logger.error("获取主板信息错误", e);
		}
		return result.trim();
	}

// 主板序列号 linux
	private static String getMainBordId_linux() {
 
		String result = "";
		String maniBord_cmd = "dmidecode | grep 'Serial Number' | awk -F ':' '{print $2}' | tail -1";
		Process p;
		try {
			p = Runtime.getRuntime().exec(new String[] { "sh", "-c", maniBord_cmd });// 管道
			BufferedReader br = new BufferedReader(new InputStreamReader(p.getInputStream()));
			String line;
			while ((line = br.readLine()) != null) {
				result += line;
				break;
			}
			br.close();
		} catch (IOException e) {
			logger.error("获取主板信息错误", e);
		}
		return result;
	}

Linux命令: 

dmidecode:获取硬件信息;

grep 'Serial Number':截取Serial Number字段内容;

awk -F ’:‘  :以“:” 分割获取到的Serial Number字段内容;

'{print $2}' “:打印分割开的内容的第二段;

想要获取其他硬件信息只需更换字段和关键符号即可,举一反三。

==============================================================================================================》

遇到的问题及解决方法(以下问题皆出自Linux系统):

问题:

有一些命令,比如 ifconfig,sudo su,dmidecode手动输入都能够打印出相关信息,但是当把这些命令写道Java代码里远程执行就不行了,显示:Cannot run program xxx:Error=2,No such file or directory;任何信息都是null,一开始我以为是系统环境变量没配置好,于是我找到了以后下解决方法:

方法一 :

在用户的 home 目录下的文件 .bashrc 加入export PATH=$PATH:/sbin ,然后执行source   .bashrc  就可以执行 ifconfig 命令了(因为基本所有命令都在sbin目录)

解决办法二:

在用户目录的.bashrc文件里加上 source /etc/profile

一般来说这两个方法确实管用,并且这两个方法误导我很长时间,以致于我一直认为是代码出错,于是修改不停的代码,结果还是报错。

==============================================================================================================》

后来偶然看到一篇文章,恍然大悟,因为我们的项目是在Linux系统的docker容器中运行的,

通过上面的代码我们不难看懂,JAVA通过

Runtime.getRuntime().exec(“xxxxxx”)

这段代码去运行了dos命令。

正常运行倒也没什么问题,重点是docker是个什么东西,它是个容器化技术,它只为服务运行提供最低标准的运行环境以降低内存消耗。这里划个重点,最低标准。相当于容器内只有残缺的linux系统和你的服务代码,只要服务的代码能跑起来,我这个系统内其他的东西都是多余的,容器内都不需要,比如说dmidecode。

我们刚才的代码中就用到了dmidecode命令。由于容器内没有dmidecode工具,那么一执行,就会报错。

解决方法(重点):

如果要在docker容器中运行dmidecode命令,那么要将宿主机的如下两个目录挂载到容器中,
1:/sbin/dmidecode --这个目录是dmidecode程序的目录,如果不挂载那么容器中识别不了dmidecode命令;
2:/dev/mem --dmidecode调用时会使用到mem这个文件,如果不挂载会找不到文件;
并且还需要在启动时增加 --privileged=ture 这个参数,让容器获得近似于宿主机root的权限。

修改完之后我也是解决了困扰我许久的问题。

Windows系统&Linux系统(Docker内)获取硬件信息Java_第1张图片

注意点:

1、/sbin/dmidecode直接挂载为容器中的/sbin/dmidecode,不要修改成其他的目录名;
2、不要注释原来的脚本,不要把/data映射注释修改了,会导致容器找不到Jar包,启动失败;
3、容器的启动函数中多或者少一个字符 ,会导致容器启动参数不完整,无法启动;

PS:对于不太了解运维及部署知识的小伙伴来说修改这个可能还是比较棘手,万一改出毛病来得不偿失,建议找运维人员修改。

总结一下,其实容器中想要用的任何东西,无论是文件还是程序还是什么别的,都可以通过挂载的形式从宿主机中挂载到容器中,让容器中可以访问到。

你可能感兴趣的:(windows,linux,docker)