日本国产亚洲-日本国产一区-日本国产一区二区三区-日本韩国欧美一区-日本韩国欧美在线-日本韩国欧美在线观看

當前位置:雨林木風下載站 > 技術(shù)開發(fā)教程 > 詳細頁面

運用加密技術(shù)保護Java源代碼

運用加密技術(shù)保護Java源代碼

更新時間:2020-02-09 文章作者:未知 信息來源:網(wǎng)絡(luò) 閱讀次數(shù):

作者: 俞良松   

來源:www.ibm.com
俞良松 (javaman@163.net)
軟件工程師,獨立顧問和自由撰稿人
2001 年 10 月

Java程序的源代碼很容易被別人偷看。只要有一個反編譯器,任何人都可以分析別人的代碼。本文討論如何在不修改原有程序的情況下,通過加密技術(shù)保護源代碼。
一、為什么要加密?
對于傳統(tǒng)的C或C++之類的語言來說,要在Web上保護源代碼是很容易的,只要不發(fā)布它就可以。遺憾的是,Java程序的源代碼很容易被別人偷看。只要有一個反編譯器,任何人都可以分析別人的代碼。Java的靈活性使得源代碼很容易被竊取,但與此同時,它也使通過加密保護代碼變得相對容易,我們唯一需要了解的就是Java的ClassLoader對象。當然,在加密過程中,有關(guān)Java Cryptography Extension(JCE)的知識也是必不可少的。

有幾種技術(shù)可以“模糊”Java類文件,使得反編譯器處理類文件的效果大打折扣。然而,修改反編譯器使之能夠處理這些經(jīng)過模糊處理的類文件并不是什么難事,所以不能簡單地依賴模糊技術(shù)來保證源代碼的安全。

我們可以用流行的加密工具加密應(yīng)用,比如PGP(Pretty Good Privacy)或GPG(GNU Privacy Guard)。這時,最終用戶在運行應(yīng)用之前必須先進行解密。但解密之后,最終用戶就有了一份不加密的類文件,這和事先不進行加密沒有什么差別。

Java運行時裝入字節(jié)碼的機制隱含地意味著可以對字節(jié)碼進行修改。JVM每次裝入類文件時都需要一個稱為ClassLoader的對象,這個對象負責把新的類裝入正在運行的JVM。JVM給ClassLoader一個包含了待裝入類(比如java.lang.Object)名字的字符串,然后由ClassLoader負責找到類文件,裝入原始數(shù)據(jù),并把它轉(zhuǎn)換成一個Class對象。

我們可以通過定制ClassLoader,在類文件執(zhí)行之前修改它。這種技術(shù)的應(yīng)用非常廣泛——在這里,它的用途是在類文件裝入之時進行解密,因此可以看成是一種即時解密器。由于解密后的字節(jié)碼文件永遠不會保存到文件系統(tǒng),所以竊密者很難得到解密后的代碼。

由于把原始字節(jié)碼轉(zhuǎn)換成Class對象的過程完全由系統(tǒng)負責,所以創(chuàng)建定制ClassLoader對象其實并不困難,只需先獲得原始數(shù)據(jù),接著就可以進行包含解密在內(nèi)的任何轉(zhuǎn)換。

Java 2在一定程度上簡化了定制ClassLoader的構(gòu)建。在Java 2中,loadClass的缺省實現(xiàn)仍舊負責處理所有必需的步驟,但為了顧及各種定制的類裝入過程,它還調(diào)用一個新的findClass方法。

這為我們編寫定制的ClassLoader提供了一條捷徑,減少了麻煩:只需覆蓋findClass,而不是覆蓋loadClass。這種方法避免了重復(fù)所有裝入器必需執(zhí)行的公共步驟,因為這一切由loadClass負責。

