学C的第十七天【指针初阶:指针是什么、指针和指针类型、野指针、指针运算】

=========================================================================

相关代码gitee自取:C语言学习日记: 加油努力 (gitee.com)

=========================================================================

接上期: 
学C的第十六天【续操作符详解、练习】_高高的胖子的博客-CSDN博客

========================================================================= 

                     

 1. 指针是什么?

                               

(1). 指针内存中一个最小单元的编号,也就是地址

                            

(2). 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量

                     

总结:指针就是地址,口语中说的指针通常指的是指针变量

                             

                                     


                                

1.1 * 指针变量

                           

可以通过 &(取地址操作符)取出变量的内存起始地址,可以把地址存放到一个变量中,这个变量就是指针变量

                     

#include <stdio.h>
int main()
{
	int a = 100; //在内存中开辟一块空间

	int* pa = &a; //取出变量a的地址,使用&(取地址操作符)
	//a变量占用4个字节的空间,这里是将a的4个字节的第一个字节的地址
	//放在pa变量中,pa就是一个指针变量
	// pa 是专门用来存放地址(指针)的,
	// int * 可以分开理解
	// * 告诉我们 pa 是 指针
	// int 告诉我们 pa 指向的 a 是 int类型

	printf("%p", pa);

	return 0;
}

                

总结:

           

