boost之spirit学习-mini_c(4)

本文探讨了boost spirit库中的error_handler.hpp,解析了result结构体在boost::result_of中的作用。error_handler是一个函数对象,用于定义规则出错时的处理方式。内容包括如何获取错误位置到行末的字符串,并展示了如何设置出错处理函数,特别是当Action为函数或函数对象时,如何使用boost::bind进行参数转换。

今天看看error_handler.hpp,先把这些周边的东西搞清楚。

    ///////////////////////////////////////////////////////////////////////////////
    //  The error handler
    ///////////////////////////////////////////////////////////////////////////////
    template <typename Iterator>
    struct error_handler
    {   
        template <typename, typename, typename>
        struct result { typedef void type; };

        error_handler(Iterator first, Iterator last)
          : first(first), last(last) {}
        //...
        Iterator first;
        Iterator last;
        std::vector<Iterator> iters;
    };

error_handler.hpp比较简单,只定义了error_handler一个模板类。包含三个成员变量:first、last和iters。first和last代表要解析的字符串的起始位置。iters比较奇怪,它在error_handler里定义,但在error_handler里一次都没用。翻翻其它代码,发现这个iters由annotation类使用,annotation类保存对这个iters数组的引用。每次出现一个需要标记的语法对象,annotation类就在iters里记录下该对象的位置,而对象的id就是iters数组的下标。从而通过对象id可以直接查到对象对代码中的确切位置。

struct result是为了boost::result_of准备的。error_handler是个函数对象,因此它的返回值需要通过这种特殊方法标记。


        template <typename Message, typename What>
        void operator()(
            Message const& message,
            What const& what,
            Iterator err_pos) const
        {
            int line;
            Iterator line_start = get_pos(err_pos, line);
            if (err_pos != last)
            {
                std::cout << message << what << " line " << line << ':' << std::endl;
                std::cout << get_line(line_start) << std::endl;
                for (; line_start != err_pos; ++line_start)
                    std::cout << ' ';
                std::cout << '^' << std::endl;
            }
            else
            {
                std::cout << "Unexpected end of file. ";
                std::cout << message << what << " line " << line << std::endl;
            }
        }

上面这个函数是函数对象方法的实现。message和what的类型都是泛型的。大概就是通过出错的位置找到出错行的起始位置和行号,再判断出错位置是否达到待解析字符串末尾。根据情况给出不同的提示。

        Iterator get_pos(Iterator err_pos, int& line) const
        {
            line = 1;
            Iterator i = first;
            Iterator line_start = first;
            while (i != err_pos)
            {
                bool eol = false;
                if (i != err_pos && *i == '\r') // CR
                {
                    eol = true;
                    line_start = ++i;
                }
                if (i != err_pos && *i == '\n') // LF
                {
                    eol = true;
                    line_start = ++i;
                }
                if (eol)
                    ++line;
                else
                    ++i;
            }
            return line_start;
        }


        Iterator get_pos(Iterator err_pos, int& line) const
        {
            line = 1;
            Iterator i = first;
            Iterator line_start = first;
            while (i != err_pos)
            {
                bool eol = false;
                if (i != err_pos && *i == '\r') // CR
                {
                    eol = true;
                    line_start = ++i;
                }
                if (i != err_pos && *i == '\n') // LF
                {
                    eol = true;
                    line_start = ++i;
                }
                if (eol)
                    ++line;
                else
                    ++i;
            }
            return line_start;
        }

        Iterator get_pos(Iterator err_pos, int& line) const
        {
            line = 1;
            Iterator i = first;
            Iterator line_start = first;
            while (i != err_pos)
            {
                bool eol = false;
                if (i != err_pos && *i == '\r') // CR
                {
                    eol = true;
                    line_start = ++i;
                }
                if (i != err_pos && *i == '\n') // LF
                {
                    eol = true;
                    line_start = ++i;
                }
                if (eol)
                    ++line;
                else
                    ++i;
            }
            return line_start;
        }


根据出错位置找出出错行的起始位置和行号。从头开始一个字符一个字符地找,很简单。
        std::string get_line(Iterator err_pos) const
        {
            Iterator i = err_pos;
            // position i to the next EOL
            while (i != last && (*i != '\r' && *i != '\n'))
                ++i;
            return std::string(err_pos, i);
        }


获取从出错位置开始到出错行末的字符串。也很简单。


error_handler用于定义rule出错时的处理。定义出错处理的函数:

on_error<Action>(rule, handler)

其中Action可以是:

Action

Description

fail

Quit and fail. Return a no_match.

retry

Attempt error recovery, possibly moving the iterator position.

accept

Force success, moving the iterator position appropriately.

rethrow

Rethrows the error.

handler可以是函数或函数对象(error_handler类即用于此处),接受4个参数:

Arg

Description

first

The position of the iterator when the rule with the handler was entered.

last

The end of input.

error-pos

The actual position of the iterator where the error occured.

what

What failed: a string describing the failure.


但error_handler只接受3个参数。因此中间需要用类似boost::bind的技巧转换一下。

bind(error_handler, "Erorr! Expecting ", _4, _3)

