_CRT_NONSTDC…与_CRT_SECURE…
目录
1.1 _CRT_NONSTDC_NO_WARNINGS 1
1.2 _CRT_NON_CONFORMING_SWPRINTFS 2
1.4 _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 4
1.5 _CRT_SECURE_CPP_OVERLOAD_SECURE_NAMES 5
第1章说明
1.1 _CRT_NONSTDC_NO_WARNINGS
使用VC++2008编译如下代码:
#include <windows.h> #include <tchar.h> #include <math.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { hypot(3,4); return 0; } |
将收到如下警告信息:
warning C4996: 'hypot': The POSIX name for this item is deprecated. Instead, use the ISO C++ conformant name: _hypot. See online help for details. |
错误原因:hypot函数是符合POSIX标准的函数,在VC++2008下这个标准已经过时,被ISO C++取代。在ISO C++标准下,该函数的名称为_hypot。
解决方法有两个:
方法一:把hypot更改为_hypot。如:#define hypot _hypot
方法二:定义宏_CRT_NONSTDC_NO_WARNINGS或_CRT_NONSTDC_NO_DEPRECATE。更改后的代码如下:
#define _CRT_NONSTDC_NO_WARNINGS #include <windows.h> #include <tchar.h> #include <math.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { hypot(3,4); return 0; } |
说明:
1、#define _CRT_NONSTDC_NO_WARNINGS必须在#include语句之前;
2、#define _CRT_NONSTDC_NO_DEPRECATE 与 #define _CRT_NONSTDC_NO_WARNINGS 完全等效。
1.2 _CRT_NON_CONFORMING_SWPRINTFS
使用VC++2008编译如下代码:
#include <windows.h> #include <tchar.h> #include <stdio.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { wchar_t wzStr[32]; swprintf(wzStr,L"%d",10); return 0; } |
将收到如下警告信息:
warning C4047: 'function' : 'size_t' differs in levels of indirection from 'unsigned short [3]' warning C4024: 'swprintf' : different types for formal and actual parameter 2 warning C4047: 'function' : 'const wchar_t *' differs in levels of indirection from 'int' warning C4024: 'swprintf' : different types for formal and actual parameter 3 |
错误原因:swprintf的声明被更改了!
下面是VC++6.0里swprintf的声明
int swprintf(wchar_t *buffer, const wchar_t *format,...);
下面是VC++2008里swprintf的声明
int swprintf(wchar_t *buffer,size_t count,const wchar_t *format,...);
可见后者多了一个参数size_t count。
解决方法有两个:
方法一:逐个修改swprintf函数。工作量大不说,遇到_stprintf函数时还得分两种情况更改代码。而且最重要的是更改之后VC++6.0就无法编译了;
方法二:定义宏 _CRT_NON_CONFORMING_SWPRINTFS,继续沿用老版本的swprintf。更改后的代码如下:
#define _CRT_NON_CONFORMING_SWPRINTFS #include <windows.h> #include <tchar.h> #include <stdio.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { wchar_t wzStr[32]; swprintf(wzStr,L"%d",10); return 0; } |
1.3 _CRT_SECURE_NO_WARNINGS
使用VC++2008编译如下代码:
#include <windows.h> #include <tchar.h> #include <stdio.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { char szBuf[10]; strcpy(szBuf,"test"); return 0; } |
将收到如下警告信息:
warning C4996: 'strcpy': This function or variable may be unsafe. Consider using strcpy_s instead. To disable deprecation, use _CRT_SECURE_NO_WARNINGS. See online help for details. |
错误原因:VC++2008认为strcpy函数不安全,建议您将其替换为strcpy_s函数。
解决方法有两个:
方法一:逐个修改strcpy为strcpy_s,工作量比较大;
方法二:定义宏 _CRT_SECURE_NO_WARNINGS或_CRT_SECURE_NO_DEPRECATE,继续使用不安全的strcpy函数。更改后的代码如下:
#define _CRT_SECURE_NO_WARNINGS #include <windows.h> #include <tchar.h> #include <stdio.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { char szBuf[10]; strcpy(szBuf,"test"); return 0; } |
1.4 _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES
上一节只是把strcpy的警告信息去除了,现在想把strcpy都替换为安全的strcpy_s函数,有什么快捷的方法?答案就是使用宏_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES和_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES_COUNT。
参考下面的代码:
#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1 #define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES_COUNT 1 #include <windows.h> #include <tchar.h> #include <stdio.h> #include <string.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { char szBuf[10]; strcpy(szBuf,"test"); strncpy(szBuf,"test",10); return 0; } |
说明:
1、因为定义了宏_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 为1,因此strcpy会被替换为strcpy_s(szBuf,sizeof(szBuf),"test");
2、因为定义了宏_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES_COUNT为1,因此strncpy(szBuf,"test",10);会被替换为strncpy_s(szBuf,sizeof(szBuf),"test",10);
3、这种替换方法只适用于静态数组,不适用于动态数组;
4、这两个宏只能用于C++代码,对于C代码是没有任何作用的。
1.5 _CRT_SECURE_CPP_OVERLOAD_SECURE_NAMES
参考下面的代码:
#include <windows.h> #include <tchar.h> #include <stdio.h> #include <string.h> int WINAPI _tWinMain(HINSTANCE hInstance ,HINSTANCE hPrevInstance,LPTSTR lpCmdLine,int nCmdShow) { char szBuf[10]; strcpy_s(szBuf,"test"); return 0; } |
strcpy_s需要三个参数,上面的代码里只有两个(程序员不熟悉strcpy_s,把它当成strcpy用)。为什么编译时没有提示错误?
因为宏_CRT_SECURE_CPP_OVERLOAD_SECURE_NAMES在默认情况下被定义为1,在这种情况下编译器会自动给strcpy_s添加缺失的参数,即修改strcpy_s(szBuf,"test");为strcpy_s(szBuf,sizeof(szBuf),"test");
说明:
1、这种替换方法只适用于静态数组,不适用于动态数组;
2、这个宏只能用于C++代码,对于C代码是没有任何作用的;
3、既然决定使用安全函数,最好还是把_CRT_SECURE_CPP_OVERLOAD_SECURE_NAMES定义为0,让编译器严格检查安全函数的各个参数。
_CRT_NONSTDC…与_CRT_SECURE…的更多相关文章
随机推荐
- Mysql-学习笔记(==》集合函数与分组四)
-- 聚集函数 配合分组语句 group by-- 显示最高分SELECT MAX(sscore) FROM db.`student`;-- 显示最高分学生的信息min maxSELECT * FRO ...
- 四个好看的CSS样式表格
文章来源 http://www.cnphp6.com/archives/58020 1. 单像素边框CSS表格 这是一个非经常常使用的表格样式. 源码: 2. 带背景图的CSS样式表格 和上面差点儿相 ...
- 多态,虚拟方法,重写,接口,类库,委托,is,as运算符,泛型集合,万能变量
多态:简而言之就是龙生九子,各有不同 有了继承,才有了多态 1.虚方法 virtual重写 override父类中的方法,在子类中并不适用,那么子类需要自主更改继承的方法或者是属性,那父类中加了vir ...
- MySQL中别名的使用
MySQL中别名的使用 为数据列或者表达式起别名时,别名紧跟数据列,中间以空格隔开,或者用关键字as隔开. #为表达式起别名 使用as关键字隔开 as T_ID from teacher_table; ...
- Codeforces 633B A Trivial Problem
B. A Trivial Problem time limit per test 2 seconds memory limit per test 256 megabytes input standar ...
- 二、java中的基本数据类型
总结: 1.java中的基本数据类型有byte.short.int.long;float.double;char;boolean. 2.基本数据类型与1相对应分别占1.2.4.8;4.8;2;1.(单 ...
- Java IO流整理Rick
Java 流IO部分: Console 控制台信息读取// ----------- Console Begin // ------------ 部分代码 Console cons ; char[] ...
- 曲线救国,解决Mac系统下,Android sdk下载失败的问题
Mac下翻_墙的问题 话说GFW屏蔽谷歌已经有一阵子了,最近打算在Mac系统下折腾个Android应用,备好了IDE,只欠SDK,无奈下载时因为GFW的缘故,总是失败,我心痛哉! 由于本人偏爱Mac系 ...
- 【转载】标准输入输出重定向(Visual C++)
原文:标准输入输出重定向(Visual C++) 引言 本人偶得在 Visual C++ 中进行输入输出重定向的办法,比通常的做法“freopen”更加的灵活和方便,特在此共享.目前,代码正在不断地摸 ...
- require或include相对路径多层嵌套引发的问题
require或include相对路径多层嵌套引发的问题 php中require/include 包含相对路径的解决办法 在PHP中require,include一个文件时,大都是用相对路径,是个 ...