基于libcurl 实现web-uploader客户端 大文件分片,断点续传(curl-uploader)

news2025/10/20 15:43:18

嵌入式设备Arm linux 基于libcurl 实现大文件分片上传,断点续传,分片大小可控,使用libcurl 实现web-upload linux c/c++ 客户端(curl-uploader)。通过wireshark 抓包网页版客户端上传文件过程,分析http post 流程。

工程代码:

客户端curl-uploader: https://download.csdn.net/download/zhujinghao09/87431650

服务器 https://gitee.com/laminae_admin/webupload/

主要流程如下:

_check_url = base_url + "/upload/fileRecord//zone/upload/md5Check"; (MD5验证)

_upload_url = base_url + "/upload/fileRecord//zone/upload"; (上传分片)

_merge_url = base_url + "/upload/fileRecord//zone/upload/merge/"; (合并分片)

核心代码:

s1 文件MD5

/*
    计算文件MD5,支持分片计算,指定计算大小
*/
int Ufile::file_md5(std::string &md5_value,int start,int end)
{
    char buf[1024 * 16];
    MD5_CTX md5Context;
    MD5_Init(&md5Context);
    int  k =0;
    int s = start;
    int e = end;

     md5_value.clear();

    //_file.seekp(0, _file.end);
    auto fos = _file.tellg();


    if(s<=0)
        s= 0;
    if(s>_file_size){
        return -1;
    }
    if(e<0){
        return -1;
    }
    if(e>=_file_size){
        e = _file_size;
    }
    k = 0;
    if(s>0)
        _file.seekg(s);

    while (_file.good() && k <= e-s) {
        _file.read(buf, sizeof(buf));
        // gcount() 返回上次从文件流提取出的字符个数
        MD5_Update(&md5Context, buf, _file.gcount());
        k += _file.gcount();
    }

    if (_file.eof())
        _file.clear();    //清空结尾eof标志,可以再次打开该文件


    unsigned char result[MD5_DIGEST_LENGTH];
    MD5_Final(result, &md5Context);

    char hex[35];
    memset(hex, 0, sizeof(hex));
    for (int i = 0; i < MD5_DIGEST_LENGTH; ++i)
    {
        sprintf(hex + i * 2, "%02x", result[i]);
    }
    hex[32] = '\0';
    md5_value = string(hex);
    //回位
    _file.seekg(fos);


    return 0;
}

S2 文件分片读取,与计算分片MD5

/*
    读取分片数据
*/
int SliceFile::do_read_slice_data(int sliceIndex)
{
    if(_cur_slice_index==sliceIndex){
        return 0;
    }
    int once_read_size = 16*1024;

    int s = sliceIndex * _slice_size;
    int k = 0;

    int to_read_size = _slice_size;

    if(s + to_read_size > file_size()){
        to_read_size = file_size() - s;
    }

    _file.seekg(s);

    while (_file.good() && k < to_read_size) {
        _file.read(_slice_data+k, once_read_size);
        // gcount() 返回上次从文件流提取出的字符个数
        k += _file.gcount();
    }

    if (_file.eof())
        _file.clear();    //清空结尾eof标志,可以再次打开该文件


    _slice_data_size = k;
    _cur_slice_index = sliceIndex;


    return 0;

}

char* SliceFile::get_slice_data(int sliceIndex)
{
    if(sliceIndex != _cur_slice_index){
        do_read_slice_data(sliceIndex);
    }
    return _slice_data;
}

int SliceFile::get_slice_data_size(int sliceIndex)
{
    if(sliceIndex != _cur_slice_index){
            do_read_slice_data(sliceIndex);
    }

    return _slice_data_size;
}
/*
        计算分片MD5

*/
int SliceFile::SliceMd5(std::string &md5,int sliceIndex){

    if(sliceIndex <0 || sliceIndex > _slice_num){
        printf("invlaid sliceIndex:%d\n",sliceIndex);
        return -1;
    }
    return file_md5(md5,sliceIndex*_slice_size,(sliceIndex+1)*_slice_size);
}


