jar包加密方案分享
前言
jar包相信大家都很熟悉,是通過打包java工程而獲得的產(chǎn)物,但是jar包是有一個致命的缺點的,那就是很容易被反編譯,只需要使用jd-gui就可以很容易的獲取到j(luò)ava源碼。
如果你想要防止別人反編譯做逆向工程,那么對jar包進行一次加密就是一項很重要的工作了。
開源jar加密分為兩種一種代碼混淆,一種字節(jié)碼轉(zhuǎn)換。
字節(jié)碼混淆,主要思路就是的對類名,變量名和方法名,局部參數(shù)名進行替換,讓其命名變得無意義,很難讀懂,但不影響其邏輯,但對于有耐心的人,還是能看明白!具體實現(xiàn)是在編譯前做的混淆,還是編譯后做的混淆,這個沒用過不太清楚。
字節(jié)碼轉(zhuǎn)換,分為兩塊,一塊是加密,對編譯后class文件進行字節(jié)碼轉(zhuǎn)換(可以采用加密算法,字節(jié)碼異或運算或自己定義規(guī)則);一塊是解密,就是在類加載的時候?qū)用艿淖止?jié)碼進行解密。加密這塊因為在本地對class文件字節(jié)碼轉(zhuǎn)換,比較簡單,方式也隨意;解密這塊主要圍繞類加載器來做文章,又可分為java版實現(xiàn)和C/C++版實現(xiàn),java版主要基于-agentJava:xxx.jar,通過Premain-Class,向Instrumentation注入ClassFileTransformer實現(xiàn),自己在ClassFileTransformer中對需要解密的class文件進行解密;C/C++主要使用-agentpath:xxx.so,基于JVMTI通過C/C++實現(xiàn),對類加載過程進行操作。
總體來說:代碼混淆和字節(jié)碼轉(zhuǎn)換可以結(jié)合,例如,先代碼混淆后,再對字節(jié)碼加密,運行時對字節(jié)碼解密。代碼混淆,上手最簡單,加密級別比較低,也容易破解。字節(jié)碼轉(zhuǎn)換,java版本對于java熟練人員上手很快,加密級別一般,個人理解該方式一個缺陷就是對于想加密的內(nèi)容加密了,但解密方式暴露了,如果能夠隱藏好解密方式,加密安全系數(shù)還是蠻高;C/C++版加密級別最高,但需要對java和C/C++都很熟練,而且需要研究JVMTI相關(guān)知識,如果真的實現(xiàn)了基本無破解風險,不過對于SpringBoot等框架,其內(nèi)部會直接分析class文件,有些坑需要解決。
另外一個問題就是,使用jar加密這塊,需要確定好是對運行的主程序jar包加密,還是第三方j(luò)ar加密,其還是有部分差異的。
基礎(chǔ)環(huán)境準備
現(xiàn)在假設(shè)你的項目是一個maven項目(目前不使用maven的項目已經(jīng)不多了),那么加密起來特別的容易,
首先就是要在你的pom文件中增加插件倉庫地址,如下:
<pluginRepositories> <pluginRepository> <id>jitpack.io</id> <url>https://jitpack.io</url> </pluginRepository> </pluginRepositories>
然后在pom文件中增加如下插件:
<plugin> <groupId>com.github.core-lib</groupId> <artifactId>xjar-maven-plugin</artifactId> <version>4.0.1</version> <executions> <execution> <goals> <goal>build</goal> </goals> <phase>install</phase> <configuration> <includes> <include>/com/huc/**/*.class</include> <include>/mapper/**/*Mapper.xml</include> <include>/*.yml</include> </includes> <excludes> <exclude>/templates/**.*</exclude> <exclude>/static/**.*</exclude> </excludes> </configuration> </execution> </executions> </plugin>
這樣,我們的準備工作就做完了,需要注意的是,includes代表我們想要加密的內(nèi)容,采用的是Ant表達式,excludes代表的是不需要加密的內(nèi)容,同樣使用的Ant表達式。
一般情況下我們建議這兩處內(nèi)容必填,如果不填寫,會加密整個jar包中的所有文件,這樣其實是沒有必要的,而且全部加密后啟動的時候也可能產(chǎn)生意料之外的錯誤。
另外要說明的是,加密后的jar包是需要通過golang環(huán)境運行的,所以我們需要提前把golang的運行環(huán)境安裝好,安裝過程請自行百度。
開始加密
現(xiàn)在我們就開始正式的加密工作了,加密過程非常簡單,只需要使用maven的install命令即可自動打包,命令如下:
mvn clean install -Dxjar.password=password -Dmaven.test.skip=true
這里的password可以自行指定密碼,是必填項。
執(zhí)行后就會得到兩個文件:一個是xjar.go的go源文件,一個是你項目的xjar包,也就是加密后的jar包。
運行加密后的jar包
運行加密后的jar包是需要先編譯xjar.go源文件生成jar包啟動器的。編譯方式如下:
go build ./xjar.go
編譯后會生成xjar.exe啟動器(王子使用的是window系統(tǒng),如果是linux系統(tǒng)就不是exe后綴了)。
之后使用如下命令即可運行加密后的jar包:
./xjar.exe java -jar ./**.xjar
可以看出,只是在使用java -jar的前邊加上啟動器即可,還是很方便的。
后記
防止反編譯只能防止jar包被逆向破解,如果想要限制用戶的使用時間,按時間付費需要怎么做呢?
這就要說到license加密技術(shù)了,我們下篇文章就來說說如何在你的項目中增加license,限制用戶的使用。
其他的加密方案參考:
ClassFinal:Mr.K/ClassFinal
基于字節(jié)碼轉(zhuǎn)換java版,國人實現(xiàn)的,蠻不錯的,對SpringBoot支持也好,其邏輯就是基于-agentJava:xxx.jar這一套原理,加密時對class文件做了兩次處理,一次是對class文件的字節(jié)碼完全加密,一次是對class文件混淆,這個混淆是保留成員和方法,對方法內(nèi)部實現(xiàn)進行隱藏;解密時,判斷如果該類是自己加密過的,找到完全加密的字節(jié)碼進行解密,如果不是自己加密的就跳過。其對class文件混淆,就是方便類似SpringBoot等三方框架直接分析class文件。好處就是,如果你是個工具包,加密后給其它人,其他人編程時引用或者編譯都不影響,但是運行時需要加密方支持,給出秘鑰之類的。比較靈活和好用,也存在該方式的統(tǒng)一缺陷,不過其支持主程序jar加密,也支持單獨的第三方工具jar加密。
JarEncrypt:https://github.com/zhikun0704/api-zxv-jvmit
基于字節(jié)碼轉(zhuǎn)換C/C++版,基于JVMIT個人實現(xiàn)的,稍微研究了下,其支持對部分class加密,對應一般java應用問題不大,但對于SpringBoot項目支持可能有些問題。因本人C++水平有限,沒有深層次研究,希望有C/C++和java都熟悉的大神有空搞一套完整的。
以上就是jar包加密方案分享的詳細內(nèi)容,更多關(guān)于jar包加密的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
使用Springboot實現(xiàn)word在線編輯保存
PageOffice目前支持的Web編程語言及架構(gòu)有:Java(JSP、SSH、MVC等),ASP.NET(C#、VB.NET、MVC、Razor等),PHP,ASP,本篇文章就帶你使用Springboot整合PageOffice實現(xiàn)word在線編輯保存2021-08-08java用LocalDateTime類獲取當天時間、前一天時間及本周/本月的開始和結(jié)束時間
這篇文章主要給大家介紹了關(guān)于java使用LocalDateTime類獲取當天時間、前一天時間及本周/本月的開始和結(jié)束時間的相關(guān)資料,文中通過代碼示例介紹的非常詳細,需要的朋友可以參考下2023-08-08java中List對象列表實現(xiàn)去重或取出及排序的方法
這篇文章主要介紹了關(guān)于java中List對象列表實現(xiàn)去重或取出以及排序的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面跟著小編來一起學習學習吧。2017-08-08Spring Boot集成Redis實戰(zhàn)操作功能
這篇文章主要介紹了Spring Boot集成Redis實戰(zhàn)操作,包括如何集成redis以及redis的一些優(yōu)點,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2018-11-11Java轉(zhuǎn)換流(InputStreamReader/OutputStreamWriter)的使用
本文主要介紹了Java轉(zhuǎn)換流(InputStreamReader/OutputStreamWriter)的使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-01-01