如何提升代码的安全性 —— 代码防御性编程的十条技巧
目录
- 一、什么是防御性编程?
- 二、防御性编程技巧
- 2.1 使用好的编码风格和合理的设计
- 2.2 不要仓促的编写代码
- 2.3 不要相信任何人
- 2.4 编码的目标要清晰,而不是简洁
- 2.5 编译时打开所有警告开关
- 2.6 使用安全的数据结构
- 2.7 检查所有的返回值
- 2.8 审慎的处理内存
- 2.9 在声明位置初始化所有变量
- 2.10 优秀的程序应该做到
- 参考
一、什么是防御性编程?
顾名思义,防御性编程是一种细致、谨慎的编程方法。为了开发可靠的软件,我们要设计系统中的每个组件,以使其尽可能的”保护”自己。我们通过明确地在代码中对设想进行检查,这是一种努力,防止我们的代码以将会展现错误行为的方式被调用。
防御性编程使我们可以尽早的发现较小的问题,而不是等到它们发展成大的灾难的时候才发现。其开发软件的过程是:
下面总结了一些防御性编程的反对和支持者的意见:
- 反对者:
- 它降低了代码的效;即使是一个很小的额外代码也需要一些额外的执行时间。它对于一个函数来说也许不要紧,但是对于一个由10万个函数组成的系统,问题就变得严重了。
- 每种防御性的做法都需要一些额外的工作;
- 支持者:
- 防御性编程可以节省大量的调试时间,使你可以去做更有意义的事情。
- 编写可以正常运行、只是速度有些慢的代码,要远远好过大多数时间都正常运行、但是有时候会崩溃的代码。
- 防御性编程避免了大量的安全性问题。
二、防御性编程技巧
2.1 使用好的编码风格和合理的设计
采用良好的编码风格,来防范大多数编码错误。如:
const关键字:
关键字const可以给读你代码的人传达非常有用的信息。例如,在函数的形参前添加const关键字意味着这个参数在函数体内不会被修改,属于输入参数。
同时,合理地使用关键字const可以使编译器很自然的保护那些不希望被修改的参数,防止其被无意的代码修改,减少bug的出现。volatile关键字:
在一些并行设备的硬件寄存器(如状态寄存器),中断服务子程序中会访问到的全局变量以及多线程应用中被几个任务共享的变量前使用volatile关键字来防止编译优化。static关键字:
函数体内static变量的作用范围为该函数体,不同于auto变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值。
在模块内的static全局变量可以被模块内的所有函数访问,但不能被模块外其它函数访问。
在模块内的static函数只可能被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内。位操作运算中,尽可能使用
<<、 >>、 &、|
等运算符,尽可能少使用/、%、*
运算符。变量和函数的命名要有意义,并且尽可能做到一个函数只做一件事情。
多采用面向对象的思想来编写代码。
在投入到编码工作之前,先考虑大体的设计方案,这也非常关键。
2.2 不要仓促的编写代码
欲速则不达,每敲一个字,都要想清楚你要输入的是什么。在写每一行时都三思而后行。可能会出现什么样的错误?你是否已经考虑了所有可能出现的逻辑分支?放慢速度,有条不紊的编程虽然看上去很平凡,但这的确是减少缺陷的好办法。
如C语言编程中,追求速度的程序员经常会出现的一个问题就是将==
错误的输入为=
,而有些编译器并不会警告,这就会造成问题。
2.3 不要相信任何人
这里是指用怀疑的眼光来审视所有的输入和所有的结果,直到你能证明这段代码是正确的时候为止。
2.4 编码的目标要清晰,而不是简洁
简单是一种美,不要让你的代码过于复杂。即编写的代码一定要逻辑清晰,可读性强。
2.5 编译时打开所有警告开关
在你的代码中产生任何警告信息,都应立即修正代码。要知道警告的出现总是有原因的。即使你认为某个警告无关紧要,也不要置之不理。
2.6 使用安全的数据结构
我们最常见的一些安全隐患大概是由缓冲溢出引起的。缓冲溢出是由于不正确的使用固定大小的数据结构而造成的。例如,如下这个代码:
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代替strcpy可以保护这个代码段 strncpy(buffer,source, 10); return buffer;
}
2.7 检查所有的返回值
如果一个函数返回一个值,他这样做肯定是有理由的。检查这个返回值,如果返回值是一个错误代码,你就必须辨别这个代码并处理所有的错误。不要让错误悄无声息的侵入你的程序;大多数难以察觉的错误都是因为程序员没有检查返回值而出现的。
2.8 审慎的处理内存
对于在执行期间所获取的任何资源,必须彻底释放。
2.9 在声明位置初始化所有变量
如果你意外的使用了一个没有初始化的变量,那么你的程序在每次运行的时候都将得到不同的结果,这取决于当时内存中的垃圾信息是什么。这样会造成很多随机的行为,给查找带来很多的麻烦。因此,需要在声明每个变量的时候就对它进行初始化。
同时,平时编码时还要注意一些细则
- 提供默认的行为:
Switch语句中将default case
的执行明示出来。同样地,如果你- 要编写一些不带else
子句的if
语句,停下来想一想,你是否该处理这个逻辑上的默认情况 - 检查数值的上下限:确保每次运算数值变量都不会溢出,即数据类型的使用要谨慎
- 注意强制转换是否合理
- 声明变量,可以使变量的声明位置与使用它的位置尽量接近,从而防止它干扰代码的其他部分
- 加合理的异常处理、日志文件
- 正确设置常量
2.10 优秀的程序应该做到
- 关心代码是否健壮
- 确保每个设想都显示地体现在防御性代码中
- 希望代码对无用信息的输入有正确的行为
- 在编程的时候认真思考自己所编写的代码
- 编写可以保护自己不受其他人的愚蠢伤害的代码。
参考
代码防御性编程的十条技巧
如何提升代码的安全性 —— 代码防御性编程的十条技巧相关推荐
- java防御性编程_代码防御性编程的十条技巧
1 什么是防御性编程? 顾名思义,防御性编程是一种细致.谨慎的编程方法.为了开发可靠的软件,我们要设计系统中的每个组件,以使其尽可能的"保护"自己.我们通过明确地在代码中对设想进行 ...
- 代码防御性编程的十条技巧
关注.星标公众号,不错过精彩内容 转自:C语言与CPP编程 1 什么是防御性编程? 顾名思义,防御性编程是一种细致.谨慎的编程方法.为了开发可靠的软件,我们要设计系统中的每个组件,以使其尽可能的&qu ...
- 追求代码质量: 用 AOP 进行防御性编程
原文出处: IBM中国 开发人员测试的主要缺点是:绝大部分测试都是在理想的场景中进行的.在这些情况下并不会出现缺陷 -- 能导致出现问题的往往是那些边界情况. 什么是边界情况呢?比方说,把 null ...
- 市场上有很多低代码开发平台,不懂编程的人可以用哪些?
市场上有很多低代码开发平台,不懂编程的人可以用哪些?这个问题一看就是外行问的啦,低代码平台主打的就是一个"全民开发",而且现在很多低代码平台都发展为零代码了,不懂编程也完全可以使用 ...
- java编程代码大全_掌握Java编程技巧,代码重构
代码重构在不改变软件系统外部行为的前提下,改善它的内部结构,通过调整程序代码改善软件的质量.性能,使其程序的设计模式和架构更趋合理,提高软件的扩展性和维护性. 代码重构目标 持续纠偏和改进软件设计 随 ...
- 我见过的最漂亮代码---通过删除代码来实现功能的提升
我曾经听一位大师级的程序员这样称赞到,"我通过删除代码来实现功能的提升."而法国著名作家兼飞行家Antoine de Saint-Exupéry的说法则更具代表性,"只有 ...
- soul刷屏编程代码_奔涌吧,编程!少儿编程教育在未来会像语文,数学一样重要!...
想必最近你的朋友圈一定被<奔涌吧,后浪>刷屏了,的确,时代在变好,我们能够更自由的学习,读书,很多孩子在年轻时就已经接触到许多的兴趣活动,他们早早的就在发展一项"事业" ...
- 让你最快速地改善代码质量的 20 条编程规范
根据学习部分极客时间 <设计模式之美>专栏 (王争 前Google工程师)和<阿里 java 规范>整理总结. 分别介绍编码规范的三个部分:命名与注释(Naming and C ...
- 超硬核!!!一篇文章搞定TCP、UDP、Socket、HTTP(详细网络编程内容+现实解释三次握手四次挥手+代码示例)【网络编程 1】
TCP.UDP.Socket 一天面试的经验: 什么是网络编程 网络编程中两个主要的问题 网络协议是什么 为什么要对网络协议分层 计算机网络体系结构 1 TCP / UDP 1.1 什么是TCP/IP ...
- 在线 OJ 项目(三) · 处理项目异常 · UUID · 校验代码的安全性 · 阶段性总结
一.处理异常 二.区分不同请求的工作目录 UUID 对 Task 类进行重构 三.校验代码的安全性 四.阶段性总结 书接上回,我们自己测试没问题,是因为使用了正常数据:万一用户输入的是非法的请求,该咋 ...
最新文章
- IBM磁带库中更换磁带的步骤
- 自然语言处理-文本分析学习记录
- Python高手必读,做一个精通规则的玩家
- boost::math模块实现图表显示使用 Lambert W 函数计算电流的测试程序
- Python基础教程学习笔记:第二章 列表和元组
- 刚刚出炉的Asp.net网站部署视频教程
- 程序自动分析(洛谷-P1955)
- mmu计算机组成原理,计算机组成原理
- linux内实践核分析模块
- 关于解决Server Tomcat v9.0 Server at localhost failed to start的问题
- iOS--UIView和UIWindow用法
- 谷歌方法 pdf txt azw3 epub mobi 百度云盘网盘下载
- Buck-Boost电路
- php中ci钩子,CodeIgniter钩子用法
- [0CTF 2016]piapiapia 1
- 阿里云IOT入门教程(三)阿里云IOT Studio自建手机App控制Wemos D1 Mini( ESP8266 )板载灯亮灭
- 07 仿网易严选微信小程序商城
- 芯擎科技正式发布“龍鹰一号”,引领“中国智造”智能座舱芯片新篇章
- 计算机英文收集(二)
- java经纬度凸包graham_凸包算法(Graham扫描法)详解
热门文章
- 几款软件需求分析工具
- 小程序--模板的使用 说明--详细版的
- 项目中用到的ws2811炫彩灯控制程序
- TMS320C6678+Kintex-7开发板——DSP程序固化操作手册
- 当前安装包签名出现异常_安卓系统手机安装应用出现应用签名异常或-22错误(联网验证失败)的应对方法...
- 乐视电视安装鸿蒙系统,乐视电视更新后无法识别apk文件怎么办?方法教程
- java1.5_Java15下载 JDK15(Java SE Development Kit 15) 15.0.2 官方正式版 Win64位 下载-脚本之家...
- 内存取证-volatility工具的使用
- 解决FireFox(火狐浏览器)占用资…
- mysql 分隔字符串的函数_Mysql 字符串分隔函数