/*
        打印文件信息,分片数量及分片md5
*/
void SliceFile::PrintInfo()
{
    printf("\n\n=====================File========================\n");
    printf("file_path: %s\n",_file_path.c_str());
    printf("file_name: %s\n",_file_name.c_str());


    printf("\n\n=====================MD5========================\n");
    printf("file_size:  %d\n",_file_size);
    printf("slice_size: %d\n",_slice_size);
    printf("slice_num:  %d\n",_slice_num);

    printf("fileMD5: %s\n",file_md5().c_str());
    for(int i=0;i<_slice_num;i++){
        std::string md5;

        SliceMd5(md5,i);
        printf("sliceIndex:%d,     zoneMd5:%s\n",i,md5.c_str());
    }

    printf("=====================MD5========================\n");

    //printf("function = %s,line = %d\n",__FUNCTION__,__LINE__);
    printf("文件使用的设备号: %lu\r\n", _file_state.st_dev); //32为int数据,高12位主设备号,低20位次设备号
    printf("索引节点号: %lu\r\n", _file_state.st_ino);
    printf("文件对应的模式,文件,目录等: %u\r\n", _file_state.st_mode);
    printf("文件的硬连接数: %lu\r\n", _file_state.st_nlink);
    printf("所有者用户识别号: %u\r\n", _file_state.st_uid);
    printf("组识别号: %u\r\n", _file_state.st_gid);
    printf("设备文件的设备号: %lu\r\n", _file_state.st_rdev);
    printf("文件大小: %ld\r\n", _file_state.st_size);
    printf("访问日期: %s", ctime(&_file_state.st_atime));
    printf("最后修改日期: %s", ctime(&_file_state.st_mtime));
    printf("文件创建时间: %s", ctime(&_file_state.st_ctime));

    printf("=====================File========================\n\n");
}

S3 libcurl POST 上传二进制分片数据


/*
 *
 * return
 *   0 sucess
 *
 */
int WebUploader::curlPostFormDataWithBin(SliceFile& file,int sliceIndex,const char* _url, struct curl_httppost* formpost,CURL_CALL_RESULT* result)
{
    CURLcode res;

    struct curl_slist * _http_headers;
    char *progress_data = (char*)NULL;

    BOOL prevent_cache = TRUE;

    // http_headers = curl_slist_append(http_headers, "Accept-Charset: utf-8");//WEB暂不识别
    // http_headers = curl_slist_append(http_headers, "Accept-Encoding: utf-8");//WEB暂不识别
    //返回为JSON时使用。
    _http_headers = curl_slist_append(_http_headers, "Content-Type: multipart/form-data;");
    //_http_headers = curl_slist_append(_http_headers, "Content-Type:application/json;charset=UTF-8");
    //http_headers = curl_slist_append(http_headers, "Content-Type: application/x-www-form-urlencoded; charset=UTF-8");
    //curl_easy_setopt(_curlHandle, CURLOPT_SSLVERSION, 3);
    curl_easy_setopt(_curlHandle, CURLOPT_SSL_VERIFYHOST, 0);
    curl_easy_setopt(_curlHandle, CURLOPT_SSL_VERIFYPEER, 0L);//忽略证书检查
    //curl_easy_setopt(_curlHandle, CURLOPT_SSLCERT,"ca.pem");
    //curl_easy_setopt(_curlHandle, CURLOPT_CAINFO,"ca.pem");
    //curl_easy_setopt(_curlHandle, CURLOPT_CAPATH, "/etc/ssl");

    curl_easy_setopt(_curlHandle, CURLOPT_URL, _url);
    curl_easy_setopt(_curlHandle, CURLOPT_POST, 1L);

    curl_easy_setopt(_curlHandle, CURLOPT_HTTPHEADER, _http_headers);    //设置发送http头部
    curl_easy_setopt(_curlHandle, CURLOPT_HTTPPOST, formpost);

    curl_easy_setopt(_curlHandle, CURLOPT_TIMEOUT, 10L);            //设置超时时间

    //重要!!!多线程下禁用控制域名解析的alarm 超时
    curl_easy_setopt(_curlHandle, CURLOPT_NOSIGNAL, prevent_cache);


    curl_easy_setopt(_curlHandle, CURLOPT_PROGRESSDATA, &file);
    curl_easy_setopt(_curlHandle, CURLOPT_NOPROGRESS, FALSE);
    curl_easy_setopt(_curlHandle, CURLOPT_PROGRESSFUNCTION,WebUploader::web_service_progress_func);

    //提交post字符串
//    if(_content_ptr && _content_size>0){
//        curl_easy_setopt(_curlHandle, CURLOPT_POSTFIELDS, _content_ptr);
//        curl_easy_setopt(_curlHandle, CURLOPT_POSTFIELDSIZE, _content_size);
//    }

    curl_easy_setopt(_curlHandle, CURLOPT_WRITEDATA, result);
    curl_easy_setopt(_curlHandle, CURLOPT_WRITEFUNCTION,WebUploader::web_service_rsp_write_func);


    curl_easy_setopt(_curlHandle, CURLOPT_VERBOSE, 1L);

    printf("enter curl ws call: curl=%p, ws_result_ptr=%p....\n",
            _curlHandle,
            result);

    //printf( "curl_easy_perform()....\n");
    res = curl_easy_perform(_curlHandle);
    result->curl_res = res;


    return (int)res;
}

