使用malloc()/ calloc()创建内存的错误概念

ryy*_*ker 3 c memory-management

malloc()/calloc()创建内存的方式概念一直是创建一个项目后,对象的地址保持不变.但是我经常用来创建一个字符串数组的函数,以及一个似乎一直运行良好的函数,最近让我质疑我的理解,也就是说,对象的内存地址可以简单地通过调用来移动calloc/malloc.

为了说明,这里是我用来为字符串数组创建内存的函数 - char **:

char ** CreateArrayOfStrings(char **a, int numWords, int maxWordLen)
{
    int i;
    a = calloc(numWords, sizeof(char *));   //create array of pointers
    if(!a) return a;                        //required caller to check for NULL
    for(i=0;i<numWords;i++)
    {
        a[i] = calloc(maxWordLen + 1, 1);   //create memory for each string 
    }
    return a;
}  
Run Code Online (Sandbox Code Playgroud)

在我的系统上,(Win7,32bit编译,ANSI C)该行:

a = calloc(numWords, sizeof(char *));   //create array of pointers  
Run Code Online (Sandbox Code Playgroud)

创建一个连续内存块,大小为numWords char *,在本例中为7,产生28个字节:

在此输入图像描述

内存跨度从地址0x03260080 + 1C(0x0326009C)
或:

a[0] is at 0x3200260080    
a[1] is at 0x3200260084    
a[2] is at 0x3200260088    
a[3] is at 0x320026008C
a[4] is at 0x3200260090    
a[5] is at 0x3200260094    
a[6] is at 0x3200260098    
Run Code Online (Sandbox Code Playgroud)

然后,我为每个numWords(7)字符串创建内存

for(i=0;i<numWords;i++)
{
    a[i] = calloc(maxWordLen + 1, 1);   //maxWordLen == 5 in this example
}  
Run Code Online (Sandbox Code Playgroud)

其结果如下:

在此输入图像描述

这表明指针的存储位置a[1] - a[6]已经改变.

有人可以解释这是怎么发生的malloc()/calloc()

das*_*ght 7

您似乎正在将苹果与橙子进行比较:

  • 打印a[i] is at ...指针时,显示数组内元素的地址a
  • 但是,当您对存储器布局进行操作时,您会在这些地址上显示,这些地址本身就是指针,因此整个图像看起来很混乱.

如果a[i]在将calloc结果分配给它们之前打印值,则应该全部为零,因为calloc NULL内存不足.在分配之后,您会看到每个指针指向6个字节的块a[i],这非常有意义.

总之,你的当你与分配内存会发生什么初步的了解malloccalloc正确的是:曾经的内存块被分配,其地址*保持不变.

*在具有虚拟内存的系统上,我应该说"它的虚拟地址".

  • @ryyker由于你有一个指针数组,这里有两个地址:(1)数组元素本身的地址,即`&a [0],&a [1],...`和(2)地址存储在该地址,即`a [0],a [1],...`.第一组地址永远不会改变,因为它们被计算为来自`a`的偏移量,它与`&a [0]`相同.将值赋给"a [i]"时,第二组地址会发生变化. (4认同)