但mini_c里用的是boost::phoenix::function,具体情况下回分解


<think>好的,我现在需要回答用户关于CMake选项中-DDOWNLOAD_BOOST-DENABLE_DO​​WNLOADS的区别的问题。首先,我得回忆一下这两个选项的作用,可能查阅相关资料确保准确性。 记得CMake项目中,有些选项用于控制依赖的下载。像Boost这样的库,有时项目会提供自动下载并构建的选项。用户提到的-DDOWNLOAD_BOOST可能是特定于项目的选项,用于允许在配置时下载Boost库。而-DENABLE_DO​​WNLOADS可能是一个更通用的选项,控制是否允许CMake在构建过程中下载任何外部依赖。 需要确认这两个选项的具体定义。通常在CMake中,选项是通过option()命令定义的。例如,项目可能有类似option(DOWNLOAD_BOOST "Download Boost instead of using system" ON),这样用户可以通过-D来设置。而ENABLE_DO​​WNLOADS可能是一个全局开关,影响所有依赖的下载行为。 可能存在的情况是,DOWNLOAD_BOOST是项目特定的,仅控制Boost的下载,而ENABLE_DO​​WNLOADS是控制整个项目是否允许下载任何文件。如果ENABLE_DO​​NLOADS设为OFF,即使DOWNLOAD_BOOST设为ON,也可能无法下载,因为全局开关关闭了下载功能。 需要进一步验证这个理解是否正确。例如,CMake的ExternalProject模块通常会有选项来控制下载,但不同项目可能有不同的实现。用户可能在编译某些项目时遇到需要同时设置这两个选项的情况,或者需要了解它们的优先级。 另外,需要说明这两个选项通常是在CMake配置阶段通过命令行传递的,例如cmake -DDOWNLOAD_BOOST=ON -DENABLE_DO​​WNLOADS=ON ..。可能还要提到,如果系统已经安装了Boost,可以关闭DOWNLOAD_BOOST来使用系统库,而ENABLE_DO​​WNLOADS关闭的话,所有依赖需要预先准备好。 最后,需要建议用户查阅项目的CMakeLists.txt或文档,因为不同项目可能有不同的实现方式,这些选项的具体行为可能因项目而异。总结时强调两者之间的区别和可能的依赖关系,帮助用户正确配置构建选项。</think>在CMake配置中,`-DDOWNLOAD_BOOST` 和 `-DENABLE_DOWNLOADS` 是与依赖管理相关的选项,具体区别如下: --- ### 1. **-DDOWNLOAD_BOOST** - **作用**: 这是一个**项目级选项**,通常用于控制是否在构建过程中自动下载并编译 **Boost 库**。 - 若设置为 `ON`(如 `-DDOWNLOAD_BOOST=ON`),CMake 会从网络下载指定版本的 Boost 源码并编译。 - 若设置为 `OFF`,则要求系统已预装 Boost 库,且CMake会尝试通过 `find_package(Boost)` 查找它。 - **适用场景**: 当项目依赖 Boost,但用户不想手动安装或系统缺少 Boost 时启用。 - **示例**: ```bash cmake -DDOWNLOAD_BOOST=ON -DBOOST_VERSION=1.80.0 .. ``` --- ### 2. **-DENABLE_DOWNLOADS** - **作用**: 这是一个**全局开关**,控制整个项目中所有通过 `ExternalProject` 或 `FetchContent` 定义的**外部依赖是否允许下载**。 - 若设置为 `ON`(默认可能为 `OFF`),允许CMake自动下载所有声明的外部依赖(如GitHub仓库、压缩包等)。 - 若设置为 `OFF`,则禁止任何下载行为,要求所有依赖已预先本地存在。 - **适用场景**: - 在离线环境中构建项目时需设为 `OFF`。 - 需要严格管控依赖来源时(如企业安全策略)。 - **示例**: ```bash cmake -DENABLE_DOWNLOADS=OFF .. ``` --- ### 3. **两者的关系** - **依赖优先级**: `-DDOWNLOAD_BOOST=ON` **必须** 配合 `-DENABLE_DOWNLOADS=ON` 使用。 若全局禁止下载(`ENABLE_DOWNLOADS=OFF`),即使指定下载Boost也会失败。 - **典型配置组合**: ```bash # 允许下载Boost,同时开启全局下载权限 cmake -DDOWNLOAD_BOOST=ON -DENABLE_DOWNLOADS=ON .. # 使用系统Boost,禁止所有下载 cmake -DDOWNLOAD_BOOST=OFF -DENABLE_DOWNLOADS=OFF .. ``` --- ### 4. **验证与调试建议** - 检查项目的 `CMakeLists.txt`,确认选项定义(如 `option(DOWNLOAD_BOOST "...")`)。 - 若下载失败,检查网络代理或尝试手动下载依赖并指定本地路径。 - 通过 `--trace` 或 `--debug-output` 参数查看CMake详细日志。 --- 总结:`-DDOWNLOAD_BOOST` 是 **针对Boost库的局部控制**,而 `-DENABLE_DOWNLOADS` 是 **全局下载开关**。两者需根据项目需求和环境配合使用。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值