都是指针,指向第一个字符所在的地址
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在文字常量区的指向。
//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';
虽然指针名就代表的是地址,但在输出时需要加&,仅限cout。
C++:
const char* p = "ppp";
cout<<&p<
C:
const char* a = "ppp";
printf("%p \n", a); //16进制地址,不过 %p的输出格式并不统一,有的编译器带0x前缀,有的不带
printf("%#X \n", a);//16进制地址,前缀统一是0x
printf("%s \n", a); //ppp
sizeof | strlen | |
性质 | 运算符 | 函数 |
功能 | 以字节为单位计算操作数占用的内存大小 | 计算字符串的长度(strlen函数遇到\0就会停止下来,返回\0前出现的字符个数,不包括\0) |
时间 | 编译时 | 运行时 |
参数 | 类型/各种object | const char*\char*\char[] |
返回值 | size_t | size_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));
区别 | string | char* |
本质 | STL中的一个容器,string封装了char*。 | 指针 |
内存管理 | 由系统管理,除非系统内存池用完,否则不会出现内存问题 | 栈/堆,如果是堆需要手动释放 |
string和const char*互转
//string转const char*
#include
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 str = "string";
//法一
char* chr = const_cast(str.c_str());
//法二
char *cstr = &str[0];
//char*转string
char* c = "ppp";
string s = c;
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<