取消收藏 libcurl在android下的移植、编译与测试以及java接口的封装
curl是利用URL语法在命令行方式下工作的文件传输工具
它支持很多协议:FTP, FTPS, HTTP, HTTPS, GOPHER, TELNET, DICT, FILE 以及 LDAP。
curl同样支持HTTPS认证,HTTP POST方法, HTTP PUT方法, FTP上传, kerberos认证,HTTP上传, 代理服务器, cookies, 用户名/密码认证, 下载文件断点续传,上载文件断点续传,,http代理服务器管道( proxy tunneling), 甚至它还支持IPv6, socks5代理服务器,,通过http代理服务器上传文件到FTP服务器等等,功能十分强大。
移植之前需做的准备工作:
1、直接到网站上下载 curl 源码
2、利用tar在android编译环境下,一般放在 external 目录下
一.移植Curl工具到Android环境步骤
1.修改cURL源码下的mk文件。源码下面的Android.mk文件最后生成的是静态库libcurl.a,做如下修改(编译成动态库)。
LOCAL_PRELINK_MODULE := false LOCAL_MODULE:= libcurl LOCAL_MODULE_TAGS := optional # Copy the licence to a place where Android will find it. # Actually, this doesn't quite work because the build system searches # for NOTICE files before it gets to this point, so it will only be seen # on subsequent builds. ALL_PREBUILT += $(LOCAL_PATH)/NOTICE $(LOCAL_PATH)/NOTICE: $(LOCAL_PATH)/COPYING | $(ACP) $(copy-file-to-target) #include $(BUILD_STATIC_LIBRARY) include $(BUILD_SHARED_LIBRARY) #########################
2.配置编译环境(cd 到Android.mk同一目录,直接在控制台输入下列代码或者把下面代码弄成sh脚本执行)红色部分根据自己源码情况
ANDROID_HOME=/home/zhoulc/android/ && \
NDK_HOME=/home/zhoulc/android/ndk && \
PATH="$ANDROID_HOME/prebuilt/linux-x86/toolchain/arm-eabi-4.4.0/bin:$PATH" \ ./configure --host=arm-linux CC=arm-eabi-gcc --with-random=/dev/urandom \ CPPFLAGS="-I$NDK_HOME/platforms/android-8/arch-arm/usr/include \ -I $ANDROID_HOME/external/curl/include/ \ -I $ANDROID_HOME/external/curl/3rd/include \ -I $ANDROID_HOME/external/curl \ -I $ANDROID_HOME/out/target/product/generic/obj/STATIC_LIBRARIES/libcurl_intermediates \ -I $ANDROID_HOME/dalvik/libnativehelper/include/nativehelper \ -I $ANDROID_HOME/system/core/include \ -I $ANDROID_HOME/hardware/libhardware/include \ -I $ANDROID_HOME/hardware/libhardware_legacy/include \ -I $ANDROID_HOME/hardware/ril/include \ -I $ANDROID_HOME/dalvik/libnativehelper/include \ -I $ANDROID_HOME/frameworks/base/include \ -I $ANDROID_HOME/frameworks/base/opengl/include \ -I $ANDROID_HOME/frameworks/base/native/include \ -I $ANDROID_HOME/external/skia/include \ -I $ANDROID_HOME/out/target/product/generic/obj/include \ -I $ANDROID_HOME/bionic/libc/arch-arm/include \ -I $ANDROID_HOME/bionic/libc/include \ -I $ANDROID_HOME/bionic/libstdc++/include \ -I $ANDROID_HOME/bionic/libc/kernel/common \ -I $ANDROID_HOME/bionic/libc/kernel/arch-arm \ -I $ANDROID_HOME/bionic/libm/include \ -I $ANDROID_HOME/bionic/libm/include/arch/arm \ -I $ANDROID_HOME/bionic/libthread_db/include \ -include $ANDROID_HOME/system/core/include/arch/linux-arm/AndroidConfig.h \ -I $ANDROID_HOME/system/core/include/arch/linux-arm/ \ -D__ARM_ARCH_5__ -D__ARM_ARCH_5T__ -D__ARM_ARCH_5E__ -D__ARM_ARCH_5TE__ -DANDROID -DNDEBUG -DNDEBUG -DHAVE_CONFIG_H" \ CFLAGS="-fno-exceptions -Wno-multichar -msoft-float -fpic -ffunction-sections \ -funwind-tables -fstack-protector -Wa,--noexecstack -Werror=format-security \ -fno-short-enums -march=armv5te -mtune=xscale -Wno-psabi -mthumb-interwork \ -fmessage-length=0 -W -Wall -Wno-unused -Winit-self -Wpointer-arith \ -Werror=return-type -Werror=non-virtual-dtor -Werror=address -Werror=sequence-point \ -g -Wstrict-aliasing=2 -finline-functions -fno-inline-functions-called-once \ -fgcse-after-reload -frerun-cse-after-loop -frename-registers -UDEBUG \ -mthumb -Os -fomit-frame-pointer -fno-strict-aliasing -finline-limit=64 \ -Wpointer-arith -Wwrite-strings -Wunused -Winline -Wnested-externs \ -Wmissing-declarations -Wmissing-prototypes -Wno-long-long -Wfloat-equal \ -Wno-multichar -Wsign-compare -Wno-format-nonliteral -Wendif-labels \ -Wstrict-prototypes -Wdeclaration-after-statement -Wno-system-headers" \ LIBS="-nostdlib -Bdynamic -Wl,-T,$ANDROID_HOME/build/core/armelf.x \ -Wl,-dynamic-linker,/system/bin/linker -Wl,--gc-sections -Wl,-z,nocopyreloc \ -L$ANDROID_HOME/out/target/product/generic/obj/lib -Wl,-z,noexecstack \ -Wl,-rpath-link=$ANDROID_HOME/out/target/product/generic/obj/lib \ -lc -llog -lcutils -lstdc++ \ -Wl,--no-undefined $ANDROID_HOME/prebuilt/linux-x86/toolchain/arm-eabi-4.4.0/lib/gcc/arm-eabi/4.4.0/libgcc.a \ $ANDROID_HOME/out/target/product/generic/obj/lib/crtend_android.o \ -lm $ANDROID_HOME/out/target/product/generic/obj/lib/crtbegin_dynamic.o \ -L$ANDROID_HOME/external/curl/3rd/libs"
3.编译libcurl.so库
cd进入android/external/curl源码目录
mm-》编译生成libcurl.so库
4.编写测试case 以及Android.mk文件并生成可执行文件
新建一个测试案例curl_test.cpp
#include "curl/curl.h" #include <stdio.h>; int main() { CURL *curl; CURLcode res; curl_global_init(CURL_GLOBAL_ALL); curl = curl_easy_init(); if (curl) { curl_easy_setopt(curl, CURLOPT_URL, "http://www.baidu.com/"); res = curl_easy_perform(curl); if (0!=res) { printf("curl error: %d\n", res); } curl_easy_cleanup(curl); } curl_global_cleanup(); return 0; }
在同一目录下写一个Android.mk文件生成curl_test可执行文件
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_C_INCLUDES += \ $(TOP)/external/curl/include/ \ LOCAL_SRC_FILES:= curl_test.cpp # No shared libraries. # No static libraries. LOCAL_SHARED_LIBRARIES := libcurl LOCAL_MODULE_TAGS := optional LOCAL_MODULE := curl_test include $(BUILD_EXECUTABLE)
生成可执行文件:curl_test
4.运行查看测试结果
运行测试case:curl_test
5.(补充)移植libcurl到android4.0,修改两个地方
1)把生成的路径改一下,一般默认为out/target/product/generic下面,我们根据系统不同(根据lunch选择不同,最终生成的路径不一样)改为系统的全局变量,
把$ANDROID_HOME/out/target/product/generic替换成$ANDROID_PRODUCT_OUT。
ANDROID_HOME_CURL=../../ && \ NDK_HOME_CURL=../../prebuilt/ndk && \ PATH="$ANDROID_HOME_CURL/prebuilt/linux-x86/toolchain/arm-eabi-4.4.0/bin:$PATH" \ ./configure --host=arm-linux CC=gcc --with-random=/dev/urandom \ CPPFLAGS="-I$NDK_HOME_CURL/platforms/android-8/arch-arm/usr/include \ -I $ANDROID_HOME_CURL/external/curl/include/ \ -I $ANDROID_HOME_CURL/external/curl/3rd/include \ -I $ANDROID_HOME_CURL/external/curl \ -I $ANDROID_HOME_CURL/out/target/product/generic/obj/STATIC_LIBRARIES/libcurl_intermediates \ -I $ANDROID_HOME_CURL/dalvik/libnativehelper/include/nativehelper \ -I $ANDROID_HOME_CURL/system/core/include \ -I $ANDROID_HOME_CURL/hardware/libhardware/include \ -I $ANDROID_HOME_CURL/hardware/libhardware_legacy/include \ -I $ANDROID_HOME_CURL/hardware/ril/include \ -I $ANDROID_HOME_CURL/dalvik/libnativehelper/include \ -I $ANDROID_HOME_CURL/frameworks/base/include \ -I $ANDROID_HOME_CURL/frameworks/base/opengl/include \ -I $ANDROID_HOME_CURL/frameworks/base/native/include \ -I $ANDROID_HOME_CURL/external/skia/include \ -I $ANDROID_HOME_CURL/out/target/product/generic/obj/include \ -I $ANDROID_HOME_CURL/bionic/libc/arch-arm/include \ -I $ANDROID_HOME_CURL/bionic/libc/include \ -I $ANDROID_HOME_CURL/bionic/libstdc++/include \ -I $ANDROID_HOME_CURL/bionic/libc/kernel/common \ -I $ANDROID_HOME_CURL/bionic/libc/kernel/arch-arm \ -I $ANDROID_HOME_CURL/bionic/libm/include \ -I $ANDROID_HOME_CURL/bionic/libm/include/arch/arm \ -I $ANDROID_HOME_CURL/bionic/libthread_db/include \ -include $ANDROID_HOME_CURL/system/core/include/arch/linux-arm/AndroidConfig.h \ -I $ANDROID_HOME_CURL/system/core/include/arch/linux-arm/ \ -D__ARM_ARCH_5__ -D__ARM_ARCH_5T__ -D__ARM_ARCH_5E__ -D__ARM_ARCH_5TE__ -DANDROID -DNDEBUG -DNDEBUG -DHAVE_CONFIG_H" \ CFLAGS="-fno-exceptions -Wno-multichar -msoft-float -fpic -ffunction-sections \ -funwind-tables -fstack-protector -Wa,--noexecstack -Werror=format-security \ -fno-short-enums -march=armv5te -mtune=xscale -Wno-psabi -mthumb-interwork \ -fmessage-length=0 -W -Wall -Wno-unused -Winit-self -Wpointer-arith \ -Werror=return-type -Werror=non-virtual-dtor -Werror=address -Werror=sequence-point \ -g -Wstrict-aliasing=2 -finline-functions -fno-inline-functions-called-once \ -fgcse-after-reload -frerun-cse-after-loop -frename-registers -UDEBUG \ -mthumb -Os -fomit-frame-pointer -fno-strict-aliasing -finline-limit=64 \ -Wpointer-arith -Wwrite-strings -Wunused -Winline -Wnested-externs \ -Wmissing-declarations -Wmissing-prototypes -Wno-long-long -Wfloat-equal \ -Wno-multichar -Wsign-compare -Wno-format-nonliteral -Wendif-labels \ -Wstrict-prototypes -Wdeclaration-after-statement -Wno-system-headers" \ LIBS="-nostdlib -Bdynamic -Wl,-T,$ANDROID_HOME_CURL/build/core/armelf.x \ -Wl,-dynamic-linker,/system/bin/linker -Wl,--gc-sections -Wl,-z,nocopyreloc \ -L$ANDROID_PRODUCT_OUT/obj/lib -Wl,-z,noexecstack \ -Wl,-rpath-link=$ANDROID_PRODUCT_OUT/obj/lib \ -lc -llog -lcutils -lstdc++ \ -Wl,--no-undefined $ANDROID_HOME_CURL/prebuilt/linux-x86/toolchain/arm-eabi-4.4.0/lib/gcc/arm-eabi/4.4.0/libgcc.a \ $ANDROID_PRODUCT_OUT/obj/lib/crtend_android.o \ -lm $ANDROID_PRODUCT_OUT/obj/lib/crtbegin_dynamic.o \ -L$ANDROID_HOME_CURL/external/curl/3rd/libs"
2)修改Android.mk
#ALL_PREBUILT += $(LOCAL_PATH)/NOTICE
#$(LOCAL_PATH)/NOTICE: $(LOCAL_PATH)/COPYING | $(ACP)
# $(copy-file-to-target)
把关于ALL_PREBUILT模块全部注释调
二.编译过程中出现的错误以及修改方法
错误1:考入源码,修改android.mk改成生成.so动态库(默认curl.a静态库)
============================================
PLATFORM_VERSION_CODENAME=REL
PLATFORM_VERSION=2.3.1
TARGET_PRODUCT=generic
TARGET_BUILD_VARIANT=eng
TARGET_SIMULATOR=
TARGET_BUILD_TYPE=release
TARGET_BUILD_APPS=
TARGET_ARCH=arm
HOST_ARCH=x86
HOST_OS=linux
HOST_BUILD_TYPE=release
BUILD_ID=GRH78
============================================
make: Entering directory `/home/zhoulc/android'
build/core/base_rules.mk:151: *** external/curl: LOCAL_BUILT_MODULE and LOCAL_INSTALLED_MODULE must not be defined by component makefiles. Stop.
解决方法:在android.mk里面把生成的静态库命令注释掉。
错误2:提示没有头文件
============================================
PLATFORM_VERSION_CODENAME=REL
PLATFORM_VERSION=2.3.1
TARGET_PRODUCT=generic
TARGET_BUILD_VARIANT=eng
TARGET_SIMULATOR=false
TARGET_BUILD_TYPE=release
TARGET_BUILD_APPS=
TARGET_ARCH=arm
HOST_ARCH=x86
HOST_OS=linux
HOST_BUILD_TYPE=release
BUILD_ID=GRH78
============================================
make: Entering directory `/home/zhoulc/android'
make: *** No rule to make target `external/curl/include/curl/curlbuild.h', needed by `out/target/product/generic/obj/include/libcurl/curl/curlbuild.h'. Stop.
make: Leaving directory `/home/zhoulc/android'
解决方法:2)cd 到 external/curl目录下,输入(红色字部分根据自己的环境做相应的更改):
参照执行步骤2
错误3:动态库生成,但出现stop问题,检测一下是不是静态库到动态库的改变,改写彻底没有。
错误4:出现超出 ALL_PREBUILT.定义
build/core/main.mk:537: *** Some files have been added to ALL_PREBUILT.
build/core/main.mk:538: *
build/core/main.mk:539: * ALL_PREBUILT is a deprecated mechanism that
build/core/main.mk:540: * should not be used for new files.
build/core/main.mk:541: * As an alternative, use PRODUCT_COPY_FILES in
build/core/main.mk:542: * the appropriate product definition.
build/core/main.mk:543: * build/target/product/core.mk is the product
build/core/main.mk:544: * definition used in all products.
build/core/main.mk:545: *
build/core/main.mk:546: * unexpected NOTICE in ALL_PREBUILT
build/core/main.mk:546: * unexpected NOTICE in ALL_PREBUILT
build/core/main.mk:547: *
build/core/main.mk:548: *** ALL_PREBUILT contains unexpected files. Stop.
make: Leaving directory `/home/zhoulc/HAndroid_4.0'
解决方法:注释掉相关信息
#ALL_PREBUILT += $(LOCAL_PATH)/NOTICE
#$(LOCAL_PATH)/NOTICE: $(LOCAL_PATH)/COPYING | $(ACP)
# $(copy-file-to-target)
三.java层通过jni调用libcurl API学习摸索经验(CURL API对应java接口封装)
1.jni模块的实现与学习
1)jvm加载c的动态库
应用层的Java类是在虚拟机(VM: Vitual Machine)上执行的,而C件不是在VM上执行,Java程式通过下面方法要求VM去载入(Load)所指定的C组件。
System.loadLibrary(*.so的档案名);
(1)告诉VM此C组件使用那一个JNI版本。如果你的*.so档没有提供JNI_OnLoad()函数,VM会默认该*.so档是使用最老的JNI 1.1版本。由于新版的JNI做了许多扩充,如果需要使用JNI的新版功能,例如JNI 1.4的java.nio.ByteBuffer,就必须藉由JNI_OnLoad()函数来告知VM。
(2)由于VM执行到System.loadLibrary()函数时,就会立即先呼叫JNI_OnLoad(),所以C组件的开发者可以藉由JNI_OnLoad()来进行C组件内的初期值之设定(Initialization) 。
执行过程如下
在jni里面找到JNI_OnLoad方法-》在JNI_OnLoad里面对不同的分支进行注册,比如register_join_easy_natives-》在该方法中把native方法与c方法一一对应registerNativeMethods在该方法中找到java对应的类以及注册对应方法
static int registerNativeMethods(JNIEnv* env, const char* className, JNINativeMethod* gMethods, int numMethods) { jclass clazz; clazz = env->FindClass(className); if (clazz == NULL) { return JNI_FALSE; } if (env->RegisterNatives(clazz, gMethods, numMethods) < 0) { return JNI_FALSE; } return JNI_TRUE; } int register_join_multi_natives(JNIEnv *env) { LOGI("register_join_easy_natives"); return registerNativeMethods(env, g_class_name, g_cls_methods, sizeof(g_cls_methods) / sizeof(g_cls_methods[0])); }
2)jni参数设置
其中比较难以理解的是第二个参数,例如
"()V"
"(II)V"
"(Ljava/lang/String;Ljava/lang/String;)V"
实际上这些字符是与函数的参数类型一一对应的。
"()" 中的字符表示参数,后面的则代表返回值。例如"()V" 就表示void Func();
"(II)V" 表示 void Func(int, int);
具体的每一个字符的对应关系如下
字符 Java类型 C类型
V void void
Z jboolean boolean
I jint int
J jlong long
D jdouble double
F jfloat float
B jbyte byte
C jchar char
S jshort short
数组则以"["开始,用两个字符表示
[I jintArray int[]
[F jfloatArray float[]
[B jbyteArray byte[]
[C jcharArray char[]
[S jshortArray short[]
[D jdoubleArray double[]
[J jlongArray long[]
[Z jbooleanArray boolean[]
上面的都是基本类型。如果Java函数的参数是class,则以"L"开头,以";"结尾,中间是用"/" 隔开的包及类名。而其对应的C函数名的参数则为jobject. 一个例外是String类,其对应的类为jstring
Ljava/lang/String; String jstring
Ljava/net/Socket; Socket jobject
如果JAVA函数位于一个嵌入类,则用$作为类名间的分隔符。
例如 "(Ljava/lang/String;Landroid/os/FileUtils$FileStatus;)Z"
3)在c中访问java的属性和方法
在此,针对andorid中c++与java中的方法互调,引用参考说明如下:
步骤1).andorid CPP调用java函数和访问其成员:原理 => CPP代码找到java的那个class里面的函数的入口地址,然后在CPP代码中调用java代码
步骤1) 用FindClass()函数找到该java类(如android.os.Binder)的实例对象的引用:
jclass clazz = env->FindClass(kBinderPathName) = env->FindClass("android.os.Binder")
步骤2) 用GetFieldID()函数获取到要访问的域(field: 实际上就是该java class中的某个成员变量的名字)的ID:
gBinderOffsets.mObject = env->GetFieldID(clazz, "mObject", "I") // mObject为java class "Binder"里的一个成员变量
-> 注意,这里将要访问的那个java对象的成员mObject的ID保存到了全局变量gBinderOffsets.mObject中,这样做的前提和优点如下:
前提: android里面,每个java进程中只允许有一个java虚拟机(sun公司原始的java架构中,一个进程中可以有多个java虚拟机)
优点: 除了第一次,以后每次要访问该java对象的成员mObject就非常快了(不用再去FindClass()和GetFieldID())
步骤3) 用GetMethodID()函数获取到要访问的方法(Method: 实际上就是该java class中的某个成员函数的名字)的ID:
gBinderOffsets.mExecTransact = env->GetMethodID(clazz, "execTransact", "(IIII)Z") // execTransact为java class "Binder"里的一个成员函数
步骤4) 用类似于GetIntField()的函数获取到该java对象的那个域(即成员)的值:
IBinder* target = (IBinder*)env->GetIntField(obj,gBinderProxyOffsets.mObject)
// 获取java android.os.Binder类型对象里面的成员mObject的值 步骤5) 用类似于CallBooleanMethod()的函数调用到该java对象的那个成员函数:
jboolean res = env->CallBooleanMethod(mObject, gBinderOffsets.mExecTransact, code, (int32_t)&data, (int32_t)reply, flags)
4)数组,字符串在jni中的定义与使用
要创建数组首先要知道类型及长度,JNI提供了一系列的数组类型及操作的函数如:
NewIntArray、NewLongArray、NewShortArray、NewFloatArray、NewDoubleArray、 NewBooleanArray、NewStringUTF、NewCharArray、NewByteArray、NewString,访问通过 GetBooleanArrayElements、GetIntArrayElements等函数。
5)libcurl API到java API转变过程中修改部分
curl_easy_setopt中根据第二个参数不同,第三个参数传递格式不同,在C中该方法第三个参数有四种类型,int、long、回调方法、回调方法参数,想在java里面一一对应,只能使用方法的重载,通过不同的参数值传递,对应上c里面的方法,至于回调函数和回调函数参数,通过在java层传递一个Object对象过去,在JNI层实现对回调函数的注册和回调函数参数的设置。
2.java部分的设计
1.java模块划分
1)curl模块有三个部分
easy_handle:为libcurl的最基础部分,所有的操作都是在easy_handle上进行的,比如发送、请求数据都是在其上进行的。如果直接在easy_handle执行操作 curl_easy_perform 函数是阻塞的(即需要等到完成才返回)
multi_handle:libcurl为异步操作提供的接口,允许调用方在一个线程中处理多个操作(就是easy_handle上的操作,注意是单线程下的),内部multi_handle采用堆栈的方式保存多个easy_handle,然后在一个线程中可以同时对多个easy_handle进行处理,multi_handle的执行操作 curl_multi_perform 函数是立即返回的,不会阻塞
share_handle:有时候多个easy_handle需要分享一些信息,比如cookie,当一个连接获取一个新的cookie,就可以将这个cookie共享到所有的连接上
分别对应java的三个类,Easy类、Multi类、Share类(实体类)
除去个别API,其他接口基本和curl提供API一一对应
2)setopt所使用的第二个参数,对应C里面的模板类型
OptValue.java里面初始化libcurl里面的常量
比如经常用到的CURLOPT_URL等
3)工具类,在进行文件上传下载、断点下载的时候,对文件经行相应操作
上传的时候,需判断需要上传的文件是否存在。
下载的时候,判断是否下载完成,如果没有完成,择可以选择断点下载,重新下载都行。如果选择重新下载,或者经过自动判断是第一次下载,则会删掉以前的文件,重新创建一个用来存放下载数据。
3.规划过程中出现的问题与解决方案
1)网络不通
在AndroidManifest.xml里面添加
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
一个对应能否访问网络,另一个SDcard写权限
2)文件无法下载
首先查看网络是否正常。
然后看下下载流程每一步是不是正常执行返回值一般为0。
Perform是否调用并且调用成功。
在进行回调函数注册之后,注册回调函数参数,看设置是否正确。
看网络访问路径是否正确。
3)文件无法读写
观察文件打开的方式,查看sdcard是否添加可操作权限。
4) 对象从java传到C出现问题。
cls = env->GetObjectClass(object);
在java层经行向上类型转换,传递Object类型到C层,然后通过JNI直接获取自动封装和解析,在java层提供的是公共接口,既所有类的父类Object。对整个模块提供扩展。
5)乱码问题(相当给力)
最开始提供了两个方法,一个从String类型到char*类型的转换,一个从char*到jstring类型的转换,把原始数据进行了编码格式的转换。
//jstring to char* char* jstringTostring(JNIEnv* env, jstring jstr) { char* rtn = NULL; jclass clsstring = env->FindClass("java/lang/String"); jstring strencode = env->NewStringUTF("utf-8"); jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B"); jbyteArray barr = (jbyteArray) env->CallObjectMethod(jstr, mid, strencode); jsize alen = env->GetArrayLength(barr); LOGI("barr = %d", alen); jbyte* ba = env->GetByteArrayElements(barr, JNI_FALSE); if (alen > 0) { rtn = (char*) malloc(alen + 1); memcpy(rtn, ba, alen); rtn[alen] = 0; LOGI("sizeof(rtn) = %d", strlen(rtn)); } env->ReleaseByteArrayElements(barr, ba, 0); return rtn; } //char* to jstring jstring stoJstring(JNIEnv* env, const char* pat) { jclass strClass = env->FindClass("java/lang/String"); jmethodID ctorID = env->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V"); jbyteArray bytes = env->NewByteArray(strlen(pat)); env->SetByteArrayRegion(bytes, 0, strlen(pat), (jbyte*) pat); jstring encoding = env->NewStringUTF("GB2312"); return (jstring) env->NewObject(strClass, ctorID, bytes, encoding); }
这个两个功能比较实用。
在整个设计改革之前,是通过c层传递流到java层,然后再由java层进行文件的读写操作,这个过程中出现了一些乱码,通过以上提供的两种方法能够解决。Java默认的String类型是UTF-8编码格式,而下下来的文件好多是用GBK编码方式,所以在java层进行文件读写的时候容易出现问题。
后来把整个模块经行了改写,读写文件是由c层完成的,没有了编码问题,乱码问题迎刃而解。
6)断点续传
断点续传需要对服务器上的文件经行比较,在文件操作的工具中提供了两个构造函数,一个需要源文件大小,如果调用这个方法当size不为0的时候,可以进行判断是否下载完成,然后再下载过程中设置断点续传,就可以实现断点续传功能,如果不初始化size或直接赋值为0,则默认为重新下载。
注:源文件大小,就是需要通过http,ftp下载的文件,需要获取服务器上文件的大小,暂时没有提供接口如何获取。
7)并发执行上传下载
由于网上multi模块资料较少,暂时只实现了multi一些简单功能,由于easy模块是单线程添加一个select接口,添加下载上传操作的handle,然后进行multi_perform
四.Libcurl使用案例与接口说明
1.http从头下载以及测试下载速度doubleValue
//http 下载以及测试下载速度 String url = "http://www.sina.com.cn/"; result = easy.easySetopt(handle, OptValue.CURLOPT_URL, url); // easy.easySetopt(handle, OptValue.CURLOPT_HEADER, 1); System.out.println("easySetopt state is "+result); easy.easySetopt(handle, OptValue.CURLOPT_WRITEFUNCTION, operate); result = easy.easyPerform(handle); double doubleValue = easy.easyGetinfo(handle, OptValue.CURLINFO_SPEED_DOWNLOAD, 0.0); System.out.println("dobleValue = "+doubleValue); easy.easyCleanUp(handle);
2.ftp上传
//ftp 上传 String url = "ftp://192.168.18.37/sina.html"; result = easy.easySetopt(handle, OptValue.CURLOPT_URL, url); easy.easySetopt(handle, OptValue.CURLOPT_USERPWD, "ipanel:ipanel123"); easy.easySetopt(handle, OptValue.CURLOPT_UPLOAD, 1L); easy.easySetopt(handle, OptValue.CURLOPT_INFILESIZE, file.length()); easy.easySetopt(handle, OptValue.CURLOPT_READFUNCTION, operate); result = easy.easyPerform(handle); easy.easySetopt(handle, OptValue.CURLOPT_INFILESIZE_LARGE, file.length()); System.out.println("easyPerform state is "+result
3.tcp_ip 通信
// //tcp_ip 通信 String url = "192.168.21.13"; result = easy.easySetopt(handle, OptValue.CURLOPT_URL, url); System.out.println("easySetopt state is "+result); easy.easySetopt(handle, OptValue.CURLOPT_PORT, 6666); easy.easySetopt(handle, OptValue.CURLOPT_CONNECT_ONLY, 1L); result = easy.easyPerform(handle); System.out.println("easyPerform state is "+result); String buf = "zhoulong潮"; int len = easy.easySend(handle, buf, buf.length()); for (int i = 1; i <=1; i++) { String value = null; value = easy.easyRecv(handle,len); System.out.println(value); } easy.easyCleanUp(handle); Log.d("","清除成功");
4.并发执行下载任务
int http_handle = easy.easyInit(); int http_handle2 = easy.easyInit(); easy.easySetopt(http_handle, OptValue.CURLOPT_URL, "http://www.sohu.com/"); easy.easySetopt(http_handle2, OptValue.CURLOPT_URL, "http://www.sina.com.cn/"); int multi_handle = multi.multiInit(); multi.multiAddHandle(multi_handle, http_handle); multi.multiAddHandle(multi_handle, http_handle2); easy.easySetopt(http_handle, OptValue.CURLOPT_WRITEFUNCTION, operate); easy.easySetopt(http_handle2, OptValue.CURLOPT_WRITEFUNCTION, operate2); /* * 调用curl_multi_perform函数执行curl请求 * url_multi_perform返回CURLM_CALL_MULTI_PERFORM(-1)时,表示需要继续调用该函数直到返回值不是CURLM_CALL_MULTI_PERFORM为止 * running_handles变量返回正在处理的easy curl数量,running_handles为0表示当前没有正在执行的curl请求 */ int still_running = 0; while(true){ int perform[] = multi.multiPerform(multi_handle); System.out.println("perform step 1"+" "+perform[0]+" "+perform[1]); if(perform[0] != -1){ still_running = perform[1]; break; } } while(still_running > 0) { if (-1 == multi.multiSelect(multi_handle)) { break; } else { // select监听到事件,调用curl_multi_perform通知curl执行相应的操作 // while(true){ int perform[] = multi.multiPerform(multi_handle); System.out.println("perform step 2"+" "+perform[0]+" "+perform[1]); if(perform[0] != -1){ still_running = perform[1]; break; } } } } multi.multiRemoveHandle(multi_handle, http_handle); multi.multiRemoveHandle(multi_handle, http_handle2); multi.multiCleanUp(multi_handle); easy.easyCleanUp(http_handle); easy.easyCleanUp(http_handle2);
5.断点续传
File file = new File("sdcard/"+path); //判断是否下载完全 FileOperate operate = new FileOperate(file,size); int handle = -1; int result = -1; handle = easy.easyInit(); if(handle != -1){ Log.d("", "初始化成功"); System.out.println(handle); } String url = "ftp://192.168.18.37/"+path; easy.easySetopt(handle, OptValue.CURLOPT_URL, url); easy.easySetopt(handle, OptValue.CURLOPT_USERPWD, "ipanel:ipanel123"); //断点续传声明 easy.easySetopt(handle, OptValue.CURLOPT_RESUME_FROM_LARGE, file.length()); easy.easySetopt(handle, OptValue.CURLOPT_WRITEFUNCTION, operate); result = easy.easyPerform(handle); easy.easySetopt(handle, OptValue.CURLOPT_INFILESIZE_LARGE, file.length()); System.out.println("easyPerform state is "+result);
6.相关参数资料
下列选项的值将被作为长整形使用(在option参数中指定): • CURLOPT_INFILESIZE : 当你上传一个文件到远程站点,这个选项告诉PHP你上传文件的大小。 • CURLOPT_VERBOSE : 如果你想CURL报告每一件意外的事情,设置这个选项为一个非零值。 • CURLOPT_HEADER : 如果你想把一个头包含在输出中,设置这个选项为一个非零值。 • CURLOPT_NOPROGRESS: 如果你不会PHP为CURL传输显示一个进程条,设置这个选项为一个非零值。注意:PHP自动设置这个选项为非零值,你应该仅仅为了调试的目的来改变这个选项。 • CURLOPT_NOBODY : 如果你不想在输出中包含body部分,设置这个选项为一个非零值。 • CURLOPT_FAILONERROR : 如果你想让PHP在发生错误(HTTP代码返回大于等于300)时,不显示,设置这个选项为一人非零值。默认行为是返回一个正常页,忽略代码。 • CURLOPT_UPLOAD: 如果你想让PHP为上传做准备,设置这个选项为一个非零值。 • CURLOPT_POST : 如果你想PHP去做一个正规的HTTP POST,设置这个选项为一个非零值。这个POST是普通的 application/x-www-from-urlencoded 类型,多数被HTML表单使用。 • CURLOPT_FTPLISTONLY : 设置这个选项为非零值,PHP将列出FTP的目录名列表。 • CURLOPT_FTPAPPEND : 设置这个选项为一个非零值,PHP将应用远程文件代替覆盖它。 • CURLOPT_NETRC : 设置这个选项为一个非零值,PHP将在你的 ~./netrc 文件中查找你要建立连接的远程站点的用户名及密码。 • CURLOPT_FOLLOWLOCATION : 设置这个选项为一个非零值(象 “Location: “)的头,服务器会把它当做HTTP头的一部分发送(注意这是递归的,PHP将发送形如 “Location: “的头)。 • CURLOPT_PUT : 设置这个选项为一个非零值去用HTTP上传一个文件。要上传这个文件必须设置CURLOPT_INFILE和CURLOPT_INFILESIZE选项. • CURLOPT_MUTE : 设置这个选项为一个非零值,PHP对于CURL函数将完全沉默。 • CURLOPT_TIMEOUT : 设置一个长整形数,作为最大延续多少秒。 • CURLOPT_LOW_SPEED_LIMIT: 设置一个长整形数,控制传送多少字节。 • CURLOPT_LOW_SPEED_TIME : 设置一个长整形数,控制多少秒传送CURLOPT_LOW_SPEED_LIMIT规定的字节数。 • CURLOPT_RESUME_FROM : 传递一个包含字节偏移地址的长整形参数,(你想转移到的开始表单)。 • CURLOPT_SSLVERSION: 传递一个包含SSL版本的长参数。默认PHP将被它自己努力的确定,在更多的安全中你必须手工设置。 • CURLOPT_TIMECONDITION : 传递一个长参数,指定怎么处理CURLOPT_TIMEVALUE参数。你可以设置这个参数为TIMECOND_IFMODSINCE 或 TIMECOND_ISUNMODSINCE。这仅用于HTTP。 • CURLOPT_TIMEVALUE : 传递一个从1970-1-1开始到现在的秒数。这个时间将被CURLOPT_TIMEVALUE选项作为指定值使用,或被默认TIMECOND_IFMODSINCE使用。
下列选项的值将被作为字符串: • CURLOPT_URL: 这是你想用PHP取回的URL地址。你也可以在用curl_init()函数初始化时设置这个选项。 • CURLOPT_USERPWD : 传递一个形如[username]:[password]风格的字符串,作用PHP去连接。 • CURLOPT_PROXYUSERPWD : 传递一个形如[username]:[password] 格式的字符串去连接HTTP代理。 • CURLOPT_RANGE : 传递一个你想指定的范围。它应该是”X-Y”格式,X或Y是被除外的。HTTP传送同样支持几个间隔,用逗句来分隔(X-Y,N-M)。 • CURLOPT_POSTFIELDS : 传递一个作为HTTP “POST”操作的所有数据的字符串。 • CURLOPT_REFERER: 在HTTP请求中包含一个”referer”头的字符串。 • CURLOPT_USERAGENT : 在HTTP请求中包含一个”user-agent”头的字符串。 • CURLOPT_FTPPORT: 传递一个包含被ftp “POST”指令使用的IP地址。这个POST指令告诉远程服务器去连接我们指定的IP地址。这个字符串可以是一个IP地址,一个主机名,一个网络界面名(在UNIX下),或是‘-’(使用系统默认IP地址)。 • CURLOPT_COOKIE : 传递一个包含HTTP cookie的头连接。 • CURLOPT_SSLCERT : 传递一个包含PEM格式证书的字符串。 • CURLOPT_SSLCERTPASSWD : 传递一个包含使用CURLOPT_SSLCERT证书必需的密码。 • CURLOPT_COOKIEFILE : 传递一个包含cookie数据的文件的名字的字符串。这个cookie文件可以是Netscape格式,或是堆存在文件中的HTTP风格的头。 • CURLOPT_CUSTOMREQUEST : 当进行HTTP请求时,传递一个字符被GET或HEAD使用。为进行DELETE或其它操作是有益的,更Pass a string to be used instead of GET or HEAD when doing an HTTP request. This is useful for doing or another, more obscure, HTTP request. 注意: 在确认你的服务器支持命令先不要去这样做。下列的选项要求一个文件描述(通过使用fopen()函数获得): • CURLOPT_FILE: 这个文件将是你放置传送的输出文件,默认是STDOUT. • CURLOPT_INFILE : 这个文件是你传送过来的输入文件。 • CURLOPT_WRITEHEADER : 这个文件写有你输出的头部分。 • CURLOPT_STDERR : 这个文件写有错误而不是stderr。用来获取需要登录的页面的例子,当前做法是每次或许都登录一次,有需要的人再做改进了.
最后附带Easy模块的部分代码:
java接口:
public class Easy { // create the handle 创建一个easy handle public int easyInit() { int res = -1; res = native_join_curl_easy_init(); if (res == -1) { Log.d("", "easyInit faild!"); } return res; } …… native int native_join_curl_easy_init(); …… }
jni接口:
joineasy.cpp
…… jint join_curl_easy_init(JNIEnv *env, jobject thiz) { int res = -1; res = (int)curl_easy_init(); // LOGE("error is the cleanup res %d" ,res); // curl_easy_cleanup((void*)res); if (res != -1) { LOGE("join_curl_easy_init convert success %d", res); } return res; } …… static JNINativeMethod g_cls_methods[] = { // { "native_join_curl_easy_init", "()I", (void*) join_curl_easy_init } } ……
注意:在写测试case的时候记得加载动态库
static { try { Log.i("","load library"); System.loadLibrary("curl_runtime");//jni部分生成了libcurl_runtime.so动态库 } catch (java.lang.Error e) { e.printStackTrace(); }
test.apk 依赖 libcurl_runtime.so同时 libcurl_runtime.so 依赖 libcurl.so