boost之spirit学习-mini_c(2)

本文详细解析了代码文件的读取、语法解析、编译成字节码、查找并执行main函数的过程,包括使用spirit进行语法解析、跳过无效字符、调用虚拟机执行字节码等内容。

main.cpp就一个main函数,倒是直接明了

main函数大概的流程:

  • 读代码文件
  • 使用spirit将代码解析成抽象语法树
  • 使用compiler把语法树编译成字节码
  • 从编译后的字节中找出main函数
  • 为main函数传入参数(通过虚拟栈传递)
  • 使用VM执行字节码
  • 输出main函数的返回值

一点点把玩其代码:

1. 读"代码文件":

点击(此处)折叠或打开

  1.     std::string source_code; // We will read the contents here.
  2.     in.unsetf(std::ios::skipws); // No white space 
  3.     std::copy(
  4.         std::istream_iterator<char>(in),
  5.         std::istream_iterator<char>(),
  6.         std::back_inserter(source_code));

居然用copy、istream_iterator、back_inserter来读文件,文节青年的代码啊

像我等普通青年,估计就用二进制方式打开文件,判断一下文件长度,直接分析好内存,一次read搞定了

2. 解析语法树

点击(此处)折叠或打开

  1. bool success = phrase_parse(iter, end, +function, skipper, ast);

其中iter和end给出了解析的字符串的起始位置

+function表示解析的语法规则。注意其中加号是不能省的,它表示匹配function一次或多次。spirit重载了+号*号,以和BNF范式统一,不过由于C++语言的限制,它们只能放在规则之前,而不是像传统一样写在规则之后。

skipper表示使用的跳过语法规则。命中skipper的字符串部分将直接跳过,不参与解析。

后面的代码如下。比较直接,不再赘述

点击(此处)折叠或打开

  1.     if (success && iter == end)
  2.     {
  3.         if (compiler(ast))
  4.         {
  5.             boost::shared_ptr<client::code_gen::function>
  6.                 p = compiler.find_function("main");
  7.             if (!p)
  8.                 return 1;

  9.             int nargs = argc-2;
  10.             if (p->nargs() != nargs)
  11.             {
  12.                 std::cerr << "Error: main function requires " << p->nargs() << " arguments." << std::endl;
  13.                 std::cerr << nargs << "supplied." << std::endl;
  14.                 return 1;
  15.             }

  16.             std::cout << "Success\n";
  17.             std::cout << "-------------------------\n";
  18.             std::cout << "Assembler----------------\n\n";
  19.             compiler.print_assembler();

  20.             // Push the arguments into our stack
  21.             for (int i = 0; i < nargs; ++i)
  22.                 vm.get_stack()[i] = boost::lexical_cast<int>(argv[i+2]);

  23.             // Call the interpreter
  24.             int r = vm.execute(compiler.get_code());

  25.             std::cout << "-------------------------\n";
  26.             std::cout << "Result: " << r << std::endl;
  27.             std::cout << "-------------------------\n\n";
  28.         }
  29.         else
  30.         {
  31.             std::cout << "Compile failure\n";
  32.         }
  33.     }
  34.     else
  35.     {
  36.         std::cout << "Parse failure\n";
  37.     }

 

注意:编译后的字节码是放在compiler对象里的。但虚拟栈是由vm对象提供的。

有点奇怪function为什么用shared_ptr来返回,貌似完全没有share的必要。或者仅仅是为了避免少写compiler的析构函数?

另一个奇怪之处是从compiler获取了main函数的指针,但仅仅用来判断传入参数的个数。vm.execute并没有指定从main函数开始执行。那么程序是怎么跳到main函数去执行的呢?直觉猜想编译出的代码里起始位置应该有直接jump到main函数的代码。翻了一下compiler.cpp,果然!

点击(此处)折叠或打开

  1.     bool compiler::operator()(ast::function_list const& x)
  2.     { 
  3.         // Jump to the main function
  4.         code.push_back(op_jump);
  5.         code.push_back(0); // we will fill this in later when we finish compiling
  6.                            // and we know where the main function is

  7.         BOOST_FOREACH(ast::function const& f, x)
  8.         { 
  9.             if (!(*this)(f))
  10.             { 
  11.                 code.clear();
  12.                 return false;
  13.             } 
  14.         } 
  15.         // find the main function
  16.         boost::shared_ptr<code_gen::function> p = 
  17.             find_function("main");

  18.         if (!p) // main function not found
  19.         { 
  20.             std::cerr << "Error: main function not defined" << std::endl;
  21.             return false;
  22.         } 
  23.         code[1] = p->get_address()-1; // jump to this (main function) address

  24.         return true;
  25.     }


其中又有一个boost的奇技淫巧的应用:BOOST_FOREACH,在<boost/foreach.hpp>中,用于简化遍历容器

不过C++11中已经从语言层面对此进行了支持,语法上略有不同:

点击(此处)折叠或打开

  1. int arr[] = {1, 2, 3, 4, 5};
  2. for (int& i : arr)
  3. {
  4.     cout << i << endl;
  5. }

main.cpp解析完了,明天看看ast.hpp

<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` 是 **全局下载开关**。两者需根据项目需求和环境配合使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值