Base64编解码Android和ios的例子,补充JNI中的例子

时间:2023-11-10 21:38:44

1.在Android中java层提供了工具类:android.util.Base64;

里面都是静态方法,方便直接使用:

使用方法如下:

  1. // Base64 编码:
  2. byte [] encode = Base64.encode("Hello, World".getBytes(), Base64.DEFAULT);
  3. String enc = new String(encode);
  4. Log.d("","base 64 encode = " + enc);
  5. // Base64 解码:
  6. byte [] result = Base64.decode("SGVsbG8sIFdvcmxk", Base64.DEFAULT);
  7. String res = new String(result);
  8. Log.d("", "base 64 result = " + res);

例子演示了将"Hello, World"编码成"SGVsbG8sIFdvcmxk",然后又解码回来。简单易懂。

2.对于ios来说,有google的提供的一个工具箱来解决。

网址:http://code.google.com/p/google-toolbox-for-mac/

需要从里面找出3个文件:GTMBase64.h,GTMBase64.m,GTMDefines.h

将这三个文件加入ios工程中即可使用了。

例如:

使用:

NSLog(@"%@", [selfencodeBase64:@"Hello, World"]);

NSLog(@"%@", [selfdecodeBase64:@"SGVsbG8sIFdvcmxk"]);

调用的自己封装的函数:

- (NSString *) encodeBase64:(NSString *) input{

NSData *data = [inputdataUsingEncoding:NSUTF8StringEncodingallowLossyConversion:YES];

data = [GTMBase64 encodeData:data];

NSString *base64String = [[NSStringalloc] initWithData:dataencoding:NSUTF8StringEncoding];

return base64String;

}

- (NSString *) decodeBase64:(NSString *) input{

NSData *data = [inputdataUsingEncoding:NSUTF8StringEncodingallowLossyConversion:YES];

data = [GTMBase64 decodeData:data];

NSString *string = [[NSStringalloc] initWithData:dataencoding:NSUTF8StringEncoding];

return string;

}

3.在Android中,我们也可以将base64的编解码算法放到jni中,这样也是比较方便的。

