JavaIO流04 4.常用的類03 4.4節點流和處理流02 4.4.5對象處理流-ObjectInputStream和ObjectOutputStream 1.序列化和反序列化 例子1: 看一個需求 將int num= 100這個int 類型的數據保存到文件中,註意不是100 數字,而是int ...
目前關於 spring native 分享的文章還比較少
寫這篇文章的主要目前是分享一下自己寫的一個 小米控制美的空調 的程式 集成 spring native 過程中碰到的一些問題和解決方法
先放地址 : https://github.com/toohandsome/xiaomi2meidi 歡迎star
對比一下速度:
上面是編譯成exe運行,下麵是jar運行 快了10倍.
Spring Native 可以通過 GraalVM 將 Spring 應用程式編譯成原生鏡像,提供了一種新的方式來部署 Spring 應用。
ps: 這篇文章主要是將其打包成exe,沒有打包成docker鏡像
註意 目前 Spring Native 已經不支持jdk8 了,這裡選用的jvm 是 graalvm-ce-java17-22.2.0, maven 選用 apache-maven-3.8.6
首先在 start.spring.io 中 選擇 spring native 和 web
下載後導入idea,把項目的sdk 和 語法 均設置為 17
按照 Visual Studio ,我這裡是vs2019 , 更高版本應該也可以,可以參考這篇文章 https://www.cnblogs.com/luguojun/p/16132521.html
環境配置好了以後,
在 resource 下 創建 META-INF/native-image/{groupId}/{artifactId}
然後在下麵創建
native-image.properties
proxy-config.json
reflect-config.json
resource-config.json
serialization-config.json
如圖所示
開始編譯主要有幾種錯誤
1. must not contain "."
must not contain ".". This can happen implicitly if the builder runs exclusively on the --module-path but specifies the com.oracle.svm.hosted.NativeImageGeneratorRunner main class without --module.
經過排除發現是 classpath 環境變數不能有 "." , 只要保留
%JAVA_HOME%\lib;
即可,如圖
2.UnsupportedFeatureError
UnsupportedFeatureError: Proxy class defined by interfaces[xxxx] not found. Generating proxy classes at runtime is not supported. Proxy classes need to be defined at image build time by specifying the list of interfaces that they implement.
會一直報錯 某某bean 不能被創建 , 需要在 proxy-config.json 中 增加 報錯信息中的 xxx 介面.
但是這裡有個問題就是 它這個錯會有很多,你改了一個bean , 下一個bean 又會有不同的介面,所以我寫了一個程式來自動分析
Pattern p = Pattern.compile("Proxy class defined by interfaces \\[(.*?)\\]");
for (int i = 0; i < 10000; i++) {
Process process = Runtime.getRuntime().exec("F:\\springnative\\start.bat");
BufferedReader bufferedReader = new BufferedReader(
new InputStreamReader(process.getInputStream(), "gbk"));
String line;
while ((line = bufferedReader.readLine()) != null) {
System.out.println(line);
}
System.out.println("success");
String s = RuntimeUtil.execForStr("\"F:\\springnative\\target\\xiaomi2meidi.exe\"");
System.out.println("exe: " + s);
Matcher matcher = p.matcher(s);
if (matcher.find()) {
var interfaces = matcher.group(1).replace("interface ", "").replace(" ", "");
String[] split = interfaces.split(",");
JSONArray jsonArray = new JSONArray();
for (String s1 : split) {
System.out.println(s1);
jsonArray.add(s1);
}
String s1 = Files.readString(Paths.get("F:\\springnative\\src\\main\\resources\\META-INF\\native-image\\com.yxd.xiaomi2meidi\\proxy-config.json"));
List<JSONArray> jsonArrays = JSON.parseArray(s1, JSONArray.class);
jsonArrays.add(jsonArray);
String pretty = JSON.toJSONString(jsonArrays, JSONWriter.Feature.PrettyFormat,
JSONWriter.Feature.WriteMapNullValue,
JSONWriter.Feature.WriteNullListAsEmpty);
try {
Files.writeString(Paths.get("F:\\springnative\\src\\main\\resources\\META-INF\\native-image\\com.yxd.xiaomi2meidi\\proxy-config.json"), pretty);
} catch (Exception e) {
e.printStackTrace();
}
} else {
break;
}
}
寫的比較粗糙
大概就是迴圈編譯運行,把運行後得到的日誌 用正則匹配出來,然後自動加到 proxy-config.json 中去,然後又重新編譯,直到它不報錯為止.
3. logback 沒有日誌或者報錯找不到 ConsoleAppender 等日誌相關的類
增加依賴
<dependency>
<groupId>org.codehaus.janino</groupId>
<artifactId>janino</artifactId>
<version>3.1.8</version>
</dependency>
在 reflect-config.json 增加配置
{
"name": "ch.qos.logback.core.ConsoleAppender",
"allPublicConstructors": true,
"allPublicMethods": true
}
這裡的類名是怎麼來的呢,其實是 logback-spring.xml 裡面的類, 還有
ch.qos.logback.core.rolling.RollingFileAppender
ch.qos.logback.classic.PatternLayout
ch.qos.logback.classic.encoder.PatternLayoutEncoder
ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy
等, 根據自己的xml配置
最後說一下 如何使用 小米控制美的空調
首先說一下思路.
想用小愛控制,那麼最好的辦法自然是接入小米的iot平臺,然而小米並不對個人開發者開發.
退而求其次,我們可以找一個第三方廠商,由他們做中間人接入.
blinker (點燈科技) 是一家物聯網技術提供商, 官網 點燈科技 (https://diandeng.tech/home) (雖然文檔爛,但是功能不含糊,該有的都有. )
雖然我之前參考 使用ESP32和Blinker實現遠程網路喚醒電腦(接入語音助手,以小愛同學為例) 這篇帖子用esp32成功控制了電腦遠程開機
但是這次我不想依賴硬體設備,不然成本太高了( 一臺空調設備就得買一個esp32),
然後我就想blinker的代碼既然能在樹莓派上面跑,自然我就能把核心邏輯摳出來用java重寫一遍
好.正文開始
首先我們在blinker官網下載 他們的app,註冊.
註冊後 登錄,顯示如下界面
你們這裡應該是空的,我加過所以有其他設備
然後我們點擊右上角的 加號 進行添加一個新的設備
點擊 獨立設備
選擇網路接入
得到 authKey, 保存好,後面要用
然後我們返回設備列表,點擊剛新加的設備. 右上角 三個點
編輯設備名稱
輸入名稱, 這個名稱就是後面你喊小愛的名稱,同時 要和 美的美居 app 裡面空調的名稱要相同
確認修改後,我們下載 github 上的 程式 ,運行後在 程式提示的配置文件中輸入
phone: 美的app手機號, password: 美的app密碼,acNameList: 空調名稱(多個用逗號隔開), blinkerKeyList: 點燈的authkey(多個用逗號隔開,需要與空調名稱一一對應)
{
"phone":"13812345678",
"password":"123456",
"acNameList":"書房空調,主卧空調,次卧空調",
"blinkerKeyList":"8*****2,2*****9,0******8",
"uid":"",
"accessToken":"",
"tokenPwd":"",
"homeId":"",
"appVersion":"",
"deviceId":"",
"deviceName":"",
"osVersion":"",
"deviceList":[]
}
程式正常運行後.
我們 打開 米家 app , 點擊 "我的" , 往下翻 選擇 "其他平臺設備"
先點添加, 找到 點燈科技. ,然後點擊 同步設備.
如果這裡出現了你剛新加的設備說明就成功了.
然後就可以用小愛控制了
有問題請聯繫[email protected] 個人網站:http://ext.123cc.cc