gdb 正在跳过行

Posted

技术标签:

【中文标题】gdb 正在跳过行【英文标题】:gdb is jumping over lines 【发布时间】:2011-11-09 12:17:10 【问题描述】:

我对 gdb 的理解有些问题。

我有一个main函数,这个main函数是我自己写的。

main 中的一些行,调用库中的一些函数,我认为库名并不重要,但它是 tesseract-ocr。

我在 main 中调用函数的行,构造函数在这里:

choiceItr = new tesseract::ChoiceIterator(itr);

我在上面一行的 gdb 处放了一个断点,然后运行,当它在那一行停止时,我使用 step 命令进入函数。

这是被调用的库函数:

ChoiceIterator::ChoiceIterator(const ResultIterator& result_it) 
  ASSERT_HOST(result_it.it_->word() != NULL);
  tesseract_ = result_it.tesseract_;
  PAGE_RES_IT res_it(*result_it.it_);
  WERD_CHOICE* best_choice = res_it.word()->best_choice;
  BLOB_CHOICE_LIST_CLIST* choices = best_choice->blob_choices();
  if (choices != NULL) 
    BLOB_CHOICE_LIST_C_IT blob_choices_it(choices);
    for (int blob = 0; blob < result_it.blob_index_; ++blob)
      blob_choices_it.forward();
    choice_it_ = new BLOB_CHOICE_IT(blob_choices_it.data());
    choice_it_->mark_cycle_pt();
   else 
    choice_it_ = NULL;
  

然后我使用 gdb 的“next”命令进入函数。

这是我的 gdb 控制台:

Breakpoint 1, pixsOfOneWord (langString=0x8049e7c "klm", 
    imageString=0x8049e71 "paket2.tif", outputData=0x8049c7b, 
    datapathString=0x8049e6f ".") at deneme234.cpp:161
161 choiceItr = new tesseract::ChoiceIterator(itr);
(gdb) step
tesseract::ChoiceIterator::ChoiceIterator (this=0x819e6f0, result_it=...)
    at resultiterator.cpp:234
234     choice_it_ = new BLOB_CHOICE_IT(blob_choices_it.data());
(gdb) next
225   ASSERT_HOST(result_it.it_->word() != NULL);
(gdb) list
220     return it_->word()->box_word->BlobPosition(blob_index_) == SP_DROPCAP;
221   return false;
222 
223 
224 ChoiceIterator::ChoiceIterator(const ResultIterator& result_it) 
225   ASSERT_HOST(result_it.it_->word() != NULL);
226   tesseract_ = result_it.tesseract_;
227   PAGE_RES_IT res_it(*result_it.it_);
228   WERD_CHOICE* best_choice = res_it.word()->best_choice;
229   BLOB_CHOICE_LIST_CLIST* choices = best_choice->blob_choices();
(gdb) next
278   // namespace tesseract.
(gdb) next
226   tesseract_ = result_it.tesseract_;
(gdb) next
278   // namespace tesseract.
(gdb) next
226   tesseract_ = result_it.tesseract_;
(gdb) next
230   if (choices != NULL) 
(gdb) 

如你所见,

tesseract_ = result_it.tesseract_;

line 被调用了两次,为什么?

PAGE_RES_IT res_it(*result_it.it_);
WERD_CHOICE* best_choice = res_it.word()->best_choice;
BLOB_CHOICE_LIST_CLIST* choices = best_choice->blob_choices();

当我“下一个”时,上面的行也没有被调用,为什么?

谢谢。

【问题讨论】:

您是否在编译时使用调试选项而不进行优化? @VJo,感谢您的回复,我必须承认我没有编译库的(tesseract)源代码,我使用的是预编译包。当我看到我可以进入 tesseract 函数时,我认为编译没问题。我正在编译我自己的源代码,其中包含只有“-g”的 main。我将搜索编译 main 而不进行优化。但是你能说,我应该在没有优化和调试选项的情况下编译 tesseract 吗?如果是这样,使用哪些调试选项,您的意思是“-g”吗?提前致谢 如果您使用的是 g++,则传递 -g -O0 以获得带有调试符号的未优化代码 【参考方案1】:

您要进入的库已使用优化和调试符号构建(很可能是-g -O2,这是 Linux 构建的默认值)。

调试优化代码有点困难,因为控制流优化会导致代码“跳来跳去”,一些变量变成“&lt;optimized out&gt;”等等。

您可以使用CXXFLAGS = -g -O0 重新构建库,或者您可以学习使用优化进行调试。

后者是一项非常有用的技能,因为很多时候您的程序只会在优化模式下崩溃,并且无论如何您都必须在该模式下对其进行调试。

【讨论】:

您好,感谢您的回复,我将库代码目录根目录中的 Makefile 更改为 O2 为 O0,卸载旧的。之后运行autoconf,配置,制作,再次安装。但是 gdb 仍然像上面的脚本一样运行。 "我更改了库代码目录根目录下的 Makefile"。是什么让你相信这就足够了?再次重新构建它,记录构建执行的所有命令,并验证resultiterator.cpp 实际上是用-O0 而不是-O2 构建的。如果它-O0构建的,那么你没有使用你刚刚构建的库。 谢谢,@Employed 俄语。我导出 CFLAGS=-g -O0 CXXFLAGS=-g -O0 并运行autoconf、配置、清理、安装。现在gdb按顺序逐行运行所有代码,现在我还有其他关于特别是tesseract的问题,也许是另一个问题。谢谢。 是的,这是 g++ 的问题,而不是 gdb/ddd 的问题。它不仅使 gdb 跳过行,还会产生错误的输出,因为它有时会错误地优化内容。我在使用 gcc-5.4.0 时遇到了 c++11 的问题,因为赋值是构造函数中的最后一行 - 当使用 -O3 时,该行只是被跳过了,但是让它 -O0 使代码正常工作. @R71 虽然编译器错误(产生错误代码)并非闻所未闻,但您不太可能在已发布的 GCC 版本中发现错误。您的代码更有可能具有未定义的行为,这允许编译器执行它想要的任何操作(包括完全跳过部分代码)。

以上是关于gdb 正在跳过行的主要内容,如果未能解决你的问题,请参考以下文章

如何在 ddd (gdb) 中跳过过去的循环

如何从mysql中的表中跳过行

在 csv 导入 pandas 期间跳过行

使用跳过行在 Spark 中读取 csv

GDB主要命令

Python Pandas,读取文件并在标题前跳过行