不過,本文的定制ClassLoader并不使用這種方法。原因很簡單。如果由默認的ClassLoader先尋找經(jīng)過加密的類文件,它可以找到;但由于類文件已經(jīng)加密,所以它不會認可這個類文件,裝入過程將失敗。因此,我們必須自己實現(xiàn)loadClass,稍微增加了一些工作量。

二、定制類裝入器
每一個運行著的JVM已經(jīng)擁有一個ClassLoader。這個默認的ClassLoader根據(jù)CLASSPATH環(huán)境變量的值,在本地文件系統(tǒng)中尋找合適的字節(jié)碼文件。

應(yīng)用定制ClassLoader要求對這個過程有較為深入的認識。我們首先必須創(chuàng)建一個定制ClassLoader類的實例,然后顯式地要求它裝入另外一個類。這就強制JVM把該類以及所有它所需要的類關(guān)聯(lián)到定制的ClassLoader。Listing 1顯示了如何用定制ClassLoader裝入類文件。


【Listing 1:利用定制的ClassLoader裝入類文件】

// 首先創(chuàng)建一個ClassLoader對象
ClassLoader myClassLoader = new myClassLoader();

// 利用定制ClassLoader對象裝入類文件
// 并把它轉(zhuǎn)換成Class對象
Class myClass = myClassLoader.loadClass( "mypackage.MyClass" );

// 最后,創(chuàng)建該類的一個實例
Object newInstance = myClass.newInstance();

// 注意,MyClass所需要的所有其他類,都將通過
// 定制的ClassLoader自動裝入




如前所述,定制ClassLoader只需先獲取類文件的數(shù)據(jù),然后把字節(jié)碼傳遞給運行時系統(tǒng),由后者完成余下的任務(wù)。

ClassLoader有幾個重要的方法。創(chuàng)建定制的ClassLoader時,我們只需覆蓋其中的一個,即loadClass,提供獲取原始類文件數(shù)據(jù)的代碼。這個方法有兩個參數(shù):類的名字,以及一個表示JVM是否要求解析類名字的標記(即是否同時裝入有依賴關(guān)系的類)。如果這個標記是true,我們只需在返回JVM之前調(diào)用resolveClass。


【Listing 2:ClassLoader.loadClass()的一個簡單實現(xiàn)】

     public Class loadClass( String name, boolean resolve )
     throws ClassNotFoundException {
   try {
     // 我們要創(chuàng)建的Class對象
      Class clasz = null;

     // 必需的步驟1:如果類已經(jīng)在系統(tǒng)緩沖之中,
     // 我們不必再次裝入它
     clasz = findLoadedClass( name );

     if (clasz != null)
       return clasz;

     // 下面是定制部分
     byte classData[] = /* 通過某種方法獲取字節(jié)碼數(shù)據(jù) */;
     if (classData != null) {
       // 成功讀取字節(jié)碼數(shù)據(jù),現(xiàn)在把它轉(zhuǎn)換成一個Class對象
       clasz = defineClass( name, classData, 0, classData.length );
     }

     // 必需的步驟2:如果上面沒有成功,
     // 我們嘗試用默認的ClassLoader裝入它
     if (clasz == null)
       clasz = findSystemClass( name );

     // 必需的步驟3:如有必要,則裝入相關(guān)的類
     if (resolve && clasz != null)

resolveClass( clasz );

     // 把類返回給調(diào)用者
     return clasz;

   } catch( IOException ie ) {
     throw new ClassNotFoundException( ie.toString() );
   } catch( GeneralSecurityException gse ) {
     throw new ClassNotFoundException( gse.toString() );
   }
}




Listing 2顯示了一個簡單的loadClass實現(xiàn)。代碼中的大部分對所有ClassLoader對象來說都一樣,但有一小部分(已通過注釋標記)是特有的。在處理過程中,ClassLoader對象要用到其他幾個輔助方法:

