指针深入解析(C语言基础)带你走进指针,了解指针

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 指针深入解析(C语言基础)带你走进指针,了解指针

内存和地址


 CPU(中央处理器)在处理数据的时候,中央的数据在内存中读取,处理后的数据也会放回内存中。


 内存会被划分为一个个单元格,每个内存单元的大小取一个字节。每个字节可以放8个比特位,每个内存单元也有一个编号,有了这个编号,CPU可以迅速找到一个内存空间。C语言中把这个编号也叫地址

C语言中给地址起了一个名字叫指针

指针变量

1.和地址

在C语言中创建变量就是向内存申请空间我们已经知道啦,比如,创建整型变量就是向内存申请4个字节,每一个字节都有地址,

2.和解引用操作符

 我们通过取地址操作符拿到是一个地址的数值,这个数值有时候需要存储下来,方便后期使用,所以我们这个放在  指针变量  中。

#include<stdio.h>
int main()
{
  int a = 0;
  int* pa = &a;
  return 0;
}

这里出现的Int *,*就是代表这个是指针变量。

 我们拿到了指针就可以找到指针指向的对象,需要用到一个操作符叫解引用操作符----*

#include<stdio.h>
int main()
{
  int a = 0;
  int* pa = &a;
  *pa = 0;
  printf("%d", a);
  return 0;
}

这里就改变了a的值。

指针变量的大小

 32位平台下地址就是32个bit,指针变量的大小的4个字节

 6位平4台下地址就是64个bit,指针变量的大小的8个字节

并且指针变量的大小与类型是无关的,管他什么整型,浮点型,在相同的平台下都是一样的。

指针变量类型的意义

 指针的类型决定了对指针解引用的时候有多大的权限,比如char类型只能1个字节,而int类型可以4个字节。

void*指针

 可以理解为无具体类型的指针,但是不可以进行指针的+-运算和解引用运算。

const修饰指针

 变量是可以被修改的,如果我们需要不可以被修改,那我们需要用到const,

对于 int*pa=&a;

    const 放在*左边,修饰的是指针指向的内容,保证指针指向的内容不能通过指针来改变,但是指针变量本身的内容可以改、

    const放在*右边,修饰是指针变量本身

const char *ptr="hello ", 本质就是把hello的首元素存到ptr中

指针运算

1.指针+-整数

2.指针-指针

3.指针的关系运算

  1.指针+-整数:数组在内存中是连续存放, 所以我们只要知道第一个元素的地址,就可以找到所有元素。

  2.指针-指针:

#include<stdio.h>
int strlen(char* s)
{
  char* p = s;
  while (*p != '\0')
  {
    p++;
  }
  return p - s;
}
int main()
{
  printf("%d\n", strlen("abc"));
  return 0;
}

野指针

 野指针就是指针指向的位置是不可知的。

原因:没有初始化,指针越界访问,指针指向的空间释放(就是可能创建的指针变量被销毁了)

数组名的理解

1.数组名就是数组首元素的地址,如下图

注意:

sizeof(数组名):计算的是整个数组的大小,单位是字节

&数组名:表示的是取出的是整个数组的地址

使用指针访问数组

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int main()
{
  int arr[10];
  int sz = sizeof(arr) / sizeof(arr[0]);
  int* p = arr;//首元素
  int i = 0;
  for (i = 0; i < sz; i++)
  {
    scanf("%d", p+i);
  }
  for (i = 0; i < sz; i++)
  {
    printf("%d ", *(p + i));
  }
  return 0;
}

一维数组传参的本质

本质来说,一维数组传参的本质传的是数组的首元素的地址

并且一维数组传参形参部分可以写成数组,也可以写成指针

#include<stdio.h>
void test(int arr[])
{
  printf("%d\n", sizeof(arr));
}
void test(int* arr)
 
{
  printf("%d\n", sizeof(arr));
 
}
 
int main()
{
  int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
  test(arr);
  return 0;
}

二级指针

二级指针就是存放一级指针的地址

因为你的一级指针存放地址的时候,也需要向内存申请一块空间,它自己本身是需要地址

#include <stdio.h>
int main()
{
  int a = 0;
  int* p = &a;
  int** pa = &p;
  return 0;
 
}

指针数组

整型数组:存放整型的数组

字符数组:存放字符的数组

指针数组:存放指针的数组,也就是说,这个数组里面的每一个元素都是指针

指针数组模拟二维数组

int main()
{
  int arr1[] = { 1,2,3,4,5 };
  int arr2 [] = {2,3,4,5,6};
  int arr3[] = { 3,4,5,6,7 };
  //数组名是元素首元素的地址
  int* p[3] = { arr1,arr2,arr3 };
  int i = 0;
  int j = 0;
  for (i = 0; i < 3; i++)
  {
    for (j = 0; j < 5; j++)
    {
      printf("%d ", p[i][j]);
    }
    printf("\n");
  }
  return 0;
}

数组指针变量

定义

整型指针变量: int*pint,存放的是整型变量的地址,指向的是整型数据的指针

浮点指针变量:float*pint,存放的是浮点型变量的地址,指向的是浮点型数据的指针

数组指针变量:int(*p)[ ] 存放的是数组的地址,能够指向数组的指针变量

数组指针变量初始化

