C语言中函数strcpy ,strncpy ,strlcpy的用法【转】

转自:http://blog.chinaunix.net/uid-20797562-id-99311.html

strcpy ,strncpy ,strlcpy的用法
好多人已经知道利用strncpy替代strcpy来防止缓冲区越界。
但是如果还要考虑运行效率的话,也许strlcpy是一个更好的方式。
1. strcpy

我们知道,strcpy 是依据 \0 作为结束判断的,如果 to 的空间不够,则会引起 buffer overflow。strcpy 常规的实现代码如下(来自 OpenBSD 3.9):

char *
strcpy(char *to, const char *from)
{
       char *save = to;

       for (; (*to = *from) != '\0'; ++from, ++to);
       return(save);
}

但通常,我们的 from 都来源于用户的输入,很可能是非常大的一个字符串,因此 strcpy 不够安全。

2. strncpy

在 ANSI C 中,strcpy 的安全版本是 strncpy。

char *strncpy(char *s1, const char *s2, size_t n);

但 strncpy 其行为是很诡异的(不符合我们的通常习惯)。标准规定 n 并不是 sizeof(s1),而是要复制的 char 的个数。一个最常见的问题,就是 strncpy 并不帮你保证 \0

结束。

char buf[8];
strncpy( buf, "abcdefgh", 8 );

看这个程序,buf 将会被 "abcdefgh" 填满,但却没有 \0 结束符了。

另外,如果 s2 的内容比较少,而 n 又比较大的话,strncpy 将会把之间的空间都用 \0 填充。这又出现了一个效率上的问题,如下:

char buf[80];
strncpy( buf, "abcdefgh", 79 );

上面的 strncpy 会填写 79 个 char,而不仅仅是 "abcdefgh" 本身。

strncpy 的标准用法为:(手工写上 \0)

strncpy(path, src, sizeof(path) - 1);
path[sizeof(path) - 1] = '\0';
len = strlen(path);

3. strlcpy

// Copy src to string dst of size siz. At most siz-1 characters
// will be copied. Always NUL terminates (unless siz == 0).
// Returns strlen(src); if retval >= siz, truncation occurred.
size_t
strlcpy(char *dst, const char *src, size_t siz);

而使用 strlcpy,就不需要我们去手动负责 \0 了,仅需要把 sizeof(dst) 告之 strlcpy 即可:

strlcpy(path, src, sizeof(path));
len = strlen(path);

if ( len >= sizeof(path) )
       printf("src is truncated.");

并且 strlcpy 传回的是 strlen(str),因此我们也很方便的可以判断数据是否被截断。

[* 一点点历史 *]

strlcpy 并不属于 ANSI C,至今也还不是标准。

strlcpy 来源于 OpenBSD 2.4,之后很多 unix-like 系统的 libc 中都加入了 strlcpy 函数,我个人在 FreeBSD、Linux 里面都找到了 strlcpy。(Linux使用的是 glibc,

glibc里面有 strlcpy,则所有的 Linux 版本也都应该有 strlcpy)

但 Windows 下是没有 strlcpy 的,对应的是strcpy_s函数 
///////////////////////////////////////////////////////////////////////////
strncpy     
      原型:extern   char   *strncpy(char   *dest,   char   *src,   int   n);   
                    
      用法:#include      
        
      功能:把src所指由NULL结束的字符串的前n个字节复制到dest所指的数组中。   
        
      说明:   
                  如果src的前n个字节不含NULL字符,则结果不会以NULL字符结束。   
                  如果src的长度小于n个字节,则以NULL填充dest直到复制完n个字节。   
                  src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串。   
                  返回指向dest的指针。   
        
      举例:   
    
    
              //   strncpy.c   
                
              #include      
              #include      
    
              main()   
              {   
                  char   *s="Golden   Global   View";   
                  char   *d="Hello,   GGV   Programmers";   
                  char   *p=strdup(s);   
                    
                  clrscr();   
                  textmode(0x00);     //   enable   6   lines   mode   
                                    
                  strncpy(d,s,strlen(s));   
                  printf("%s\n",d);   
                    
                  strncpy(p,s,strlen(d));   
                  printf("%s",p);   
                    
    
                  getchar();   
                  return   0;   
              }   
------------------------------   
memcpy     
      原型:extern   void   *memcpy(void   *dest,   void   *src,   unsigned   int   count);   
    
      用法:#include      
        
      功能:由src所指内存区域复制count个字节到dest所指内存区域。   
        
      说明:src和dest所指内存区域不能重叠,函数返回指向dest的指针。   
        
      举例:   
    
              //   memcpy.c   
                
              #include      
              #include      
    
              main()   
              {   
                  char   *s="Golden   Global   View";   
                  char   d[20];   
                    
                  clrscr();   
                    
                  memcpy(d,s,strlen(s));   
                  d[strlen(s)]=0;   
                  printf("%s",d);   
    
                  getchar();   
                  return   0;   
              }  

函数名: strdup 
功  能: 将串拷贝到新建的位置处 
用  法: char *strdup(char *str); 
程序例:

#include  
#include  
#include

int main(void) 
 { 
    char *dup_str, *string = "abcde";

    dup_str = strdup(string); 
    printf("%s\n", dup_str); 
    free(dup_str);

    return 0; 
 }

