目前關于 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轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/506536.html
標籤:其他
上一篇:day37-IO流04
下一篇:java多執行緒基礎學習
