Raw*_*awa 17 java android decompiling
我创建了一个简单的应用程序,一个计数器应用程序,按下按钮会增加一个整数并更新文本视图.代码如下:
public class MainActivity extends Activity {
public static int count = 0;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final TextView textView = (TextView) findViewById(R.id.count);
textView.setText(Integer.toString(count));
final Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
count++;
textView.setText(Integer.toString(count));
}
});
}
...
}
Run Code Online (Sandbox Code Playgroud)
在使用dex2jar和jd-gui反编译同一个应用程序后,我收到了以下代码:
public class MainActivity extends Activity {
public static int count = 0;
protected void onCreate(final Bundle paramBundle) {
super.onCreate(paramBundle);
setContentView(2130903040);
paramBundle = (TextView)findViewById(2131296257);
paramBundle.setText(Integer.toString(count));
((Button)findViewById(2131296256)).setOnClickListener(new View.OnClickListener() {
public void onClick(View paramAnonymousView) {
MainActivity.count += 1;
paramBundle.setText(Integer.toString(MainActivity.count));
}
});
}
...
}
Run Code Online (Sandbox Code Playgroud)
在以下行:
paramBundle = (TextView)findViewById(2131296257);
paramBundle.setText(Integer.toString(count));
Run Code Online (Sandbox Code Playgroud)
系统如何将textview设置为paramBundle?为什么会发生这种情况?paramBundle是Bundle类型,TextView不是Bundle的子类,根据反编译版本,更多Bundle是最终的.反编译后出了什么问题?来自反编译器的信息是错误的还是为什么我们得到这个结果?
编辑:
# virtual methods
.method protected onCreate(Landroid/os/Bundle;)V
.locals 3
.param p1, "savedInstanceState" # Landroid/os/Bundle;
.prologue
.line 17
invoke-super {p0, p1}, Landroid/app/Activity;->onCreate(Landroid/os/Bundle;)V
.line 18
const/high16 v2, 0x7f030000
invoke-virtual {p0, v2}, Lcom/example/rawa/helloworld/MainActivity;->setContentView(I)V
.line 20
const v2, 0x7f090001
invoke-virtual {p0, v2}, Lcom/example/rawa/helloworld/MainActivity;->findViewById(I)Landroid/view/View;
move-result-object v1
check-cast v1, Landroid/widget/TextView;
.line 21
.local v1, "textView":Landroid/widget/TextView;
sget v2, Lcom/example/rawa/helloworld/MainActivity;->count:I
invoke-static {v2}, Ljava/lang/Integer;->toString(I)Ljava/lang/String;
move-result-object v2
invoke-virtual {v1, v2}, Landroid/widget/TextView;->setText(Ljava/lang/CharSequence;)V
.line 22
const/high16 v2, 0x7f090000
invoke-virtual {p0, v2}, Lcom/example/rawa/helloworld/MainActivity;->findViewById(I)Landroid/view/View;
move-result-object v0
check-cast v0, Landroid/widget/Button;
.line 23
.local v0, "button":Landroid/widget/Button;
new-instance v2, Lcom/example/rawa/helloworld/MainActivity$1;
invoke-direct {v2, p0, v1}, Lcom/example/rawa/helloworld/MainActivity$1;-><init>(Lcom/example/rawa/helloworld/MainActivity;Landroid/widget/TextView;)V
invoke-virtual {v0, v2}, Landroid/widget/Button;->setOnClickListener(Landroid/view/View$OnClickListener;)V
.line 30
return-void
.end method
Run Code Online (Sandbox Code Playgroud)
我绝对不是一个专家,只是一个新手.但我也使用apktool解码了应用程序并收到了上面的smali代码.根据我的理解,已保存的实例(paramBundle)被加载到p1(= v3)并在onCreate中使用,并且在第20行或第21行中没有以任何方式使用它.对我来说,这是针对反编译错误的?请记住,apktool允许再次构建应用程序,因此在反编译时不会丢失任何数据.
原因是局部变量的类型发生了变化,但有些反编译器未能正确处理。
这是onCreate
使用 dex2jar + javap反编译的代码:
protected void onCreate(android.os.Bundle);
Code:
0: aload_0
1: aload_1
2: invokespecial #20 // Method android/app/Activity.onCreate:(Landroid/os/Bundle;)V
5: aload_0
6: ldc #21 // int 2130903040
8: invokevirtual #25 // Method setContentView:(I)V
11: aload_0
12: ldc #26 // int 2131230720
14: invokevirtual #30 // Method findViewById:(I)Landroid/view/View;
17: checkcast #32 // class android/widget/TextView
20: astore_1
21: aload_1
22: getstatic #12 // Field count:I
25: invokestatic #38 // Method java/lang/Integer.toString:(I)Ljava/lang/String;
28: invokevirtual #42 // Method android/widget/TextView.setText:(Ljava/lang/CharSequence;)V
31: aload_0
32: ldc #43 // int 2131230721
34: invokevirtual #30 // Method findViewById:(I)Landroid/view/View;
37: checkcast #45 // class android/widget/Button
40: new #6 // class com/example/test/MainActivity$1
43: dup
44: aload_0
45: aload_1
46: invokespecial #48 // Method com/example/test/MainActivity$1."<init>":(Lcom/example/test/MainActivity;Landroid/widget/TextView;)V
49: invokevirtual #52 // Method android/widget/Button.setOnClickListener:(Landroid/view/View$OnClickListener;)V
52: return
Run Code Online (Sandbox Code Playgroud)
这里aload_0
是MainActivity对象的局部变量,aload_1
也是Bundle对象的局部变量。问题的根源出现在代码 #20 处,它将刚刚检索到的 TextView 对象的引用存储到局部变量 1 ( astore_1
) 中,而该变量之前存储的是 Bundle 对象!
这样做是因为 Bundle 对象不用于该方法的其余部分,因此重用其局部变量而不是新变量会更有效。然而,这也意味着反编译器需要额外努力才能生成正确的 Java 代码。
归档时间: |
|
查看次数: |
906 次 |
最近记录: |