【高手进来指导2个问题】被调函数返回值是malloc的字符指针,free函数该怎么写,有经验的朋友帮忙啊【有简单例程】

flycsdn 2008-04-06 12:19:37
【高手进来指导2个问题】被调函数返回值是malloc的字符指针,free函数该怎么写,有经验的朋友帮忙啊【有简单例程】

被调函数:char* func( ... )
{
......
char* b =NULL;
b = (char*)malloc( length );
......
return b;
}

问题:
1.主调函数如何获取这个被调函数的返回值(是动态分配的变量啊),能直接在主调函数里面char * a=func(...);
还是需要 a = (char*)malloc( length );之后,才能 a=func(...);

2.怎么写被调函数的free函数,我的策略是在主调里面写,不知道对吗,另外怎么写啊

...全文
146 15 打赏 收藏 转发到动态 举报
写回复
用AI写文章
15 条回复
切换为时间正序
请发表友善的回复…
发表回复
lin_style 2008-04-06
  • 打赏
  • 举报
回复
#include <vld.h>
#include <iostream>

using namespace std;

char* func()
{

char* b =NULL;
b = (char*)malloc( 1 );

return b;
}

int main(int argc, char** argv)
{
char *recv;
recv = func();
free(recv);

return 0;
}


检测出来是
No memory leaks detected.

我怀疑你在参数里面做了手脚
HelloDan 2008-04-06
  • 打赏
  • 举报
回复
[Quote=引用 7 楼 flycsdn 的回复:]
至于4楼的哥们,我也是这么做的,但怎么发生了内存泄漏,这让我有点怀疑这个做法,能给予理论论证指导吗
[/Quote]

都说了只要有就行了,但你要保证指针是指向那个malloc出来的内存。如果你在使用中改了指向,又没有其他的指向同一块内存,那么就出现了内存泄漏
lin_style 2008-04-06
  • 打赏
  • 举报
回复
可以把你的代码发一段出来我调调看吗。
flycsdn 2008-04-06
  • 打赏
  • 举报
回复
至于4楼的哥们,我也是这么做的,但怎么发生了内存泄漏,这让我有点怀疑这个做法,能给予理论论证指导吗
flycsdn 2008-04-06
  • 打赏
  • 举报
回复
3楼和5楼的哥们,我似乎没理解你们的意思,因为我觉得你们不理解我的问题,有点答非所问了,能否单就我的问题给予帮助,不要改变现在的状况,给予一些经验,多谢哦
ttkk_2007 2008-04-06
  • 打赏
  • 举报
回复
char* func( ... )
{
......
char* b =NULL;
b = (char*)malloc( length );
......
return b;
}

问题:
1.主调函数如何获取这个被调函数的返回值(是动态分配的变量啊),能直接在主调函数里面char * a=func(...);
还是需要 a = (char*)malloc( length );之后,才能 a=func(...);

2.怎么写被调函数的free函数,我的策略是在主调里面写,不知道对吗,另外怎么写啊
============================================================
1.直接char *a = func();
2.free(a);
wangxipu 2008-04-06
  • 打赏
  • 举报
回复
C
的话就像ls的说的那样,自己定义个释放的函数

如果是C++
的话,析构函数会自动帮你来做这些事情的
lin_style 2008-04-06
  • 打赏
  • 举报
回复
1。你只要知道你那快申请空间的地址就行了。
2。
假设主设函数里有个
char* pT = func();;;

只要最后有free(pT)就好
agaric 2008-04-06
  • 打赏
  • 举报
回复

通常我们都会做一些小小的抽象。
比如要malloc一个RECT结构的时候,你的func我这里就叫作CreateRect

RECT *CreateRect()
{
return (RECT *)mallc( sizeof(RECT) );

}

如果我要释放的话,我会写一个函数,叫做FreeRect的函数
void FreeRect( RECT *rect )
{
free(rect );

}

这就叫做配对。


如果在加上初始化和最后销毁处理的动作,实际上就很像c++里面的构造函数和析构函数了。
flycsdn 2008-04-06
  • 打赏
  • 举报
回复
楼上的大哥,首先谢谢你

1.你的意思是接收获取被调函数的动态分配的返回值,是不需要在主调中动态分配的,直接一个简单的字符指针就可以了???

2.晕,都可以是什么意思啊,被调函数中可以写吗????哪里写???,而主调函数中又该如何写,是不是free(a)就可以了???【注意不是free(b)】

HelloDan 2008-04-06
  • 打赏
  • 举报
回复
char * a=func(...);
//这样就行了。