/*
 * {"success":true,"code":10000,"message":"...............","data":{"zoneNowIndex":1,"fileZone":{"id":"1619576327126519808","zoneName":"8c6ca01dcd6c57160a83e7e8d675b3e1.mp4.temp","zonePath":"d://temp/webupload/fileData/webuploadfile/temp/cbe607627b5e27ee460acde1005f160d","zoneMd5":"8c6ca01dcd6c57160a83e7e8d675b3e1","zoneRecordDate":"2023-01-29 06:01:01","zoneCheckDate":null,"zoneTotalCount":2,"zoneTotalMd5":"cbe607627b5e27ee460acde1005f160d","zoneNowIndex":1,"zoneStartSize":5242880,"zoneEndSize":7271720,"zoneTotalSize":7271720,"zoneSuffix":".mp4","fileRecordId":"1619576324489478146"},"isExist":false}}
 *
return
     0  already upload sucess
    -1  error
*/
int WebUploader::curlPostSlice(SliceFile& file,int sliceIndex,string &zoneMd5)
{
    struct curl_httppost *formpost = NULL;
    struct curl_httppost *lastptr = NULL;

    CURLFORMcode  resultForm;
    int ret =0;

    struct curl_httppost *p = NULL;


    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "id",
            CURLFORM_COPYCONTENTS, (const char *) "WU_FILE_0", CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "name",
            CURLFORM_COPYCONTENTS, (const char *) file.file_name().c_str(), CURLFORM_END );

    //application/octet-stream
    //video/mp4

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "type",
            CURLFORM_COPYCONTENTS, (const char *) file.content_type().c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "lastModifiedDate",
            CURLFORM_COPYCONTENTS, (const char *) file.lastModifiedDate().c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "size",
            CURLFORM_COPYCONTENTS, (const char *) to_string(file.file_size()).c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "chunk",
            CURLFORM_COPYCONTENTS, (const char *) to_string(sliceIndex).c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "fileMd5",
            CURLFORM_COPYCONTENTS, (const char *) file.file_md5().c_str(), CURLFORM_END );


    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "contentType",
            CURLFORM_COPYCONTENTS, (const char *) file.content_type().c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneTotalMd5",
            CURLFORM_COPYCONTENTS, (const char *) file.file_md5().c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneMd5",
            CURLFORM_COPYCONTENTS, (const char *) zoneMd5.c_str(), CURLFORM_END );


    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneTotalCount",
            CURLFORM_COPYCONTENTS, (const char *) to_string(file.slice_num()).c_str(), CURLFORM_END );


    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneNowIndex",
            CURLFORM_COPYCONTENTS, (const char *) to_string(sliceIndex).c_str(), CURLFORM_END );

    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneTotalSize",
            CURLFORM_COPYCONTENTS, (const char *) to_string(file.file_size()).c_str(), CURLFORM_END );


    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneStartSize",
            CURLFORM_COPYCONTENTS, (const char *) to_string(file.slice_size()*sliceIndex).c_str(), CURLFORM_END );

    //Content-Disposition: form-data; name="zoneEndSize"
    //
    //386726
    int zoneEndSize = file.slice_size()*(sliceIndex+1) > file.file_size()? file.file_size(): file.slice_size()*(sliceIndex+1);
    resultForm = curl_formadd( &formpost, &lastptr, CURLFORM_COPYNAME, "zoneEndSize",
            CURLFORM_COPYCONTENTS, (const char *) to_string(zoneEndSize).c_str(), CURLFORM_END );

    //Content-Disposition: form-data;name="file";filename="yolov5s_zhc_foodSmokePhone_13330_300_960_2022-05-12.rknn"
    //Content-Type:    application/octet-stream
    //二进制流上传:
    resultForm = curl_formadd(&formpost,&lastptr,
            CURLFORM_COPYNAME,     "file",
            CURLFORM_BUFFER,        file.file_name().c_str(),
            CURLFORM_BUFFERPTR,     file.get_slice_data(sliceIndex),
            CURLFORM_BUFFERLENGTH,  file.get_slice_data_size(sliceIndex),
            CURLFORM_CONTENTTYPE,    "application/octet-stream",
            CURLFORM_END);
    //curl_formadd

    if(resultForm!=CURL_FORMADD_OK){
        printf("curl_formadd faild:%d\n",resultForm);
        return -1;
    }
    p = formpost;
    do{
        printf("%s,%s,%d\n",p->name, p->contents,p->contentslength);
        p = p->next;
    }while(p!=lastptr);

    clear_ws_reault(&_ws_result);
    ret = curlPostFormDataWithBin(file,sliceIndex,_upload_url.c_str(),formpost,&_ws_result);

    if(ret == 0 && strstr(_ws_result.json_data,"\"success\":true,\"code\":10000,")){
        //upload sucess
        ret = 0;
    }else{
        printf("curlPostFormDataWithBin fiald ret:%d,%d\n",ret,_ws_result.curl_res);
    }


    return ret;

}

