Android jni调试打印char阵列的实例详解

Android jni调试打印char阵列的实例详解

前言:

在android开发中,用jni有时候需要打印某一个字符串的二进制格式输出,比较友好的输出格式是一个四列,八列,十六列的矩阵格式。类似在错误删除野指针时出现如下错误:

pid: 2721, tid: 3005, name: pool-5-thread-5 >>> onxmaps.hunt <<<
signal 11 (SIGSEGV), code 1 (SEGV_MAPERR), fault addr deadbaad
Abort message: 'invalid address or address of corrupt block 0x7e31e028 passed to dlfree'
r0 00000000 r1 4011917a r2 deadbaad r3 4011cd0d
r4 7e31e028 r5 40127190 r6 41b54000 r7 7e31e030
r8 00000003 r9 7ed97bb5 sl 00000001 fp 7ed97bb9
ip 00000001 sp 82a7d9c0 lr 400ea873 pc 400ea874 cpsr 600f0030
d0 2064696c61766e69 d1 2073736572646461
d2 657264646120726f d3 6f6320666f207373
d4 3fd34413509f79fb d5 41568f570e698a86
d6 412e848000000000 d7 00000400fb561fc7
d8 7ff0000000000000 d9 41568c0b304b0668
d10 408f400000000000 d11 0000000000000000
d12 0000000000000000 d13 0000000000000000
d14 0000000000000000 d15 0000000000000000
d16 c07422af5ad9a77f d17 010001ff0d000013
d18 6743a514430fcb23 d19 657fcd52992ddb94
d20 4820450ad34fbe9e d21 a2fe0391c1ee451b
d22 bf5544b8ce928c56 d23 d4404b0a8749e7f1
d24 3fd5555555555555 d25 391377ce858a5d48
d26 bca0000000000000 d27 3940000000000000
d28 3ff0000000000000 d29 bef375cbdb605373
d30 412e848000000000 d31 3fd5555555555563
scr 60000013 

backtrace:
#00 pc 00011874 /system/lib/libc.so (dlfree+1191)
#01 pc 0000dd13 /system/lib/libc.so (free+10)
#02 pc 00082485 /system/lib/libcrypto.so (CRYPTO_free+24)
#03 pc 0002aa85 /system/lib/libssl.so (ssl_parse_serverhello_tlsext+244)
#04 pc 00016bbd /system/lib/libssl.so (ssl3_get_server_hello+904)
#05 pc 000196bf /system/lib/libssl.so (ssl3_connect+642)
#06 pc 00024f55 /system/lib/libssl.so (SSL_do_handshake+72)
#07 pc 0000c67f /system/lib/libjavacrypto.so
#08 pc 00020bcc /system/lib/libdvm.so (dvmPlatformInvoke+112)
#09 pc 00051927 /system/lib/libdvm.so (dvmCallJNIMethod(unsigned int const*, JValue*, Method const*, Thread*)+398)
#10 pc 0002a060 /system/lib/libdvm.so
#11 pc 00031510 /system/lib/libdvm.so (dvmMterpStd(Thread*)+76)
#12 pc 0002eba8 /system/lib/libdvm.so (dvmInterpret(Thread*, Method const*, JValue*)+184)
#13 pc 00063e75 /system/lib/libdvm.so (dvmCallMethodV(Thread*, Method const*, Object*, bool, JValue*, std::__va_list)+336)
#14 pc 00063e99 /system/lib/libdvm.so (dvmCallMethod(Thread*, Method const*, Object*, JValue*, ...)+20)
#15 pc 00058b6b /system/lib/libdvm.so
#16 pc 0000d278 /system/lib/libc.so (__thread_entry+72)
#17 pc 0000d410 /system/lib/libc.so (pthread_create+240) 

