美文网首页
使用C++17标准库递归拷贝文件目录

使用C++17标准库递归拷贝文件目录

作者: FredricZhu | 来源:发表于2021-06-13 11:48 被阅读0次

相关注意事项可以参考前一篇,《使用C++17 标准库处理文件系统文件》,
直接上代码,
CMakeLists.txt

cmake_minimum_required(VERSION 2.6)
project(chat_room)

add_definitions(-std=c++17)


find_package(Boost REQUIRED COMPONENTS
    system
    filesystem
    serialization
    program_options
    thread
    )

include_directories(${Boost_INCLUDE_DIRS})

file( GLOB APP_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/*.cpp ${CMAKE_CURRENT_SOURCE_DIR}/*.h)
foreach( sourcefile ${APP_SOURCES} )
        file(RELATIVE_PATH filename ${CMAKE_CURRENT_SOURCE_DIR} ${sourcefile})
    if( "${filename}" STREQUAL "main.cpp")
        string(REPLACE ".cpp" "" file ${filename})
        add_executable(${file}  ${APP_SOURCES})
        target_link_libraries(${file} ${Boost_LIBRARIES})
        target_link_libraries(${file} pthread)
    endif()
endforeach( sourcefile ${APP_SOURCES})

main.cpp

#include "f_copy.h"
#include <filesystem>
#include <iostream>
#include <fstream>
#include <string>

int main(int argc, char* argv[]) {
    std::filesystem::path cur_path = std::filesystem::current_path();
    std::filesystem::path srcDir {cur_path};
    std::filesystem::path dstDir {cur_path.parent_path()};
    srcDir /= "CMakeFiles";
    dstDir /= "target_cmake";
    
    CopyDir(srcDir, dstDir);
}

f_copy.h

#ifndef _FREDRIC_F_COPY_H_
#define _FREDRIC_F_COPY_H_

#include <string>

void TextFileCopy(const std::string& srcF, const std::string& dstF); 
void BinaryFileCopy(const std::string& srcF, const std::string& dstF);
void CopyDir(const std::string& srcDir, const std::string& dstDir);

#endif

f_copy.cpp

#include "f_copy.h"

#include <filesystem>
#include <fstream>
#include <iostream>
#include <string>

void TextFileCopy(const std::string& srcF, const std::string& dstF) {
    std::ifstream input{srcF};
    if (!input) {
        std::cout << "Source File not found" << std::endl;
        return;
    }

    std::filesystem::path destPath{dstF};
    if (std::filesystem::exists(destPath)) {
        std::cout << "File already exists, will override the exist file: "
                  << dstF << std::endl;
    }

    std::ofstream output{dstF};
    if (!output) {
        std::cout << "Could not open the dest file" << std::endl;
        return;
    }

    std::string line;
    while (std::getline(input, line)) {
        output << line << std::endl;
    }

    input.close();
    output.close();
}

void BinaryFileCopy(const std::string& srcF, const std::string& dstF) {
    std::ifstream input{srcF, std::ios::in | std::ios::binary};
    if (!input) {
        std::cout << "Could not open the source file" << std::endl;
        return;
    }

    if (std::filesystem::exists(dstF)) {
        std::cout << "File Already exists, will override the exists file: "
                  << dstF << std::endl;
    }

    std::ofstream output{dstF, std::ios::out | std::ios::binary};
    if (!output) {
        std::cout << "Could not open the destination file" << std::endl;
        return;
    }

    auto file_size = std::filesystem::file_size(srcF);
    const unsigned BufferSize = 512;
    char buffer[BufferSize]{};

    std::cout << "Copying: " << srcF;
    // 文件不到一个Buffer大
    if (file_size < BufferSize) {
        if (!input.read(buffer, file_size)) {
            throw std::runtime_error("Error occurred during read operation");
        }
        if (!output.write(buffer, file_size)) {
            throw std::runtime_error("Error occurred during write operation");
        }
    } else {
        // 大文件分为多个块来拷,并打印进度,进度乘10,是为了不让结果看起来都是0,一直不打印
        auto chunks = file_size / BufferSize;
        // 这里用int是因为 剩余大小小于 512,用int就可以了,不需要 std::size_t
        int remaining = file_size % BufferSize;
        int progress{}, oldProgress{};
        for (int i = 0; i < chunks; ++i) {
            if (!input.read(buffer, BufferSize)) {
                throw std::runtime_error(
                    "Error occurred during read operation");
            }

            if (!output.write(buffer, BufferSize)) {
                throw std::runtime_error(
                    "Error occurred during write operation");
            }

            // 先转浮点避免全零,再转整型便于比较
            // 类型转换建议使用 static_cast,不要直接强转
            // 因为static_cast会做类型检查
            progress = static_cast<int>((10 * static_cast<float>(i) / chunks));
            if (progress != oldProgress) {
                std::cout << '.';
            }
            oldProgress = progress;
        }

        // 下一次的读取将会小于 BufferSize
        // 但是buffer 里面可能还有上次读取的残存数据,
        // 所以需要使用memset清空
        // 比较底层的C操作,但是没办法,因为读写二进制必须按char* [byte]处理
        memset(buffer, '\0', BufferSize);

        if (remaining > 0) {
            if (!input.read(buffer, remaining)) {
                throw std::runtime_error(
                    "Error occurred during read operation");
            }
            if (!output.write(buffer, remaining)) {
                throw std::runtime_error(
                    "Error occurred during write operation");
            }
            std::cout << '.';
        }

        std::cout << "Done! " << std::endl;

        input.close();
        output.close();
    }
}


void CopyDir(const std::string& srcDir, const std::string& dstDir) {
    // 使用std::filesystem::directory_iterator类 迭代目录下的所有文件
    std::filesystem::directory_iterator d_begin {srcDir};
    std::filesystem::directory_iterator d_end;

    // begin != end 说明还有文件
    while(d_begin != d_end) {
        // 目标path
        std::filesystem::path dstPath{dstDir};
        // 获取源文件名,拼到目标path后面
        auto dstDirName = d_begin->path().filename();
        dstPath /= dstDirName;
        // 如果是目录
        if(d_begin->is_directory()) {
            // 目录不存在
            if(!std::filesystem::exists(dstPath)) {
                // 新建目录
                std::filesystem::create_directory(dstPath);
            }
            // 递归拷贝子目录
            CopyDir(d_begin->path(), dstPath);
        } else {
            // 二进制模式拷贝文件
            BinaryFileCopy(d_begin->path(), dstPath);
        }
        ++ d_begin;
    }
}

程序输出如下,


image.png

拷贝目录的结果如下,


image.png

相关文章

网友评论

      本文标题:使用C++17标准库递归拷贝文件目录

      本文链接:https://www.haomeiwen.com/subject/luaweltx.html