自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

談?wù)勅绾卧谶\行時獲取「環(huán)境」信息

開發(fā) 開發(fā)工具
軟件工程師,特別是開發(fā)客戶端產(chǎn)品,App這一類,都免不了需要判斷當(dāng)前所處環(huán)境。比如客戶端產(chǎn)品要判斷是Windows/Linux系統(tǒng),x86還是x64等等。 App則需要判斷安卓、iOS的版本,當(dāng)前環(huán)境是否WIFI之類的。

軟件工程師,特別是開發(fā)客戶端產(chǎn)品,App這一類,都免不了需要判斷當(dāng)前所處環(huán)境。比如客戶端產(chǎn)品要判斷是Windows/Linux系統(tǒng),x86還是x64等等。 App則需要判斷安卓、iOS的版本,當(dāng)前環(huán)境是否WIFI之類的。

[[233843]]

對于 Java 應(yīng)用, 無論 Web 還是 桌面應(yīng)用,也會遇到需要判斷當(dāng)前所使用的 JDK 版本,當(dāng)前應(yīng)用對應(yīng)的操作系統(tǒng)等等。

比如我們在幾年前做應(yīng)用服務(wù)器集群管理與監(jiān)控時,需要判斷應(yīng)用服務(wù)器所屬物理機(jī)器的CPU以及內(nèi)存的使用率、對于服務(wù)器實例進(jìn)行操作,對于不同的JVM 實現(xiàn),采用不同的attach機(jī)制等等。當(dāng)時是使用 Sigar 進(jìn)行這些硬件信息的獲取。

由于提供的是All in One 的版本,所以需要判斷操作系統(tǒng)類型,來判斷加載 Sigar 的Windows支持 dll 文件還是Linux 支持 so 文件。

當(dāng)然獲取操作系統(tǒng)類型、版本都較容易,直接通過System 的getProperty再加上對應(yīng)的名稱就能拿到,比如「os.name」,「os.arch」等。對于 JVM 的廠商,則可以通過「java.vm.vendor」來得到,完整的可以通過System.getProperties全部拿到。

如果習(xí)慣使用JMX 讀取Platform 的 MBean,也可以通過 JMX來獲取,便捷操作類似這樣:

  1. OperatingSystemMXBean os = ManagementFactory.getOperatingSystemMXBean(); 
  2. System.out.println(os.getName()); 

本質(zhì)上也都是一樣的。

那除了這種方式,還有哪些方式可以獲取到呢?一般開源軟件又是怎么做的呢?

我們來看 Tomcat 內(nèi)部怎么實現(xiàn)的。

我們發(fā)現(xiàn), Tomcat 內(nèi)部,對于通過 System 的 properties獲取到的,基本都是用來打印 Log 和輸出使用,對于應(yīng)用內(nèi)的控制,基本沒看到使用。

那 Tomcat在判斷版本時,是怎么做的呢?為什么不直接用這個呢?

首先看怎么做的。

前面的文章提到過, Tomcat 對于內(nèi)存泄漏做了一些努力,比如先Hold 一塊內(nèi)存這種(Tomcat與內(nèi)存泄露處理),也比如防止內(nèi)存泄漏的PreventionListener,會先將可能共用的 class加載到Common classLoader里。

那這里加載到 Common classLoader這些常用 class時,就需要判斷當(dāng)前是Java 哪個版本,因為有些 class 是某些版之后才出現(xiàn)的。