code around pc:
400ea854 6a014478 62021e4a f7fdb95a e008fd39
400ea864 4621482a 44784a2a f001447a 4a13f9b3
400ea874 49286014 f8d14479 079a31bc f501d51c
400ea884 e8bd70e0 f02c40f8 4823b895 f7fd4478
400ea894 4822fd0d e7fa4478 42b7688f ae10f43f
400ea8a4 481fe611 e7f24478 4478481e 6888e7ef
400ea8b4 f43f4298 e606aed4 bf00bdf8 deadbaad
400ea8c4 0003cdae 0003cda0 0003cd08 0003283b
400ea8d4 0003cc7c 0003cc6a 0003cbf2 0003cbd0
400ea8e4 0003cb74 0003cb5e 0003caf8 0003cae0
400ea8f4 0003cace 0003ca76 0003ca50 0003c9c6
400ea904 0003c970 0003c956 0003c938 0002e90c
400ea914 0003249d 0003c914 00032479 00032471
400ea924 00032461 0003245b 460db538 b1704601
400ea934 0200ea45 f405fb00 04030c10 4620b143
400ea944 ef24f028 bf1842a8 34fff04f 4604e000 

code around lr:
400ea850 482e61a3 6a014478 62021e4a f7fdb95a
400ea860 e008fd39 4621482a 44784a2a f001447a
400ea870 4a13f9b3 49286014 f8d14479 079a31bc
400ea880 f501d51c e8bd70e0 f02c40f8 4823b895
400ea890 f7fd4478 4822fd0d e7fa4478 42b7688f
400ea8a0 ae10f43f 481fe611 e7f24478 4478481e
400ea8b0 6888e7ef f43f4298 e606aed4 bf00bdf8
400ea8c0 deadbaad 0003cdae 0003cda0 0003cd08
400ea8d0 0003283b 0003cc7c 0003cc6a 0003cbf2
400ea8e0 0003cbd0 0003cb74 0003cb5e 0003caf8
400ea8f0 0003cae0 0003cace 0003ca76 0003ca50
400ea900 0003c9c6 0003c970 0003c956 0003c938
400ea910 0002e90c 0003249d 0003c914 00032479
400ea920 00032471 00032461 0003245b 460db538
400ea930 b1704601 0200ea45 f405fb00 04030c10
400ea940 4620b143 ef24f028 bf1842a8 34fff04f

谷歌的工程师非常老道,在code around pc以下就是一个五列矩阵,这种打印格式的可读性比较强。最近项目中需要使用加密算法,因此调试时打印矩阵是一种不错的选择。由于android jni提供的接口时

__android_log_write

每次打印都会一行,不会像printf方便。因此需要对__android_log_write进行二次封装。思路就是先申请一段空间,然后把打印的内容存储在该内存中,最后log输出。

具体代码如下:

#include <android/log.h>
#include <cstring>
#include <cstdlib> 

// 一般定义在公共文件
#define ldebug(tag, format, ...) {__android_log_write(tag, format, ##__VA_ARGS__);}
#define TAG "345" 

void print_matrix(char *text, size_t size) { // 打印16列的矩阵
  char temp[16] = {0};
  size_t lines = (size + 15) / 16; // 保证打印的整行的矩阵
  lines = lines > 0 ? lines : 1; // 最小为一
  const size_t LEN = lines * 16 * 3 + 1; // 给打印buf申请足够的buffer。 乘3是因为打印时传入的每个字符char字符占三个位置。见注释AB
  char *buf = (char*)malloc(LEN * sizeof(char));
  if (NULL == buf) {
    return;
  }
  memset(buf, 0, LEN);
  int n = 0;
  for (size_t i = 0; i < lines * 16; i++) {
    if (16 == n) {
      strcat(buf, "\n");//注释A:占一个字符
      n = 0;
    }
    if (n > 0 && i > 0) {
      strcat(buf, " ");//注释A:占一个字符
    }
    memset(temp, 0, 16);
    if (i < size) { // 在text字符串内则打印字符串内容,超过text长度则打印00
      snprintf(temp, 16, "%02x", *(text + i));//注释B:占两个字符
    } else {
      snprintf(temp, 16, "%02x", 0);//注释B:占两个字符
    }
    strcat(buf, temp);
    ++n;
  }
  *(buf + LEN - 1) = '\0'; // 注意字符串结束
  ldebug(TAG, "%s", buf);
  free(buf);
  buf = NULL;
}

测试调用代码

void testPrintMatrix() {
  char temp[] = "Hello, this is print_matrix's test case.";
  print_matrix(temp, sizeof(temp));
}

输出结果

08-03 18:46:03.101 D/345 (30611): testPrintMatrix
08-03 18:46:03.101 D/345 (30611): 48 65 6c 6c 6f 2c 20 74 68 69 73 20 69 73 20 70
08-03 18:46:03.101 D/345 (30611): 72 69 6e 74 5f 6d 61 74 72 69 78 27 73 20 74 65
08-03 18:46:03.101 D/345 (30611): 73 74 20 63 61 73 65 2e 00 00 00 00 00 00 00 00

说明,因为使用<android/log.h>记得在Android.mk添加

LOCAL_LDLIBS := -L$(SYSROOT)/usr/lib -llog

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android在JNI中使用ByteBuffer的方法

    本文实例讲述了Android在JNI中使用ByteBuffer的方法.分享给大家供大家参考.具体如下: 一.ByteBuffer 定义 在NIO中,数据的读写操作始终是与缓冲区相关联的(读取时信道(SocketChannel)将数据读入缓冲区,写入时首先要将发送的数据按顺序填入缓冲区) 缓冲区是定长的,基本上它只是一个列表,它的所有元素都是基本数据类型.ByteBuffer是最常用的缓冲区,它提供了读写其他数据类型的方法,且信道的读写方法只接收ByteBuffer. ByteBuffer有以下几

  • 从源码编译Android系统的Java类库和JNI动态库的方法

    利用源码编译Android系统Java类库 1.编写Java项目和Android.mk文件 ├── Android.mk └── src └── com └── lhw └── framework └── led └── Led.java Led.java文件 package com.lhw.framework.led; /** * LED操作库 * @author Micky Liu */ public class Led { public boolean turnOn() { return

  • Android JNI c/c++调用java的实例

    Android JNI c/c++调用java的实例 近期通过研究SDL源码 得出Android JNI  c/c++调用Java 无需新建虚拟机,这样省去很多步骤,这样调用Android JNI会变得非常容易,大家看下简单实例: 具体步骤如下 第一步获得:两个参数 JNIEnv和jclass void Java_com_Test_Audio_Init( JNIEnv* env,jclass cls, jobject thiz ) { InitJNI(env,cls); } bool InitJ

  • Android通过JNI实现守护进程

    开发一个需要常住后台的App其实是一件非常头疼的事情,不仅要应对国内各大厂商的ROM,还需要应对各类的安全管家...虽然不断的研究各式各样的方法,但是效果并不好,比如任务管理器把App干掉,服务就起不来了... 网上搜寻一番后,主要的方法有以下几种方法,但都是治标不治本: 1.提高Service的优先级:这个,也只能说在系统内存不足需要回收资源的时候,优先级较高,不容易被回收,然并卵... 2.提高Service所在进程的优先级:效果不是很明显 3.在onDestroy方法里重启service:

  • Ubuntu中为Android HAL编写JNI方法提供JAVA访问硬件服务接口

    在上两篇文章中,我们介绍了如何为Android系统的硬件编写驱动程序,包括如何在Linux内核空间实现内核驱动程序和在用户空间实现硬件抽象层接口.实现这两者的目的是为了向更上一层提供硬件访问接口,即为Android的Application Frameworks层提供硬件服务.我们知道,Android系统的应用程序是用Java语言编写的,而硬件驱动程序是用C语言来实现的,那么,Java接口如何去访问C接口呢?众所周知,Java提供了JNI方法调用,同样,在Android系统中,Java应用程序通过

  • 浅谈Android Studio JNI生成so库

    1.新建Android studio工程 2.新建class:AppKey.java.主要为了保存密钥 代码块 package com...adminapp.lib.utils.jni; /** * Created by seven on 16/9/8. */ public class AppKey { static { System.loadLibrary("AppKey"); } public static native String WechatId(); public stat

  • Android Studio中导入JNI生成的.so库的实现方法

    Android Studio中导入JNI生成的.so库的实现方法 由于在原来的ADT的Eclipse环境中,用ndk_build工具生成了相应的各个.so库文件之后,eclipse工具就会自动把这些库导入到apk中.而Android Studio目前为止(0.86版本)还无法做到那么自动,但是我们可以通过以下方式进行. 首先在Android Studio工程的app目录下创建整个jni目录,jni目录里写Android.mk.Application.mk以及各类C/C++和汇编源文件.然后跟原来

  • Android jni调试打印char阵列的实例详解

    Android jni调试打印char阵列的实例详解 前言: 在android开发中,用jni有时候需要打印某一个字符串的二进制格式输出,比较友好的输出格式是一个四列,八列,十六列的矩阵格式.类似在错误删除野指针时出现如下错误: pid: 2721, tid: 3005, name: pool-5-thread-5 >>> onxmaps.hunt <<< signal 11 (SIGSEGV), code 1 (SEGV_MAPERR), fault addr dea

  • Android自定义View中attrs.xml的实例详解

    Android自定义View中attrs.xml的实例详解 我们在自定义View的时候通常需要先完成attrs.xml文件 在values中定义一个attrs.xml 然后添加相关属性 这一篇先详细介绍一下attrs.xml的属性. <?xml version="1.0" encoding="utf-8"?> <resources> //自定义属性名,定义公共属性 <attr name="titleText" for

  • Android ListView中动态添加RaidoButton的实例详解

    Android ListView中动态添加RaidoButton的实例详解 这里讲解的内容是:从数据库中取得数据,将这些数据的value值赋值给Radiobutton的text属性,将这些数据的key值赋值给radiobutton的key值.同时实现点击一整行,更换radiobutton选择. XML代码:主要是添加一个ListView控件 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android&q

  • Android 监听软键盘状态的实例详解

    Android 监听软键盘状态的实例详解 近日遇到要检测软键盘是否显示或隐藏的问题,搜了一下网上,最后找到一个很简单的,记录一下. activityRoot是activity的根view,就是xml里面的第一个view,给它设置一个id. final View activityRootView = findViewById(R.id.activityRoot); activityRootView.getViewTreeObserver().addOnGlobalLayoutListener(ne

  • Android实现定时器的五种方法实例详解

    一.Timer Timer是Android直接启动定时器的类,TimerTask是一个子线程,方便处理一些比较复杂耗时的功能逻辑,经常与handler结合使用. 跟handler自身实现的定时器相比,Timer可以做一些复杂的处理,例如,需要对有大量对象的list进行排序,在TimerTask中执行不会阻塞子线程,常常与handler结合使用,在处理完复杂耗时的操作后,通过handler来更新UI界面. timer.schedule(task, delay,period); task: Time

  • Android 动态注册监听网络变化实例详解

    Android 动态注册监听网络变化实例详解 新建一个BroadcastTest项目,然后修改MainActivity中的代码,如下: public class MainActivity extends AppCompatActivity { private IntentFilter intentFilter; private NetworkChangeReceiver networkChangeReceiver; @Override protected void onCreate(Bundle

  • Android顶部(toolbar)搜索框实现的实例详解

    Android顶部(toolbar)搜索框实现的实例详解 本文介绍两种SearchView的使用情况,一种是输入框和搜索结果不在一个activity中,另一种是在一个activity中. 首先编写toolbar的布局文件 toolbar中图标在menu文件下定义一个布局文件实现 示例代码: <menu xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.

  • Android 中ViewPager重排序与更新实例详解

    Android 中ViewPager重排序与更新实例详解 最近的项目中有栏目订阅功能,在更改栏目顺序以后需要更新ViewPager.类似于网易新闻的频道管理. 在重新排序之后调用了PagerAdapter的notifyDataSetChanged方法,发现ViewPager并没有更新,于是我开始跟踪源码,在调用PagerAdapter的notifyDataSetChanged方法后,会触发Viewpager的dataSetChanged方法. void dataSetChanged() { //

  • Android 判断是否能真正上网的实例详解

    Android 判断是否能真正上网的实例详解 检测网络是否连接 实现代码: /** * 检测网络是否连接 * * @return */ private boolean isNetworkAvailable() { // 得到网络连接信息 ConnectivityManager manager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE); // 去进行判断网络是否连接 if (manager.getA

  • Android 图片存入系统相册更新显示实例详解

    Android 图片存入系统相册更新显示实例详解 在开发android的过程中,我们避免不了可能会涉及到做一个自定义相册或则会去本地创建一个文件夹来存储我们需要的图片.拿相册来说,比如我们创建一个test的文件夹,拍完一张照片后存储到这个指定的test文件夹里,然后在相册里面显示出来,就像微信的效果一样.拍完即可立即显示.但是,在实际开发过程中我们保存完一张图片后并不能立即更新显示出来这个图片,需要我们重启手机才能在系统相册中显示出来. 这里先提供一个插入系统图库的方法: MediaStore.

随机推荐