strcpy与strncpy的区别

咱们先来看看strcpy, 下面的程序没有问题:ios

#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = {0};
    char *p = "abc";
    strcpy(str, p);
    cout << str << endl;
 
    return 0;
}
小程序

      可是,我运行下面程序的时候,就有问题了:
#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = {0};
    char *p = "abcdefg";
    strcpy(str, p);
    cout << str << endl;
 
    return 0;
}
     结果以下:
安全


       也许你要说, 这是程序猿不当心, str的空间过小, p指向的串太大,程序猿当心注意一下就行了。真的是这样吗?马虎的程序猿还少吗?反正,我有时也是马虎程序猿之一。谁也不敢保证本身在用strcpy的时候,str的空间足够大。 你想,系统那么复杂, p指向的串,有时你根本没法预料,对不对? 因此,要避免这类程序崩溃,不能仅仅依赖于程序猿的细心和认真,尽管咱们能够尽量细心。函数

      那怎么办?咱们看看更安全的strncpy函数吧,原型:char *strncpy(char *dest, char *src, size_t num);spa

上述有问题的程序应该改成:.net


#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = {0};
    char *p = "abcdefg";
    strncpy(str, p, sizeof(str) - 1);
    cout << str << endl;
 
    return 0;
}
      或许,你要说, 上面这样拷贝后, str中获得的不是想要的"abcdefg"啊!对,但这样至少不会致使系统崩溃。最后的问题是:系统不崩溃,但串不对,这样就相对好定位了。
blog


另外看一下下面这个小程序:内存


#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = {0};
    char *p = "a\0bcdefg";
    strncpy(str, p, sizeof(str) - 1);
    cout << str << endl;
 
    return 0;
}
     结果为a
字符串


绝对绝对要注意的是: strncpy并无拷贝串后的\0字符,而strcpy却拷贝了。这充分说明,strncpy是为拷贝字符而生的,而strcpy是拷贝字符串而生的。但二者都不能越界拷贝。只要正确使用strncpy, 那就比strcpy安全。原型

看看这个程序:


#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = "xyz";
    str[3] = 'w'; //故意将最后的\0换成w
    
    char *p = "abc";
    strcpy(str, p); //把p指向的串拷贝到str中去(恰好能够容纳)
    cout << str << endl;
 
    return 0;
}
      结果为:abc
再看:

#include <iostream>
using namespace std;
 
int main()
{
    char str[4] = "xyz";
    str[3] = 'w'; //故意将最后的'\0'换成'w'
    
    char *p = "abc";
    strncpy(str, p, sizeof(str) - 1); //只拷贝了3个字符,没有拷贝'\0'
    cout << str << endl;
 
    return 0;
}
     结果为:abcw? 异常的结果,固然,在你的机器上极可能是其余的异常结果。从这个例子能够看出, 定义str的时候,若是不进行初始化(char str[4] = {0};),那是很是很是危险的!


总结一下:

1. 好的程序猿用strncpy, 先定义并初始化char str[MAX + 1] = {0};, 若是在程序中间须要再往str中拷贝串,必定要用memset清零。 拷贝范式为:strncpy(str, p, sizeof(str) - 1); , 流氓程序猿用strcpy, 固然我常常在写博客时这么干.

2. strncpy拷贝的是字符,不拷贝串,因此最后一个\0没有拷贝,因此,拷贝的时候须要对串进行清零处理,必定要养成好习惯。

3. strncpy拷贝时,可能会致使截断,但程序不会崩溃。

4. 根据上面strncpy的原型,其实strncpy也并不必定拷贝num个字符,有特殊状况,好比:


#include <iostream> using namespace std;   int main() {     char str[4] = {'w', 'x', 'y', 'z'};     char *p = "a\0bcdefg";     strncpy(str, p, sizeof(str) - 1);       cout << int(str[0]) << endl; // 'a'     cout << int(str[1]) << endl; // '\0'     cout << int(str[2]) << endl; // '\0'   //确实要注意这个值, 不是'b'       cout << int(str[3]) << endl; // 'z'       return 0; }

相关文章
相关标签/搜索