久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術(shù)文章
文章詳情頁

最好的Java 反編譯工具的使用對比分析

瀏覽:5日期:2022-08-12 10:30:38
前言

Java 反編譯,一聽可能覺得高深莫測,其實(shí)反編譯并不是什么特別高級的操作,Java 對于 Class 字節(jié)碼文件的生成有著嚴(yán)格的要求,如果你非常熟悉 Java 虛擬機(jī)規(guī)范,了解 Class 字節(jié)碼文件中一些字節(jié)的作用,那么理解反編譯的原理并不是什么問題。甚至像下面這樣的 Class 文件你都能看懂一二。

最好的Java 反編譯工具的使用對比分析

一般在逆向研究和代碼分析中,反編譯用到的比較多。不過在日常開發(fā)中,有時(shí)候只是簡單的看一下所用依賴類的反編譯,也是十分重要的。

恰好最近工作中也需要用到 Java 反編譯,所以這篇文章介紹目前常見的的幾種 Java 反編譯工具的使用,在文章的最后也會通過編譯速度語法支持以及代碼可讀性三個(gè)維度,對它們進(jìn)行測試,分析幾款工具的優(yōu)缺點(diǎn)

Procyon

Github 鏈接:https://github.com/mstrobel/procyonProcyon 不僅僅是反編譯工具,它其實(shí)是專注于 Java 代碼的生成和分析的一整套的 Java 元編程工具。主要包括下面幾個(gè)部分:

Core Framework Reflection Framework Expressions Framework Compiler Toolset (Experimental) Java Decompiler (Experimental)

可以看到反編譯只是 Procyon 的其中一個(gè)模塊,Procyon 原來托管于 bitbucket,后來遷移到了 GitHub,根據(jù) GitHub 的提交記錄來看,也有將近兩年沒有更新了。不過也有依賴 Procyon 的其他的開源反編譯工具如** decompiler-procyon**,更新頻率還是很高的,下面也會選擇這個(gè)工具進(jìn)行反編譯測試。

使用 Procyon

<!-- https://mvnrepository.com/artifact/org.jboss.windup.decompiler/decompiler-procyon --><dependency> <groupId>org.jboss.windup.decompiler</groupId> <artifactId>decompiler-procyon</artifactId> <version>5.1.4.Final</version></dependency>

寫一個(gè)簡單的反編譯測試。

package com.wdbyte.decompiler;import java.io.IOException;import java.nio.file.Path;import java.nio.file.Paths;import java.util.Iterator;import java.util.List;import org.jboss.windup.decompiler.api.DecompilationFailure;import org.jboss.windup.decompiler.api.DecompilationListener;import org.jboss.windup.decompiler.api.DecompilationResult;import org.jboss.windup.decompiler.api.Decompiler;import org.jboss.windup.decompiler.procyon.ProcyonDecompiler;/** * Procyon 反編譯測試 * * @author https://github.com/niumoo * @date 2021/05/15 */public class ProcyonTest { public static void main(String[] args) throws IOException {Long time = procyon('decompiler.jar', 'procyon_output_jar');System.out.println(String.format('decompiler time: %dms', time)); } public static Long procyon(String source,String targetPath) throws IOException {long start = System.currentTimeMillis();Path outDir = Paths.get(targetPath);Path archive = Paths.get(source);Decompiler dec = new ProcyonDecompiler();DecompilationResult res = dec.decompileArchive(archive, outDir, new DecompilationListener() { public void decompilationProcessComplete() {System.out.println('decompilationProcessComplete'); } public void decompilationFailed(List<String> inputPath, String message) {System.out.println('decompilationFailed'); } public void fileDecompiled(List<String> inputPath, String outputPath) { } public boolean isCancelled() {return false; }});if (!res.getFailures().isEmpty()) { StringBuilder sb = new StringBuilder(); sb.append('Failed decompilation of ' + res.getFailures().size() + ' classes: '); Iterator failureIterator = res.getFailures().iterator(); while (failureIterator.hasNext()) {DecompilationFailure dex = (DecompilationFailure)failureIterator.next();sb.append(System.lineSeparator() + ' ').append(dex.getMessage()); } System.out.println(sb.toString());}System.out.println('Compilation results: ' + res.getDecompiledFiles().size() + ' succeeded, ' + res.getFailures().size() + ' failed.');dec.close();Long end = System.currentTimeMillis();return end - start; }}

Procyon 在反編譯時(shí)會實(shí)時(shí)輸出反編譯文件數(shù)量的進(jìn)度情況,最后還會統(tǒng)計(jì)反編譯成功和失敗的 Class 文件數(shù)量。

....五月 15, 2021 10:58:28 下午 org.jboss.windup.decompiler.procyon.ProcyonDecompiler$3 call信息: Decompiling 650 / 783五月 15, 2021 10:58:30 下午 org.jboss.windup.decompiler.procyon.ProcyonDecompiler$3 call信息: Decompiling 700 / 783五月 15, 2021 10:58:37 下午 org.jboss.windup.decompiler.procyon.ProcyonDecompiler$3 call信息: Decompiling 750 / 783decompilationProcessCompleteCompilation results: 783 succeeded, 0 failed.decompiler time: 40599ms

Procyon GUI

對于 Procyon 反編譯來說,在 GitHub 上也有基于此實(shí)現(xiàn)的開源 GUI 界面,感興趣的可以下載嘗試。Github 地址:https://github.com/deathmarine/Luyten

CFR

GitHub 地址:https://github.com/leibnitz27/cfrCFR 官方網(wǎng)站:http://www.benf.org/other/cfr/(可能需要FQ)Maven 倉庫: https://mvnrepository.com/artifact/org.benf/cfr

CFR(Class File Reader) 可以支持 Java 9、Java 12、Java 14 以及其他的最新版 Java 代碼的反編譯工作。而且 CFR 本身的代碼是由 Java 6 編寫,所以基本可以使用 CFR 在任何版本的 Java 程序中。值得一提的是,使用 CFR 甚至可以將使用其他語言編寫的的 JVM 類文件反編譯回 Java 文件。

CFR 命令行使用

使用 CFR 反編譯時(shí),你可以下載已經(jīng)發(fā)布的 JAR 包,進(jìn)行命令行反編譯,也可以使用 Maven 引入的方式,在代碼中使用。下面先說命令行運(yùn)行的方式。

直接在 GitHub Tags 下載已發(fā)布的最新版 JAR. 可以直接運(yùn)行查看幫助。

# 查看幫助java -jar cfr-0.151.jar --help

如果只是反編譯某個(gè) class.

# 反編譯 class 文件,結(jié)果輸出到控制臺java -jar cfr-0.151.jar WindupClasspathTypeLoader.class# 反編譯 class 文件,結(jié)果輸出到 out 文件夾java -jar cfr-0.151.jar WindupClasspathTypeLoader.class --outputpath ./out

