java實(shí)現(xiàn)反編譯的方法

反編譯的過程與編譯剛好相反,就是將已編譯好的編程語言還原到未編譯的狀態(tài),也就是找出程序語言的源代碼。就是將機(jī)器看得懂的語言轉(zhuǎn)換成程序員可以看得懂的語言。Java語言中的反編譯一般指將class文件轉(zhuǎn)換成java文件。

創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:做網(wǎng)站、網(wǎng)站制作、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的阿拉山口網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

Java常用反編譯工具

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

JAVAP

javap是jdk自帶的一個(gè)工具,可以對(duì)代碼反編譯,也可以查看java編譯器生成的字節(jié)碼。javap和其他兩個(gè)反編譯工具最大的區(qū)別是他生成的文件并不是java文件,也不像其他兩個(gè)工具生成代碼那樣更容易理解。拿一段簡單的代碼舉例,如我們想分析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;
        }
    }
}

執(zhí)行以下兩個(gè)命令:

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并沒有將字節(jié)碼反編譯成java文件,而是生成了一種我們可以看得懂字節(jié)碼。其實(shí)javap生成的文件仍然是字節(jié)碼,只是程序員可以稍微看得懂一些。如果你對(duì)字節(jié)碼有所掌握,還是可以看得懂以上的代碼的。其實(shí)就是把String轉(zhuǎn)成hashcode,然后進(jìn)行比較。

JAD

JAD是一個(gè)比較不錯(cuò)的反編譯工具,只要下載一個(gè)執(zhí)行工具,就可以實(shí)現(xiàn)對(duì)class文件的反編譯了。還是上面的源代碼,使用jad反編譯后內(nèi)容如下:

命令:jad.exe Decompilation.class 會(huì)生成一個(gè)Decompilation.jad的文件

JAD反編譯的結(jié)果如下:

// 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;
        }
    }
}

看上面的代碼這不就是標(biāo)準(zhǔn)的java的源代碼么。這個(gè)就很清楚的可以看到原來字符串的switch是通過equals()和hashCode()方法來實(shí)現(xiàn)的。

CFR

JAD很好用,但是無奈的是很久沒更新了,所以只能用一款新的工具替代他,CFR是一個(gè)不錯(cuò)的選擇,相比JAD來說,他的語法可能會(huì)稍微復(fù)雜一些,但是好在他可以用.

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

建議大家手動(dòng)通過javac Decompilation.java命令來編譯生成Decompilation.class文件,再做測試.

成功的反編譯結(jié)果如下:

/*
 * 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有很多參數(shù),還是剛剛的代碼,如果我們使用以下命令,輸出結(jié)果就會(huì)不同:
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表示對(duì)于switch支持string的細(xì)節(jié)進(jìn)行解碼。

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

--decodelambdas可以對(duì)lambda表達(dá)式進(jìn)行反編譯。

JD-GUI

JD-GUI 是一個(gè)用 C++ 開發(fā)的 Java反編譯工具,由 Pavel Kouznetsov開發(fā),支持Windows、Linux和蘋果Mac Os三個(gè)平臺(tái)。而且提供了Eclipse平臺(tái)下的插件JD-Eclipse。JD-GUI 基于GPLv3開源協(xié)議,對(duì)個(gè)人使用是完全免費(fèi)的。JD-GUI主要的是提供了可視化操作,直接拖拽文件到窗口既可,效果圖如下

java實(shí)現(xiàn)反編譯的方法

JadClipse

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

所需要資源: net.sf.jadclipse_3.3.0.jar插件jar和JAD.exe反編譯軟件(在文末有下載地址)

JadClipse下載地址在官網(wǎng)下載插件的jar包,然后將jar包放到eclipse的plugins目錄下;在打開Eclipse,Eclipse->Window->Preferences->Java,此時(shí)你會(huì)發(fā)現(xiàn)會(huì)比原來多了一個(gè)JadClipse的選項(xiàng)如下圖配置JadClipse:

java實(shí)現(xiàn)反編譯的方法

java實(shí)現(xiàn)反編譯的方法

基本配置完畢后,我們可以設(shè)置一下class文件的默認(rèn)打開方式:

Eclipse->Window->Preferences->General->Editors->File Associations 我們可以看到class文件的打開方式有兩個(gè),這里設(shè)置JadClipse和Eclipse自帶的Class File Viewer,而JadClipse是默認(rèn)的。

全部配置完成,下面我們可以查看源碼了,選擇需要查看的類,按F3即可查看源碼.如果JadClipse不是默認(rèn)設(shè)置,設(shè)置成默認(rèn)設(shè)置既可.

以上就是java如何進(jìn)行反編譯的詳細(xì)內(nèi)容,更多請(qǐng)關(guān)注創(chuàng)新互聯(lián)其它相關(guān)文章!

標(biāo)題名稱:java實(shí)現(xiàn)反編譯的方法
當(dāng)前網(wǎng)址:http://bm7419.com/article40/jccjho.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站建設(shè)、網(wǎng)站策劃商城網(wǎng)站、定制網(wǎng)站、定制開發(fā)域名注冊(cè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

營銷型網(wǎng)站建設(shè)