S4 分片上传主流程

/*
 *
 *
 *
   return 0  already upload sucess
   return 1  no upload
   return -1  system error
 **/
int WebUploader::CheckSlice(SliceFile& file,int sliceIndex,string &zoneMd5)
{
    int ret = -1;

    string ctx = "checkType=2&zoneTotalMd5=";
    ctx += file.file_md5();
    ctx += "&zoneMd5=";
    ctx += zoneMd5;

    char* content_type = "Content-Type: application/x-www-form-urlencoded; charset=UTF-8";
    clear_ws_reault(&_ws_result);
    curlPost(_check_url.c_str(),content_type,ctx.c_str(),ctx.size(),&_ws_result);

    if(_ws_result.curl_res != CURLE_OK){
        ret = -1;
        goto end;
    }
    if(strstr(_ws_result.json_data,"\"success\":true,\"code\":10000,")){
        //already upload
        ret = 0;
        goto end;
    }else if(strstr(_ws_result.json_data,"\"success\":false,\"code\":99999,")){
        //no upload
        ret = 1;
        goto end;
    }else{
        ret = -1;
    }
end:
    return ret;
}

/*
 *
 * return:
 *  0 sucess
 *  1 faild
 */
int WebUploader::MergeFileSlices(SliceFile& file)
{
    int ret = -1;
    string zoneMd5;
    string ctx="";

    if(!file.remote_path().empty()){
        ctx += "remotePath=";
        ctx += file.remote_path();
    }
    string url = string(_merge_url.c_str());
    url += file.file_md5();
    char* content_type = "Content-Type: application/x-www-form-urlencoded; charset=UTF-8";
    clear_ws_reault(&_ws_result);
    curlPost(url.c_str(),content_type,ctx.c_str(),ctx.size(),&_ws_result);

    if(_ws_result.curl_res != CURLE_OK){
        ret = 1;
        goto end;
    }
    if(strstr(_ws_result.json_data,"\"success\":true,\"code\":10000,")){
        //Merge  sucess
         ret = 0;
        goto end;
    }else{
        //Merge  faild
        ret = 1;
        goto end;
    }
end:
    return ret;
}