指针变量,用来存放地址的变量。(存放在指针中的值都被当成地址处理
                 


                                 

1.2 * 内存单元的编址

            

查看往期文章:

学C的第五天(初识指针,内存产生内存单元地址过程,指针函数介绍和其大小;结构体补充)_高高的胖子的博客-CSDN博客

                      

总结:

                        

(1). 在32位机器上,地址是32个0或1组成的二进制序列,那地址就得用4个字节的空间来存储,所以一个指针变量的大小就应是4个字节

        

(2). 如果在64位机器上,如果有64个地址线,那一个指针变量的大小是8个字节,才能存放一个地址。

        

(3). 指针变量是用来存放地址的,地址唯一标示一个内存单元

        

(4). 指针的大小32位平台是4个字节,在64位平台是8个字节

                           


                                  

2. 指针和指针类型

                     

指针的定义方式:

             

type + * 

              

(如:int类型 -- int * 变量)

               

不同 类型* 的指针是为了存放 该类型类型变量地址

(如:char* 类型 的指针是为了存放 char类型变量地址

                       


                  

 2.1 * 指针类型的意义(不同指针类型访问的对象的大小)

               

int* 类型 的指针变量:
//指针类型的意义
#include <stdio.h>
int main()
{
	int a = 0x11223344; //0x开头的是16进制数字
	int* pa = &a;
	*pa = 0;

	return 0;
}

              

char* 类型 的指针变量:
//指针类型的意义:char*
#include <stdio.h>
int main()
{
	int a = 0x11223344; //0x开头的是16进制数字
	char* pa = &a;
	*pa = 0;

	return 0;
}

               

               

总结:

                             

对于指针变量:        type        *        p;

                 

(1).   *  :说明 p 是指针变量

               

(2). type:说明 p 指向的对象类型

               

(3). typep 解引用的时候访问的对象的大小sizeof(type) --> 重点!

                      

(4). 指针的类型决定了指针解引用时有多大的权限(能操作几个字节

                        


                  

 2.2 * 指针类型的意义(指针 +- 整数)

             

//指针类型的意义2:
#include <stdio.h>
int main()
{
	int a = 0x11223344; //0x开头的是16进制数字

	int* pa = &a;
	char* pc = &a;

	printf("%p\n", pa);
	printf("%p\n", pc);

	printf("%p\n", pa+1);
	printf("%p\n", pc+1);

	return 0;
}

                         

总结:

                

指针的类型决定了指针向前或者向后走一步有多大(步长)

                     


                            

3. 野指针

                   

概念:野指针就是这种指向的位置时不可知的随机的不正确的没有明确限制的

                 


                       

3.1 * 野指针成因

                

1. 指针未初始化

                

               

2. 指针越界访问

                       

                    

3. 指针指向的空间释放

            


                    

3.2 * 如何规避野指针

                  

1. 指针初始化

                

(1). 明确知道这种应该初始化为谁的地址,就直接初始化

                 

(2). 不知道这种初始化为何值时,暂时初始化为NULL(空指针),没有指向任何有效的空间,该指针不能直接使用

                

             

2. 小心指针越界

              

3. 指针指向的空间释放,及时将它置为NULL(空指针)

               

4. 避免返回局部变量的指针

                  

5. 指针使用之前检查有效性

               

                   


                     

4. 指针运算

              

  • 指针        +-        整数

          

  • 指针        -        指针

            

  • 指针的关系运算

              


               

4.1 * 指针        +-        整数

//指针运算
#include <stdio.h>
int main()
{
	int arr[10] = { 0 };
	//不使用下标访问数组
	int* p = &arr[0]; //相当于arr
	int i = 0;
	int sz = sizeof(arr) / sizeof(arr[0]); //数组长度

	//使用指针运算循环赋值:
	//第一种方法:
	for (i = 0; i < sz; i++)
	{
		*p = i;
		p++; //指针运算
		// p = p + 1; 指针加一,移到数组中的下一位
	}
	
	//第二种方法:
	//for (i = 0; i < sz; i++)
	//{
	//	*(p + i) = i; 
	//}

	//使用指针打印
	p = arr; //地址位置还原到数组第一个元素位置进行打印
	for (i = 0; i < sz; i++)
	{
		printf("%d ", *(p + i));
	}

	return 0;
}

             

补充:

                  

                    

4.2 * 指针        -        指针

//地址-地址(指针-指针)
#include <stdio.h>
int main()
{
	int arr[10] = { 0 };
	printf("%d\n", &arr[9] - &arr[0]);
	printf("%d\n", &arr[0] - &arr[9]);

	// 指针-指针 得到的数值的绝对值:是指针和指针之间的元素个数
	
	// 指针-指针 运算的前提条件:指针和指针指向了同一块空间(同一数组)

	return 0;
}

                   

               

应用:使用 指针 - 指针 求字符串长度

//应用:
#include <stdio.h>

//第一种版本:正常思路
//int my_strlen(char* s)
//{
//	int count = 0; //统计字符串个数
//	if (*s != '\0')//数组首元素不是结束标志
//	{
//		count++; //能进来说明有元素,个数++
//		s++; //判断下一位
//	}
//	return count;
//}

//第二种版本:递归
//int my_strlen(char* s)
//{
//	if (*s == '\0')
//	{
//		return 0;
//	}
//	else
//	{
//		return 1 + my_strlen(s + 1);
//	}
//}

//第三种版本:指针减指针求数组元素个数
int my_strlen(char* s)
{
	char* start = s;
	while (*s != '\0') //也可以写成 while(*s) ,\0的ASCII码值是0
	{
		s++;
	} //循环到找到 \0 的地址
	return s - start; //指针-指针:\0地址 - 首地址 = 元素个数
}


int main()
{
	char arr[] = "abcdef";
	int len = my_strlen(arr);
	printf("%d\n", len);

	return 0;
}

             

           

4.3 * 指针的关系运算

//指针的关系运算(比较大小)
#include <stdio.h>

#define N_VALUES 5 //定义宏:数组长度为5

int main()
{
	float values[N_VALUES]; //浮点数数组
	float* vp;

	for (vp = &values[N_VALUES]; vp > &values[0];)
	// vp 等于 数组第五个元素的地址;vp 大于 数组第0个元素的地址
	{
		*--vp = 0; // vp先 -- ,再解引用 vp,把第四个元素赋值为0,然后是第三个、第二个……
	}
	//实现倒着给数组赋值

	int i = 0;
	for (i = 0; i < N_VALUES; i++)
	{
		printf("%d ", *(vp + i));
	}

	return 0;
}

               

标准规定:

允许指向数组元素的指针指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较

  • 13
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
C语言中的智能指针是一种自动管理内存的指针,可以帮助开发人员避免内存泄漏和悬空指针等问题。但是,C语言本身并没有提供智能指针的实现,需要程序员自己手动编写。 C++中的智能指针是用来管理动态分配的内存的类,它们使用 RAII(资源获取即初始化)技术来确保在指针超出作用域时,内存被正确释放。C++标准库中提供了两种智能指针:std::shared_ptr 和 std::unique_ptr。Boost库中也提供了类似的智能指针实现。 std::shared_ptr 是基于引用计数的智能指针,它允许多个 shared_ptr 指向同一个对象。每当一个新的 shared_ptr 被创建时,其引用计数会加1,每当一个 shared_ptr 被销毁时,其引用计数会减1。当引用计数为0时,对象被自动销毁。 std::unique_ptr 是一种独占式的智能指针,它不支持拷贝和赋值操作,但支持移动语义。unique_ptr 只能有一个指向同一对象的实例,当 unique_ptr 被销毁时,它所指向的对象也会被销毁。 Boost库中的智能指针实现与C++标准库类似,但提供了更多的选项和扩展功能。例如,Boost库中的 shared_ptr 支持自定义删除器,可以在对象销毁时执行自定义操作。此外,Boost库还提供了一些其他类型的智能指针,如 scoped_ptr 和 intrusive_ptr 等。 总体来说,C++标准库和Boost库中的智能指针实现都非常强大和灵活,可以大大简化内存管理和错误处理。程序员可以根据自己的需求选择最适合的智能指针类型

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

高高的胖子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值