coco2dx c++ 断点续传实现
实现效果如下
iPhone截图
android 日志截图
流程图如下
功能主要通过CURL c pthread 实现 我实现的不是多线程断点(如果要实现可以根据我这个进行添加任务序列,可参考 cocos2d-x 中AssetsManager的实现,其实我的部分也是参考这个写的 为什么写这个呢 原因就是 AssetsManager是不支持断点续传的)
博客地址:http://blog.csdn.net/vpingchangxin/article/details/22309067
具体可以去CURL官网或者找资料科普一下
PS:如果是版本发布最后设置超时时间20秒左右否则下载会占用更多下载实现效率等问题 我是为了测试 设置超时时间为2秒
1.先创建一个界面进行控制进行下载、停止、删除、进度 并绑定事件
2.在进行下载中开一个线程进行下载 (因为牵涉到UI,不开线程UI会卡着堵塞UI线程直到下载完成)下面是事件中的控制 HelloWorldSecene.cpp中的实现
void HelloWorld::menuCallback(CCObject* pSender) { CCMenuItem *item = (CCMenuItem *)pSender; switch (item->getTag()) { case 1: // down start CCDirector::sharedDirector()->getScheduler()->scheduleSelector(schedule_selector(HelloWorld::updateUI), this, 0, false); // HttpClient中参考 isStop = false; this->threadStart(); break; case 2: // down stop isStop = true; break; case 3: if (isStop) { CCLog("downFilePath:%s",downFilePath.c_str()); if (access(downFilePath.c_str(), 0) == 0) { remove(downFilePath.c_str()); CCMessageBox("删除成功", "温馨提示"); }else{ CCMessageBox("没有找到文件目录", "温馨提示"); } }else{ CCMessageBox("下载中或没有文件下载", "温馨提示"); } break; default: break; } }
3。实现线程类并回调设置
// 启动线程的方法 int HelloWorld::threadStart() { pthread_mutex_init(&g_downloadMutex, NULL); int errCode=0; pthread_t th_curlDown; // 线程初始化 do { pthread_attr_t tAttr; errCode=pthread_attr_init(&tAttr); CC_BREAK_IF(errCode!=0); errCode=pthread_attr_setdetachstate(&tAttr, PTHREAD_CREATE_DETACHED); if(errCode!=0) { pthread_attr_destroy(&tAttr); break; } errCode=pthread_create(&th_curlDown, &tAttr, thread_funcation, this); } while (0); return errCode; } // 需要线程来完成的功能都写在这个函数里 void* HelloWorld::thread_funcation(void *arg) { CCLOG("thread started..."); HelloWorld *hw = (HelloWorld*)arg; hw->ccc = new CurlDown("http://developer.baidu.com/map/static/doc/output/BaiduMap_AndroidSDK_v2.4.0_All.zip",hw->downFilePath); // ccc->mDownloadUrl = "http://developer.baidu.com/map/static/doc/output/BaiduMap_AndroidSDK_v2.4.0_All.zip"; // int leng = ccc->getDownloadFileLenth(); hw->ccc->setDelegate(hw); hw->ccc->downloadControler(); return NULL; }
4.实现回调进度、成功、错误(里面用到线程锁对数据进度更新UI,本来对线程就不熟悉,问了群里面的大牛,看了不少资料)
void HelloWorld::onError(CurlDown::ErrorCode errorCode){ CCLog("error"); pthread_mutex_lock(&g_downloadMutex); updateStr = "error"; pthread_mutex_unlock(&g_downloadMutex); CCDirector::sharedDirector()->getScheduler()->unscheduleSelector(schedule_selector(HelloWorld::updateUI), this); } void HelloWorld::onProgress(double percent, void *delegate, string filefullPath){ // 下载进度 CCLog("donw progress:%.2f%%",percent); if (isStop) { CurlDown * cd = (CurlDown *)delegate; // pthread_mutex_lock(&g_downloadMutex); cd->setStopDown(); // pthread_mutex_unlock(&g_downloadMutex); } pthread_mutex_lock(&g_downloadMutex); const char * per =CCString::createWithFormat("donw progress:%.2f%%",percent)->getCString(); updateStr = per; downFilePath = filefullPath; pthread_mutex_unlock(&g_downloadMutex); } void HelloWorld::onSuccess(string filefullPath){ CCLog("success"); pthread_mutex_lock(&g_downloadMutex); updateStr = "success"; downFilePath = filefullPath; pthread_mutex_unlock(&g_downloadMutex); }
5.CurlDown.h CurlDown.cpp类实现 (可以直接抽取出来用于任何地方,没有牵涉到cocos2d-x部分,cocos2d-x 部分可以删除没关系)
1)对类初始化
static pthread_mutex_t g_downloadMutex_1; CurlDown::~CurlDown(){ mFileLenth = 0; } CurlDown::CurlDown():isStop(false),mDownloadUrl(""),timeout(2){ // test timeout 2 seconds. if release timeout 20 seconds mFileLenth = 0; mFilePath = ""; pthread_mutex_init(&g_downloadMutex_1, NULL); } CurlDown::CurlDown(string downUrl,string filePath):mFileLenth(0),isStop(false),mDownloadUrl(downUrl),timeout(2),mFilePath(filePath){ // test timeout 2 seconds. if release timeout 20 seconds mDownloadUrl = downUrl; pthread_mutex_init(&g_downloadMutex_1, NULL); } void CurlDown::setDelegate(CurlDownDelegate * delegate) { mDelegate = delegate; }
2)控制下载方法
void CurlDown::downloadControler() { CCLog("--1-"); mFileLenth = getDownloadFileLenth(); // 获取远程文件大小 if (mFileLenth <= 0) { cout << "download file fail..." << endl; mDelegate->onError(kNetwork); return; } vector<string> searchPaths = CCFileUtils::sharedFileUtils()->getSearchPaths(); vector<string>::iterator iter = searchPaths.begin(); searchPaths.insert(iter, mFilePath); CCFileUtils::sharedFileUtils()->setSearchPaths(searchPaths); CCLog("--2-mFileLenth:%f",mFileLenth); mFileName = mDownloadUrl.substr(mDownloadUrl.rfind(‘/‘) + 1); CCLog("--3-"); CCLog("mFileName:%s;",mFileName.c_str()); // mFilePath = CCFileUtils::sharedFileUtils()->getWritablePath(); // CCLog("--5-"); mFilePath = mFilePath + mFileName; CCLog("mFilePath:%s",mFilePath.c_str()); CCLog("--6-"); bool ret = false; while (true){ // 循环下载 每30秒进行下载 避免断网情况 ret = download(); //直接下载 进行堵塞线程 CCLog("----stop---%s------",isStop?"true":"false"); if (isStop) { // 如果进行停止 break CCLog("----stop---------"); break; } if (ret ){ //下载完成 break; } sleep(0.5); //每次下载中间间隔0.5秒 } if (ret) { CCLog("download ok"); mDelegate->onSuccess(mFilePath); } else { CCLog("download fail"); mDelegate->onError(kUncompress); } }
3)核心下载
#pragma mark 进行下载 bool CurlDown::download() { FILE *fp = NULL; if(access(mFilePath.c_str(), 0)==0) { // 以二进制形式追加 fp = fopen(mFilePath.c_str(), "ab+"); } else { // 二进制写 fp = fopen(mFilePath.c_str(), "wb"); } if (fp == NULL) {// 如果文件初始化失败进行返回 return false; } // 读取本地文件下载大小 long localFileLenth = getLocalFileLength(); //已经下载的大小 CCLog("filePath:%s;leng:%ld",mFilePath.c_str() , localFileLenth ); //4397779 //3377875 CURL *handle = curl_easy_init(); std::string packageUrl = mDownloadUrl; //下载地址+下载文件名 curl_easy_setopt(handle, CURLOPT_URL, packageUrl.c_str()); // http://curl.haxx.se/libcurl/c/fopen.html curl_easy_setopt(handle, CURLOPT_TIMEOUT, timeout); curl_easy_setopt(handle, CURLOPT_WRITEFUNCTION, my_write_func); //写文件回调方法 curl_easy_setopt(handle, CURLOPT_WRITEDATA, fp); // 写入文件对象 curl_easy_setopt(handle, CURLOPT_RESUME_FROM, localFileLenth); // 从本地大小位置进行请求数据 // curl_easy_setopt(handle, CURLOPT_RESUME_FROM_LARGE, localFileLenth); // 坑 curl_easy_setopt(handle, CURLOPT_NOPROGRESS, 0L); curl_easy_setopt(handle, CURLOPT_PROGRESSFUNCTION, my_progress_func ); //下载进度回调方法 curl_easy_setopt(handle, CURLOPT_PROGRESSDATA, this); // 传入本类对象 CURLcode res = curl_easy_perform(handle); fclose(fp); return res == CURLE_OK; }
下面大家要问道的就是求源码(^..^)源码已经上传github https://github.com/pingchangxin/BPDownload cesd 下载位置:http://download.csdn.net/detail/vpingchangxin/7108649
我这边就再mac上门跑了下 windows的没有进行跑(对win的配置繁琐的头疼了了)