/*
   return 0  already upload sucess
   return -1   error
 */
int WebUploader::UploadFileSlice(SliceFile& file,int sliceIndex)
{
    int ret = 0;
    std::string zoneMd5;

    printf("\n\n");

    zoneMd5.clear();
    file.SliceMd5(zoneMd5,sliceIndex);

    printf("========================= step 2.%d-1 CheckSlice==========================\n",sliceIndex);

    ret = CheckSlice(file,sliceIndex,zoneMd5);


    printf("CheckSlice:%d result:%d\n",sliceIndex,ret);
    //already upload
    if(ret==0){
        return 0;
    }else if(ret==1){
        printf("========================= step 2.%d-2 curlPostSlice==========================\n",sliceIndex);
        //no upload
        //do cur post
        ret = curlPostSlice(file,sliceIndex,zoneMd5);
        printf("curlPostSlice:%d result:%d\n",sliceIndex,ret);
        return ret;
    }else{
        printf("CheckSlice sys error sliceIndex:%d,zoneMd5:%s\n",sliceIndex,zoneMd5.c_str());
        return -1;
    }

    return 0;
}
/*
   return 0  already upload sucess
   return -1   error
 */
int WebUploader::UploadFile(SliceFile& file)
{
    int ret = 0;
    //"contentType" = "image/jpeg"
    do{
        //step1

        printf("========================= step 1 CheckFile==========================\n");
        int checkRet = CheckFile(file);
        if(checkRet==0){
            ret = 0;
            break;
        }else if(checkRet==-1){
            printf("CheckFile sys error...\n");
            ret = -1;
            break;
        }

        //step2
        printf("========================= step 2 UploadFileSlice==========================\n");
        for(int i =0;i<file.slice_num();i++){
            ret = UploadFileSlice(file,i);
            if(ret !=0){
                printf("upload file:%s,slice:%d faild\n",file.file_path().c_str(),i);
                break;
            }
        }
        if(ret != 0){
            break;
        }
        //step3
        printf("========================= step 3 MergeFileSlices==========================\n");
        ret = MergeFileSlices(file);
        if(ret != 0){
            printf("upload file:%s,Merge faild\n",file.file_path().c_str());
            break;
        }
    }while(0);

    return ret;
}

/*
   return 0  already upload sucess
   return -1   error
 */
int WebUploader::UploadFile(string filePathName,string remotePath,ON_FTP_POST_FILE_PROGRESS fn,int sliceSize)
{
    int ret = -1;

    SliceFile file(this,fn,remotePath,sliceSize);

    ret = file.open(filePathName);
    if(ret!=0){
        printf("file open faild");
        return -1;
    }
    file.PrintInfo();


    ret = UploadFile(file);
    if(ret != 0){
        printf("upload file faild");
        goto end;
    }

end:
    file.close();

    return ret;
}

分析工具:

wireshark

参考链接:

1 web-uploader: http://fex.baidu.com/webuploader/ 百度

2 后台服务器: https://gitee.com/laminae_admin/webupload/

3 文件md5: https://www.shuzhiduo.com/A/6pdDlkoOdw/

4 libcurl上传文件 https://juejin.cn/post/6901995720740765704

5 libcurl curl_formadd https://wenku.baidu.com/view/db8e88c26194dd88d0d233d4b14e852458fb39d7.html?_wkts_=1675932603111&bdQuery=libcurl+curl_formadd+%E6%96%87%E4%BB%B6

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/334081.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【排序算法】快速排序(Quick Sort)