反編譯某個(gè) JAR.

# 反編譯 jar 文件,結(jié)果輸出到 output_jar 文件夾➜ Desktop java -jar cfr-0.151.jar decompiler.jar --outputdir ./output_jarProcessing decompiler.jar (use silent to silence)Processing com.strobel.assembler.metadata.ArrayTypeLoaderProcessing com.strobel.assembler.metadata.ParameterDefinitionProcessing com.strobel.assembler.metadata.MethodHandleProcessing com.strobel.assembler.metadata.signatures.FloatSignature.....

反編譯結(jié)果會按照 class 的包路徑寫入到指定文件夾中。

最好的Java 反編譯工具的使用對比分析

CFR 代碼中使用

添加依賴這里不提。

<!-- https://mvnrepository.com/artifact/org.benf/cfr --><dependency> <groupId>org.benf</groupId> <artifactId>cfr</artifactId> <version>0.151</version></dependency>

實(shí)際上我在官方網(wǎng)站和 GitHub 上都沒有看到具體的單元測試示例。不過沒有關(guān)系,既然能在命令行運(yùn)行,那么直接在 IDEA 中查看反編譯后的 Main 方法入口,看下命令行是怎么執(zhí)行的,就可以寫出自己的單元測試了。

package com.wdbyte.decompiler;import java.io.IOException;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import org.benf.cfr.reader.api.CfrDriver;import org.benf.cfr.reader.util.getopt.OptionsImpl;/** * CFR Test * * @author https://github.com/niumoo * @date 2021/05/15 */public class CFRTest { public static void main(String[] args) throws IOException {Long time = cfr('decompiler.jar', './cfr_output_jar');System.out.println(String.format('decompiler time: %dms', time));// decompiler time: 11655ms } public static Long cfr(String source, String targetPath) throws IOException {Long start = System.currentTimeMillis();// source jarList<String> files = new ArrayList<>();files.add(source);// target dirHashMap<String, String> outputMap = new HashMap<>();outputMap.put('outputdir', targetPath);OptionsImpl options = new OptionsImpl(outputMap);CfrDriver cfrDriver = new CfrDriver.Builder().withBuiltOptions(options).build();cfrDriver.analyse(files);Long end = System.currentTimeMillis();return (end - start); }}JD-Core

GiHub 地址:https://github.com/java-decompiler/jd-coreJD-core 官方網(wǎng)址:https://java-decompiler.github.io/JD-core 是一個(gè)的獨(dú)立的 Java 庫,可以用于 Java 的反編譯,支持從 Java 1 至 Java 12 的字節(jié)碼反編譯,包括 Lambda 表達(dá)式、方式引用、默認(rèn)方法等。知名的 JD-GUI 和 Eclipse 無縫集成反編譯引擎就是 JD-core。JD-core 提供了一些反編譯的核心功能,也提供了單獨(dú)的 Class 反編譯方法,但是如果你想在自己的代碼中去直接反編譯整個(gè) JAR 包,還是需要一些改造的,如果是代碼中有匿名函數(shù),Lambda 等,雖然可以直接反編譯,不過也需要額外考慮。

使用 JD-core

<!-- https://mvnrepository.com/artifact/org.jd/jd-core --><dependency> <groupId>org.jd</groupId> <artifactId>jd-core</artifactId> <version>1.1.3</version></dependency>

為了可以反編譯整個(gè) JAR 包,使用的代碼我做了一些簡單改造,以便于最后一部分的對比測試,但是這個(gè)示例中沒有考慮內(nèi)部類,Lambda 等會編譯出多個(gè) Class 文件的情況,所以不能直接使用在生產(chǎn)中。

package com.wdbyte.decompiler;import java.io.File;import java.io.IOException;import java.io.InputStream;import java.nio.file.Files;import java.nio.file.Path;import java.nio.file.Paths;import java.util.Enumeration;import java.util.HashMap;import java.util.jar.JarFile;import java.util.zip.ZipEntry;import java.util.zip.ZipFile;import org.apache.commons.io.IOUtils;import org.apache.commons.lang3.StringUtils;import org.jd.core.v1.ClassFileToJavaSourceDecompiler;import org.jd.core.v1.api.loader.Loader;import org.jd.core.v1.api.printer.Printer;/** * @author https://github.com/niumoo * @date 2021/05/15 */public class JDCoreTest { public static void main(String[] args) throws Exception {JDCoreDecompiler jdCoreDecompiler = new JDCoreDecompiler();Long time = jdCoreDecompiler.decompiler('decompiler.jar','jd_output_jar');System.out.println(String.format('decompiler time: %dms', time)); }}class JDCoreDecompiler{ private ClassFileToJavaSourceDecompiler decompiler = new ClassFileToJavaSourceDecompiler(); // 存放字節(jié)碼 private HashMap<String,byte[]> classByteMap = new HashMap<>(); /** * 注意:沒有考慮一個(gè) Java 類編譯出多個(gè) Class 文件的情況。 * * @param source * @param target * @return * @throws Exception */ public Long decompiler(String source,String target) throws Exception {long start = System.currentTimeMillis();// 解壓archive(source);for (String className : classByteMap.keySet()) { String path = StringUtils.substringBeforeLast(className, '/'); String name = StringUtils.substringAfterLast(className, '/'); if (StringUtils.contains(name, '$')) {name = StringUtils.substringAfterLast(name, '$'); } name = StringUtils.replace(name, '.class', '.java'); decompiler.decompile(loader, printer, className); String context = printer.toString(); Path targetPath = Paths.get(target + '/' + path + '/' + name); if (!Files.exists(Paths.get(target + '/' + path))) {Files.createDirectories(Paths.get(target + '/' + path)); } Files.deleteIfExists(targetPath); Files.createFile(targetPath); Files.write(targetPath, context.getBytes());}return System.currentTimeMillis() - start; } private void archive(String path) throws IOException {try (ZipFile archive = new JarFile(new File(path))) { Enumeration<? extends ZipEntry> entries = archive.entries(); while (entries.hasMoreElements()) {ZipEntry entry = entries.nextElement();if (!entry.isDirectory()) { String name = entry.getName(); if (name.endsWith('.class')) {byte[] bytes = null;try (InputStream stream = archive.getInputStream(entry)) { bytes = IOUtils.toByteArray(stream);}classByteMap.put(name, bytes); }} }} } private Loader loader = new Loader() {@Overridepublic byte[] load(String internalName) { return classByteMap.get(internalName);}@Overridepublic boolean canLoad(String internalName) { return classByteMap.containsKey(internalName);} }; private Printer printer = new Printer() {protected static final String TAB = ' ';protected static final String NEWLINE = 'n';protected int indentationCount = 0;protected StringBuilder sb = new StringBuilder();@Override public String toString() { String toString = sb.toString(); sb = new StringBuilder(); return toString;}@Override public void start(int maxLineNumber, int majorVersion, int minorVersion) {}@Override public void end() {}@Override public void printText(String text) { sb.append(text); }@Override public void printNumericConstant(String constant) { sb.append(constant); }@Override public void printStringConstant(String constant, String ownerInternalName) { sb.append(constant); }@Override public void printKeyword(String keyword) { sb.append(keyword); }@Override public void printDeclaration(int type, String internalTypeName, String name, String descriptor) { sb.append(name); }@Override public void printReference(int type, String internalTypeName, String name, String descriptor, String ownerInternalName) { sb.append(name); }@Override public void indent() { this.indentationCount++; }@Override public void unindent() { this.indentationCount--; }@Override public void startLine(int lineNumber) { for (int i=0; i<indentationCount; i++) sb.append(TAB); }@Override public void endLine() { sb.append(NEWLINE); }@Override public void extraLine(int count) { while (count-- > 0) sb.append(NEWLINE); }@Override public void startMarker(int type) {}@Override public void endMarker(int type) {} };}JD-GUI

GitHub 地址:https://github.com/java-decompiler/jd-guiJD-core 也提供了官方的 GUI 界面,需要的也可以直接下載嘗試。

最好的Java 反編譯工具的使用對比分析

Jadx

GitHub 地址:https://github.com/skylot/jadxJadx 是一款可以反編譯 JAR、APK、DEX、AAR、AAB、ZIP 文件的反編譯工具,并且也配有 Jadx-gui 用于界面操作。Jadx 使用 Grade 進(jìn)行依賴管理,可以自行克隆倉庫打包運(yùn)行。

git clone https://github.com/skylot/jadx.gitcd jadx./gradlew dist# 查看幫助 ./build/jadx/bin/jadx --help jadx - dex to java decompiler, version: devusage: jadx [options] <input files> (.apk, .dex, .jar, .class, .smali, .zip, .aar, .arsc, .aab)options: -d, --output-dir - output directory -ds, --output-dir-src - output directory for sources -dr, --output-dir-res - output directory for resources -r, --no-res- do not decode resources -s, --no-src- do not decompile source code --single-class - decompile a single class --output-format - can be ’java’ or ’json’, default: java -e, --export-gradle - save as android gradle project -j, --threads-count - processing threads count, default: 6 --show-bad-code - show inconsistent code (incorrectly decompiled) --no-imports- disable use of imports, always write entire package name --no-debug-info - disable debug info --add-debug-lines - add comments with debug line numbers if available --no-inline-anonymous - disable anonymous classes inline --no-replace-consts - don’t replace constant value with matching constant field --escape-unicode - escape non latin characters in strings (with u) --respect-bytecode-access-modifiers - don’t change original access modifiers --deobf - activate deobfuscation --deobf-min - min length of name, renamed if shorter, default: 3 --deobf-max - max length of name, renamed if longer, default: 64 --deobf-cfg-file - deobfuscation map file, default: same dir and name as input file with ’.jobf’ extension --deobf-rewrite-cfg - force to save deobfuscation map --deobf-use-sourcename - use source file name as class name alias --deobf-parse-kotlin-metadata - parse kotlin metadata to class and package names --rename-flags - what to rename, comma-separated, ’case’ for system case sensitivity, ’valid’ for java identifiers, ’printable’ characters, ’none’ or ’all’ (default) --fs-case-sensitive - treat filesystem as case sensitive, false by default --cfg - save methods control flow graph to dot file --raw-cfg - save methods control flow graph (use raw instructions) -f, --fallback - make simple dump (using goto instead of ’if’, ’for’, etc) -v, --verbose - verbose output (set --log-level to DEBUG) -q, --quiet - turn off output (set --log-level to QUIET) --log-level - set log level, values: QUIET, PROGRESS, ERROR, WARN, INFO, DEBUG, default: PROGRESS --version - print jadx version -h, --help - print this helpExample: jadx -d out classes.dex

根據(jù) HELP 信息,如果想要反編譯 decompiler.jar 到 out 文件夾。

./build/jadx/bin/jadx -d ./out ~/Desktop/decompiler.jar INFO - loading ...INFO - processing ...INFO - doneress: 1143 of 1217 (93%)

Fernflower

GitHub 地址:https://github.com/fesh0r/fernflowerFernflower 和 Jadx 一樣使用 Grade 進(jìn)行依賴管理,可以自行克隆倉庫打包運(yùn)行。

➜ fernflower-master ./gradlew buildBUILD SUCCESSFUL in 32s4 actionable tasks: 4 executed➜ fernflower-master java -jar build/libs/fernflower.jarUsage: java -jar fernflower.jar [-<option>=<value>]* [<source>]+ <destination>Example: java -jar fernflower.jar -dgs=true c:mysource c:my.jar d:decompiled➜ fernflower-master mkdir out➜ fernflower-master java -jar build/libs/fernflower.jar ~/Desktop/decompiler.jar ./outINFO: Decompiling class com/strobel/assembler/metadata/ArrayTypeLoaderINFO: ... doneINFO: Decompiling class com/strobel/assembler/metadata/ParameterDefinitionINFO: ... doneINFO: Decompiling class com/strobel/assembler/metadata/MethodHandle...➜ fernflower-master ll outtotal 1288-rw-r--r-- 1 darcy staff 595K 5 16 17:47 decompiler.jar➜ fernflower-master

Fernflower 在反編譯 JAR 包時(shí),默認(rèn)反編譯的結(jié)果也是一個(gè) JAR 包。Jad

反編譯速度

到這里已經(jīng)介紹了五款 Java 反編譯工具了,那么在日常開發(fā)中我們應(yīng)該使用哪一個(gè)呢?又或者在代碼分析時(shí)我們又該選擇哪一個(gè)呢?我想這兩種情況的不同,使用時(shí)的關(guān)注點(diǎn)也是不同的。如果是日常使用,讀讀代碼,我想應(yīng)該是對可讀性要求更高些,如果是大量的代碼分析工作,那么可能反編譯的速度和語法的支持上要求更高些。為了能有一個(gè)簡單的參考數(shù)據(jù),我使用 JMH 微基準(zhǔn)測試工具分別對這五款反編譯工具進(jìn)行了簡單的測試,下面是一些測試結(jié)果。

測試環(huán)境

環(huán)境變量 描述 處理器 2.6 GHz 六核Intel Core i7 內(nèi)存 16 GB 2667 MHz DDR4 Java 版本 JDK 14.0.2 測試方式 JMH 基準(zhǔn)測試。 待反編譯 JAR 1 procyon-compilertools-0.5.33.jar (1.5 MB) 待反編譯 JAR 2 python2java4common-1.0.0-20180706.084921-1.jar (42 MB)

反編譯 JAR 1:procyon-compilertools-0.5.33.jar (1.5 MB)

Benchmark Mode Cnt Score Units cfr avgt 10 6548.642 ± 363.502 ms/op fernflower avgt 10 12699.147 ± 1081.539 ms/op jdcore avgt 10 5728.621 ± 310.645 ms/op procyon avgt 10 26776.125 ± 2651.081 ms/op jadx avgt 10 7059.354 ± 323.351 ms/op

反編譯 JAR 2: python2java4common-1.0.0-20180706.084921-1.jar (42 MB)

JAR 2 這個(gè)包是比較大的,是拿很多代碼倉庫合并到一起的,同時(shí)還有很多 Python 轉(zhuǎn) Java 生成的代碼,理論上代碼的復(fù)雜度會更高。

Benchmark Cnt Score Cfr 1 413838.826ms fernflower 1 246819.168ms jdcore 1 Error procyon 1 487647.181ms jadx 1 505600.231ms 語法支持和可讀性

如果反編譯后的代碼需要自己看的話,那么可讀性更好的代碼更占優(yōu)勢,下面我寫了一些代碼,主要是 Java 8 及以下的代碼語法和一些嵌套的流程控制,看看反編譯后的效果如何。

package com.wdbyte.decompiler;import java.util.ArrayList;import java.util.List;import java.util.stream.IntStream;import org.benf.cfr.reader.util.functors.UnaryFunction;/** * @author https://www.wdbyte.com * @date 2021/05/16 */public class HardCode <A, B> { public HardCode(A a, B b) { } public static void test(int... args) { } public static void main(String... args) {test(1, 2, 3, 4, 5, 6); } int byteAnd0() {int b = 1;int x = 0;do { b = (byte)((b ^ x));} while (b++ < 10);return b; } private void a(Integer i) {a(i);b(i);c(i); } private void b(int i) {a(i);b(i);c(i); } private void c(double d) {c(d);d(d); } private void d(Double d) {c(d);d(d); } private void e(Short s) {b(s);c(s);e(s);f(s); } private void f(short s) {b(s);c(s);e(s);f(s); } void test1(String path) {try { int x = 3;} catch (NullPointerException t) { System.out.println('File Not found'); if (path == null) { return; } throw t;} finally { System.out.println('Fred'); if (path == null) { throw new IllegalStateException(); }} } private final List<Integer> stuff = new ArrayList<>();{stuff.add(1);stuff.add(2); } public static int plus(boolean t, int a, int b) {int c = t ? a : b;return c; } // Lambda Integer lambdaInvoker(int arg, UnaryFunction<Integer, Integer> fn) {return fn.invoke(arg); } // Lambda public int testLambda() {return lambdaInvoker(3, x -> x + 1);//return 1; } // Lambda public Integer testLambda(List<Integer> stuff, int y, boolean b) {return stuff.stream().filter(b ? x -> x > y : x -> x < 3).findFirst().orElse(null); } // stream public static <Y extends Integer> void testStream(List<Y> list) {IntStream s = list.stream() .filter(x -> {System.out.println(x);return x.intValue() / 2 == 0;}) .map(x -> (Integer)x+2) .mapToInt(x -> x);s.toArray(); } // switch public void testSwitch1(){int i = 0;switch(((Long)(i + 1L)) + '') { case '1':System.out.println('one');} } // switch public void testSwitch2(String string){switch (string) { case 'apples':System.out.println('apples');break; case 'pears':System.out.println('pears');break;} } // switch public static void testSwitch3(int x) {while (true) { if (x < 5) {switch ('test') { case 'okay':continue; default:continue;} } System.out.println('wow x2!');} }}

此處本來貼出了所有工具的反編譯結(jié)果,但是礙于文章長度和閱讀體驗(yàn),沒有放出來,不過我在個(gè)人博客的發(fā)布上是有完整代碼的,個(gè)人網(wǎng)站排版比較自由,可以使用 Tab 選項(xiàng)卡的方式展示。如果需要查看可以訪問 https://www.wdbyte.com 進(jìn)行查看。

Procyon

看到 Procyon 的反編譯結(jié)果,還是比較吃驚的,在正常反編譯的情況下,反編譯后的代碼基本上都是原汁原味。唯一一處反編譯后和源碼語法上有變化的地方,是一個(gè)集合的初始化操作略有不同。

// 源碼 public HardCode(A a, B b) { } private final List<Integer> stuff = new ArrayList<>();{ stuff.add(1); stuff.add(2); }// Procyon 反編譯private final List<Integer> stuff; public HardCode(final A a, final B b) { (this.stuff = new ArrayList<Integer>()).add(1); this.stuff.add(2);}

而其他部分代碼, 比如裝箱拆箱,Switch 語法,Lambda 表達(dá)式,流式操作以及流程控制等,幾乎完全一致,閱讀沒有障礙。

裝箱拆箱操作反編譯后完全一致,沒有多余的類型轉(zhuǎn)換代碼。

// 源碼private void a(Integer i) { a(i); b(i); c(i);}private void b(int i) { a(i); b(i); c(i);}private void c(double d) { c(d); d(d);}private void d(Double d) { c(d); d(d);}private void e(Short s) { b(s); c(s); e(s); f(s);}private void f(short s) { b(s); c(s); e(s); f(s);}// Procyon 反編譯private void a(final Integer i) { this.a(i); this.b(i); this.c(i);}private void b(final int i) { this.a(i); this.b(i); this.c(i);}private void c(final double d) { this.c(d); this.d(d);}private void d(final Double d) { this.c(d); this.d(d);}private void e(final Short s) { this.b(s); this.c(s); this.e(s); this.f(s);}private void f(final short s) { this.b(s); this.c(s); this.e(s); this.f(s);}

Switch 部分也是一致,流程控制部分也沒有變化。

// 源碼 switchpublic void testSwitch1(){ int i = 0; switch(((Long)(i + 1L)) + '') {case '1': System.out.println('one'); }}public void testSwitch2(String string){ switch (string) {case 'apples': System.out.println('apples'); break;case 'pears': System.out.println('pears'); break; }}public static void testSwitch3(int x) { while (true) {if (x < 5) { switch ('test') {case 'okay': continue;default: continue; }}System.out.println('wow x2!'); }}// Procyon 反編譯public void testSwitch1() { final int i = 0; final String string = (Object)(i + 1L) + ''; switch (string) {case '1': { System.out.println('one'); break;} }}public void testSwitch2(final String string) { switch (string) {case 'apples': { System.out.println('apples'); break;}case 'pears': { System.out.println('pears'); break;} }} public static void testSwitch3(final int x) { while (true) {if (x < 5) { final String s = 'test'; switch (s) {case 'okay': { continue;}default: { continue;} }}else { System.out.println('wow x2!');} }}

Lambda 表達(dá)式和流式操作完全一致。

// 源碼// Lambdapublic Integer testLambda(List<Integer> stuff, int y, boolean b) { return stuff.stream().filter(b ? x -> x > y : x -> x < 3).findFirst().orElse(null);}// streampublic static <Y extends Integer> void testStream(List<Y> list) { IntStream s = list.stream().filter(x -> { System.out.println(x); return x.intValue() / 2 == 0; }).map(x -> (Integer)x+2).mapToInt(x -> x); s.toArray();}// Procyon 反編譯public Integer testLambda(final List<Integer> stuff, final int y, final boolean b) { return stuff.stream().filter(b ? (x -> x > y) : (x -> x < 3)).findFirst().orElse(null);}public static <Y extends Integer> void testStream(final List<Y> list) { final IntStream s = list.stream().filter(x -> {System.out.println(x);return x / 2 == 0; }).map(x -> x + 2).mapToInt(x -> x); s.toArray();}

流程控制,反編譯后發(fā)現(xiàn)丟失了無意義的代碼部分,閱讀來說并無障礙。

// 源碼void test1(String path) { try {int x = 3; } catch (NullPointerException t) {System.out.println('File Not found');if (path == null) { return; }throw t; } finally {System.out.println('Fred');if (path == null) { throw new IllegalStateException(); } }}// Procyon 反編譯void test1(final String path) { try {} catch (NullPointerException t) {System.out.println('File Not found');if (path == null) { return;}throw t; } finally {System.out.println('Fred');if (path == null) { throw new IllegalStateException();} }}

鑒于代碼篇幅,下面幾種的反編譯結(jié)果的對比只會列出不同之處,相同之處會直接跳過。

CFR

CFR 的反編譯結(jié)果多出了類型轉(zhuǎn)換部分,個(gè)人來看沒有 Procyon 那么原汁原味,不過也算是十分優(yōu)秀,測試案例中唯一不滿意的地方是對 while continue 的處理。

// CFR 反編譯結(jié)果// 裝箱拆箱private void e(Short s) { this.b(s.shortValue()); // 裝箱拆箱多出了類型轉(zhuǎn)換部分。 this.c(s.shortValue()); // 裝箱拆箱多出了類型轉(zhuǎn)換部分。 this.e(s); this.f(s);}// 流程控制void test1(String path) { try {int n = 3;// 流程控制反編譯結(jié)果十分滿意,原汁原味,甚至此處的無意思代碼都保留了。 } catch (NullPointerException t) {System.out.println('File Not found');if (path == null) { return;}throw t; } finally {System.out.println('Fred');if (path == null) { throw new IllegalStateException();} }}// Lambda 和 Stream 操作完全一致,不提。// switch 處,反編譯后功能一致,但是流程控制有所更改。public static void testSwitch3(int x) { block6: while (true) { // 源碼中只有 while(true),反編譯后多了 block6if (x < 5) { switch ('test') {case 'okay': { continue block6; // 多了 block6} } continue;}System.out.println('wow x2!'); }}JD-Core

JD-Core 和 CFR 一樣,對于裝箱拆箱操作,反編譯后不再一致,多了類型轉(zhuǎn)換部分,而且自動優(yōu)化了數(shù)據(jù)類型。個(gè)人感覺,如果是反編譯后自己閱讀,通篇的數(shù)據(jù)類型的轉(zhuǎn)換優(yōu)化影響還是挺大的。

// JD-Core 反編譯private void d(Double d) { c(d.doubleValue()); // 新增了數(shù)據(jù)類型轉(zhuǎn)換 d(d);}private void e(Short s) { b(s.shortValue()); // 新增了數(shù)據(jù)類型轉(zhuǎn)換 c(s.shortValue()); // 新增了數(shù)據(jù)類型轉(zhuǎn)換 e(s); f(s.shortValue()); // 新增了數(shù)據(jù)類型轉(zhuǎn)換}private void f(short s) { b(s); c(s); e(Short.valueOf(s)); // 新增了數(shù)據(jù)類型轉(zhuǎn)換 f(s);}// Stream 操作中,也自動優(yōu)化了數(shù)據(jù)類型轉(zhuǎn)換,閱讀起來比較累。public static <Y extends Integer> void testStream(List<Y> list) { IntStream s = list.stream().filter(x -> {System.out.println(x);return (x.intValue() / 2 == 0); }).map(x -> Integer.valueOf(x.intValue() + 2)).mapToInt(x -> x.intValue()); s.toArray();}Jadx

首先 Jadx 在反編譯測試代碼時(shí),報(bào)出了錯誤,反編譯的結(jié)果里也有提示不能反編 Lambda 和 Stream 操作,反編譯結(jié)果中變量名稱雜亂無章流程控制幾乎陣亡,如果你想反編譯后生物肉眼閱讀,Jadx 肯定不是一個(gè)好選擇。

// Jadx 反編譯private void e(Short s) { b(s.shortValue());// 新增了數(shù)據(jù)類型轉(zhuǎn)換 c((double) s.shortValue());// 新增了數(shù)據(jù)類型轉(zhuǎn)換 e(s); f(s.shortValue());// 新增了數(shù)據(jù)類型轉(zhuǎn)換}private void f(short s) { b(s); c((double) s);// 新增了數(shù)據(jù)類型轉(zhuǎn)換 e(Short.valueOf(s));// 新增了數(shù)據(jù)類型轉(zhuǎn)換 f(s);}public int testLambda() { // testLambda 反編譯失敗 /*r2 = this;r0 = 3r1 = move-resultjava.lang.Integer r0 = r2.lambdaInvoker(r0, r1)int r0 = r0.intValue()return r0 */ throw new UnsupportedOperationException('Method not decompiled: com.wdbyte.decompiler.HardCode.testLambda():int');}// Stream 反編譯失敗public static <Y extends java.lang.Integer> void testStream(java.util.List<Y> r3) { /*java.util.stream.Stream r1 = r3.stream()r2 = move-resultjava.util.stream.Stream r1 = r1.filter(r2)r2 = move-resultjava.util.stream.Stream r1 = r1.map(r2)r2 = move-resultjava.util.stream.IntStream r0 = r1.mapToInt(r2)r0.toArray()return */ throw new UnsupportedOperationException('Method not decompiled: com.wdbyte.decompiler.HardCode.testStream(java.util.List):void');}public void testSwitch2(String string) { // switch 操作無法正常閱讀,和源碼出入較大。 char c = 65535; switch (string.hashCode()) {case -1411061671: if (string.equals('apples')) {c = 0;break; } break;case 106540109: if (string.equals('pears')) {c = 1;break; } break; } switch (c) {case 0: System.out.println('apples'); return;case 1: System.out.println('pears'); return;default: return; }}Fernflower

Fernflower 的反編譯結(jié)果總體上還是不錯的,不過也有不足,它對變量名稱的指定,以及 Switch 字符串時(shí)的反編譯結(jié)果不夠理想。

//反編譯后變量命名不利于閱讀,有很多 var 變量int byteAnd0() { int b = 1; byte x = 0; byte var10000; do { int b = (byte)(b ^ x); var10000 = b; b = b + 1; } while(var10000 < 10); return b;}// switch 反編譯結(jié)果使用了hashCodepublic static void testSwitch3(int x) { while(true) { if (x < 5) { String var1 = 'test'; byte var2 = -1; switch(var1.hashCode()) { case 3412756: if (var1.equals('okay')) { var2 = 0; } default: switch(var2) { case 0: } } } else { System.out.println('wow x2!'); } }}總結(jié)

五種反編譯工具比較下來,結(jié)合反編譯速度和代碼可讀性測試,看起來 CFR 工具勝出,Procyon 緊隨其后。CFR 在速度上不落下風(fēng),在反編譯的代碼可讀性上,是最好的,主要體現(xiàn)在反編譯后的變量命名裝箱拆箱類型轉(zhuǎn)換流程控制上,以及對 Lambda 表達(dá)式、Stream 流式操作和 Switch語法支持上,都非常優(yōu)秀。根據(jù) CFR 官方介紹,已經(jīng)支持到 Java 14 語法,而且截止寫這篇測試文章時(shí),CFR 最新提交代碼時(shí)間實(shí)在 11 小時(shí)之前,更新速度很快。

文章中部分代碼已經(jīng)上傳 GitHub :github.com/niumoo/lab-notes/tree/master/java-decompiler

以上就是最好的Java 反編譯工具的使用對比分析的詳細(xì)內(nèi)容,更多關(guān)于Java 反編譯工具的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 国产一级免费视频 | 黄色毛片在线看 | 久久久久国产精品 | 亚洲大奶网 | 久久成人国产精品 | 欧美一级二级片 | 久久久久国产一区二区三区 | 成人1区 | 午夜精品在线 | 欧美精品一区久久 | 欧美日韩不卡在线 | 欧美成人免费在线视频 | 午夜在线电影 | 91精品国产综合久久久亚洲 | 久草视频免费看 | 国产高清精品在线 | 老司机午夜免费精品视频 | 久久一级| 老汉色影院 | 日日操av | 日韩中文字幕一区 | 一二区精品 | 99精品欧美一区二区蜜桃免费 | 日韩免费激情视频 | 中文字幕在线观看免费视频 | 亚洲91精品| 91在线网址 | 久久高清 | 美女逼网站 | 日韩中文字幕 | 美女高潮久久久 | 福利视频一区 | 国产欧美日韩精品一区 | 亚洲一区二区三区视频 | 国产精品第一国产精品 | 亚洲精品一区二区三区蜜桃下载 | 日韩精品一区二区三区中文字幕 | 免费人成电影 | 午夜欧美一区二区三区在线播放 | 亚洲精品二区三区 | 亚洲一区| 亚洲91精品 | 国产视频2021 | 日韩精品一区二区三区中文在线 | 丁香婷婷久久久综合精品国产 | 亚洲第一视频 | 久久久久亚洲 | 爱爱网av| 午夜午夜精品一区二区三区文 | 精品一区二区三区在线观看 | 中文久久| 成人在线视频免费观看 | 精品国产一级片 | 久久精品国产一区二区三区不卡 | 国产成人一区 | 欧美日韩国产综合视频 | 99精品九九| 日韩精品一二三区 | 影音先锋久久 | 精品亚洲一区二区三区 | 欧美日韩一级电影 | 夏同学福利网 | 国产亚洲一区二区精品 | 亚洲精品麻豆 | 性色视频免费观看 | 亚洲 国产 另类 精品 专区 | 香蕉久久av一区二区三区 | 国产精品影院在线观看 | 色婷婷亚洲一区二区三区 | 亚洲一区视频在线 | 色网在线看 | 欧美成人精品一区二区男人看 | 免费国产成人 | 午夜精品久久久久久 | 私人毛片免费高清视频 | 一级毛片免费视频 | 久久电影一区 | 国产中文字幕在线观看 | 欧美,日韩,国产精品免费观看 | 偷拍电影一区二区三区 | 久久久精品视频免费观看 | 激情小网站 | 成人黄色在线视频 | 欧美在线综合 | 久久天堂 | 欧美一区二区三区在线观看视频 | 黄色在线 | 亚洲精品在线视频观看 | 久久日韩| 色婷婷综合久久久久中文一区二区 | 中文字幕在线三区 | 在线第一页 | 午夜视 | 影音先锋亚洲资源 | 韩日免费视频 | 日韩一二三 | 成人免费视频网站在线看 | 成人福利视频 | 精品久久久久久久久久久 | 自拍偷拍第一页 | 国产成人精品久久二区二区91 | 色综合久久久久 | 九色av| 91综合网| 久久久久久亚洲一区二区三区蜜臀 | 男女羞羞视频免费观看 | 一区二区三区亚洲视频 | www.9191 | 婷婷午夜激情 | 精品久| 久久久久久网址 | 亚洲狠狠爱一区二区三区 | 91精品中文字幕一区二区三区 | 一区二区三区精品视频 | 亚洲三区在线观看 | 成人精品国产 | 国产精品美女一区二区三区四区 | 草草视频在线观看 | 亚洲精品久久久久久久久久久 | 综合五月激情 | 91嫩草在线 | 一区不卡| caoporn视频 | 99久久99久久| 日韩一区在线播放 | 91精品国产综合久久久久久 | 成人在线免费视频 | 91观看在线视频 | 国产一区二区免费 | 欧美日韩在线视频观看 | 黄色毛片在线观看 | 欧美在线三级 | 日韩成人在线一区 | 久久精品网 | 欧美一区二区免费 | 欧美日韩免费一区二区三区 | 欧美日韩国产一区二区三区不卡 | 国产免费视频在线 | 欧美三级视频在线播放 | 伊人影院久久 | 成年无码av片在线 | 日韩av在线一区 | 亚洲精品一区二区三区蜜桃久 | 欧美一区二区另类 | 日本精品一区二 | 天天精品| 中文字幕久久精品 | 国产亚洲精品精品国产亚洲综合 | 欧美一级片免费观看 | 日本一区二区高清视频 | 国产一区日韩欧美 | 国产精品美女视频一区二区三区 | 91在线播放视频 | 狠狠搞狠狠干 | 日本成人一二三区 | 北条麻妃99精品青青久久 | 成人免费视频视频在线观看 免费 | 成人 在线| 国产亚洲精品成人av久久ww | 色网在线观看 | 成人在线网址 | 欧美在线一级 | 天天干欧美 | 美女又黄又免费 | 91视频免费网站 | 久久狠狠| 亚洲国产成人一区二区精品区 | 精品一区二区三区在线观看视频 | 国产一区二区在线看 | 日本精品免费 | 精精国产xxxx视频在线 | 免费国产网站 | 国产精品69毛片高清亚洲 | 欧美视频免费在线 | 五月婷亚洲 | www.久久.com | 国内久久精品 | 91麻豆精品国产91久久久更新时间 | 欧美与黑人午夜性猛交久久久 | 国产精品一区二区三区在线播放 | 亚洲视频1区 | 国产免费一区二区三区四区五区 | 91av久久| 亚洲乱码一区二区 | 精品乱码久久久 | 久久伊人久久 | av在线综合网| 国产馆一区二区 | 亚洲九九 | 久久tv在线观看 | 操操网 | 精品国产一区二区三区久久久蜜 | 岛国a视频 | 欧美激情| 午夜伦理影院 | 亚洲欧美日韩在线 | 亚洲色图p| 日本免费在线视频 | 狠狠躁天天躁夜夜添人人 | 国产a视频 | 国产日产精品一区二区三区四区 | 国产艳妇av视国产精选av一区 | 欧美成人一区二区三区片免费 | 欧美二区在线观看 | 亚洲精品国品乱码久久久久 | 日韩视频一区在线观看 | 国产麻豆乱码精品一区二区三区 | 日韩在线成人 | 在线免费观看av片 | 色综合久久88色综合天天 | 97视频网站 | 91 久久| 一色一黄视频 | 精品乱码一区二区 | 国产综合视频在线观看 | 一区免费视频 | 亚洲第一黄 | 国产精品视频 | 日韩三级在线 | 欧美精品久久久 | 欧美日韩电影一区二区 | 久久久中文字幕 | 香蕉视频成人在线观看 | 日韩欧美在线播放 | 亚洲精品综合在线 | 午夜久久网站 | 69黄在线看片免费视频 | 欧美日韩精品在线一区 | 中文字幕av一区 | 国产精品视频免费观看 | 久久中文字幕一区二区三区 | 黄网站涩免费蜜桃网站 | 一级免费片 | 看亚洲a级一级毛片 | 91精品久久久久久久久中文字幕 | 亚洲电影一区二区三区 | 免费av黄色| 91高清视频在线观看 | 亚洲电影在线观看 | 免费中文字幕 | 成年人在线观看 | www.成人在线视频 | 国产91色在线 | 亚洲 | 久久久久久中文字幕 | 欧美精品综合 | 欧美激情欧美激情在线五月 | 欧美一区免费 | 自拍视频在线 | 精品在线一区 | 91成人免费在线观看 | 一级大片一级一大片 | 五月天婷婷精品 | 天天天堂| 少妇激烈床戏视频 | 精品成人佐山爱一区二区 | 日本一区二区三区四区 | 一级片手机免费看 | 亚洲精品亚洲人成人网 | 自拍偷拍亚洲欧洲 | 亚洲国产精品99久久久久久久久 | 午夜视频在线观看网站 | 欧美一级在线 | 色综合视频 | 久久久久久久久久久久久久av | 日韩精品一区二区三区在线播放 | 日韩精品一区在线 | 国产精品一区亚洲二区日本三区 | 国产色在线 | 天天看夜夜 | 后人极品翘臀美女在线播放 | 亚洲一区二区视频在线观看 | 日韩在线资源 | 国产精品久久久久久久久久99 | 粉嫩在线 | 91在线成人 | 在线视频亚洲 | 欧美日韩综合视频 | 91精品国产综合久久久久久丝袜 | 久章操| 久久毛片 | 国产精品视频99 | 国产精品久久久久久久久动漫 | 国产欧美日韩综合精品 | 国产精品视频久久久 | 91视频8mav| 精品久久久久久亚洲综合网 | 久久精品久久久久电影 | 国产成人在线看 | 玖玖在线精品 | 国产高清视频在线 | 亚洲aⅴ天堂av在线电影软件 | 国产在线一区二区三区 | 亚洲一本 | 97久久精品午夜一区二区 | 久在线 | 久久精品国产久精国产 | 一区二区福利 | 一区二区三区在线观看国产 | av黄在线观看 | 自拍视频在线观看 | 久久精品久久久久久 | www.99热这里只有精品 | 亚洲一区二区三区高清 | 亚洲精品国产综合 | 中文字幕在线观看亚洲 | 在线观看亚洲一区二区三区 | 一区二区在线免费观看 | 狠狠亚洲 | 四虎网址| 国产一区二区三区久久久久久久久 | 91精品久久久久久久久久 | 美日韩精品视频 | 日韩在线中文字幕 | 91精品国产一区二区 | 国产一级在线观看 | 91麻豆精品国产91久久久资源速度 | a亚洲精品| 精品国产一区二区三区四区 | 黄色免费一级 | 尹人成人| 99亚洲 | 亚洲国产精品一区二区久久 | 视频在线一区二区三区 | 91aiai| av中文字幕第一页 | 精品无码久久久久国产 | 亚洲综合色视频在线观看 | 亚洲中字幕 | 亚洲欧美第一页 | 男人天堂av网 | 久久久久女人精品毛片九一韩国 | 亚州中文字幕 | 天天射影院| 久久精品中文 | 成人国产精品一级毛片视频 | 色爱av| 天天夜碰日日摸日日澡 | 日韩小视频在线播放 | 最新国产精品精品视频 | 久福利| 精品久久久久久久久久久 | 97视频人人澡人人爽 | 福利91 | 日本免费视频在线观看 | 欧美一级在线免费观看 | 久久国产综合 | 亚洲精品一区中文字幕乱码 | 国产成人 综合 亚洲 | 大胆裸体gogo毛片免费看 | 国产欧美日本 | 91精品国产综合久久福利软件 | 国产精品2| 国产精品成人一区二区三区夜夜夜 | 亚洲视频一区二区三区四区 | 成人免费视频视频在线观看 免费 | 日韩国产精品视频 | 欧美激情自拍偷拍 | 中文字幕在线看第二 | 亚洲精品一区二三区不卡 | 日韩综合 | 国产高清免费 | 精品久久香蕉国产线看观看亚洲 | 欧美日韩国产在线 | 福利片在线观看 | 天天草av| 成人午夜精品久久久久久久蜜臀 | 免费一区二区三区 | 综合色婷婷一区二区亚洲欧美国产 | 天天操天天拍 | 天天操夜夜操 | 亚洲午夜电影在线 | 国内精品久久久久 | 久久99视频 | 久草成人| 日韩欧美精品一区二区三区 | 成人亚洲精品久久久久软件 | 午夜在线 | 日韩欧美综合 | 国产偷自视频区视频 | 天久久 | 日韩久久一区二区 | 日韩成人不卡 | 狠狠爱天天操 | 国语精品久久 | 亚洲综合影院 | 亚洲欧美日韩精品久久奇米色影视 | 成人福利在线 | 日韩一级黄色大片 | 国产拍拍视频 | 亚洲一级毛片 | 精品亚洲一区二区三区 | 美女天堂| 欧美一级片在线 | 欧美精品一区在线发布 | 色噜噜一区二区 | 三级视频网站 | 久久亚洲天堂 | 男女羞羞视频免费在线观看 | 国产一区二区在线播放 | 国产成人av在线 | 国产欧美日韩综合精品一区二区 | 亚洲天堂电影网 | 日韩精品一区二区三区第95 | 亚洲网站在线观看 | 九色自拍 | 999精品视频 | 君岛美绪一区二区三区在线视频 | 在线日韩一区 | 亚洲瑟瑟| 国产综合视频在线观看 | 国产一级免费视频 | 久久国产欧美一区二区三区精品 | 中文字幕一区二区三区乱码图片 | 成年人免费网站 | 日本久久久一区二区三区 | 午夜在线小视频 | 在线视频中文字幕 | 国产亚洲精品久久久久动 | 国产区免费观看 | 欧美性猛交一区二区三区精品 | 韩国av一区二区 | 国产精品一区二区久久久久 | 久久精品这里热有精品 | 天堂久久爱资源站www | 久久99精品视频 | 综合一区| 国产视频一二三区 | av一区二区在线播放 | 国产成人影视 | 日韩精品视频在线播放 | 欧美在线视频网 | 婷婷在线观看视频 | 欧美一区二区三区视频 | 97视频免费在线观看 | 日韩免费视频 | 欧美国产视频一区 | 日日爽 | 国产亚洲精品精品国产亚洲综合 | www.夜夜骑 | 精久久 | 91精品久久久久久久久久入口 | 日本久久久久 | 成人午夜激情 | 国产91久久久 | 亚洲综合在线播放 | 在线观看av片 | 午夜国产一区 | a成人| 久久9999久久 | 国产羞羞视频 | 国产精品大片在线观看 | 午夜激情视频免费 | 亚洲视频一 | 久久99精品视频 | 欧美亚洲日本 | 中文字幕免费在线 | 日韩欧美在线综合 | av女人的天堂 | 久久精品综合 | 欧洲另类交| 国产成人在线一区二区 | 北条麻妃一区二区在线 | 亚洲一区中文字幕在线观看 | 欧美精品一区二区三区四区 | 在线色网| 亚洲小视频| 亚洲成人精品 | gogo熟少妇大胆尺寸 | 久久国产婷婷国产香蕉 | 色视频久久 | 国产精品久久久久久久久久妇女 | 国产视频自拍一区 | 久久亚洲一区 | 黄色免费高清视频 | 欧美激情 | 91午夜视频 | 亚洲精品乱码久久久久久9色 | 在线欧美日韩 | 亚洲精品视频在线 | 九九综合久久 | av黄色一级片 | 亚洲精品久久久久久久久久久 | 日本女人高潮视频 | 日韩一级免费在线观看 | 精品不卡| av中文字幕在线 | 成人在线国产 | av午夜电影| 欧美亚洲国产一区 | 99久久99热这里只有精品 | 男人的天堂亚洲 | 视频一区 中文字幕 | 国产精品一区二区无线 | 久久久精品网站 | 欧美一级在线免费观看 | 一级免费片 | 久99视频| 亚洲一区二区三区国产 | 午夜免费福利电影 | 成人在线小视频 | 成人久久久 | 国产精品综合一区二区 | 激情一区 | 蜜月久综合久久综合国产 | 国产成人涩涩涩视频在线观看 | 亚洲自啪 | 亚洲国产一区二区三区 | 午夜三级在线 | 国产一区999| 亚洲九九九| 毛片免费在线观看 | 北条麻妃99精品青青久久 | 男人的天堂在线视频 | 国产精品久久久久久久久久妞妞 | 在线观看亚洲视频 | 精品二区 | 欧洲一区二区三区 | 欧美一区二区三区精品 | 午夜色视频在线观看 | 韩日视频在线观看 | 亚洲成人精品 | 久久精品国产一区二区三 | 中文字幕第80页 | 久久久一区二区 | 激情综合久久 | 日韩欧美在线不卡 | 真人一级毛片 | www.亚洲精品 | 午夜家庭影院 | 亚洲欧美国产一区二区 | 天天干在线影院 | 中文字幕av亚洲精品一部二部 | 91九色视频在线 | 亚洲第一天堂 | 亚洲成人中文字幕 | 日韩一区二区三区在线 | 日韩精品一区二区三区在线观看 | 81精品国产乱码久久久久久 | 国产大毛片 | 亚洲色图p | 美女久久 | 亚洲啊v在线 | av资源首页 | www中文字幕 | 久久久国色 | 国产视频一区二区在线 | 99精品视频免费在线观看 | 成人免费小视频 | 久久久国产精品免费 | 亚洲黄页 | 精品av| 久久精品网| 在线看国产| 亚洲综合电影 | 亚洲成a人v欧美综合天堂麻豆 | www.久久精品 | 一级欧美一级日韩 | 欧美啊v| 精品视频网站 | 精品九九 | 国产一区在线看 | 一区二区亚洲 | 亚洲精品视频在线观看网站 | 国产成人精品一区二区三区视频 | 美女黄网 | 四虎影院网站 | 亚洲精选一区 | 91精品国产综合久久久蜜臀粉嫩 | 精品日本久久 | 久久久久久亚洲精品 | 在线一区二区三区视频 | av网址在线播放 | 欧美成人第一页 | 国产成人福利在线观看 | 成人在线观看h | 99免费精品 | 亚洲电影二区 | 成人精品一区 | 亚洲精品久久久久一区二区三区 | 91观看| 欧美日日 | 欧美激情一区二区三级高清视频 | 日韩在线观看网站 | 久久国产精品一区 | 日韩在线小视频 | 日韩三区| 81精品国产乱码久久久久久 | 中文字幕在线三区 | 久久精品视频亚洲 | 欧美区在线 | 国产精选一区二区 | 亚洲a网站 | 免费a视频 | 日韩精品一区二区三区四区视频 | 春色导航 | 国产欧美精品一区二区色综合朱莉 | 久久久久久九九 | 久久精品国产99国产精品 | 成年免费视频 | 国产美女av在线 | 亚洲cb精品一区二区三区 | 欧美日韩电影一区二区三区 | 亚洲精品成人 | 日韩视频在线观看一区 | 中文字幕在线三区 | 中文字幕一区二区三区四区 | www.久久视频 | 国产片av| 午夜精品久久久久久久99黑人 | 久久精品欧美 | 黄在线看v| 日韩午夜电影在线观看 | 免费av电影网站 | 人人人艹| 男人天堂视频在线观看 |