对应的c中算法如下:

  1. #include "com_example_base64test_JniTest.h"
  2. #include <stdlib.h>
  3. #include <android/log.h> // 这个是输出LOG所用到的函数所在的路径
  4. #define LOG_TAG    "JNILOG" // 这个是自定义的LOG的标识
  5. #undef LOG // 取消默认的LOG
  6. #define LOGD(...)  __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  7. #define LOGI(...)  __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  8. #define LOGW(...)  __android_log_print(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  9. #define LOGE(...)  __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  10. #define LOGF(...)  __android_log_print(ANDROID_LOG_FATAL,LOG_TAG,__VA_ARGS__) // 定义LOG类型
  11. const char base[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  12. char* base64_encode(const char* data, int data_len);
  13. char *base64_decode(const char* data, int data_len);
  14. static char find_pos(char ch);
  15. /*
  16. * Class:     com_example_base64test_JniTest
  17. * Method:    encode
  18. * Signature: (Ljava/lang/String;)Ljava/lang/String;
  19. */
  20. JNIEXPORT jstring JNICALL Java_com_example_base64test_JniTest_encode
  21. (JNIEnv *env, jobject obj, jstring string)
  22. {
  23. // 先将jstring转换成char*
  24. char *t = 0;
  25. jclass clsstring = env->FindClass("java/lang/String");
  26. jstring strencode = env->NewStringUTF("utf-8");
  27. jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B");
  28. jbyteArray barr= (jbyteArray)env->CallObjectMethod(string, mid, strencode);
  29. jsize alen = env->GetArrayLength(barr);
  30. jbyte* ba = env->GetByteArrayElements(barr, JNI_FALSE);
  31. if (alen > 0)
  32. {
  33. t = (char*)malloc(alen + 1);
  34. memcpy(t, ba, alen);
  35. t[alen] = 0;
  36. }
  37. env->ReleaseByteArrayElements(barr, ba, 0);
  38. // 此时的t里面有了jstring的内容
  39. int i = 0;
  40. int j = strlen(t);
  41. char *enc = base64_encode(t, j);
  42. int len = strlen(enc);
  43. char *dec = base64_decode(enc, len);
  44. LOGD("\noriginal: %s\n", t);
  45. LOGD("\nencoded : %s\n", enc);
  46. LOGD("\ndecoded : %s\n", dec);
  47. free(enc);
  48. free(dec);
  49. // 将base64编码后的char转换成jstring返回给java层
  50. //    jclass strClass = env->FindClass("Ljava/lang/String;");
  51. jclass strClass = env->FindClass("java/lang/String");
  52. jmethodID ctorID = env->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");
  53. jbyteArray bytes = env->NewByteArray(strlen(enc));
  54. env->SetByteArrayRegion(bytes, 0, strlen(enc), (jbyte*)enc);
  55. jstring encoding = env->NewStringUTF("UTF-8");
  56. //    jchar encoding_name[] = { 'U', 'T', 'F', '-', '8'};
  57. //    jstring encoding = env->NewString(encoding_name, 5);
  58. return (jstring)env->NewObject(strClass, ctorID, bytes, encoding);
  59. //    jbyte buffer[] = /* UTF8 encoding buffer */
  60. //
  61. //    jbyteArray bytes = env->NewByteArray(sizeof(buffer));
  62. //
  63. //    env->SetByteArrayRegion(bytes, 0, sizeof(buffer), buffer);
  64. //    return bytes;
  65. }
  66. /*
  67. * Class:     com_example_base64test_JniTest
  68. * Method:    decode
  69. * Signature: (Ljava/lang/String;)Ljava/lang/String;
  70. */
  71. JNIEXPORT jstring JNICALL Java_com_example_base64test_JniTest_decode
  72. (JNIEnv *env, jobject obj, jstring base)
  73. {
  74. // 先将jstring转换成char*
  75. char *t = 0;
  76. jclass clsstring = env->FindClass("java/lang/String");
  77. jstring strencode = env->NewStringUTF("utf-8");
  78. jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B");
  79. jbyteArray barr= (jbyteArray)env->CallObjectMethod(base, mid, strencode);
  80. jsize alen = env->GetArrayLength(barr);
  81. jbyte* ba = env->GetByteArrayElements(barr, JNI_FALSE);
  82. if (alen > 0)
  83. {
  84. t = (char*)malloc(alen + 1);
  85. memcpy(t, ba, alen);
  86. t[alen] = 0;
  87. }
  88. env->ReleaseByteArrayElements(barr, ba, 0);
  89. // 此时的t里面有了jstring的内容
  90. int i = 0;
  91. int j = strlen(t);
  92. //    char *enc = base64_encode(t, j);
  93. //    int len = strlen(enc);
  94. char *dec = base64_decode(t, j);
  95. LOGD("\noriginal: %s\n", t);
  96. //    LOGD("\nencoded : %s\n", enc);
  97. LOGD("\ndecoded : %s\n", dec);
  98. //    free(enc);
  99. free(dec);
  100. // 将base64编码后的char转换成jstring返回给java层
  101. //    jclass strClass = env->FindClass("Ljava/lang/String;");
  102. jclass strClass = env->FindClass("java/lang/String");
  103. jmethodID ctorID = env->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");
  104. jbyteArray bytes = env->NewByteArray(strlen(dec));
  105. env->SetByteArrayRegion(bytes, 0, strlen(dec), (jbyte*)dec);
  106. jstring encoding = env->NewStringUTF("utf-8");
  107. jobject result = env->NewObject(strClass, ctorID, bytes, encoding);
  108. return (jstring)env->NewObject(strClass, ctorID, bytes, encoding);
  109. //    return result;
  110. //    return bytes;
  111. }
  112. /* */
  113. char *base64_encode(const char* data, int data_len)
  114. {
  115. //int data_len = strlen(data);
  116. int prepare = 0;
  117. int ret_len;
  118. int temp = 0;
  119. char *ret = NULL;
  120. char *f = NULL;
  121. int tmp = 0;
  122. char changed[4];
  123. int i = 0;
  124. ret_len = data_len / 3;
  125. temp = data_len % 3;
  126. if (temp > 0)
  127. {
  128. ret_len += 1;
  129. }
  130. ret_len = ret_len*4 + 1;
  131. ret = (char *)malloc(ret_len);
  132. if ( ret == NULL)
  133. {
  134. LOGD("No enough memory.\n");
  135. exit(0);
  136. }
  137. memset(ret, 0, ret_len);
  138. f = ret;
  139. while (tmp < data_len)
  140. {
  141. temp = 0;
  142. prepare = 0;
  143. memset(changed, '\0', 4);
  144. while (temp < 3)
  145. {
  146. //printf("tmp = %d\n", tmp);
  147. if (tmp >= data_len)
  148. {
  149. break;
  150. }
  151. prepare = ((prepare << 8) | (data[tmp] & 0xFF));
  152. tmp++;
  153. temp++;
  154. }
  155. prepare = (prepare<<((3-temp)*8));
  156. //printf("before for : temp = %d, prepare = %d\n", temp, prepare);
  157. for (i = 0; i < 4 ;i++ )
  158. {
  159. if (temp < i)
  160. {
  161. changed[i] = 0x40;
  162. }
  163. else
  164. {
  165. changed[i] = (prepare>>((3-i)*6)) & 0x3F;
  166. }
  167. *f = base[changed[i]];
  168. //printf("%.2X", changed[i]);
  169. f++;
  170. }
  171. }
  172. *f = '\0';
  173. return ret;
  174. }
  175. /* */
  176. static char find_pos(char ch)
  177. {
  178. char *ptr = (char*)strrchr(base, ch);//the last position (the only) in base[]
  179. return (ptr - base);
  180. }
  181. /* */
  182. char *base64_decode(const char *data, int data_len)
  183. {
  184. int ret_len = (data_len / 4) * 3;
  185. int equal_count = 0;
  186. char *ret = NULL;
  187. char *f = NULL;
  188. int tmp = 0;
  189. int temp = 0;
  190. char need[3];
  191. int prepare = 0;
  192. int i = 0;
  193. if (*(data + data_len - 1) == '=')
  194. {
  195. equal_count += 1;
  196. }
  197. if (*(data + data_len - 2) == '=')
  198. {
  199. equal_count += 1;
  200. }
  201. if (*(data + data_len - 3) == '=')
  202. {//seems impossible
  203. equal_count += 1;
  204. }
  205. switch (equal_count)
  206. {
  207. case 0:
  208. ret_len += 4;//3 + 1 [1 for NULL]
  209. break;
  210. case 1:
  211. ret_len += 4;//Ceil((6*3)/8)+1
  212. break;
  213. case 2:
  214. ret_len += 3;//Ceil((6*2)/8)+1
  215. break;
  216. case 3:
  217. ret_len += 2;//Ceil((6*1)/8)+1
  218. break;
  219. }
  220. ret = (char *)malloc(ret_len);
  221. if (ret == NULL)
  222. {
  223. LOGD("No enough memory.\n");
  224. exit(0);
  225. }
  226. memset(ret, 0, ret_len);
  227. f = ret;
  228. while (tmp < (data_len - equal_count))
  229. {
  230. temp = 0;
  231. prepare = 0;
  232. memset(need, 0, 4);
  233. while (temp < 4)
  234. {
  235. if (tmp >= (data_len - equal_count))
  236. {
  237. break;
  238. }
  239. prepare = (prepare << 6) | (find_pos(data[tmp]));
  240. temp++;
  241. tmp++;
  242. }
  243. prepare = prepare << ((4-temp) * 6);
  244. for (i=0; i<3 ;i++ )
  245. {
  246. if (i == temp)
  247. {
  248. break;
  249. }
  250. *f = (char)((prepare>>((2-i)*8)) & 0xFF);
  251. f++;
  252. }
  253. }
  254. *f = '\0';
  255. return ret;
  256. }

不过这个例子里面,log打印的都是正确的,可是返回到java层的确是乱码,这个问题暂时还没有解决。希望有明白的同志告知一下。谢谢。工程附件中。