快速排序(Quick Sort)使用分治法算法思想。快速排序介绍它的基本思想是: 选择一个基准数&#xff0c;通过一趟排序将要排序的数据分割成独立的两部分&#xff1b;其中一部分的所有数据都比另外一部分的所有数据都要小。然后&#xff0c;再按此方法对这两部分数据分别进行快速排…

SCADA软件平台数据库功能的应用

一、前言 SCADA系统是以计算机为基础的过程控制和调度自动化系统&#xff0c;它对工业现场的设备进行监视和控制&#xff0c;并实现数据采集与分析、设备运行控制、异常报警等。数据库系统作为SCADA系统中的重要组成部分&#xff0c;是进行数据采集、处理和存储的基础&#xf…

软件设计(八)

软件设计&#xff08;七&#xff09;https://blog.csdn.net/ke1ying/article/details/128927149 71、计算机系统主要是由 DRAM 构成的。 解析&#xff1a;随机访问存储器RAM有两类。 DRAM:(dynamic)动态随机访问存储器。SRAM比DRAM贵很多&#xff0c;DRAM电容很小&#xff0…

基于深度学习的安全帽监管系统

摘 要 安全生产管理是建筑、重工业等高危企业发展的重要方针,安全帽在施工生产环境中对人员头部防护起着关键作用,因此加强安全帽佩戴监管十分必要。近年来,基于图像视觉的安全帽佩戴监测方法成为了企业实施管理的主要手段,如何提高安全帽佩戴检测精度和检测速度是应用的关键难…

Linux系列文章 —— 进程概念(图解PCB、进程的描述、状态、创建、组织、查看详解)

系列文章目录 文章目录系列文章目录前言一、冯诺依曼体系结构二、操作系统三、进程概念进程概念描述进程&#xff08;PCB&#xff09;组织进程查看进程创建进程&#xff08;fork&#xff09;进程队列进程状态运行状态 - R睡眠状态 - S停止状态 - T僵尸进程 - Z孤儿进程进程优先…

不会做UI自动化测试?一起设计框架再实践吧

目的相信做过测试的同学都听说过自动化测试&#xff0c;而UI自动化无论何时对测试来说都是比较吸引人的存在。相较于接口自动化来说它可以最大程度的模拟真实用户的日常操作与特定业务场景的模拟&#xff0c;那么存在即合理&#xff0c;自动化UI测试自然也是广大测试同学职业道…

Jmeter之实现参数化的不同方式详解

参数化简介 定义&#xff1a;动态的获取、设置或生成数据&#xff0c;是一种由程序驱动代替人工驱动的数据设计方案&#xff0c;提高脚本的编写效率以及编写质量 适用场景&#xff1a;当提交的数据量较大时&#xff0c;每次修改太麻烦&#xff0c;可以使用参数化 本文介绍实现…

DAMA数据管理知识体系指南之数据仓库和商务智能管理

第9章 数据仓库和商务智能管理 9.1简介 数据仓库&#xff08;Data Warehouse,DW)由两个主要部分构成&#xff1a;首先是一个整合的决策支持数据库&#xff0c;其次是用于收集、清洗、转换、存储来自于各种操作型数据源和外部数据源数据的相关软件程序。两者结合以支持历史的、…

mycat学习

安装部署下载1&#xff1a;mycat2-install-template-1.21.zip下载2&#xff1a;mycat2-1.21-release-jar-with-dependencies.jar解压mycat2-install-template-1.21.zipunzip mycat2-install-template-1.21.zip把mycat2-1.21-release-jar-with-dependencies.jar放在mycat/lib中修…

【逐步剖C】-第五章-指针初阶

一、指针的基本知识 1. 预备知识&#xff1a; &#xff08;1&#xff09;内存的简单概念&#xff1a; 内存是电脑上特别重要的存储器&#xff0c;计算机中程序的运行都是在内存中进行。 为了有效的使用内存&#xff0c;就把内存划分成一个个小的内存单元&#xff0c;经过仔细…

js方法总结:join 、split、splice、async await

