草庐IT

c++ - 发布和调试编译程序的源代码是否有所不同? [C/C++]

coder 2024-02-20 原文

到目前为止,我已经对C++编程有了更多的了解,并一直运行到整个“调试与发行”编译版本。现在,我觉得我对已发布和调试版本的已编译代码之间的某些差异有了相当不错的了解。对于代码的调试版本,编译器不会尝试优化代码,以便您可以运行调试器并逐行浏览程序。本质上,编译后的代码在执行方式上与源代码非常相似。在 Release模式下编译时,编译器会尝试优化程序,使其具有相同的功能,但效率更高。

但是,我很好奇发行版本和调试版本之间的源代码是否可以不同的实例。也就是说,当我们提到调试与发布时,我们是否总是在谈论编译后的代码,或者源代码中是否存在差异?

出现此问题是由于我使用的是专有的编程语言,在该语言中,不存在正式的逐步调试器,但确实存在串行监视器。因此,我们的许多“调试”与“发布”代码是通过#defines实现的,它们看起来像这样:

#ifdef _DEBUG

    check that error didn't occur...

    SerialPrint("Error occurred")

#endif

因此,总结一下我的问题,取决于您的IDE,通常是否有一些设置可以实现我所说明的内容?也就是说,当您尝试编译为调试版本时,它可以与源代码中的更改集成吗?还是发布vs调试通常只引用编译的二进制文件?

谢谢!

最佳答案

Is there a difference in source code for release and debug compiled program?



它取决于源代码以及用于编译库或程序的选项。以下是我知道的一些差异。

ASSERTS

最简单的“调试和诊断”是assert。当未定义NDEBUG时,它们有效。断言创建自我调试代码,并且在遇到意外情况时会捕捉。诀窍是您必须声明所有内容。在验证参数和状态的任何地方,都应该看到一个断言。到处都有一个断言,您应该看到一个if来验证参数和状态。

当我看到没有断言的代码库时,我会笑。我对自己说,如果开发人员在调试器中浪费时间,他们的时间就太多了。我经常问为什么你不使用断言,而它们通常会回答以下问题...

Posix assert很烂,因为它调用abort。如果要调试程序,则通常需要逐步执行代码,以查看代码如何处理导致assert触发的负面条件。终止程序的目的在于“调试和诊断”。它必须是C/C++历史上最愚蠢的决定之一。似乎没有人记得中止的原因(几年前,我试图在各种C/C++标准列表中追踪血统书)。

通常,您将无用的Posix断言替换为更有用的内容,例如assert在Linux上引发SIGTRAP或在Windows上调用DebugBreak。例如,参见示例 trap.h 。您用断言替换Posix assert,以确保您正在使用的库获得更新的行为(如果它们已被编译,则为时已晚)。

当诸如ISC的BIND(为Internet提供动力的DNS服务器),DoS本身及其断言(它们有自己的断言;它们不使用Posix断言)之类的项目时,我也会笑。有许多CVE反对BIND自行实现的DoS。 DoS在自己身边,“让中止正在调试的程序”。

为了完整起见,Microsoft基础类(MFC)曾经有大约16,000或20,000个断言来帮助及早发现错误。那是在1990年代末或2000年代中期。我不是今天的状态。

API

存在一些专门为“调试和诊断”而构建的API。即使不一定在生产中使用它们,也可以使用其他API。

前者(故意构建)的一个示例是Logging和DebugPrint API。苹果成功地使用它来导出用户的FileVault密码和 key 。另请参阅os x filevault debug print

后者的一个示例(不安全生产)是Windows IsBadReadPointer IsBadWritePointer 。它的生产不安全,因为它遇到了竞争状况。但是它通常对开发很好,因为您需要额外的检查。

当我们执行安全检查和审核时,我们通常会要求/建议删除所有不必要的日志记录;并确保无法在运行时更改日志记录级别。当应用正式投入生产时,调试时间就结束了。没有理由记录所有内容。



有时有一些特殊的库可用于帮助调试诊断程序。我想到了Linux的Electric Fence和Microsoft的CRT Library。两者都是带有API的内存检查器。在这种情况下,您的链接命令也将有所不同。

选项

有时,您需要其他选项或定义来帮助进行调试和诊断。我想到了Glibc++和-D_GLIBCXX_DEBUG。另一个是概念检查,它曾经由define -D_GLIBCXX_CONCEPT_CHECKS启用。它的Boost代码及其损坏,因此您不应使用它。在这些情况下,您的编译标志将有所不同。

我经常 mock 的另一个版本是缺少NDEBUG定义的Release版本。出于政策考虑,其中包括Debian和Ubuntu。 NSA,GHCQ和其他3个字母的代理机构感谢他们获取敏感信息(如服务器 key ),剥离加密(将其写入不 protected 文件中),然后输出敏感信息(将它们发送给Windows错误报告,分配错误)报告等)。

初始化

当未明确初始化值时,某些开发环境会使用特殊的位模式执行初始化。它实际上只是工具(例如编译器或链接器)的功能。我想到了微软的工具。看到When and why will an OS initialise memory to 0xCD, 0xDD, etc. on malloc/free/new/delete? GCC对此有功能要求,但是我认为它还没有做任何事情。

当我拆卸生产DLL并看到Microsoft调试位模式时,我常常会笑,因为我知道它们正在交付调试DLL。我之所以大笑是因为它通常表明Release DLL出现了开发团队无法清除的内存错误。 Adobe为此而臭名昭著(即使它们不提供Apple或Microsoft这样的操作系统,也并不奇怪Adobe supplies some of the most insecure software on the planet)。
#ifdef _DEBUG

    check that error didn't occur...

    SerialPrint("Error occurred")

#endif

这让我想哭,但是您仍然必须在2016年这样做。GDB在(Aarch64),X32和S/390下损坏了(是?),因此您必须使用printf来调试代码。

关于c++ - 发布和调试编译程序的源代码是否有所不同? [C/C++],我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39158125/

有关c++ - 发布和调试编译程序的源代码是否有所不同? [C/C++]的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  4. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  5. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  6. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  7. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  8. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  9. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  10. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

随机推荐