关注、星标公众,不错过精彩内容

转自:C语言与CPP编程

1 什么是防御性编程?

顾名思义,防御性编程是一种细致、谨慎的编程方法。为了开发可靠的软件,我们要设计系统中的每个组件,以使其尽可能的”保护”自己。我们通过明确地在代码中对设想进行检查,这是一种努力,防止我们的代码以将会展现错误行为的方式被调用。

防御性编程使我们可以尽早的发现较小的问题,而不是等到它们发展成大的灾难的时候才发现。其开发软件的过程是:

下面总结了一些防御性编程的反对和支持者的意见:

反对者:

  1. 它降低了代码的效;即使是一个很小的额外代码也需要一些额外的执行时间。它对于一个函数来说也许不要紧,但是对于一个由10万个函数组成的系统,问题就变得严重了。

  2. 每种防御性的做法都需要一些额外的工作;

支持者:

  1. 防御性编程可以节省大量的调试时间,使你可以去做更有意义的事情。

  2. 编写可以正常运行、只是速度有些慢的代码,要远远好过大多数时间都正常运行、但是有时候会崩溃的代码。

  3. 防御性编程避免了大量的安全性问题。

防御性编程技巧

  1. 使用好的编码风格和合理的设计

采用良好的编码风格,来防范大多数编码错误。如:

  • const关键字:

关键字const可以给读你代码的人传达非常有用的信息。例如,在函数的形参前添加const关键字意味着这个参数在函数体内不会被修改,属于输入参数。

同时,合理地使用关键字const可以使编译器很自然的保护那些不希望被修改的参数,防止其被无意的代码修改,减少bug的出现。

  • volatile关键字:

在一些并行设备的硬件寄存器(如状态寄存器),中断服务子程序中会访问到的全局变量以及多线程应用中被几个任务共享的变量前使用volatile关键字来防止编译优化。

  • static关键字:

  1. 函数体内static变量的作用范围为该函数体,不同于auto变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值。

  2. 在模块内的static全局变量可以被模块内的所有函数访问,但不能被模块外其它函数访问。

  3. 在模块内的static函数只可能被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内。

  • 位操作运算中,尽可能使用<<、 >>、 &、|等运算符,尽可能少使用/、%、*运算符。

  • 变量和函数的命名要有意义,并且尽可能做到一个函数只做一件事情。

  • 多采用面向对象的思想来编写代码。

  • 在投入到编码工作之前,先考虑大体的设计方案,这也非常关键。

  1. 不要仓促的编写代码

欲速则不达,每敲一个字,都要想清楚你要输入的是什么。在写每一行时都三思而后行。可能会出现什么样的错误?你是否已经考虑了所有可能出现的逻辑分支?放慢速度,有条不紊的编程虽然看上去很平凡,但这的确是减少缺陷的好办法。

如C语言编程中,追求速度的程序员经常会出现的一个问题就是将”==”错误的输入为”=”,而有些编译器并不会警告,这就会造成问题。

  1. 不要相信任何人

这里是指用怀疑的眼光来审视所有的输入和所有的结果,直到你能证明这段代码是正确的时候为止。

  1. 编码的目标要清晰,而不是简洁

简单是一种美,不要让你的代码过于复杂。即编写的代码一定要逻辑清晰,可读性强。

  1. 编译时打开所有警告开关

在你的代码中产生任何警告信息,都应立即修正代码。要知道警告的出现总是有原因的。即使你认为某个警告无关紧要,也不要置之不理。

  1. 使用安全的数据结构

我们最常见的一些安全隐患大概是由缓冲溢出引起的。缓冲溢出是由于不正确的使用固定大小的数据结构而造成的。例如,如下这个代码:

char * unsafe_copy(const char * source)
{char *buffer = new char[10];strcpy(buffer,source);return buffer;
}

如果source中的数据长度超过10个字符,它就会造成其它问题。我们可以改成如下形式:

char * safe_copy(const char * source)
{char *buffer = new char[10];strncpy(buffer,source,10);  //用strncpy代替strcpy可以保护这个代码段return buffer;
}
  1. 检查所有的返回值

如果一个函数返回一个值,他这样做肯定是有理由的。检查这个返回值,如果返回值是一个错误代码,你就必须辨别这个代码并处理所有的错误。不要让错误悄无声息的侵入你的程序;大多数难以察觉的错误都是因为程序员没有检查返回值而出现的。

  1. 审慎的处理内存

对于在执行期间所获取的任何资源,必须彻底释放。

  1. 在声明位置初始化所有变量

如果你意外的使用了一个没有初始化的变量,那么你的程序在每次运行的时候都将得到不同的结果,这取决于当时内存中的垃圾信息是什么。这样会造成很多随机的行为,给查找带来很多的麻烦。因此,需要在声明每个变量的时候就对它进行初始化。

  1. 同时,平时编码时还要注意一些细则

  • 提供默认的行为:Switch语句中将default case的执行明示出来。同样地,如果你要编写一些不带else子句的if语句,停下来想一想,你是否该处理这个逻辑上的默认情况

  • 检查数值的上下限:确保每次运算数值变量都不会溢出,即数据类型的使用要谨慎

  • 注意强制转换是否合理

  • 声明变量,可以使变量的声明位置与使用它的位置尽量接近,从而防止它干扰代码的其他部分

  • 加合理的异常处理、日志文件

  • 正确设置常量