async await 异步里面有同步操作 执行完一个&#xff0c;再去执行另一个是同步&#xff0c;如promise().then() 所有的一起执行是异步 同步和异步操作的区别就是是否阻碍后续代码的执行。 下面例子是同步&#xff0c;执行完 test再去执行test2最后test3&#xff0c;&#xff…

Android 性能优化之内存优化与泄漏分析工具LeakCanary

一、背景 在Android应用中&#xff0c;除了正常的业务开发&#xff0c;我们也要关注性能问题。卡顿、内存溢出、内存泄漏等问题&#xff0c;直接的表现会反馈到用户体验上&#xff0c;用户体验不好导致应用被卸载或者换到其他平台。 在性能优化&#xff0c;各大公司都会付出一些…

sass预处理中文教程(2023开学版)

Sass安装步骤 一、Sass的安装 由于Sass是使用Ruby编写的&#xff0c;所以我们在使用Sass之前需要安装一下Ruby。下面是Windows平台的安装步骤。对于Mac平台&#xff0c;请查看这里&#xff1a;Mac平台安装Sass。 【步骤1“下载Ruby”】&#xff1a;大家到Ruby的官网&#xf…

MySQL----内置函数

MySQL---->内置函数 函数&#xff1a;将经常使用的代码封装起来&#xff0c;需要的时候直接调用就可以 从函数定义角度&#xff0c;函数可分为&#xff1a; 内置函数&#xff1a;系统内置的通用函数自定义函数&#xff1a;需要根据需求编写的函数 MySQL提供的内置函数从…

【游戏逆向】RPG游戏自动打怪之取得最近怪物

RPG游戏辅助想要自动打怪&#xff0c;首要的任务就是需要获得最近的怪物,否则打怪的顺序会非常混乱。 想要取得最近怪物信息 要2个步骤 第一步 计算出所有怪物的距离 第二步 对所有怪物的距离进行判断,筛选出距离最近的 第一步怪物距离计算 如果是以2D的平面来看 2个点的距…

[Tomcat下载安装以及配置(详细教程)]

下载及安装Tomcat进入tomcat官网&#xff0c;Tomcat官网Apache Tomcat - Welcome!选择需要下载的版本&#xff0c;点击下载下载路径一定要记住&#xff0c;并且路径中尽量不要有中文下载后是压缩包 .zip&#xff0c;解压后tomcat系统各个文件夹目录是什么意义&#xff1a;bin&a…

二进制部署K8S集群

目录 一、架构图 二、部署步骤 1、实验环境 2、操作系统初始化配置 3、部署 docker引擎 4、部署 etcd 集群 5、部署 Master 组件 一、架构图 二、部署步骤 1、实验环境 服务器类型IP地址master192.168.80.5node01192.168.80.8node02192.168.80.9 2、操作系统初始化配置…

【C++】类和对象--类的6个默认成员函数

目录1.类的6个默认成员函数2.构造函数2.1概念2.2特性3.析构函数3.1概念3.2特性4.拷贝构造函数4.1概念4.2特征5.赋值运算符重载5.1运算符重载5.2赋值运算符重载5.3前置和后置重载5.4流插入和流提取运算符重载6.const成员7.取地址重载和const取地址操作符重载1.类的6个默认成员函…

“搜索大战”正式打响,微软发布ChatGPT版搜索引擎和浏览器

微软公司宣布推出由ChatGPT支持的最新版本Bing&#xff08;必应&#xff09;搜索引擎和Edge浏览器&#xff0c;今天上线&#xff0c;免费使用&#xff01; 自去年开始&#xff0c;Stable Diffusion、ChatGPT 等 AI 工具的横空出世&#xff0c;貌似在告诉人们“AI 正在准备重塑整…

常见的编程语言有哪些?

一、编程语言定义 编程语言的定义是指主要用于人和计算机之间通信的语言&#xff0c;它既能够让程序员能够准确的定义计算机所需数据&#xff0c;也能让计算机精准的识别人的意图。 二、编程语言排行榜以上排名信息来源于Tiobe公司公布的编程语言排行榜&#xff0c;近两年由于大…