findLoadedClass:用來進行檢查,以便確認被請求的類當前還不存在。loadClass方法應(yīng)該首先調(diào)用它。
defineClass:獲得原始類文件字節(jié)碼數(shù)據(jù)之后,調(diào)用defineClass把它轉(zhuǎn)換成一個Class對象。任何loadClass實現(xiàn)都必須調(diào)用這個方法。
findSystemClass:提供默認ClassLoader的支持。如果用來尋找類的定制方法不能找到指定的類(或者有意地不用定制方法),則可以調(diào)用該方法嘗試默認的裝入方式。這是很有用的,特別是從普通的JAR文件裝入標準Java類時。
resolveClass:當JVM想要裝入的不僅包括指定的類,而且還包括該類引用的所有其他類時,它會把loadClass的resolve參數(shù)設(shè)置成true。這時,我們必須在返回剛剛裝入的Class對象給調(diào)用者之前調(diào)用resolveClass。

三、加密、解密
Java加密擴展即Java Cryptography Extension,簡稱JCE。它是Sun的加密服務(wù)軟件,包含了加密和密匙生成功能。JCE是JCA(Java Cryptography Architecture)的一種擴展。

JCE沒有規(guī)定具體的加密算法,但提供了一個框架,加密算法的具體實現(xiàn)可以作為服務(wù)提供者加入。除了JCE框架之外,JCE軟件包還包含了SunJCE服務(wù)提供者,其中包括許多有用的加密算法,比如DES(Data Encryption Standard)和Blowfish。

為簡單計,在本文中我們將用DES算法加密和解密字節(jié)碼。下面是用JCE加密和解密數(shù)據(jù)必須遵循的基本步驟:

步驟1:生成一個安全密匙。在加密或解密任何數(shù)據(jù)之前需要有一個密匙。密匙是隨同被加密的應(yīng)用一起發(fā)布的一小段數(shù)據(jù),Listing 3顯示了如何生成一個密匙。
【Listing 3:生成一個密匙】

// DES算法要求有一個可信任的隨機數(shù)源
SecureRandom sr = new SecureRandom();

// 為我們選擇的DES算法生成一個KeyGenerator對象
KeyGenerator kg = KeyGenerator.getInstance( "DES" );
kg.init( sr );

// 生成密匙
SecretKey key = kg.generateKey();

// 獲取密匙數(shù)據(jù)
byte rawKeyData[] = key.getEncoded();

/* 接下來就可以用密匙進行加密或解密,或者把它保存
    為文件供以后使用 */
doSomething( rawKeyData );



步驟2:加密數(shù)據(jù)。得到密匙之后,接下來就可以用它加密數(shù)據(jù)。除了解密的ClassLoader之外,一般還要有一個加密待發(fā)布應(yīng)用的獨立程序(見Listing 4)。
【Listing 4:用密匙加密原始數(shù)據(jù)】

   // DES算法要求有一個可信任的隨機數(shù)源
   SecureRandom sr = new SecureRandom();

   byte rawKeyData[] = /* 用某種方法獲得密匙數(shù)據(jù) */;

   // 從原始密匙數(shù)據(jù)創(chuàng)建DESKeySpec對象
   DESKeySpec dks = new DESKeySpec( rawKeyData );

   // 創(chuàng)建一個密匙工廠,然后用它把DESKeySpec轉(zhuǎn)換成
   // 一個SecretKey對象
   SecretKeyFactory keyFactory = SecretKeyFactory.getInstance( "DES" );
   SecretKey key = keyFactory.generateSecret( dks );

   // Cipher對象實際完成加密操作
   Cipher cipher = Cipher.getInstance( "DES" );

   // 用密匙初始化Cipher對象
   cipher.init( Cipher.ENCRYPT_MODE, key, sr );

   // 現(xiàn)在,獲取數(shù)據(jù)并加密
   byte data[] = /* 用某種方法獲取數(shù)據(jù) */

   // 正式執(zhí)行加密操作
   byte encryptedData[] = cipher.doFinal( data );

   // 進一步處理加密后的數(shù)據(jù)
   doSomething( encryptedData );