clrscr在conio.h里面。 
函数名: clrscr 
功能: 清除文本模式窗口 
用法: void clrscr(void); 

程序例: 
#include 
int main(void) 
{int i; 
clrscr(); 
for (i = 0; i < 20; i++) 
cprintf("%d\r\n", i); 
cprintf("\r\nPress any key to clear screen"); 
getch(); 
clrscr(); 
cprintf("The screen has been cleared!"); 
getch(); 
return 0; 
}

时间: 2024-09-23 15:23:15

C语言中函数strcpy ,strncpy ,strlcpy的用法【转】的相关文章

c-C语言中函数数组调用问题

问题描述 C语言中函数数组调用问题 5C 定义的二维数组为全局变量,在主函数中有多个子函数要用到它.请问下大家,子函数中计算的此二维数组的值间能传递么?还是也需要调用呢?~~ 解决方案 可以传引用或者指针,或者定义为全局变量.那就直接用不用传了. 解决方案二: 传递数组地址或指针,C中没有引用 解决方案三: 你用全局变量,每个函数里面都可以改啊. 解决方案四: 全局的直接访问就行了,就是要注意不要把里面的数据乱改,导致读取出问题 解决方案五: 你使用全局变量,每个函数都可以去访问它并且修改它,会

非递归二叉树遍历-c语言中函数指针作为参数与函数的嵌套

问题描述 c语言中函数指针作为参数与函数的嵌套 函数指针作为另一函数的参数和函数的嵌套的区别,感觉都是调用,有什么不一样呢?他们都适用在什么情况下!(我是在学非递归遍历二叉树时看到的) Status Visit(TElemType e){ printf("%cn",e); return OK; } Status InOrderTraverse(BiTree T ,Status(*Visit)(TElemType e)){ SqStack S; InitStack(S); Push(S,

函数调用-c语言中函数返回值类型的问题

问题描述 c语言中函数返回值类型的问题 看一道改错题: #include #include #include proc(int k)//很显然根据主调函数和返回值类型知道这里缺少类型 { int n;float s,w,p,q; n=1; s=1.0; while(n<=k){ w=2.0*n; p=w-1.0; q=w+1.0; s=s*w*w/p/q; n++; } return s; } void main(){ system("CLS"); printf("%f

c语言中函数中变量范围的问题?

问题描述 c语言中函数中变量范围的问题? ATT 测试环境:vs2013 win7 直接上代码 # include <stdio.h> # include <stdlib.h> void switch_a () //声明switch_a函数 void menu()//声明menu函数 int main() { int a, b; //参加运算的两个数字 int c; //判断进行什么运算的变量 printf("欢迎使用四则运算器n"); printf("

c语言中函数形参为空指针,调用时可以不用带实参吗?

问题描述 c语言中函数形参为空指针,调用时可以不用带实参吗? 最近在看代码,有地方不太懂 void map_hash_init(void) { map_hash_head = hash_create ((uint32_t (*)(void ))(con_key), (int ()(void *, void *))(Con_comp)); } 这是一个hash表的初始化 hash_creat的原型: hash_create (uint32_t (*hash_key) (), int32_t (*h

请求帮忙,非常感谢关于 C语言中函数的!

问题描述 请求帮忙,非常感谢关于 C语言中函数的! #include int M,N; void pr(int w[M][N]) { printf("%d",w[M][N]); } void main() { int a[3][3]; int i,j; for(i=0;i<3;i++) for(j=0;j<3;j++) { printf("a[%d][%d]=",i,j); scanf("%d",&a[i][j]); pr(a

C语言中函数与指针的应用总结_C 语言

1. 首先,在C语言中函数是一种function-to-pointer的方式,即对于一个函数,会将其自动转换成指针的类型. 复制代码 代码如下: #include<stdio.h> void fun(){} int main(void){   printf("%p %p %p\n", &fun, fun, *fun);   return 0;} -------------------------------------------------------------

深入解析C语言中函数指针的定义与使用_C 语言

1.函数指针的定义    函数是由执行语句组成的指令序列或者代码,这些代码的有序集合根据其大小被分配到一定的内存空间中,这一片内存空间的起始地址就成为函数的地址,不同的函数有不同的函数地址,编译器通过函数名来索引函数的入口地址,为了方便操作类型属性相同的函数,c/c++引入了函数指针,函数指针就是指向代码入口地址的指针,是指向函数的指针变量. 因而"函数指针"本身首先应该是指针变量,只不过该指针变量指向函数.这正如用指针变量可指向整形变量.字符型.数组一样,这里是指向函数.C在编译时,

C语言中函数的声明、定义及使用的入门教程_C 语言

对函数的"定义"和"声明"不是一回事.函数的定义是指对函数功能的确立,包括指定函数名,函数值类型.形参及其类型以及函数体等,它是一个完整的.独立的函数单位.而函数的声明的作用则是把函数的名字,函数类型以及形参的类型.个数和顺序通知编译系统,以便在调用该函数时进行对照检查(例如,函数名是否正确,实参与形参的类型和个数是否一致),它不包括函数体.--谭浩强 ,<C程序设计>(第四版),清华大学出版社,2010年6月,p182 这段论述包含了许多概念性错误,这