优秀的程序应该做到:

  • 关心代码是否健壮

  • 确保每个设想都显示地体现在防御性代码中

  • 希望代码对无用信息的输入有正确的行为

  • 在编程的时候认真思考自己所编写的代码

  • 编写可以保护自己不受其他人的愚蠢伤害的代码。

免责声明:本文来源网络,版权归原作者所有。如涉及作品版权问题,请与我联系删除。

推荐阅读:

树莓派基于 Linux 的 Windows XP

嵌入式软件测试的10条秘诀

如何减少编码、测试和调试三个阶段的bug?

关注微信公众号『strongerHuang』,后台回复“1024”查看更多内容,回复“加群”按规则加入技术交流群。

长按前往图中包含的公众号关注

代码防御性编程的十条技巧相关推荐

  1. java防御性编程_代码防御性编程的十条技巧

    1 什么是防御性编程? 顾名思义,防御性编程是一种细致.谨慎的编程方法.为了开发可靠的软件,我们要设计系统中的每个组件,以使其尽可能的"保护"自己.我们通过明确地在代码中对设想进行 ...

  2. 追求代码质量: 用 AOP 进行防御性编程

    原文出处: IBM中国 开发人员测试的主要缺点是:绝大部分测试都是在理想的场景中进行的.在这些情况下并不会出现缺陷 -- 能导致出现问题的往往是那些边界情况. 什么是边界情况呢?比方说,把 null  ...

  3. 细节决定成败,聊聊防御性编程

    点击上方"小麦大叔",选择"置顶/星标公众号" 福利干货,第一时间送达 什么是防御性编程? 防御性编程是一种细致.谨慎的编程方法.为了开发可靠的软件,我们要设计 ...

  4. 才从Guava中明白了一件事:不可变特性与防御性编程

    一.面试常谈:String类与不可变特性 问:String类是可变的吗? 答:emm--由于String类的底层是final关键字修饰,因此它是不可变的. 问:它被设计为不可变的好处有哪些呢? 答: ...

  5. 如何写出优质干净的代码,这6个技巧你不能错过!

    编写干净的代码并不是一件容易的事情,这需要尝试不同的技巧和实践. 作为一名开发者,编写一手干净的代码很重要. 先列举出编写干净代码的一些好处,再提出6个技巧用于编写干净代码,供开发者进行参考学习. 开 ...

  6. 2021-01-20 Python编程特殊小技巧汇集

    Python编程特殊小技巧汇集 Python作为一种高级编辑语言,有很多使用的小技巧,分享一期. 1.变量值互换 a = 0b = 1a,b = b, a 2.连续赋值 a, b = 2, 1 3.自 ...

  7. 常见的8个前端防御性编程方案

    大家好,我是若川.欢迎加我微信 ruochuan12,长期交流学习.今天分享几个常见的前端防御性编程方案.点击下方卡片关注我,或者查看源码等系列文章.学习源码整体架构系列.年度总结.JS基础系列 关于 ...

  8. JDBC 防御性编程

    防御性编程(Defensive Programming) 什么是Defensive Programming[1]? 原文:Defensive programming is a form of defe ...

  9. 用Vim编程——配置与技巧

    本文介绍如何用强大的文本编辑器 Vim编程.只要配置得当,效果堪比一个IDE. 配置 基本配置: "开启语法高亮 syntax on "依文件类型设置自动缩进 filetype i ...

最新文章

  1. Linux的文件系统及其硬盘分区挂载原理
  2. poj 2063 Investment(01背包变形)
  3. WIFI-MESH + 蓝牙MESH在智能家居领域有着广泛的应用场景,他们的区别到底在哪里呢
  4. 混合云模式助力斗鱼搭建混搭大数据架构
  5. 携带token的ajax请求方法封装
  6. HTML5在线编辑器Wysihtml5之Bootstrap扩展
  7. 使用IDEA创建Maven项目教程
  8. HCIE Security 防火墙反病毒 备考笔记(幕布)
  9. ArcMap水文分析系列教程
  10. 深度IP转换器安卓版APP怎么修改OPPO手机IP地址
  11. Protel转PADS
  12. java两级缓存框架J2cache
  13. 百度搜索引擎的工作原理
  14. 机器学习二:K均值聚类算法(k-means clustering algorithm)
  15. 用FPGA产生正弦信号
  16. Centos 安装FTP配置目录权限,iptables设置ftp服务
  17. 10个有关快门的必学知识
  18. 摄像头各参数的意义_手机摄像头的一些主要参数表达了什么意思?
  19. qmail邮件系统(五)vpopmail和qmailadmin对用户的管理
  20. Emacs之缩进(五十三)

热门文章

  1. DTT的生活就是对吃的一种细细品味
  2. unity3d学习笔记(一)方向键移动物体
  3. 全面解析枚举+联合体~
  4. caffe 中的损失函数分析
  5. pandas读取csv文件UnicodeDecodeError: ‘utf-8‘ codec can‘t decode byte 0xca in position 37: invalid contin
  6. celebA数据集(StarGAN)分享
  7. @keyframes简单使用
  8. java生僻字解决方案
  9. 数字通信调制方式的误码率matlab仿真,包括OOK,PRK,FSK以及QAM
  10. 资源管理Placement部署(Nova)