• CLion 的 Debug 模式是怎么回事


    我思考这个问题的起因是 Codeoforces Avito Challenges 2018 的 E 题,我想到了正解,但写得太慢,最后一刻才提交。有个地方写错,结果是 Runtime error on pretest 13。
    那个错误是个数组越界。代码片段如下:

        #define rng(i, a, b) for(int i = (a); i < (b); ++i)
    
        vv<pii> fact(2e5+5);
    
        // 计算 fact
        
        
        vv<bool> dp(m, vb(ma));    // m 和 ma 是正整数
    
        vv<int> pre(m, vi(ma));
    
        rng (i, 0, SZ(fact[a[0]])) {
            dp[0][i] = true;
        }
    
        rng(i, 0, m) {  // 错误出在这里
            auto &cur = fact[a[i]];
            auto &nxt = fact[a[i + 1]]; 
            rng(j, 0, SZ(fact[a[i]])) {
                if(dp[i][j]) {
                    int t = cur[j].second;
                    for (int k = 0; k < SZ(nxt); k++) {
                        if(t < nxt[k].first) {
                            dp[i+1][k] = true;
                            pre[i+1][k] = j;
                        }
                    }
                }
            }
        }
    
    

    那个 for 循环本应是 rng(i, 0, m-1)。我的写法通过了三个样例,交上去在 pretest 13 RE 了。仓促改了另外一个地方又交上去结果在 pretest 2 就 RE 了。数组越界这个问题什么时候会触发 RE 是不确定的。

    赛后我想起曾经看到过 std::vector 会自动检查数组越界,可是我在 CLion 中编译运行上面的代码为何运行时没提示数组越界呢,况且 CLion 的 Run/Debug Configurations 还显示处于 Debug 模式。


    -1

    图中红线框出的那里,左边的 algo 是当前选择的 target 的名字,target 是在 CMakeLists.txt 中定义的。红框圈起来的东西,官方名称 是 run/debug configuration selector。


    -2

    于是我搜了一下怎样开启数组越界检查。在 StackOverflow 上找到一个答案。办法就是编译时加上选项 _GLIBCXX_DEBUG 其实这是个宏定义,不必作为 g++ 的命令行选项, 在代码里加上 #define _GLIBCXX_DEBUG 也可以,和其他宏定义没区别,只不过会被编译器识别,并对编译器产生影响。

    我加了这个宏定义再编译运行原来的代码,果然就报错了。不过并没有指明错误是由哪一行代码引起的,对于 debug 帮助不大,这个问题请移步这里。不过虽然未能指出是那一行代码出的错,但是指出了是访问 3 这个位置越界了,那么就说明对应的 vector 的 size 应该是 3,这个信息对于这道题目的 debug 仍然很有帮助,据此可以知道这个 3 是 m 的值。


    -3

    我的疑问是 CLion 的 Build/Debug Configurations 框里的那个 Debug 到底是什么含义(What does it imply?)先看看这个 Debug 究竟是什么含义。

    首先要介绍 CLion 中的一个重要概念 Run/Debug Configrations

    To run or debug your code in CLion, you can use numerous run/debug configurations. Each run/debug configuration represents a named set of run/debug startup properties. When you perform run, debug, or test operations with CLion, you always start a process based on one of the existing configurations using its parameters.

    CLion comes with a number of run/debug configuration types for the various running, debugging and testing issues. You can create your own run/debug configurations of specific types.

    关于 Run/Debug Configurations 的更多细节,见 Create and Edit Configurations

    其实 -1 红框里的那个 Debug 是当前所选择的 configuration 所采用的 CMake profile 的名字。可以在下图中的加号(+)上面看到 “profiles” 字样。

    **图**-4

    上图是 CMake settings dialog。

    当然这个 name 是可以改的,比如改成 debug,那么第一幅图红框里也会相应地由 Debug 变成 debug。

    现在我们收集到的信息有

    • -1 中的 Debug 字样代表的是当前的 configuration 所采用的 CMake profile。
    • 这个 CMake profile 名为 "Debug" 是因为所选的 build type 是 Debug。
    • CMake profile 中的 build options 是传给 build tool 的。在我采用的 toolchain 里,build tool 是 make。

    现在我们要问:当 build type 为 Debug 时,是否会把类似于 _GLIBCXX_DEBUG 这样的选项传递给编译器呢?

    答案是不会。

    我们可以从 CLion 文档中关于 build type 的页面上示例代码中推测出来


    -5

    关于 -4 中的 build option -j 6,CLion 文档是这样解释的:

    Build options In this text field, specify the options to be passed to the build tool used by CMake.
    Find more information about the available build options in the CMake documentation.

    If nothing is specified, the default settings are used. Note, that default settings depend on the selected environment. For example, if make generator is selected, the default value is -j <number_of_cpu>, while for Microsoft Visual C++ the field is empty.

    我采用的 toolchain 是 MinGW,所用的 generator 自然是 make,另外我的 laptop 的 CPU 是 Intel Core i7-8750H,# of Cores = 6,因此默认的 build option 是 -j 6

    在 CMake 文档里关于 CMAKE_BUILD_TYPE 这个 variable 的说明页面,我找到线索

    ... For example, in a build tree configured to build type Debug, CMake will see to having CMAKE_C_FLAGS_DEBUG settings get added to the CMAKE_C_FLAGS settings.

    我猜当 build type 是 Debug 时 CMake 只是把 -g 之类的调试选项传给编译器了。

    结论

    Run/Debug Configurations 是控制可执行文件生成以后的执行与调试的,而 Build Configurations 是控制从源代码生成可执行文件的过程的。

    Reference

    Catching silly mistakes with GCC
    C++ и bounds checking

  • 相关阅读:
    Apache Solr入门教程(初学者之旅)
    Codeforces 631 (Div. 2) E. Drazil Likes Heap 贪心
    Codeforces 631 (Div. 2) D. Dreamoon Likes Sequences 位运算^ 组合数 递推
    Codeforces 631 (Div. 2) C. Dreamoon Likes Coloring 思维or构造
    python中的类型转换
    MVC3.0在各个版本IIS中的部署
    get和post的区别
    Vue和React对比
    谈谈你对web标注和W3c的理解和认识
    js中的undefined 和null
  • 原文地址:https://www.cnblogs.com/Patt/p/10136138.html
Copyright © 2020-2023  润新知