首页 > 代码库 > 指针知识梳理5-字符串与指针,程序内存总结

指针知识梳理5-字符串与指针,程序内存总结

一、文字常量区域

#include <stdio.h>
int main()
{
	char *p = "hello";
	int i = 0;
	for(i = 0;i<6;i++)
	{
		printf("%c",*(p+i));
	}
	*p = '1';
}

大家分析验证以上代码,p 是一个指针变量,存储地址,“hello”是一个字符串占6个字节,当我们用printf的时候能够输出"hello",我们得出结论,p存储的是存放‘h’的内存的地址,当我们*p = ‘1‘,去写这块内存的时候,会出现段错误。

根据以上现象,我们引出下面概念:

"hello"存放在 文字常量区,这个区用来存放代码中出现的字符串常量,并且这块内存区域只能读不能写,以下对程序内存做总结:

二、程序内存总结

程序运行起来后分为一下几块内存区域:

1、代码区

存放各种语句生成的指令。

2、文字常量区

字符串常量放在这里。

比如 char *p = "hello";

3、栈区

局部变量,由系统自动申请释放,生存周期为模块开始到模块结束。

4、堆区

malloc free,由程序员自己申请释放。

5、静态存储区

全局变量  static 变量放在这里,程序开始申请出来,程序结束释放。

三、结合代码综合分析 

#include <stdio.h>
char a[10] = "hello"; //“hello”字符串存储在全局数组中,在 静态存储区。
int main()
{
	char b[10] = "hello"; //“hello”存储在数组b中,在静态存储区。
	char *p1 = "hello"; //“hello”存储在文字常量区,p1存储了存放‘h’的内存的地址,p1在栈区。
	char *p2 = NULL;//p2 在栈区
	*p = '1';//错误,这块区域只能读。
	p2 = (char*)malloc(10);// 在堆中申请10个字节的内存,并把这块内存的地址存放在p2中。
	strcpy(p2,"hello");//把"hello"字符串拷贝到堆内存中。
	free(p2);
}


四、字符串操作函数的基本原理

1、printf

原型  int  printf(const char* p,...);

关于const与指针  参考 http://blog.csdn.net/xiaoliu0518/article/details/32943713

printf函数是一个可变参数函数,参数个数不定,第一个参数是const char* p ,p 存放字符串的地址,通过这个字符串的内容去解析后面的变量然后输出相应的内容。

比如 printf("%c",...);

当字符串为"%c"的时候,printf原型可简单理解为 printf("%c",char c),打印字符。

在比如printf("%s",...)

当字符串为"%s"的时候,printf的原型可以简单理解为 printf("%s",const char *p),在printf中会一个一个打印字符,简单逻辑如下:

int printf("%s",const char *p)
{
     while(*p != '\0')
     {
           putchar(*p);
           p++;
     }
}

所以这里知道为什么,%s的时候,后面可以传char 数组、字符串常量、char* str了吧,因为形参是const char *p。

char a[10] = "hello";

printf("%s","hello");

printf("%s",a);

2、strlen 

strlen简单逻辑如下,依然是判断‘\0‘结束

int  strlen(const char *str)
{
      int len = 0;
      while(*str != '\0')
     {
          len++;
          str++;
     }
     return len;
}
根据形参,可以有以下调用


strlen(字符串常量)

strlen(数组名)

strlen(char*指针)


3、strcpy

strcpy简单逻辑如下,依然是判断‘\0‘结束

void strcpy(char *dest,const char* src)
{
        while(*src != '\0')
        {
		*dest = *src;
		dest++;
 		src++;
	}
	*dest = '\0';
}


通过以上三个例子,简单了解字符串相关函数的操作原理,注意都是要判断‘\0‘,如果没有‘\0‘,则地址一直往后加,就会访问到非法内存。


以上3个例子只写简单逻辑,并没有考虑安全性,严谨性,大家可以自己实现下  strlen   strcpy。



再考虑 如下:

char *p = "hello";
char a[10] = "hello";
printf("%s",p);
strcpy(p,"bye");
printf("%s",a);
strcpy(a,"bye");
哪些代码有问题?