C语言:函数返回字符串的四种方式 作者:马育民 • 2022-05-28 10:57 • 阅读:10327 # 预备知识 函数中,无论是 **传参**,还是在 **return 返回值**,都有一个 **拷贝的过程**。 传参: - 传进来的参数是个值,函数在工作之前要把这个值拷贝一份供自己使用 - 传进来的是个地址,函数也就会拷贝该地址供自己使用 return返回时: - 如果返回一个值,函数会将该值拷贝一份以提供给主调函数使用 - 返回的是一个指针(也就是地址),自然拷贝的就是一个地址,供主调函数使用。 # 有四种方式: 1. (推荐)函数参数传递指针,返回该指针 2. (一般)使用堆空间,返回申请的堆地址,注意释放 3. (不推荐)返回函数内定义的静态变量(共享) 4. (不推荐)返回全局变量 # 错误的例子 ``` #include #include 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 #include #include 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 #include #include 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 原文出处:http://malaoshi.top/show_1IX3OPtQMaum.html