C语言-数组练习进阶

1,整型数组

2,字符数组

3,二维数组

4,复杂指针运算


1,整型数组

复制代码
#include<stdio.h>
//数组名的理解
//指针的运算和指针类型的意义
//
// 整型数组
int main() {
	int a[] = { 1,2,3,4 };
	printf("%d\n", sizeof(a));
	//sizeof(数组名),数组名表示整个数组,计算的是整个数组的大小,单位是字节
	printf("%d\n", sizeof(a+0));
	//a不是单独放在sizeof内部,也没有取地址,所以a就是首元素的地址,a+0还是首元素的地址
	//是地址,大小就是4/8 个字节
	printf("%d\n", sizeof(*a));
	//*a中的a是数组首元素的地址,*a就是对首元素的地址解引用,找到的就是首元素
	//首元素的大小就是4个字节
	printf("%d\n", sizeof(a+1));
	//这里的a是数组首元素的地址
	//a+1是第二个元素的地址
	//sizeof(a+1)就是地址的大小
	printf("%d\n", sizeof(a[1]));
	//计算的是第二个元素的大小
	printf("%d\n", sizeof(&a));
	//&a取出的数组的地址,数组的地址,也就是个地址
	printf("%d\n", sizeof(*&a));
	//第一种理解方法
	//&a ---> int(*)[4]
	//&a拿到的是数组名的地址,类型是 int(*)[4] ,是一种数组指针
	//数组指针解引用找到的是数组
	//*&a ----> a
	//第二种理解方法
	//&和*抵消了
	//*&a ---> a
	printf("%d\n", sizeof(&a+1));
	//&a取出的是数组的地址
	//&a --> int(*)[4]
	//&a+1  是从数组a的地址向后跳过了一个(4个整型元素的)数组的大小
	//&a+1 还是地址,是地址就是4/8字节
	printf("%d\n", sizeof(&a[0]));
	//&a[0]就是第一个元素的地址
	//计算的是地址的大小
	printf("%d\n", sizeof(&a[0]+1));
	//&a[0]+1 是第二个元素的地址
	//大小是4/8个字节
	//&a[0]+1  --->  &a[1]
	return 0;
}

2,字符数组

复制代码
#include<stdio.h>
//字符数组
//int main() {
//	char arr[] = { 'a','b','c','d','e','f' };
//	printf("%d\n", sizeof(arr));//6
//	//sizeof(数组名)
//	printf("%d\n", sizeof(arr+0));
//	//arr + 0  是数组首元素的地址 
//	printf("%d\n", sizeof(*arr));
//	//*arr就是数组的首元素,大小是1字节
//	//*arr  -->  arr[0]
//	//*(arr+0) --> arr[0]
//	printf("%d\n", sizeof(arr[1]));
//	printf("%d\n", sizeof(&arr));
//	//&arr是数组的地址,是地址就是4/8个字节
//	printf("%d\n", sizeof(&arr+1));
//	//&arr+1 是数组后的地址
//
//	printf("%d\n", sizeof(&arr[0]+1));
//	//&arr[0]+1是第二个元素的地址
//
//	printf("%d\n", sizeof(arr[0] + 1)); 
//
//	return 0;
//}
//#include<string.h>
//int main() {
//	char arr[] = { 'a','b','c','d','e','f' };
//	printf("%d\n", strlen(arr));//随机值
//	printf("%d\n", strlen(arr + 0));//随机值
//
//	//printf("%d\n", strlen(*arr));//strlen('a');--->strlen(97);//野指针
//	//printf("%d\n", strlen(arr[1]));//同上
//	
//	printf("%d\n", strlen(&arr));//随机值
//	printf("%d\n", strlen(&arr + 1));//随机值 - 6
//	printf("%d\n", strlen(&arr[0] + 1));//随机值 - 1
//
//	return 0;
//}

