C语言:函数返回字符串的四种方式

预备知识

函数中,无论是 传参,还是在 return 返回值,都有一个 拷贝的过程

传参:

  • 传进来的参数是个值,函数在工作之前要把这个值拷贝一份供自己使用
  • 传进来的是个地址,函数也就会拷贝该地址供自己使用

return返回时:

  • 如果返回一个值,函数会将该值拷贝一份以提供给主调函数使用
  • 返回的是一个指针(也就是地址),自然拷贝的就是一个地址,供主调函数使用。

有四种方式:

  1. (推荐)函数参数传递指针,返回该指针
  2. (一般)使用堆空间,返回申请的堆地址,注意释放
  3. (不推荐)返回函数内定义的静态变量(共享)
  4. (不推荐)返回全局变量

错误的例子

#include <stdio.h>
#include <string.h>

char * retstring();
int main()
{
    char * name2;
    name2 = retstring();
    printf("%s\n",name2);
    return 0;
}

char * retstring()
{
    char name[10];
    strcpy(name,"汉青");
    return name;
}

编译报错:返回了一个局部变量的地址。这个程序的输出结果是不确定的

原因,局部变量的生存期是就在 函数内部,当该函数执行后,name 的内存空间已经被回收。

方法一:将字符串指针作为函数参数传入,并返回该指针

典型的strcpy()函数应该就是采用的这种方法,第一个参数为指向目的字符串的指针,返回值也为这个指针。

char* strcpy(char* des,const char* source)
 
{
 
 char* r=des;

  assert((des != NULL) && (source != NULL));

 while((*r++ = *source++)!='\0');

 return des;
 
}

方法二:使用malloc函数动态分配

但是一定要注意在主调函数中将其释放,应为malloc动态分配的内存位于堆区,而堆区的内存是要程序员自己释放的。

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

char * retstring();
int main()
{
    char * name2;
    name2 = retstring();
    printf("%s\n",name2);
    //记住一定要用free释放,否则会造成内存泄露
    free(name2);
    return 0;
}

char * retstring()
{
    char * name;
    name = (char *)malloc(10); 
    strcpy(name,"张汉青");
    return name;
}

方法三:返回一个静态局部变量。

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

char * retstring();
int main()
{
    char * name2;
    name2 = retstring();
    printf("%s\n",name2);
    return 0;
}

char * retstring()
{
    static char name[10];
    strcpy(name,"张汉青");
    return name;
}

这种方法有一个问题: 由于采用了静态局部变量(位于静态区,程序结束时由系统进行释放),这就导致,如果多次调用这个函数,下一次调用会将上一次调用的结果覆盖掉。

C语言中的库函数,tmpnam()函数、getenv()函数等应该都是采用的这种方法,这也就是为什么,使用这样的函数的时候应该立即将返回结果拷贝一份的原因。

方法四: 使用全局变量。

一个例子如下:

char   g_s[100]; 
char*   fun() 
{ 
        strcpy(g_s,   "abc "); 
        return   g_s; 
}

感谢:
https://www.cnblogs.com/qingergege/p/6496683.html


原文出处:https://malaoshi.top/show_1IX3OPtQMaum.html