首頁 >Java >Java基礎 >java如何進行反編譯

java如何進行反編譯

尚
原創
2019-12-07 15:06:078954瀏覽

java如何進行反編譯

反編譯的過程與編譯剛好相反,就是將已編譯好的程式語言還原到未編譯的狀態,也就是找出程式語言的原始碼。就是將機器看得懂的語言轉換成程式設計師可以看得懂的語言。 Java語言中的反編譯一般指將class檔案轉換成java檔。

Java常用反編譯工具

本文主要介紹4個Java的反編譯工具:javap、jad和cfr以及視覺化反編譯工具JD-GUI

#JAVAP

javap是jdk自帶的工具,可以對程式碼反編譯,也可以查看java編譯器產生的字節碼。 javap和其他兩個反編譯工具最大的差別在於他產生的文件並不是java文件,也不像其他兩個工具產生程式碼那麼容易理解。拿一段簡單的程式碼舉例,如我們想分析Java 7中的switch是如何支援String的,我們先有以下可以編譯通過的原始碼:

public class switchDemoString {
    public static void main(String[] args) {
        String str = "world";
        switch (str) {
            case "hello":
                System.out.println("hello");
                break;
            case "world":
                System.out.println("world");
                break;
            default:
                break;
        }
    }
}

執行以下兩個指令:

javac Decompilation.java
javap -c Decompilation.class

生成程式碼如下:

Compiled from "Decompilation.java"
public class Decompilation {
  public Decompilation();
    Code:
       0: aload_0
       1: invokespecial #8                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: ldc           #16                 // String world
       2: astore_1
       3: aload_1
       4: dup
       5: astore_2
       6: invokevirtual #18                 // Method java/lang/String.hashCode:()I
       9: lookupswitch  { // 2
              99162322: 36
             113318802: 48
               default: 82
          }
      36: aload_2
      37: ldc           #24                 // String hello
      39: invokevirtual #26                 // Method java/lang/String.equals:(Ljava/lang/Object;)Z
      42: ifne          60
      45: goto          82
      48: aload_2
      49: ldc           #16                 // String world
      51: invokevirtual #26                 // Method java/lang/String.equals:(Ljava/lang/Object;)Z
      54: ifne          71
      57: goto          82
      60: getstatic     #30                 // Field java/lang/System.out:Ljava/io/PrintStream;
      63: ldc           #24                 // String hello
      65: invokevirtual #36                 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      68: goto          82
      71: getstatic     #30                 // Field java/lang/System.out:Ljava/io/PrintStream;
      74: ldc           #16                 // String world
      76: invokevirtual #36                 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      79: goto          82
      82: return
}

javap並沒有將字節碼反編譯成java文件,而是產生了一種我們可以看得懂字節碼。其實javap產生的檔案還是字節碼,只是程式設計師可以稍微看得懂一些。如果你對字節碼有所掌握,還是可以看得懂以上的程式碼的。其實就是把String轉成hashcode,然後再比較。

JAD

JAD是比較不錯的反編譯工具,只要下載執行工具,就可以實現對class檔案的反編譯了。還是上面的原始程式碼,使用jad反編譯後內容如下:

指令:jad.exe Decompilation.class 會產生一個Decompilation.jad的檔案

JAD反編譯的結果如下:

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Jad home page: http://www.kpdus.com/jad.html
// Decompiler options: packimports(3) 
// Source File Name:   Decompilation.java

package com.yveshe;

import java.io.PrintStream;

public class Decompilation
{

    public Decompilation()
    {
    }

    public static void main(String args[])
    {
        String str = "world";
        String s;
        switch((s = str).hashCode())
        {
        default:
            break;

        case 99162322: 
            if(s.equals("hello"))
                System.out.println("hello");
            break;

        case 113318802: 
            if(s.equals("world"))
                System.out.println("world");
            break;
        }
    }
}

看上面的程式碼這不就是標準的java的原始碼麼。這個就很清楚的可以看到原來字串的switch是透過equals()和hashCode()方法來實現的。

CFR

JAD很好用,但無奈的是很久沒更新了,所以只能用一款新的工具取代他,CFR是一個不錯的選擇,比起JAD來說,他的語法可能會稍微複雜一些,但是好在他可以用.

CFR將反編譯現代Java特性–Java 8 lambdas(Java和更早版本中的Java beta 103),已經反編譯Java 7 String,但CFR是完全用Java 6寫的.

建議大家手動透過javac Decompilation.java指令來編譯產生Decompilation.class檔,再做測試.

成功的反編譯結果如下:

/*
 * Decompiled with CFR 0_125.
 */
package com.yveshe;

import java.io.PrintStream;

public class Decompilation {
    public static void main(String[] args) {
        String str;
        String s = str = "world";
        switch (s.hashCode()) {
            default: {
                break;
            }
            case 99162322: {
                if (!s.equals("hello")) break;
                System.out.println("hello");
                break;
            }
            case 113318802: {
                if (!s.equals("world")) break;
                System.out.println("world");
            }
        }
    }
}

比較Jad來說,CFR有很多參數,還是剛剛的程式碼,如果我們使用以下指令,輸出結果就會不同:
E :\CRF>java -jar cfr_0_125.jar Decompilation.class

/*
 * Decompiled with CFR 0_125.
 */
package com.yveshe;

import java.io.PrintStream;

public class Decompilation {
    public static void main(String[] args) {
        String str;
        String s = str = "world";
        switch (s.hashCode()) {
            default: {
                break;
            }
            case 99162322: {
                if (!s.equals("hello")) break;
                System.out.println("hello");
                break;
            }
            case 113318802: {
                if (!s.equals("world")) break;
                System.out.println("world");
            }
        }
    }
}

--decodestringswitch表示對於switch支援string的細節進行解碼。

類似的還有--decodeenumswitch、--decodefinally、--decodelambdas等。

--decodelambdas可以對lambda表達式進行反編譯。

JD-GUI

JD-GUI 是一個用C 開發的Java反編譯工具,由Pavel Kouznetsov開發,支援Windows、Linux和蘋果Mac Os三個平台。而且提供了Eclipse平台下的插件JD-Eclipse。 JD-GUI 是基於GPLv3開源協議,對個人使用是完全免費的。 JD-GUI主要的是提供了視覺化操作,直接拖曳檔案到視窗既可,效果圖如下

java如何進行反編譯

JadClipse

java如何進行反編譯

java如何進行反編譯

# #在Eclipse中安裝Jad插件,注意這裡是安裝的是Jad插件不是Jd插件~

所需要資源: net.sf.jadclipse_3.3.0.jar插件jar和JAD.exe反編譯軟體(在文末有下載地址)JadClipse下載地址在官網下載插件的jar包,然後將jar包放到eclipse的plugins目錄下;在打開Eclipse,Eclipse->Window->Preferences-> Java,此時你會發現會比原來多了一個JadClipse的選項如下圖配置JadClipse:

##################基本配置完畢後,我們可以設定一下class檔案的預設開啟方式:######Eclipse->Window->Preferences->General->Editors->File Associations 我們可以看到class檔案的開啟方式有兩個,這裡設定JadClipse和Eclipse自帶的Class File Viewer,而JadClipse是預設的。 ######全部配置完成,下面我們可以查看源碼了,選擇需要查看的類,按F3即可查看源碼.如果JadClipse不是默認設置,設置成默認設置既可.######更多java知識請關注###java基礎教學###欄。 ######

以上是java如何進行反編譯的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn