说明
在之前的Android中Jni编程,跨模块传递string空指针,会导致程序崩溃异常。在VS不同的版本之间调用动态库,其中string作为参数传递,会出现另一种错误。调试过程中,监控string的传递值,会提示:读取该字符串的字符出错
解析
string和STL模板出现异常的基本原理是一样的,传递的内存结构和动态库的内存结构是不一致的,因为无法确保当前的程序和动态库使用的VS版本是一致的,为了避免这个问题,传递char数组是一种非常保险的做法
结论
Android Jni编程中不允许跨模块返回一个空的string类型
例子
当前有两个模块A,B,在不同的库文件中
using namespace A
{
std::string GetString()
{
std::string str = "";
return str;
}
}
using namespace B
{
void GetResult()
{
std::string str = A::GetString();
}
}
一旦调用GetResult,就会发生错误,按照崩溃的信息
signal 11 (SIGSEGV), code 2 (SEGV_ACCERR), fault addr 0xdeadbaad
Abort message: 'invalid address or address of corrupt block 0x358e4 passed to dlfree'
r0 00000000 r1 f7643dec r2 deadbaad r3 00000000
r4 000358e4 r5 f76450f8 r6 aaff3000 r7 e41a28c4
r8 12d080e0 r9 ab00b490 sl 12d46500 fp 12d46500
ip 00000080 sp fffddb48 lr f7614fe7 pc f7614fe8 cpsr 600f0030
d0 0000000000000000 d1 6120726f2073736c
d2 6f20737365726466 d3 707572726f632072
d4 ab1e7e80ab1e7e80 d5 0000000000000000
d6 00740065004e002e d7 006b0072006f0077
d8 0000000000000000 d9 0000000000000000
d10 0000000000000000 d11 0000000000000000
d12 0000000000000000 d13 0000000000000000
d14 0000000000000000 d15 0000000000000000
d16 0000000000000000 d17 0000278c00000000
d18 6ffeabe86ffeabe8 d19 6ffeabe86ffeabe8
d20 6ffeabe86ffeabe8 d21 6ffeabe86ffeabe8
d22 6ffeabe86ffeabe8 d23 6ffeabe86ffeabe8
d24 6ffeabe86ffeabe8 d25 6ffeabe86ffeabe8
d26 6ffeabe86ffeabe8 d27 6ffeabe86ffeabe8
d28 6ffeabe86ffeabe8 d29 6ffeabe86ffeabe8
d30 6ffeabe86ffeabe8 d31 6ffeabe86ffeabe8
scr 60000011
backtrace:
#00 pc 00028fe8 /system/lib/libc.so (dlfree+1239)
#01 pc 0000f2c3 /system/lib/libc.so (free+10)
#02 pc 0000c630 /data/app/cn.test/lib/arm/libnative.so
大致猜测为当GetString退出当前的模块A作用域,就已经释放了string指向的内存空间,当GetResult函数调用结束的时候,重新释放导致了问题的出现。
在这里如果在同一个模块(同一个库文件)使用,就是GetString和GetResult处于同一个模块,就不会有任何的问题。
或者GetString中定义的str不是一个空字符串,也不会有任何的问题。
目前在java层,jstring 为空,传递给jni层出现错误:
signal 11 (SIGSEGV), code 1 (SEGV_MAPERR), fault addr 0xb38800ec
r0 00000002 r1 0000002e r2 ffffffd0 r3 b6d0f2f0
r4 b6d0f2e8 r5 b38b0f6c r6 b4cc2000 r7 b3880000
r8 b4cc9008 r9 b6d02594 sl 00000001 fp 00000000
ip b38800b8 sp be8e53a8 lr b6ce434b pc b6ce13ca cpsr 200f0030
d0 0000000000000000 d1 0000000000000000
d2 000003e800000000 d3 0000000000000014
d4 424831a6528a632c d5 5aea5aea5aea52e9
d6 5aea5aea5aea5aea d7 5aea5aea5aea5aea
d8 0000000000000000 d9 0000000000000000
d10 0000000000000000 d11 0000000000000000
d12 0000000000000000 d13 0000000000000000
d14 0000000000000000 d15 0000000000000000
d16 0000043800000780 d17 0000000100000004
d18 0000004800000002 d19 0000000400000000
d20 0000000000000033 d21 0000000000000000
d22 0000000000000000 d23 4ffd51265a458900
d24 0000000000000001 d25 0001000100050005
d26 8003800380078007 d27 0023002300030003
d28 0000000000000000 d29 0000000000000000
d30 0000000000000000 d31 0000000000000000
scr 80000012
backtrace:
#00 pc 000553ca /system/lib/libc.so (ifree+49)
#01 pc 00058347 /system/lib/libc.so (je_free+374)
#02 pc 0000fe1f /data/app/test/lib/arm/libnative.so
#03 pc 0000ac61 /data/app/test/lib/arm/libnative.so
已经开始进展的活动:在Linux跨模块测试string空串问题,没有重现,说明Linux没有这个问题,而Android采用的C库是GOOGLE编写的,应该是string的源码处理上有问题,但是在同一个模块是不会有这个问题,而是在跨模块才会有这个问题,又将注意力转移到编译器优化上。