一杯清酒邀明月
天下本无事,庸人扰之而烦耳。

在编写C++程序时,不可避免会遇到strcpy()函数和其安全版本strcpy_s()函数,其实之所以会推出_s版本的函数,就是为了使编程更加安全,然而为了保证安全,也就会更容易使我们编写的代码“被报错”。所以这里来简略说一下strcpy()函数和strcpy_s()函数的使用及注意事项。

首先,我们知道原函数strcpy()函数和安全版本strcpy_s()函数都是存在于头文件<cstring>中的,所以程序一开始必须要有以下语句:

#include <cstring>

其次,原函数strcpy()函数是存在于标准名称空间std中的成员,所以要使用strcpy()函数,还需要加上以下语句:

using namespace std;

或者:

using std::strcpy;

或者在每次使用strcpy()函数时,前面加上名称空间:

std::strcpy(str1, str2);

但是对于最新的编辑器,往往你正常使用strcpy()函数,还是会报错,例如下面这个简单的例子:

 1 // strcpy.cpp -- test the strcpy function and strcpy_s function
 2  
 3 #include "stdafx.h"
 4 #include <iostream>
 5 #include <cstring>
 6  
 7  
 8 int main()
 9 {
10     char str1[20];
11     char str2[20];
12     std::cout << "Please enter str2: ";
13     std::cin.get(str2, 20);
14     std::strcpy(str1, str2);
15     std::cout << "str1 is " << "\" " << str1 << "\".\n";
16     system("pause");
17     return 0;
18 }

正常来说,语法上没有任何问题,但是运行时,Visual Studio 2017 会报错,显示如下:

意义很简单,就是告诉你,strcpy()函数不安全,必须改为使用strcpy_s()函数,首先不管改成strcpy_s()函数之后会发生什么后续问题,其实从理论上来说,上面的代码语法上和逻辑上来说都是对的,那么怎么避免编辑器强制要求你使用安全版本呢?

其实解决方法有很多,单单是避免上图中的错误代码4996的情况,可以使用编辑器的选择性提供warning功能,在include语句前面加上下句:

#pragma warning( disable : 4996)

但是这种解决方法有时候不能解决问题,例如VS2017我就试过好像不行,那么我们就索性关闭warning功能就好了,完成这个任务的方法是在#include<stdio.h>的前面加上一句,如下所示:

1 #define _CRT_SECURE_NO_WARNINGS
2 #include <stdio.h>

VS2017里面,这句应该是加在头文件“stdafx.h”里面。

加完之后,再次运行相同的程序,就可以正常运行了,显示结果如下图所示:

那么解决了不使用安全版本的问题,接下来就来说一下使用安全版本的情况。

如果我们选择相信编辑器,那么我们就会把strcpy()函数改成使用strcpy_s()函数,改完之后运行首先会遇到以下错误:

这是因为之前使用strcpy()函数时,我们知道该函数是标准名称空间std的成员,而安全版本strcpy_s并不是该名称空间的成员,所以“std::”应该被去掉,去掉之后再次运行,就会正常运行了。

但是有时候,编辑器在你去掉“std::”还是会出现错误,错误说明是:1.没有与参数列表匹配的 重载函数"strcpy_s"实例;2."strcpy_s":函数不接受2个参数。

如下图所示:

这是因为strcpy_s()函数是有两个版本,用两个参数、三个参数都可以,只要可以保证缓冲区大小。
三个参数时:

1 errno_t strcpy_s( 
2 char *strDestination, 
3 size_t numberOfElements, 
4 const char *strSource 
5 );

两个参数时:

1 errno_t strcpy_s( 
2 char (&strDestination)[size], 
3 const char *strSource 
4 ); // C++ only 

所以,若我们使用new来分配储存空间时,就会出现上面说的不能保证缓冲区大小的问题了。

看下面的代码:

1 char * str;
2 str = new char[4];
3 strcpy_s(str, "C++");

语法来说没有什么问题,但是因为str的储存空间是使用new临时分配的,所以并不能保证缓冲区大小,点击运行就会出现上述的两种错误了。

这种情况的解决方法其实很简单,那就是不符合2个参数的版本就使用3个参数的版本呗。在两个str之间,加上一个参数,标识长度。

所以完整代码如下:

 1 // strcpy.cpp -- test the strcpy function and strcpy_s function
 2  
 3 #include "stdafx.h"
 4 #include <iostream>
 5 #include <cstring>
 6  
 7  
 8 int main()
 9 {
10     char str1[20];
11     char str2[20];
12     std::cout << "Please enter str2: ";
13     std::cin.get(str2, 20);
14     strcpy_s(str1, str2);
15     std::cout << "str1 is " << "\"" << str1 << "\".\n";
16     char * str;
17     str = new char[20];
18     strcpy_s(str, strlen(str1)+1, str1);
19     std::cout << "str is " << "\"" << str << "\".\n";
20     system("pause");
21     return 0;
22 }

这里分别使用了2个参数和3个参数的strcpy_s()函数版本。

后面使用3个参数的版本时,一般的做法就是将长度定为被复制的字符串长度+1,因为strlen()返回字符串长度,但是不包括字符串末尾的空字符,所以+1。

上述代码运行结果如下图所示:

以上就是strcpy()函数和strcpy_s()函数的基本使用和注意事项了,希望对大家有所帮助~

posted on 2020-12-22 11:14  一杯清酒邀明月  阅读(4492)  评论(0编辑  收藏  举报