具體實現(xiàn)是這個樣子:

  1. // Trigger a call to sun.awt.AppContext.getAppContext(). This 
  2. // will pin the system class loader in memory but that shouldn't 
  3. // be an issue. 
  4. if (appContextProtection && !JreCompat.isJre8Available()) { 
  5.     ImageIO.getCacheDirectory(); 
  6.  
  7. // Trigger the creation of the AWT (AWT-Windows, AWT-XAWT, 
  8. // etc.) thread 
  9. if (awtThreadProtection && !JreCompat.isJre9Available()) { 
  10.     java.awt.Toolkit.getDefaultToolkit(); 

什么時候 Available了?

  1. static { 
  2.     // This is Tomcat 8 with a minimum Java version of Java 7. The latest 
  3.     // Java version the optional features require is Java 9. 
  4.     // Look for the highest supported JVM first 
  5.     if (Jre9Compat.isSupported()) { 
  6.         instance = new Jre9Compat(); 
  7.         jre9Available = true
  8.         jre8Available = true
  9.     } 
  10.     else if (Jre8Compat.isSupported()) { 
  11.         instance = new Jre8Compat(); 
  12.         jre9Available = false
  13.         jre8Available = true
  14.     } else { 
  15.         instance = new JreCompat(); 
  16.         jre9Available = false
  17.         jre8Available = false
  18.     } 

具體是不是支持是直接通過加載特定版JDK 對應(yīng)的 class 來判斷

  1. static { 
  2.     Method m1 = null
  3.     try { 
  4.         // The class is Java6+... 
  5.         Class<?> c1 = Class.forName("javax.net.ssl.SSLParameters"); 
  6.         // ...but this method is Java8+ 
  7.         m1 = c1.getMethod("setUseCipherSuitesOrder", boolean.class); 
  8.     } catch (SecurityException e) { 
  9.         // Should never happen 
  10.     } catch (NoSuchMethodException e) { 
  11.         // Expected on Java < 8 
  12.     } catch (ClassNotFoundException e) { 
  13.         // Should never happen 
  14.     } 
  15.     setUseCipherSuitesOrderMethod = m1

通過這種加載 class 的方式,和 我們前面通過 System.property獲取,有啥區(qū)別呢?

要知道, System 的 Property 是個系統(tǒng)屬性,是可配置的,也就是說,誰都可以進(jìn)行setProperty的操作。如果不巧被別人改了,那你的程序可能就會出現(xiàn)不符合你預(yù)期的行為。

而加載類的形式,并不受其他人的影響。是最真實的一線聲音。

這樣的判斷不方式,不僅 Tomcat 自己,其他框架也有在用。我們來看 Spring Boot。

Boot 執(zhí)行的時候,有時候會判斷當(dāng)前應(yīng)用是否需要支持 Web, 類似于是否加載了 Spring MVC 這種。

  1. private static final String[] WEB_ENVIRONMENT_CLASSES = { "javax.servlet.Servlet", 
  2.     "org.springframework.web.context.ConfigurableWebApplicationContext" }; 
  3.      
  4.         private boolean deduceWebEnvironment() { 
  5.             for (String className : WEB_ENVIRONMENT_CLASSES) { 
  6.                 if (!ClassUtils.isPresent(className, null)) { 
  7.                     return false; 
  8.                 } 
  9.             } 
  10.             return true; 
  11.         } 

然后在啟動時候,會調(diào)用deduceWebEnvironment 方法,從而決定一些執(zhí)行的邏輯。

還有其他方式么? 如何不想自己手工處理,可以使用一些工具類,比如 Apache 的 commons-lang, 提供了一個 SystemUtils 可以直接。例如判斷操作系統(tǒng)和 Java 版本

  1. if (SystemUtils.IS_JAVA_1_7) { 
  2.     System.out.println("Hello 1.7"); 

工具類是怎么做的呢?

  1. public static final String JAVA_VERSION = getSystemProperty("java.version"); 
  2. private static String getJavaVersionTrimmed() { 
  3.     if(JAVA_VERSION != null) { 
  4.         for(int i = 0; i < JAVA_VERSION.length(); ++i) { 
  5.             char ch = JAVA_VERSION.charAt(i); 
  6.             if(ch >= 48 && ch <= 57) { 
  7.                 return JAVA_VERSION.substring(i); 
  8.             } 
  9.         } 
  10.     } 
  11.  
  12.     return null; 

說到底,也還是System.getProperty.

所以,在檢測環(huán)境的時候,我們可以通過System.getProperty,也可以通過 Tomcat 和 Spring 這種加載 class 的方式。或者干脆直接用工具類。 你還用過啥辦法呢?

【本文為51CTO專欄作者“侯樹成”的原創(chuàng)稿件,轉(zhuǎn)載請通過作者微信公眾號『Tomcat那些事兒』獲取授權(quán)】

戳這里,看該作者更多好文

責(zé)任編輯:趙寧寧 來源: 51CTO專欄
相關(guān)推薦

2015-09-09 10:10:35

運行時改變圖標(biāo)

2020-07-15 11:17:04

云計算云安全云原生

2015-07-20 15:44:46

Swift框架MJExtension反射

2021-08-18 08:32:09

代碼運行時間示波器

2023-04-03 13:01:14

UbuntuCRI-O

2024-03-21 09:15:58

JS運行的JavaScrip

2018-04-08 14:27:45

Linuxuptime系統(tǒng)運行時間

2020-04-06 11:47:44

Linux命令腳本

2019-07-12 09:30:12

DashboardDockerDNS

2021-09-11 15:38:23

容器運行鏡像開放

2022-10-08 00:00:00

V8channel對象

2009-06-17 15:46:36

Java運行時本機(jī)內(nèi)存

2023-11-21 16:31:51

C++語言

2022-10-08 00:06:00

JS運行V8

2024-01-08 08:24:02

PythonSys 模塊工具

2014-09-02 10:39:53

Go語言C語言

2009-11-02 12:25:30

2013-11-26 16:49:55

Android開發(fā)運行時KitKat

2020-12-07 13:31:43

GoMutex開發(fā)者

2023-01-03 09:10:21

點贊
收藏

51CTO技術(shù)棧公眾號