#include<stdio.h>
int main()
{
  int arr[10] = { 0 };
  int(*p)[10] = &arr;
  return 0;
}

&arr与*p的类型是完全一致的

二维数组传参的本质

  二维数组的首元素就是第一行,是一个一维数组

在本质上传的也是地址,传递的就是这个一维数组的地址

函数指针变量

 函数是有地址的,函数名就是函数的地址,也可以用   &函数名   来表示

int       (*p)      (int x, int y)........p指向函数的参数类型和个数的交代

.               .

.               .

.               .

.            函数指针变量名字

.

p指向函数返回的类型

函数指针变量的使用

#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
int main()
{
  int (*p)(int, int) = Add;
  printf("%d\n", (*p)(2, 3));
  printf("%d\n", p(2, 3));
 
  return 0;
}

typeof关键字

可以将复杂的类型简单化

比如,你觉得unsigned int 写起来不方便

可以改成 typeof unsigned int uint;

之后你使用就可以直接使用 unit

指针类型也可以改名字

typeof int * ptr-t;

回调函数

 回调函数就是通过函数指针调用的函数

 如果你把函数的指针作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数,被调用的函数就是回调函数

回调前

#include<stdio.h>
int add(int a, int b)
{
  return a + b;
}
int sub(int a, int b)
{
  return a - b;
}
int mul(int a, int b)
{
  return a * b;
}
int div(int a, int b)
{
  return a / b;
}
int main()
{
  int x, y;
  int input = 1;
  int ret = 0;
  do
  {
    printf("******************************\n");
    printf("****  1. add     2. sub   ****\n");
    printf("****  3. mul     4. div   ****\n");
    printf("****  0. exit             ****\n");
    printf("******************************\n");
    scanf("%d", &input);
    switch (input)
    {
    case 1:
      printf("输入操作数:");
      scanf("%d%d", &x, &y);
      ret = add(x, y);
      printf("ret=%d\n", ret);
      break;
    case 2:
      printf("输入操作数:");
      scanf("%d%d", &x, &y);
      ret = sub(x, y);
      printf("ret=%d\n", ret);
      break;
    case 3:
      printf("输入操作数:");
      scanf("%d%d", &x, &y);
      ret = mul(x, y);
      printf("ret=%d\n", ret);
      break;
    case 4:
      printf("输入操作数:");
      scanf("%d%d", &x, &y);
      ret = div(x, y);
      printf("ret=%d\n", ret);
      break;
    case 0:
      break;
    }
 
  } while (input);
  return 0;
}

回调后

#include<stdio.h>
int add(int a, int b)
{
  return a + b;
}
int sub(int a, int b)
{
  return a - b;
}
int mul(int a, int b)
{
  return a * b;
}
int div(int a, int b)
{
  return a / b;
}
void calc(int(*pf)(int, int))
{
  int ret = 0;
  int x, y;
  pintf("输入操作数:");
  scanf("%d%d", &x, &y);
  ret = pf(x, y);
  printf("ret=%d\n", ret);
 
}
int main()
{
  int x, y;
  int input = 1;
  int ret = 0;
  do
  {
    printf("******************************\n");
    printf("****  1. add     2. sub   ****\n");
    printf("****  3. mul     4. div   ****\n");
    printf("****  0. exit             ****\n");
    printf("******************************\n");
    scanf("%d", &input);
    switch (input)
    {
    case 1:
      calc(add);
      break;
    case 2:
      calc(sub);
      break;
    case 3:
      calc(mul);
      break;
    case 4:
      calc(div);
      break;
    case 0:
      break;
    }
 
  } while (input);
  return 0;
}

qsort函数使用举例

 函数实现排序

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
int int_cmp(const void* p1, const void* p2)
{
  return (*(int*)p1 - *(int*)p2);
}
 
int main()
{
  int arr[] = { 1,3,5,6,8,2,4 };
  int i = 0;
  qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof(int), int_cmp);
  for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
  {
    printf("%d ", arr[i]);
  }
  printf("\n");
 
  return 0;
}

非常感谢您的阅读,多多关注呀,后面我也会尽心尽力为大家带来优质好文的


相关文章
|
1月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
174 14
|
1月前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
88 1
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
97 9
|
1月前
|
存储 编译器 C语言
【C语言】数据类型全解析:编程效率提升的秘诀
在C语言中,合理选择和使用数据类型是编程的关键。通过深入理解基本数据类型和派生数据类型,掌握类型限定符和扩展技巧,可以编写出高效、稳定、可维护的代码。无论是在普通应用还是嵌入式系统中,数据类型的合理使用都能显著提升程序的性能和可靠性。
66 8
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
56 7
|
1月前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
686 6
|
1月前
|
存储 网络协议 算法
【C语言】进制转换无难事:二进制、十进制、八进制与十六进制的全解析与实例
进制转换是计算机编程中常见的操作。在C语言中,了解如何在不同进制之间转换数据对于处理和显示数据非常重要。本文将详细介绍如何在二进制、十进制、八进制和十六进制之间进行转换。
72 5
|
1月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
54 5
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
117 2
|
1月前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是"将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。创建型模式分为5种:单例模式、工厂方法模式抽象工厂式、原型模式、建造者模式。
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析

推荐镜像

更多