步驟3:解密數(shù)據(jù)。運行經(jīng)過加密的應(yīng)用時,ClassLoader分析并解密類文件。操作步驟如Listing 5所示。
【Listing 5:用密匙解密數(shù)據(jù)】

   // DES算法要求有一個可信任的隨機數(shù)源
   SecureRandom sr = new SecureRandom();

   byte rawKeyData[] = /* 用某種方法獲取原始密匙數(shù)據(jù) */;

   // 從原始密匙數(shù)據(jù)創(chuàng)建一個DESKeySpec對象
   DESKeySpec dks = new DESKeySpec( rawKeyData );

   // 創(chuàng)建一個密匙工廠,然后用它把DESKeySpec對象轉(zhuǎn)換成
   // 一個SecretKey對象
   SecretKeyFactory keyFactory = SecretKeyFactory.getInstance( "DES" );
   SecretKey key = keyFactory.generateSecret( dks );

   // Cipher對象實際完成解密操作
   Cipher cipher = Cipher.getInstance( "DES" );

   // 用密匙初始化Cipher對象
   cipher.init( Cipher.DECRYPT_MODE, key, sr );

// 現(xiàn)在,獲取數(shù)據(jù)并解密
   byte encryptedData[] = /* 獲得經(jīng)過加密的數(shù)據(jù) */

   // 正式執(zhí)行解密操作
   byte decryptedData[] = cipher.doFinal( encryptedData );

   // 進一步處理解密后的數(shù)據(jù)
   doSomething( decryptedData );


四、應(yīng)用實例
前面介紹了如何加密和解密數(shù)據(jù)。要部署一個經(jīng)過加密的應(yīng)用,步驟如下:

步驟1:創(chuàng)建應(yīng)用。我們的例子包含一個App主類,兩個輔助類(分別稱為Foo和Bar)。這個應(yīng)用沒有什么實際功用,但只要我們能夠加密這個應(yīng)用,加密其他應(yīng)用也就不在話下。
步驟2:生成一個安全密匙。在命令行,利用GenerateKey工具(參見GenerateKey.java)把密匙寫入一個文件: % java GenerateKey key.data


步驟3:加密應(yīng)用。在命令行,利用EncryptClasses工具(參見EncryptClasses.java)加密應(yīng)用的類: % java EncryptClasses key.data App.class Foo.class Bar.class

該命令把每一個.class文件替換成它們各自的加密版本。
步驟4:運行經(jīng)過加密的應(yīng)用。用戶通過一個DecryptStart程序運行經(jīng)過加密的應(yīng)用。DecryptStart程序如Listing 6所示。
【Listing 6:DecryptStart.java,啟動被加密應(yīng)用的程序】

import java.io.*;
import java.security.*;
import java.lang.reflect.*;
import javax.crypto.*;
import javax.crypto.spec.*;

