C语言中函数返回字符串的方法汇总

 更新时间:2017年05月21日 10:06:35   作者:qingergege  
C语言返回字符串函数共有四种方式,分别如下:使用堆空间,返回申请的堆地址,注意释放、函数参数传递指针,返回该指针、返回函数内定义的静态变量(共享)、返回全局变量

在讨论着四种方法之前,首先要对函数有一个简单的认识,无论是在形实结合时,还是在return语句返回时,都有一个拷贝的过程。你传进来的参数是个值,自然函数在工作之前要把这个值拷贝一份供自己使用,你传进来的是个地址,函数也就会拷贝该地址供自己使用。同样return返回时,如果返回一个值,函数会将该值拷贝一份以提供给主调函数使用,返回的是一个指针(也就是地址),自然拷贝的就是一个地址,供主调函数使用。

先给出一个错误的例子:

#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;
}

编译一下代码,会发现提示一个警告,大概意思就是说返回了一个局部变量的地址。这个程序的输出结果是不确定的,因为我们都知道,局部变量的生存期是就在块内部,这里也就是在函数retstring()的内部,在main函数中,name的内存空间已经被回收。

所以不能返回一个自动变量的字符串。。。

下面给出四种返回字符串的方法:

1、 将字符串指针作为函数参数传入,并返回该指针。

2、 使用malloc函数动态分配内存,注意在主调函数中释放。

3、 返回一个静态局部变量。

4、 使用全局变量。

下面是详细解释:

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

典型的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  s; 
} 

就写到这里啦,希望对你有所帮助。。有错误的地方还请指正,谢谢~~

相关文章

  • C++读取文件的四种方式总结

    C++读取文件的四种方式总结

    C++可以根据不同的目的来选取文件的读取方式,C++中有四种常见的读取方式,本文主要介绍了这四种方法的具体实现,需要的可以参考一下
    2023-04-04
  • C++实现LeetCode(152.求最大子数组乘积)

    C++实现LeetCode(152.求最大子数组乘积)

    这篇文章主要介绍了C++实现LeetCode(152.求最大子数组乘积),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C语言中文件常见操作的示例详解

    C语言中文件常见操作的示例详解

    FILE为C语言提供的文件类型,它是一个结构体类型,用于存放文件的相关信息。本文主要为大家介绍下C语言中几个常见的文件操作的实现,需要的可以参考一下
    2022-12-12
  • C语言运算符的优先级和结合性实例详解

    C语言运算符的优先级和结合性实例详解

    本文主要介绍C语言运算符的知识,这里对运算符的优先级和结合性做出了详解,并附实例代码,希望能帮助有需要的小伙伴
    2016-07-07
  • C++超详细分析顺序表

    C++超详细分析顺序表

    程序中经常需要将一组数据元素作为整体管理和使用,需要创建这种元素组,用变量记录它们,传进传出函数等。一组数据中包含的元素个数可能发生变化,顺序表则是将元素顺序地存放在一块连续的存储区里,元素间的顺序关系由它们的存储顺序自然表示
    2022-03-03
  • C语言中隐藏结构体的细节

    C语言中隐藏结构体的细节

    以笔者粗浅的认识,有两种最常用的方法,可以实现库内结构体定义的隐藏:接口函数形参使用结构体指针,接口函数形参使用句柄。
    2017-05-05
  • c++中拷贝构造函数的参数类型必须是引用

    c++中拷贝构造函数的参数类型必须是引用

    如果拷贝构造函数中的参数不是一个引用,即形如CClass(const CClass c_class),那么就相当于采用了传值的方式(pass-by-value),而传值的方式会调用该类的拷贝构造函数,从而造成无穷递归地调用拷贝构造函数。因此拷贝构造函数的参数必须是一个引用
    2013-07-07
  • C语言简明讲解归并排序的应用

    C语言简明讲解归并排序的应用

    这篇文章主要介绍了 c语言排序之归并排序,归并就是把两个或多个序列合并,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-05-05
  • 详解C++模拟实现priority_queue(仿函数)

    详解C++模拟实现priority_queue(仿函数)

    本文主要介绍了关于C++中优先级队列的模拟实现,以及仿函数的使用,优先级队列是一种容器适配器,其第一个元素总是最大的,仿函数本质是一个类,重载了operator(),可改变比较逻辑,同时,文中还介绍了如何进行类的比较,如日期类或其指针的比较,以及库中sort函数的使用方法
    2024-10-10
  • C语言直接插入排序算法

    C语言直接插入排序算法

    大家好,本篇文章主要讲的是C语言直接插入排序算法,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2022-01-01

最新评论