2
// 都可以。有就行了。
ysuliu 2008-04-06
  • 打赏
  • 举报
回复
10楼和12楼的回复已经很清楚了。
这么写肯定不会leak的。。

原理:char* func( ... ) 的返回值是你分配内存的地址,这部分内存是在堆上的,所以不涉及局部变量的作用域问题,你如果不释放,它就一直在。
char * a=func(...); 这么写就把返回值赋给了a,也就是a现在指向了你在func里分配的内存。

释放的话就直接free(a); 就行了,因为a就是你分配的内存地址,所以不会泄漏。
长安宁 2008-04-06
  • 打赏
  • 举报
回复
char* func( ... )
{ ......
char* b =NULL;
b = (char*)malloc( length );
//free(b);这样没有问题吧;
//char *c=b;
//free(c);这样没有问题吧;
// free;是释放的指针所指向的空间;所以在func()的返回值,是个指针,你在调用处,直接free就可以了;
......
return b;
}


星羽 2008-04-06
  • 打赏
  • 举报
回复

给你个例子吧


#include <stdio.h>

char* rev_string(const char* str) {
char* r = (char*)malloc(strlen(str) + 1);
strcpy(r, str);
strrev(r);
return r;
}

int main()
{
char* p = rev_string("hello world");
printf("%s\n", p);
free(p);

return 0;
}



flycsdn 2008-04-06
  • 打赏
  • 举报
回复
但是的确泄漏了,我是通过java的JNI调用dll做的字符串转换发现的

1.你用的是什么检测工具啊,很喜欢哦

2.不说健壮性,free过的内存,不但可以使用,而且还可以free多次,对不对啊?

