1、char*和char[]的共同点

都是指针,指向第一个字符所在的地址

2、char*的用法

char a[] = "aaa";
char* p1 = a;  //char*是常量指针(常量的指针),刚好a就是个指针常量。
char* p2 = "ppp"; //不建议
const char* p2 = "ppp";//等效于 char const* p2 = "ppp";

char*是一个常量指针(常量的指针),即它指向的内存中的内容不能发生改变,但它可以改变自身的指向。

另外,不加const,直接将一个常量字符串赋值给char*是有隐患的(即char*p = "ppp"),它会做如下的操作:

1、在栈区开辟内存放char* p。

2、在文字常量区(即常量存储区)开辟内存放"ppp"。

3、将"ppp"中第一个字符的内存地址赋值给p。

崩溃隐患:当编译strcmp(p, "hello");时系统不会报错,因为符合第一个参数char*第二个参数const char*的参数限制。实际运行时,系统将试图修改文字常量区的内容,而文字常量区属于只读区,进而引发崩溃。若加上const就会避免它进入该类函数哈哈~

如果一定要改变值,应该写char* p  = "ccc";改变p在文字常量区的指向。

 3、char[]的用法

//char[]
char a[20] = "aaa";
char b[] = "bbbbb";
char c[] = {'s','t','r','i','n','g','\0'};
char d[] = {"dddd"};

//一般这样初始化
char num[20] = {0};
//初始化的目的是为了添加\0,否则\0会跑到莫名其妙的地方导致很多错误。
//ASCII码中0对应的是nul符,即\0。这一行的意义就是给0~19都赋\0。

第一行中,"aaa"中的每一个字符分别赋值给数组中的每一个元素并存储在栈上,数组位置不够的字符以"\0"填充,且a[19]一定是"\0"。 

另外,由于char[]是一个指针常量,它等同于char *const a(但这玩意儿不能用于读入,暂时理解是没有指定长度),因此指针中所保存的地址不能发生变化,下面的写法就是错误的。

char a[] = "aaa";
a = "hello"; // "hello"会在文字常量区开辟一片新内存

应该改变地址内保存的值

a[0] = 'b';

4、怎么输出char*地址

虽然指针名就代表的是地址,但在输出时需要加&,仅限cout。

C++:

const char* p = "ppp"; 
cout<<&p<<endl; //地址 方法一
cout<<(void*)p<<endl;//地址 方法二
cout<<p<<endl; //ppp

C:

const char* a = "ppp";
printf("%p \n", a); //16进制地址,不过 %p的输出格式并不统一,有的编译器带0x前缀,有的不带
printf("%#X \n", a);//16进制地址,前缀统一是0x
printf("%s \n", a);  //ppp

5、strlen和sizeof

sizeofstrlen
性质运算符函数
功能以字节为单位计算操作数占用的内存大小计算字符串的长度(strlen函数遇到\0就会停止下来,返回\0前出现的字符个数,不包括\0)
时间编译时运行时
参数类型/各种objectconst char*\char*\char[]
返回值size_tsize_t
头文件运算符没有头文件string.h

ps:

(1)它们返回的都是size_t,这是一种记录操作数大小的数据类型,属于无符号长整型,因此不要用减法比较返回值,因为结果始终是无符号长整型(非负数)。

(2)sizeof常用参数一览

struct test{
    int num;
}
int a;
sizeof(int);
sizeof(a);
sizeof(test);//sizeof(struct test);
sizeof(fun());//该函数必须有返回值

(3)sizeof的主要功能是参与存储分配,例如:

void* malloc(size_t size);
size_t fread(void* ptr, size_t size, size_t nmemb, FILE* stream);
void* memset(void* s, int c, sizeof(s));

6、char*和string

区别stringchar*
本质STL中的一个容器,string封装了char*。指针
内存管理由系统管理,除非系统内存池用完,否则不会出现内存问题栈/堆,如果是堆需要手动释放

string和const char*互转

//string转const char*
#include <string>
string s = "sss";
const char* c = s.c_str();

//const char*转string
const char* c = "ppp";
string s = c;

string和char*互转

//string转char*
#include <string>
string str = "string";
//法一
char* chr = const_cast<char*>(str.c_str());
//法二
char *cstr = &str[0];


//char*转string
char* c = "ppp";
string s = c;

7、赋NULL时究竟在做什么

C++:

/* Define NULL pointer value */
#ifndef NULL
    #ifdef __cplusplus
        #define NULL    0
    #else  /* __cplusplus */
        #define NULL    ((void *)0)
    #endif  /* __cplusplus */
#endif  /* NULL */

C:

#define NULL    ((void *)0)

小结

1、const在*前修饰指定的类型,const在*之后修饰指针。

2、 读入只能是char [],因为读入需要明确的地址,注意必须标定长度。

        改值还得是char*,因为方便。

//一般读入
char num[20];//必须标定长度
cin>>num;
scanf("%s", num);

//还可以这样读入
char num[20];//必须标定长度
char* p = num;
cin>>p;

3、char[]在运行时刻赋值,char*在编译时刻赋值

参考文章:

char * 与char []区别总结_bitcarmanlee的博客-CSDN博客_char和char*的区别1.先上结论c++代码里头经常见到char * 与char []的写法,这两种写法都可以表示一个字符串。比如:void charcode() { char* a = "c1"; char b[] = "c2"; printf("a=%s, b=%s", a, b);}上面这段代码,最后输出的结果为a=c1, b=c2所以这两种方式都可以表示字符串,那么区别到底是什么?我们先说结论,后面再来分析:char * 这种方式表示常量指针,char[] 这种表示指针常量!2https://blog.csdn.net/bitcarmanlee/article/details/124166842char数组的地址怎么输出_JINGpuzzle的博客-CSDN博客_char数组的地址对于int这样的数据类型如int num[3];cout<<num<<endl;输出第一个元素的地址cout<<&num[1]<<endl;输出第二个元素的地址但是我们知道char字符串数组会比较特殊我们先来看这样一个例子按理说数组名不就应该是数组的第一个元素的地址吗,为什么他把整个字符串都给输出了?好,下面我们再来看一个例子因此我们可以得出这样的答案,cout在遇到char的指针时,会输出整个字符串,直..https://blog.csdn.net/Jing_JiaNan/article/details/121170349c++指针(一)——指针的定义以及写法_大豆木南的博客-CSDN博客_c++定义指针本系列讲述指针的文章一共分为六篇文章目录如下:一、c++中指针的定义以及写法二、定义变量时的内存分配三、指针与数组名的关系四、常量指针与常量指针五、指针的指针六、指针和引用好了,按照这个顺序开启探索c++指针之旅~一、c++中指针的定义int *p; //*号前有空格,可以是任意个空格int* p; //*号后有空格,可以是任意个空格int*p; //*号前后都没有空格int * p; //...https://blog.csdn.net/qq_32103261/article/details/80221014

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