#include<string.h>
//int main() {
//	//char arr[] = { 'a','b','c','d','e','f' };
//	char arr[] = "abcdef";
//	//[a b c d e f \0]
//	printf("%d\n", sizeof(arr));//7
//	printf("%d\n", sizeof(arr + 0));//4/8
//	printf("%d\n", sizeof(*arr));//1
//	printf("%d\n", sizeof(arr[1]));//1
//	printf("%d\n", sizeof(&arr));//4/8
//	printf("%d\n", sizeof(&arr + 1));//4/8
//	printf("%d\n", sizeof(&arr[0] + 1));//4/8
//
//	return 0;
//}

//注意:
// strlen 是求字符串长度的,关注的是字符串中的\0,计算的是\0之前出现的字符的个数
// strlen是库函数,只针对字符串
//sizeof只关注占用内存空间的大小,不在乎内存中放的是什么
// sizeof是操作符
//
//int main() {
//	//char arr[] = { 'a','b','c','d','e','f' };
//	char arr[] = "abcdef";
//	//[a b c d e f \0]
//	printf("%d\n", strlen(arr));//6
//	printf("%d\n", strlen(arr + 0));//6
//	//printf("%d\n", strlen(*arr));//err
//	//printf("%d\n", strlen(arr[1]));//err
//	printf("%d\n", strlen(&arr));//6
//	printf("%d\n", strlen(&arr + 1));//随机值
//	printf("%d\n", strlen(&arr[0] + 1));//5
//
//	return 0;
//}


int main() {
	char* p = "abcdef";
	printf("%d\n", sizeof(p));//4/8
	printf("%d\n", sizeof(p+1));//4/8
	printf("%d\n", sizeof(*p));//1
	printf("%d\n", sizeof(p[0]));//1
	printf("%d\n", sizeof(&p));//4/8
	printf("%d\n", sizeof(&p+1));//4/8
	printf("%d\n", sizeof(&p[0]+1));//4/8

	printf("%d\n", strlen(p));//6
	printf("%d\n", strlen(p + 1));//5
	printf("%d\n", strlen(*p));//err
	printf("%d\n", strlen(p[0]));//err
	printf("%d\n", strlen(&p));//随机值
	printf("%d\n", strlen(&p + 1));//随机值
	printf("%d\n", strlen(&p[0] + 1));//5

	return 0;
}

3,二维数组

复制代码
#include<stdio.h>
//二维数组
int main() {
	
	int a[3][4] = { 0 };
	printf("%d\n", sizeof(a));//48
	printf("%d\n", sizeof(a[0][0]));//4
	printf("%d\n", sizeof(a[0]));
	//a[0]是第一行这个一维数组的数组名,单独放在sizeof内部,a[0]表示第一个整个这个一维数组
	//sizeof(a[0])计算的就是第一行的大小
	printf("%d\n", sizeof(a[0]+1));
	//a[0]并没有单独放在sizeof内部,也没取地址,a[0]就表示首元素的地址
	//就是第一行这个一维数组的第一个元素的地址,a[0]+1 就是第一行的第二个元素的地址
	printf("%d\n", sizeof(*(a[0]+1)));
	//a[0]+1 就是第一行第二个元素的地址
	//*(a[0]+1)就是第一行第二个元素
	printf("%d\n", sizeof(a + 1));//4/8
	//a虽然是二维数组的地址,但是并没有单独放在sizeof内部,也没取地址
	//a表示首元素的地址,二维数组的首元素是它的第一行,a就是第一行的地址
	//a+1就是跳过第一行,表示第二行的地址
	printf("%d\n", sizeof(*(a + 1)));//16
	//*(a+1)是对第二行地址的解引用,拿到的是第二行
	//*(a+1)  --> a[1]

	printf("%d\n", sizeof(&a[0] + 1));//4/8
	//&a[0]  对第一行的数组名取地址,拿出的是第一行的地址
	//&a[0]+1  得到的是第二行的地址
	printf("%d\n", sizeof(*(&a[0] + 1)));//16
	printf("%d\n", sizeof(*a));//16
	//a 表示首元素的地址,就是第一行的地址
	//*a 就是对第一行地址的解引用,拿到的就是第一行
	printf("%d\n", sizeof(a[3]));//16
	//printf("%d\n", sizeof(a[0]));//16



	return 0;
}

4,复杂指针运算

复制代码
#include<stdio.h>

//int main() {
//	int a[5] = { 1,2,3,4,5 };
//	int* ptr = (int*)(&a + 1);
//
//	printf("%d %d", *(a + 1), *(ptr - 1));//2  5
//	//
//	//
//	return 0;
//}


//
//struct Test
//{
//	int num;
//	char* pcName;
//	short sDate;
//	char cha[2];
//	short sBa[4];
//}*p=(struct Test*)0x100000;
////假设p的值为0x100000, 如下表表达式的值分别为多少
////已知,结构体Test类型的变量大小是20个字节
//int main() {
//	printf("%p\n", p + 0x1);//0x1  是16进制1
//	//0x100000  + 20  -->  0x100014   (x86的运行环境)
//	printf("%p\n", (unsigned long)p + 0x1);
//	//1048576   + 1   -->  1048577
//	printf("%p\n", (unsigned int*)p + 0x1);
//	//0x100000 + 4 -->0x100004
//	return 0;
//}

//int main() {
//
//	int a[4] = { 1,2,3,4 };
//	int* ptr1 = (int*)(&a + 1);
//	int* ptr2 = (int*)((int)a + 1);
//	printf("%x %x", ptr1[-1], *ptr2);
//
//	return 0;
//}

//int main() {
//	int a[3][2] = { (0,1),(2,3),(4,5) };//注意:逗号表达式  其实就是1 3 5
//	int* p;
//	p = a[0];//第一行的数组名,首元素的地址
//	printf("%d", p[0]);
//
//	return 0;
//}

//int main() {
//	int a[5][5];
//	int(*p)[4];
//	p = a;
//	printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
//	return 0;
//}

//int main() {
//	int aa[2][5] = { 1,2,33,4,5,6,7,8,9,10 };
//	int* ptr1 = (int*)(&aa + 1);
//	int* ptr2 = (int*)(*(aa + 1));
//	printf("%d , %d ", *(ptr1 + 1), *(ptr2 - 1));
//	//                      10            5
//	return 0;
//}

//int main() {
//	char* a[] = { "work","at"."alibaba" };
//	char** pa = a;
//	pa++;
//	printf("%s\n", *pa);//at
//
//	return 0;
//}

int main() {
	char* c[] = { "enter","new","point","first" };
	char** cp[] = { c + 3,c + 2,c + 1,c };
	char*** cpp = cp;
	printf("%s\n", **++cpp);
	printf("%s\n", *-- * ++cpp + 3);
	printf("%s\n", *cpp[-2]+3);
	printf("%s\n", cpp[-1][-1]+1);

	return 0;
}
相关推荐
超级大福宝2 小时前
N皇后问题:经典回溯算法的一些分析
数据结构·c++·算法·leetcode
Wect2 小时前
LeetCode 530. 二叉搜索树的最小绝对差:两种解法详解(迭代+递归)
前端·算法·typescript
Rabbit_QL2 小时前
【BPE实战】从零实现 BPE 分词器:训练、编码与解码
python·算法·nlp
Railshiqian2 小时前
给android源码下的模拟器添加两个后排屏的修改
android·开发语言·javascript
小O的算法实验室2 小时前
2024年IEEE TII SCI1区TOP,面向动态多目标多AUV路径规划的协同进化计算算法,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
Charlie_lll2 小时前
力扣解题-88. 合并两个有序数组
后端·算法·leetcode
菜鸡儿齐3 小时前
leetcode-最小栈
java·算法·leetcode
雪人不是菜鸡3 小时前
简单工厂模式
开发语言·算法·c#
岛雨QA3 小时前
常用十种算法「Java数据结构与算法学习笔记13」
数据结构·算法