记一次使用修改字节码的方法解决java.lang.NoSuchMethodError
接兔兔国际sdk ane
充值界面选择兔币充值就会闪退,
观察logcat ,NoSuchMethodError: com.tutu.common.a.b.getContext 原来是因为没有方法找不到
04-19 10:10:54.224: E/AndroidRuntime(20315): FATAL EXCEPTION: main
04-19 10:10:54.224: E/AndroidRuntime(20315): Process: com.tutusdk.global.demo, PID: 20315
04-19 10:10:54.224: E/AndroidRuntime(20315): java.lang.NoSuchMethodError: com.tutu.common.a.b.getContext
04-19 10:10:54.224: E/AndroidRuntime(20315): at com.tutu.common.a.b.a(TutuAlertDialog.java:78)
04-19 10:10:54.224: E/AndroidRuntime(20315): at com.tutu.common.a.b.onCreateView(TutuAlertDialog.java:66)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.Fragment.performCreateView(Fragment.java:1786)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:947)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1126)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:739)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1489)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.support.v4.app.FragmentManagerImpl$1.run(FragmentManager.java:454)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.os.Handler.handleCallback(Handler.java:733)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.os.Handler.dispatchMessage(Handler.java:95)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.os.Looper.loop(Looper.java:136)
04-19 10:10:54.224: E/AndroidRuntime(20315): at android.app.ActivityThread.main(ActivityThread.java:5113)
04-19 10:10:54.224: E/AndroidRuntime(20315): at java.lang.reflect.Method.invokeNative(Native Method)
04-19 10:10:54.224: E/AndroidRuntime(20315): at java.lang.reflect.Method.invoke(Method.java:515)
04-19 10:10:54.224: E/AndroidRuntime(20315): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
04-19 10:10:54.224: E/AndroidRuntime(20315): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:609)
04-19 10:10:54.224: E/AndroidRuntime(20315): at dalvik.system.NativeStart.main(Native Method)
可以用附件中的jar包练习,下载的是zip 解压之后就有jar包。用 jd-gui 反编译找到b class, 介绍一个更好用的反编译工具:apktoolbox,下载:http://www.52pojie.cn/thread-429318-1-1.html
package com.tutu.common.a; import android.graphics.drawable.ColorDrawable;
import android.os.Bundle;
import android.support.v4.app.DialogFragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.TextView;
import com.feng.android.i.e;
import com.feng.android.i.f; /* compiled from: TutuAlertDialog */
public class b extends DialogFragment {
private TextView a;
private Button b;
private Button c;
private String d;
private String e;
private String f;
private String g;
private a h; /* compiled from: TutuAlertDialog */
public interface a {
void c(String str); void c_(String str);
} public void onCreate(Bundle bundle) {
super.onCreate(bundle);
} public static b a(String str, String str2, String str3, String str4, a aVar) {
b bVar = new b();
Bundle bundle = new Bundle();
bundle.putString("left_text", str);
bundle.putString("right_text", str2);
bundle.putString("dialog_tips", str3);
bundle.putString("dialog_tag", str4);
bVar.setArguments(bundle);
bVar.a(aVar);
return bVar;
} public View onCreateView(LayoutInflater layoutInflater, ViewGroup viewGroup, Bundle bundle) {
getDialog().requestWindowFeature(1);
getDialog().getWindow().setBackgroundDrawable(new ColorDrawable(0));
View inflate = layoutInflater.inflate(e.c(getActivity(), "tutu_alert_dialog_layout"), viewGroup, false);
getDialog().getWindow().setGravity(17);
if (getArguments() != null) {
this.d = getArguments().getString("left_text");
this.e = getArguments().getString("right_text");
this.f = getArguments().getString("dialog_tips");
this.g = getArguments().getString("dialog_tag");
}
a(inflate);
setCancelable(false);
return inflate;
} public void a(a aVar) {
this.h = aVar;
} private void a(View view) {
this.b = (Button) view.findViewById(e.a(getActivity(), "tutu_alert_dialog_ignore_btn"));
this.c = (Button) view.findViewById(e.a(getActivity(), "tutu_alert_dialog_recharge_btn"));
this.a = (TextView) view.findViewById(e.a(getContext(), "tutu_alert_dialog_tips"));
if (!f.c(this.d)) {
this.b.setText(this.d);
}
if (!f.c(this.e)) {
this.c.setText(this.e);
}
if (!f.c(this.f)) {
this.a.setText(this.f);
}
this.b.setOnClickListener(new OnClickListener(this) {
final /* synthetic */ b a; {
this.a = r1;
} public void onClick(View view) {
if (this.a.h != null) {
this.a.h.c_(this.a.g);
}
this.a.dismiss();
}
});
this.c.setOnClickListener(new OnClickListener(this) {
final /* synthetic */ b a; {
this.a = r1;
} public void onClick(View view) {
if (this.a.h != null) {
this.a.h.c(this.a.g);
}
this.a.dismiss();
}
});
}
}
问题就出在标红的位置,通过这个
http://stackoverflow.com/questions/36116606/nosuchmethoderrorcom-android-app-fragment-getcontext-in-android
知道, 新版的sdk才支持getContext方法,但是adobe air sdk 要跟进比较慢,没有该方法,要改成 getActivity
因为这是第三方sdk , 没有源代码, 只能修改 class文件。
google 一番 如何修改字节码, 找到几篇文章。 放在最后。
下载 jclasslib ,安装。打开jclasslib bytecode viewer 用于查看 b.class 文件的字节码
解压jar包, 将 com/tutu/common/a文件夹内的b.class 文件 拖拽进bytecode viewer
根据上面反编译出来的源码,我们要修改的地方在 private void a(View view) 这个方法中,展开左边的Methods, 一个个看图中名称为a的方法, 观察右边的Access flags, 如果不是private的迅速跳过,是的话观察右边Descriptor ,这里面是参数列表。 觉得像的就展开它, 选中 [0]Code, 观察右边的byte code
图中 22行 就是要找的 getContext。 鼠标点击前面的 #56 , 跳的下面这张图
看到使用的 划红线的部分是 #80 再回到 bytecode中 , 看13行 getActivity的地方时 #54, 鼠标点击它, 跳到下面这张图
看到 划红线的部分是 #78 , 滑动左边的列表, 发现这个东西是在Constant Pool 分类下。 好下面编代码修改字节码, 将指向#80 改成指向 #78
在eclipse中新建一个java project。
进入jclasslib 安装目录, 进入 modules/data/src/main/java ,拷贝 org 及其内容到 java project src 目录下
新建一个 App 类,放main方法
import java.io.DataInput;
import java.io.DataInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException; import org.gjt.jclasslib.io.ClassFileWriter;
import org.gjt.jclasslib.structures.CPInfo;
import org.gjt.jclasslib.structures.ClassFile;
import org.gjt.jclasslib.structures.InvalidByteCodeException;
import org.gjt.jclasslib.structures.MethodInfo;
import org.gjt.jclasslib.structures.constants.ConstantMethodrefInfo;
import org.gjt.jclasslib.structures.constants.ConstantNameAndTypeInfo; public class App {
public static void main(String[] args) throws InvalidByteCodeException, IOException {
String filePath = "C:/Users/fp/Documents/goalPlatformClientV1/ANE/tutu/package/Android-ARM/b.class";
FileInputStream fis = new FileInputStream(filePath);
DataInput di = new DataInputStream(fis);
ClassFile cf = new ClassFile();
cf.read(di); CPInfo[] infos = cf.getConstantPool(); MethodInfo[] ms = cf.getMethods();
MethodInfo m = ms[5]; // System.out.println(m.getAccessFlags());
// System.out.println(m.getAccessFlagsVerbose());
// System.out.println(m.getName());
// System.out.println(m.getNameIndex());
// System.out.println(m.getDescriptor());
// System.out.println(m.getDescriptorIndex());
// AttributeInfo[] getAttributes = m.getAttributes();
// System.out.println(m.getAttributes()); ConstantMethodrefInfo uInfo = (ConstantMethodrefInfo) infos[56]; //刚刚那里是CONSTANT_Methodref_info所以这里要用这个
ConstantNameAndTypeInfo nt = uInfo.getNameAndTypeInfo();
String s = String.format("%s\n%s\n%s\n%s", nt.getName(), nt.getTag(), nt.getTagVerbose() , nt.getVerbose());
System.out.println(s);
uInfo.setNameAndTypeIndex(78);
infos[56] = uInfo; // int count = infos.length;
// for (int i = 0; i < count; i++) {
// if (infos[i] != null) {
// System.out.print(i);
// System.out.print(" = ");
// System.out.print(infos[i].getVerbose());
// System.out.print(" = ");
// System.out.println(infos[i].getTagVerbose());
// if (i == 160) {//刚刚找到的是21位置
// ConstantUtf8Info uInfo = (ConstantUtf8Info) infos[i]; //刚刚那里是CONSTANT_Utf-8_info所以这里要用这个
//// System.out.println("xxxxxxxxxxxxxxxxxxxx");
//// uInfo.setString("getActivity"); // 如果用这种方式直接修改string ,不行, Name and Type 后面的type 还是Landroid/content/Context
//// infos[i] = uInfo;
// }
// }
// }
//这种方式也可以,一样的
/* if(infos[count] != null) {
ConstantUtf8Info uInfo = (ConstantUtf8Info) infos[i]; //刚刚那里是CONSTANT_Utf-8_info所以这里要用这个
uInfo.setBytes("baidu".getBytes());
infos[count] = uInfo;
}*/ cf.setConstantPool(infos);
fis.close();
File f = new File(filePath);
ClassFileWriter.writeToFile(f, cf);
}
}
用于练手的附件:
http://files.cnblogs.com/files/lonkiss/java.library.tutu.zip
解压之后有个jar包。 博客园不让直接上传jar包
参考资料:
直接修改别人jar包里面的class文件 工具:jclasslib - hexin373的专栏 - 博客频道 - CSDN.NET http://blog.csdn.net/hexin373/article/details/6669813
如何利用JClassLib修改.class文件 - “羊习习”的专栏 - 博客频道 - CSDN.NET http://blog.csdn.net/betterandroid/article/details/14520667
记一次使用修改字节码的方法解决java.lang.NoSuchMethodError的更多相关文章
- android apk 防止反编译技术第二篇-运行时修改字节码
上一篇我们讲了apk防止反编译技术中的加壳技术,如果有不明白的可以查看我的上一篇博客http://my.oschina.net/u/2323218/blog/393372.接下来我们将介绍另一种防止a ...
- JVM 字节码指令手册 - 查看 Java 字节码
JVM 字节码指令手册 - 查看 Java 字节码 jdk 进行的编译生成的 .class 是 16 进制数据文件,不利于学习分析.通过下命令 javap -c Demo.class > Dem ...
- 记一次部署时报java.lang.NoSuchMethodError:javax.persistence.spi.PersistenceUnitInfo.getValidationMode()Ljavax / persistence / ValidationMode;的解决办法
楼主在部署war包的时候,本地启动不报错,服务器商报如下问题: Error creating bean with name 'entityManagerFactory' defined in clas ...
- 动态修改字节码以替换用反射调用get set方法的形式
1. 起因 在前两天,为了解决websphere和JDK8上部署的应用发起webservice调用(框架用的cxf)时报错的问题,跟了一些代码,最终发现可以通过加上参数-Dcom.sun.xml.bi ...
- Python 文件编译为字节码的方法
一般情况下 python 不需要手动编译字节码.但是如果不想直接 release 源代码给其他人,将文件编译成字节码,可以实现一定程度的信息隐藏. 1) 使用模块 py_compile 编译一个单文件 ...
- 从字节码层面,解析 Java 布尔型的实现原理
最近在系统回顾学习 Java 虚拟机方面的知识,其中想到一个很有意思的问题:布尔型在虚拟机中到底是什么类型? 要想解答这个问题,我们看 JDK 的源码是无法解决源码的,我们必须深入到 class 文件 ...
- 从字节码的角度看Java内部类与外部类的互相访问
Java中non-static内部类为何可以访问外部类的变量?Java中外部类又为何可以访问内部类的private变量?这两个问题困扰过我一段时间,查了一些网上的答案,大多从“闭包”概念入手,理解起来 ...
- 记一次netty版本冲突,报java.lang.NoSuchMethodError: io.netty.util.internal.ObjectUtil.checkPositive的问题
elasticsearch 5.6中使用TransportClient初始化抛异常 在引入elasticsearch5.6的transportclient包中,会引入netty进行通信. <!- ...
- 记一次线上环境的内存溢出(java.lang.OutOfMemoryError)
事故背景 今天客户说风控项目有个别用户查询不到数据不是报错就是一直卡在那里,我就去那个接口看了下. 一看项目日志今天的都几个g了,平常也就几百兆吧,很明显出了问题. 请求接口后使用命令tail -f ...
随机推荐
- Node.js + React + MongoDB 实现 TodoList 单页应用
之前用 Ant Design 开发了一个项目,因此对 React 的特性有了一定的了解,React 使用封装组件的思想,组件各自维护自己的状态和 UI, 组件之间通过 props 传递数据和方法.当状 ...
- STM32F103RC进入串口3接收中断产生HardFault_Hander问题解决!
最近在以前的项目上添加串口3通讯后,程序一进入接收中断后就产生HardFault_Hander.串口3发送数据一切正常,当打开串口3接收功能时,程序就处于HardFault_Hander状态,而导致死 ...
- 我的java学习笔记
最近一直在自学C#和js,想着想把以前学的java学习笔记整理下发上来.
- iOS开发之MapKit
1.概述 MapKit框架使用前提: 导入框架: 导入主头文件: #import <MapKit/MapKit.h> MapKit框架使用须知: MapKit框架中所有数据类型的前缀都是M ...
- win32 htmlayout点击按钮创建新窗口,以及按钮图片样式
最近在做一个C++ win32的桌面图形程序,我不是C++程序员,做这个只是因为最近没什么java的活. windows api,之前接触的时候,还是大学,那时用这个开发打飞机游戏纯粹是娱乐.现在基本 ...
- synchronized关键字
最近重新梳理了下java的synchronized相关内容,希望能帮助到有需要的朋友们. 主要阐述以下几个问题: 1.非static方法前加synchronized class Demo{ synch ...
- 【C语言】浅谈可变参数与printf函数
一.何谓可变参数 int printf( const char* format, ...); 这是使用过C语言的人所再熟悉不过的printf函数原型,它的参数中就有固定参数format和可变参数(用& ...
- 实现标签的添加与删除(tags)
在项目中会遇到,标签(tags)的添加与去除的需求 demo:我们有 tags '专利','商标','版权','域名' demand:在发布内容的时候,要求可以添加tag,(实现tag的增加与删除 ...
- 【Flex】去除外边框,底背景透明,改变exe的icon
一.去除程序外边框 1.在 xx-app.xml文件里,找到 <!-- <systemChrome></systemChrome> --> 这句话,然后删掉注释 ...
- 老李分享:robotium常用API 2
断言: 具体请查看官网 断言方法assert(robotium特有的断言方式,实际项目中和Junit的assert方法配合使用) void assertCurrentActivity (String ...