若可以,第一次free,那段内存就给了os,第二次free,os就一点控制都没有吗,若是如此,岂不是任意个小程序的指针都能随便指向其他区域了吗
用变量a给出下面的定义 a) 一个整型数(An integer) b) 一个指向整型数的指针(A pointer to an integer) c) 一个指向指针的的指针,它指向的指针是指向一个整型数(A pointer to a pointer to an integer) d) 一个有10个整型数的数组(An array of 10 integers) e) 一个有10个指针的数组,该指针是指向一个整型数的(An array of 10 pointers to integers) f) 一个指向有10个整型数数组的指针(A pointer to an array of 10 integers) g) 一个指向函数指针,该函数有一个整型参数并返回一个整型数(A pointer to a function that takes an integer as an argument and returns an integer) h) 一个有10个指针的数组,该指针指向一个函数,该函数有一个整型参数并返回一个整型数( An array of ten pointers to functions that take an integer argument and return an integer ) 答案是: a) int a; // An integer b) int *a; // A pointer to an integer c) int **a; // A pointer to a pointer to an integer d) int a[10]; // An array of 10 integers e) int *a[10]; // An array of 10 pointers to integers f) int (*a)[10]; // A pointer to an array of 10 integers g) int (*a)(int); // A pointer to a function a that takes an integer argument and returns an integer h) int (*a[10])(int); // An array of 10 pointers to functions that take an integer argument and return an integer 13.下面的数据声明都代表什么? (1)float (**def)[10]; def是一个二级指针,它指向的是一个一维数组的指针,数组的元素都是float. (2)double*(*gh)[10]; gh是一个指针,它指向一个一维数组,数组元素都是double*. (3)double(*f[10])(); f是一个数组,f有10个元素,元素都是函数指针,指向的函数类型是没有参数且返回double的函数. (4)int *((*b)[10]); 就跟int *(*b)[10]是一样的,b是一维数组的指针. (5)Long (*fun)(int); 函数指针. (6)int (*(*F)(int,int))(int); F是一个函数指针,指向的函数的类型是有两个int参数并且返回一个函数指针函数,返回的函数指针指向有一个int参数且返回int的函数.  所有的ASCII码都可以用“\”加数字(一般是8进制数字)来表示。而C中定义了一些字母前加"\"来表示常见的那些不能显示的ASCII字符,如\0,\t,\n等,就称为转义字符,因为后面的字符,都不是它本来的ASCII字符意思了。   转义字符 意义 ASCII码值(十进制)   \a 响铃(BEL) 007   \b 退格(BS) 008   \f 换页(FF) 012   \n 换行(LF) 010   \r 回车(CR) 013   \t 水平制表(HT) 009   \v 垂直制表(VT) 011   \\ 反斜杠 092   \? 问号字符 063   \' 单引号字符 039   \" 双引号字符 034   \0 空字符(NULL) 000   \ddd 任意字符 三位八进制   \xhh 任意字符 二位十六进制   字符型常量   字符型常量所表示的值是字符型变量所能包含的值。我们可以用ASCII表达式来表示一个字符型常量,或者用单引号内加反斜杠表示转义字符。   'A', '\x2f', '\013';   其中:\x表示后面的字符是十六进制数,\0表示后面的字符是八进制数。   注意:在Turbo C 2.0中,字符型常量表示数的范围是-128到127,除非你把它声明为unsigned,这样就是0到255。   上面我们见到的\x,\n,\a等等都是叫转义字符,它告诉编译器需要用特殊的方式进行处理。下面给出所有的转义字符和所对应的意义:   转义字符 描述   \' 单引号   \ 双引号   \\ 反斜杠   \0 空字符   \ccc (最多三位)八进制数   \a 声音符   \b 退格符   \f 换页符   \n 换行符   \r 回车符   \t 水平制表符   \v 垂直制表符   \xhh(最多两位)十六进制符 下面的代码输出是什么,为什么? 1、sizeof()和strlen()的使用? 答: 1.从功能定义上,strlen函数,用来求字符串的长度,sizeof函数是用来求指定变量或变量类型等所占用内存的 大小; 2.sizeof是运算符,而strlen是C库函数strlen只能用char*做参数,且以'\0'结尾的; 对于静态数组处理: char str[20]="0123456789"; strlen(str)=10; //表示数组中字符串的长度 sizeof(str)=20; //表示数组变量分配的长度 对于指针处理: char *str="0123456789"; strlen(str)=10; //表示字符串的长度 sizeof(str)=4; //表示指针变量的所占内存大小 sizeof(*str)=1; //表示'0'这个字符变量的所占内存大小 2、 struct 和 class的区别? 答: (1)类中的成员默认是private的,当是可以声明为public,private 和protected,结构中定义的成员默认的都是public. (2)结构中不允许定义成员函数,当是类中可以定义成员函数. (3)结构不可以存在继承关系,当是类可以存在继承关系. 3、 strcpy(), strcat(), strcmp,strlen(), strchr(), memcpy(), memcmp(), memset()的编? 4、 内存的分配方式的分配方式有几种? 答: (1)从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量。 (2)在栈上创建。在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 (3)从堆上分配,亦称动态内存分配。程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用非常灵活,但问题也最多。 5,、解释堆和栈的区别? 答: (1)栈区(stack):由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈。 (2)堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表.(主要有:malloc/free,new/delete) 6,、枚举的用法,以及它与宏的区别? 答: 枚举(enum)作用:定义一组常量,而且这些常量组成了一个集合。当我们将emun参数传递给函数时,如果传递了一个不属于这个集合的参数,编译器报错。 区别: 枚举是类型,而宏没有类型。 枚举将检查其语法,而宏是简单的文本替换。 7、 const的用法,以及声明const变量与宏的区别? 答: const的用法有四种: a): const对象和const类型的对象; b): const指针 和 指向const的指针 及两者结合; c): const 参数修饰 和 参数返回类型的const修饰; d): const类成员 和 const 成员函数; 区别: const常量有数据类型, 而宏常量没有数据类型。编译器可以对前者进行类型安全检查,而对后者只能进行字符 替换,没有类型 安全检查。而且字符替换可能会带来料想不到的边界效应。有些集成化工具可以对const常量进行试, 但不能对宏量进行试。 8、函数assert的用法? 答:断言assert是仅在debug版本起作用的宏,用于检查“不应该“发生的情况。程序员可以把assert看成一个在任何系统状态下都可以安全使用的无害测试手段。 9、用预处理指令#define 声明一个常数,用以表明1年中有多少秒(忽略闰年问题)? 答: #define SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL 10、 一个"标准"宏MIN,这个宏输入两个参数并返回较小的一个? 答: #define MIN(A,B) ((A) <= (B) ? (A) : (B)) 11、嵌入式系统中经常要用到无限循环,你怎么样用C编死循环呢? 答: 这个问题用几个解决方案。我首选的方案是: while(1) { } 一些程序员更喜欢如下方案: for(;;) { } 13、关于内存对齐的问题以及sizof()的输出? 答: 编译器自动对齐的原因:为了提高程序的性能,数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;然而,对齐的内存访问仅需要一次访问。 14、 static有什么用途?(请至少说明两种) 答: (1)在函数体,一个被声明为静态的变量在这一函数用过程中维持其值不变。 (2) 在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问。它是一个本地的全局变量。 (3) 在模块内,一个被声明为静态的函数只可被这一模块内的其它函数用。那就是,这个函数被限制在声明它的模块的本地范围内使用。 15、引用与指针有什么区别? 答: (1) 引用必须被初始化,指针不必。 (2) 引用初始化以后不能被改变,指针可以改变所指的对象。 (3) 不存在指向空值的引用,但是存在指向空值的指针。 16、什么是平衡二叉树? 答: 左右子树都是平衡二叉树 且左右子树的深度差值的绝对值不大于1。 17、堆栈溢出一般是由什么原因导致的? 答: 没有回收垃圾资源。 18、什么是预编译?何时需要预编译? 答: (1)总是使用不经常改动的大型代码体。 (2)程序由多个模块组成,所有模块都使用一组标准的包含文件和相同的编译选项。在这种情况下,可以将所有包含文件预编译为一个预编译头。 19、队列和栈有什么区别? 答: 队列先进先出;栈后进先出。 20、数组和链表的区别? 答: 数组:数据顺序存储,固定大小; 链表:数据可以随机存储,大小可动态改变; 21、为什么数组名作为参数,会改变数组的内容,而其它类型如int却不会改变变量的值? 答: 当数组名作为参数时,传递的实际上是地址。而其他类型如int作为参数时,由于函数参数值实质上是实参的一份拷贝,被函数内部对形参的改变并不影响实参的值。 22、数据结构和算法: A:查找: (1)二分法查找; B:出下列算法的时间复杂度和实现排序: (1)冒泡排序; (2)选择排序; (3)插入排序; (4)快速排序; (5)堆排序; (6)归并排序; 23、编gbk_strlen函数,计算含有汉字的字符串的长度,汉字作为一个字符处理;已知:汉字编码为双字节,其中 参数里涉及指针,就要考虑该指针是不是一个需要修改的量,如果是,则参数应采用指向指针指针。 (C语言里参数传递都是传值,是一个拷贝,修改指针,只是改变了拷贝的指向,原指针指向并没有改变,而修改指针的内容则是可以的。)如果函数的参数是一个指针,不要指望用该指针去申请动态内存。(即上面所说的修改该指针) 编译器总是要为函数的每个参数制作临时副本,指针参数p的副本是 _p,指针参数也不例外。 如果非得要用指针参数去申请内存,那么应该改用“指向指针指针”,由于“指向指针指针”这个概念不容易理解,我们也可以用函数返回值来传递动态内存。 常常有人把return语句用错了。这里强不要用return语句返回指向“栈内存”的指针,因为该内存在函数结束时自动消亡。 c语言里,利用函数返回值来表明它是否执行成功的做法最常见。 2.如果要在单链表中删除或插入一个元素,就必须知道指向删除点或插入点前面那个元素的指针才行。 数组具有随机存取性(可以立即存取任一元素) char *str = "the word" 常量字符串,静态存储区,具有程序生命期。 char *str = (char *)malloc(....); 堆上分配,手动释放,否则内存泄漏。 char str[] = "the word" 栈上分配,函数生命期。 4. '\0'不是指ASCII码值,它是字符,它的ASCII码值是0,也就是ASCII码表的第一个字符。'\0'并非是空格,空格的ASCII码值是32,只不过是输出类似于空格而已。 5. 在32位程序里,所有类型的指针的值都是一个32位整数,因为32位程序里内存地址全都是32位长,指针本身占据了4个字节的长度,即用一个字的空间保存一个地址。地址是按字节表示的,一个字节一个地址。地址+1,移动一个字节。 6.从语法上看,你只须把指针声明语句中的指针名字和名字左边的指针声明符*去掉,剩下的就是指针所指向的类型。 *p的结果是p所指向的东西 &a的运算结果是一个指针,指针所指向的类型是a的类型,指针所指向的地址嘛,那就是a的地址。 7. 声明了一个数组TYPE array[n],则数组名称array就有了两重含义:第一,它代表整个数组,它的类型是TYPE[n];第二 ,它是一个指针,该指针的类型是TYPE*,该指针指向的内存区就是数组第0号单元,该指针自己占有单独的内存区. 在表达式sizeof(array)中,数组名array代表数组本身,故这时sizeof函数测出的是整个数组的大小。 sizeof(array+n)测出的是指针类型的大小。 char类型的长度是一个字节,int类型的长度是4个字节,即一个字。 sizeof()得到的值是以字节为单位,所以sizeof(int) = 4 8. char* 相当于string ,char* arr[10]即字符串数组

64,643

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