public class DecryptStart extends ClassLoader
{
// 這些對象在構(gòu)造函數(shù)中設(shè)置,
// 以后loadClass()方法將利用它們解密類
private SecretKey key;
private Cipher cipher;

// 構(gòu)造函數(shù):設(shè)置解密所需要的對象
public DecryptStart( SecretKey key ) throws GeneralSecurityException,
     IOException {
   this.key = key;

   String algorithm = "DES";
   SecureRandom sr = new SecureRandom();
   System.err.println( "[DecryptStart: creating cipher]" );
   cipher = Cipher.getInstance( algorithm );
   cipher.init( Cipher.DECRYPT_MODE, key, sr );
}

// main過程:我們要在這里讀入密匙,創(chuàng)建DecryptStart的
// 實例,它就是我們的定制ClassLoader。
// 設(shè)置好ClassLoader以后,我們用它裝入應(yīng)用實例,
// 最后,我們通過Java Reflection API調(diào)用應(yīng)用實例的main方法
static public void main( String args[] ) throws Exception {
   String keyFilename = args[0];
   String appName = args[1];

    // 這些是傳遞給應(yīng)用本身的參數(shù)
   String realArgs[] = new String[args.length-2];
   System.arraycopy( args, 2, realArgs, 0, args.length-2 );

   // 讀取密匙
   System.err.println( "[DecryptStart: reading key]" );
   byte rawKey[] = Util.readFile( keyFilename );
   DESKeySpec dks = new DESKeySpec( rawKey );
   SecretKeyFactory keyFactory = SecretKeyFactory.getInstance( "DES" );
   SecretKey key = keyFactory.generateSecret( dks );

   // 創(chuàng)建解密的ClassLoader
   DecryptStart dr = new DecryptStart( key );

   // 創(chuàng)建應(yīng)用主類的一個實例
   // 通過ClassLoader裝入它
   System.err.println( "[DecryptStart: loading "+appName+"]" );
   Class clasz = dr.loadClass( appName );

   // 最后,通過Reflection API調(diào)用應(yīng)用實例
   // 的main()方法

   // 獲取一個對main()的引用
   String proto[] = new String[1];
   Class mainArgs[] = { (new String[1]).getClass() };
   Method main = clasz.getMethod( "main", mainArgs );

   // 創(chuàng)建一個包含main()方法參數(shù)的數(shù)組
   Object argsArray[] = { realArgs };
   System.err.println( "[DecryptStart: running "+appName+".main()]" );

   // 調(diào)用main()
   main.invoke( null, argsArray );
}

public Class loadClass( String name, boolean resolve )
     throws ClassNotFoundException {
   try {
     // 我們要創(chuàng)建的Class對象
     Class clasz = null;

     // 必需的步驟1:如果類已經(jīng)在系統(tǒng)緩沖之中
     // 我們不必再次裝入它
     clasz = findLoadedClass( name );

if (clasz != null)
       return clasz;

     // 下面是定制部分
     try {
       // 讀取經(jīng)過加密的類文件
       byte classData[] = Util.readFile( name+".class" );

       if (classData != null) {
         // 解密...
         byte decryptedClassData[] = cipher.doFinal( classData );

         // ... 再把它轉(zhuǎn)換成一個類
         clasz = defineClass( name, decryptedClassData,
           0, decryptedClassData.length );
         System.err.println( "[DecryptStart: decrypting class "+name+"]" );
       }
     } catch( FileNotFoundException fnfe ) {
     }

     // 必需的步驟2:如果上面沒有成功
     // 我們嘗試用默認的ClassLoader裝入它
     if (clasz == null)
       clasz = findSystemClass( name );

     // 必需的步驟3:如有必要,則裝入相關(guān)的類
     if (resolve && clasz != null)
       resolveClass( clasz );

     // 把類返回給調(diào)用者
     return clasz;
   } catch( IOException ie ) {
     throw new ClassNotFoundException( ie.toString()
);
   } catch( GeneralSecurityException gse ) {
     throw new ClassNotFoundException( gse.toString()
);
   }
}
}

對于未經(jīng)加密的應(yīng)用,正常執(zhí)行方式如下: % java App arg0 arg1 arg2

對于經(jīng)過加密的應(yīng)用,則相應(yīng)的運行方式為:% java DecryptStart key.data App arg0 arg1 arg2


DecryptStart有兩個目的。一個DecryptStart的實例就是一個實施即時解密操作的定制ClassLoader;同時,DecryptStart還包含一個main過程,它創(chuàng)建解密器實例并用它裝入和運行應(yīng)用。示例應(yīng)用App的代碼包含在App.java、Foo.java和Bar.java內(nèi)。Util.java是一個文件I/O工具,本文示例多處用到了它。完整的代碼請從本文最后下載。

五、注意事項
我們看到,要在不修改源代碼的情況下加密一個Java應(yīng)用是很容易的。不過,世上沒有完全安全的系統(tǒng)。本文的加密方式提供了一定程度的源代碼保護,但對某些攻擊來說它是脆弱的。

雖然應(yīng)用本身經(jīng)過了加密,但啟動程序DecryptStart沒有加密。攻擊者可以反編譯啟動程序并修改它,把解密后的類文件保存到磁盤。降低這種風險的辦法之一是對啟動程序進行高質(zhì)量的模糊處理。或者,啟動程序也可以采用直接編譯成機器語言的代碼,使得啟動程序具有傳統(tǒng)執(zhí)行文件格式的安全性。

另外還要記住的是,大多數(shù)JVM本身并不安全。狡猾的黑客可能會修改JVM,從ClassLoader之外獲取解密后的代碼并保存到磁盤,從而繞過本文的加密技術(shù)。Java沒有為此提供真正有效的補救措施。

不過應(yīng)該指出的是,所有這些可能的攻擊都有一個前提,這就是攻擊者可以得到密匙。如果沒有密匙,應(yīng)用的安全性就完全取決于加密算法的安全性。雖然這種保護代碼的方法稱不上十全十美,但它仍不失為一種保護知識產(chǎn)權(quán)和敏感用戶數(shù)據(jù)的有效方案。

參考資源

在運行時刻更新功能模塊。介紹了一個利用類庫加載器ClassLoader 實現(xiàn)在運行時刻更新部分功能模塊的Java程序,并將其與C/C++中實現(xiàn)同樣功能的動態(tài)鏈接庫方案進行了比較。
Java 技巧 105:利用 JWhich 掌握類路徑。展示一個簡單的工具,它可以清楚地確定類裝載器從類路徑中載入了什么 Java 類。
要了解更多的 Java 安全信息,請閱讀 java.sun.com的 Java Security API 頁。
如何封鎖您的(或打開別人的) Java 代碼。Java 代碼反編譯和模糊處理的指南。
使您的軟件運行起來:擺弄數(shù)字。真正安全的軟件需要精確的隨機數(shù)生成器。

下載本文代碼:EncryptedJavaClass_code.zip

關(guān)于作者
俞良松,軟件工程師,獨立顧問和自由撰稿人。最初從事PB和Oracle開發(fā),現(xiàn)主要興趣在于Internet開發(fā)。您可以通過 javaman@163.net 和我聯(lián)系。

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統(tǒng)下載排行

主站蜘蛛池模板: 日韩色视频在线观看 | 久久国产精品-久久精品 | 亚洲国产日韩在线 | 国产在线观看午夜不卡 | 欧美在线观看一区二区三 | 欧美午夜成年片在线观看 | 亚洲视频在线观看 | 日韩欧美一区二区在线观看 | 亚洲免费大全 | 亚洲精品亚洲人成毛片不卡 | 国产日韩欧美自拍 | 中文字幕在线观看日韩 | 日韩精品在线播放 | 欧美亚洲综合另类型色妞 | 99re6国产精品免费播放 | 99精品视频在线视频免费观看 | 嘿嘿嘿视频在线观看网站 | 狠狠操成人 | 国产99r视频精品免费观看 | 91精品国产肉丝高跟在线 | www.在线免费观看 | 成人精品一区二区久久 | 国产高清在线视频一区二区三区 | 欧美国产精品日韩在线 | 91亚洲精品在看在线观看高清 | youjizz日本人 | 国产高清在线看免费视频观 | 中文在线观看永久免费 | 香港三级日本三级人妇网站 | 免费国产一区二区三区 | 中国美女一级毛片 | 伊人久久在线视频 | 国产尤物 | 一区二区三区亚洲 | 在线观看日本www | 思思久久99er热只有频精品66 | 91精品国产手机在线版 | 成人在线免费播放 | 欧美bbbbbxxxxx | 日韩在线三级视频 | 免费人成在线观看网站 |