整理C语言中各种类型指针的特性与用法

指针为什么要区分类型:

在同一种编译器环境下,一个指针变量所占用的内存空间是固定的。比如,在16位编译器环境 下,任何一个指针变量都只占用8个字节,并不会随所指向变量的类型而改变。

虽然所有的指针都只占8个字节,但不同类型的变量却占不同的字节数。

一个int占用4个字节,一个char占用1个字节,而一个double占用8字节;

现在只有一个地址,我怎么才能知道要从这个地址开始向后访问多少个字节的存储空间呢,是4个,是1个,还是8个。

所以指针变量需要它所指向的数据类型告诉它要访问多少个字节存储空间。

也就是说,如果不指定指针的类型,那么当指针指向一个变量的时候,她从首地址开始,但是它不知道什么时候停止,它不知道要访问多少个存储空间。比如有一个char类型的变量,我用一个指针指向它,但是这个指针我设置成int类型,这样一来这个指针就会向后访问四个字节的存储空间,很明显得到的结果不是char类型应该得到的,所以要区分类型。

只有指针是可以运算(移动)的,数组名是不可以的。

 int x[10];
 x++; //illegal
 int* p = x;
 p++; //legal

两指针变量相减所得之差是两个指针所指数组元素之间相差的元素个数。

实际上是两个指针值(地址)相减之差再除以该数组元素的长度(字节数)。
(pointer2地址值 - pointer地址值) / sizeof(所指向数据类型)
指针之间可以相减,但不可以相加(相加无意义)。
定义字符串:

字符数组:

char string[] = "hello";
printf("%s\n",string);

字符串指针指向字符串:

char *str = "hello"

使用字符数组来保存的字符串是存在”栈”里的,所以它是可读可写的,所以我们可以修改字符数组里的某个元素的值。

但是,使用字符指针来保存字符串,它保存的是字符串常量地址,"常量区"是只读的,所以是不可改的。

char *str = "hello";
*(str+1) = 'w'; // 错误

使用注意:

char *str;
scanf("%s", str); 

/* str是一个野指针,他并没有指向某一块内存空间,所以不允许这样写。如果给str分配内存空间是可以这样用的 */

 /********* 数组的方法****************/ 

 char name[20];
 scanf("%s",name);  

/************* 给字符针针分配内存空间的办法***********/ 

 char *name;
 name=(char*)malloc(50);   //此时name已经指向一个刚刚分配的地址空间。
 scanf("%s",name);

指针函数(是函数,返回值是指针)注意:

如果函数返回一个字符串,那么如果用一个数组以下面的形式来接的话,是会报错的:

char *test() {
  return "hello";
}

int main(int argc, const char * argv[]) {

  char names[10];

  names = test();

  return 0;
}

这是因为,返回的字符串相当于一个这样的数组:{‘h', ‘e', ‘l', ‘l', ‘o', ‘\0'},但是前面我们说过,数组如果在定义的时候没有用{}这种方式初始化,那么后面就不能再用这种方式初始化了,所以会出错。

解决方法:将char names[10]改为char *names或者char names[10]直接等于test()。

函数指针(是指针,指向函数):

格式:函数的返回值类型 (*指针变量名) (形参1, 形参2, ...);

int sum(int a,int b)
{
 return a + b;
}

int (*p)(int,int);
p = sum;

应用场景:

调用函数

将函数作为参数在函数间传递

函数指针能更灵活:

int minus(int a, int b)
{
  return (a - b);
}

int add(int a, int b)
{
  return (a + b);
}

int myFunction(int a, int b, int (*funcP) (int, int))
{
  return funcP(a, b);
}

int main()
{
  int minusResult = myFunction(10, 20, minus);
  int addResult = myFunction(10, 20, add);
   ...
  return 0;
}

/*
  函数指针能让程序更灵活,比如后续有乘、除函数的时候,只需实现这两个函数然后在主函数调用myFunction函数即可。如果是多人协作,不同的人写不同的功能,如果我们来写myFunction那么基本就不用修改就可以一直使用,非常灵活。
*/

技巧:

1、把要指向函数头拷贝过来

2、把函数名称使用小括号括起来

3、在函数名称前面加上一个*

4、修改函数名称

使用注意:

由于这类指针变量存储的是一个函数的入口地址,所以对它们作加减运算(比如p++)是无意义的。

如上例,如果想使用p这个函数指针,可以直接向使用sum一样:

int result = p(10, 10);

也可以这样:

int result = (*p)(10, 10);

结构体是一种自定义数据类型,注意,它是数据类型。

struct Student {
   char *name;
   int age;
 };

 struct Student stu;

注意,结构体的后面是有 ; 的。

在使用结构体类型的时候,要加上struct关键字。

定义结构体类型的同时定义变量:

struct Student {
  char *name;
  int age;
} stu;

这种在定义的同时也定义了变量,就相当于:

struct Student {
   char *name;
   int age;
 };

 struct Student stu;

定义结构体类型的同时定义变量,以后如果想继续使用这个结构体类型,仍然可以使用常规的方式定义:

struct Student newStu;

匿名结构体定义结构体变量:

struct {
  char *name;
  int age;
} stu;

这种匿名方式与上面的方式相比,虽然看起来更简洁(省去了结构名),但是要注意,这只能定义一个stu变量,而不能再定义新的变量,因为结构名没有了。

(0)

相关推荐

  • 详解C语言编程中的函数指针以及函数回调

    函数指针: 就是存储函数地址的指针,就是指向函数的指针,就是指针存储的值是函数地址,我们可以通过指针可以调用函数. 我们先来定义一个简单的函数: //定义这样一个函数 void easyFunc() { printf("I'm a easy Function\n"); } //声明一个函数 void easyFunc(); //调用函数 easyFunc(); //定义这样一个函数 void easyFunc() { printf("I'm a easy Function\n

  • C语言中的函数指针基础学习教程

    顾名思义,函数指针就是函数的指针.它是一个指针,指向一个函数.看例子: A) char * (*fun1)(char * p1,char * p2); B) char * *fun2(char * p1,char * p2); C) char * fun3(char * p1,char * p2); 看看上面三个表达式分别是什么意思? C)这很容易,fun3是函数名,p1,p2是参数,其类型为char *型,函数的返回值为char *类型. B) 也很简单,与C)表达式相比,唯一不同的就是函数的

  • C语言 字符串指针详解及示例代码

    C语言中没有特定的字符串类型,我们通常是将字符串放在一个字符数组中,这在<C语言字符数组和字符串>中已经进行了详细讲解,这里不妨再来演示一下: #include <stdio.h> int main(){ char str[] = "http://c.biancheng.net"; int len = strlen(str), i; //直接输出字符串 printf("%s\n", str); //每次输出一个字符 for(i=0; i<

  • 简要说明C语言中指针函数与函数指针的区别

    指针函数一般是指返回指针的函数: #include <stdio.h> int* fun(int *a) { return a; } int main(int argc, char **argv) { int a = 3; printf("%d", *(fun(&a))); return 0; } 函数指针是表示指向函数开始地址的指针: 首先要了解函数的调用过程: #include <stdio.h> int fun(int i) { return i

  • 详解C语言结构体中的函数指针

    结构体是由一系列具有相同类型或不同类型的数据构成的数据集合.所以,标准C中的结构体是不允许包含成员函数的,当然C++中的结构体对此进行了扩展.那么,我们在C语言的结构体中,只能通过定义函数指针的方式,用函数指针指向相应函数,以此达到调用函数的目的. 函数指针 函数类型 (*指针变量名)(形参列表):第一个括号一定不能少. "函数类型"说明函数的返回类型,由于"()"的优先级高于"*",所以指针变量名外的括号必不可少.  注意指针函数与函数指针表示

  • 深入解析C语言中函数指针的定义与使用

    1.函数指针的定义     函数是由执行语句组成的指令序列或者代码,这些代码的有序集合根据其大小被分配到一定的内存空间中,这一片内存空间的起始地址就成为函数的地址,不同的函数有不同的函数地址,编译器通过函数名来索引函数的入口地址,为了方便操作类型属性相同的函数,c/c++引入了函数指针,函数指针就是指向代码入口地址的指针,是指向函数的指针变量. 因而"函数指针"本身首先应该是指针变量,只不过该指针变量指向函数.这正如用指针变量可指向整形变量.字符型.数组一样,这里是指向函数.C在编译时

  • 对C语言中指针的理解与其基础使用实例

    C语言的指针,关键意思在于"指". "指"是什么意思? 其实完全可以理解为指示的意思.比如,有一个物体,我们称之为A.正是这个物体,有了这么个称谓,我们才能够进行脱离这个物体的实体而进行一系列的交流.将一个物体的指示,是对这个物体的抽象.有了这种抽象能力,才有所谓的智慧和文明.所以这就是"指示"这种抽象方法的威力. 退化到C语言的指针,指针是一段数据/指令(在冯诺易曼体系中,二者是相通,在同一空间中的)的指示.这是指示,也就是这段数据/指令的起始

  • 举例理解C语言二维数组的指针指向问题

    之前对数组的概念一直没有理解透彻,只觉得数组名就是个常量指针而已,用法和基本的指针差不多.所以当我尝试用二级指针去访问二维数组时,就经常会出错.下面就是刚开始写的一个错误的程序: #include <stdio.h> int main() { int iArray[2][3] = {{1,2,3},{4,5,6}}; int **pArray = NULL; pArray = iArray; printf("array[0][0] = %d\n", pArray[0][0]

  • C语言 数组指针详解及示例代码

    数组(Array)是一系列具有相同类型的数据的集合,每一份数据叫做一个数组元素(Element).数组中的所有元素在内存中是连续排列的,整个数组占用的是一块内存.以int arr[] = { 99, 15, 100, 888, 252 };为例,该数组在内存中的分布如下图所示: 定义数组时,要给出数组名和数组长度,数组名可以认为是一个指针,它指向数组的第 0 个元素.在C语言中,我们将第 0 个元素的地址称为数组的首地址.以上面的数组为例,下图是 arr 的指向: 下面的例子演示了如何以指针的方

  • 直观理解C语言中指向一位数组与二维数组的指针

    一维数组和指针: 对于一位数组和指针是很好理解的: 一维数组名: 对于这样的一维数组:int a[5];  a作为数组名就是我们数组的首地址, a是一个地址常量 . 首先说说常量和变量的关系, 对于变量来说, 用箱子去比喻再好不过了, 声明一个变量就声明一个箱子,比如我们开辟出一个苹果类型的箱子, 给这个变量赋值就是把盛放苹果的箱子中放入一个实实在在的苹果, 这就是变量的赋值.  而对于数组来说, 就是一组类型相同的箱子中,一组苹果箱子, 可以放入不同的苹果. 一维数组空间: 变量被声明后, 我

  • C语言中的函数指针学习笔记

    一.定义函数指针 return_type (*func_pointer)(parameter_list) 普通指针变量的定义 int * p; char * pointer; 类型的限定都在变量前面: 函数指针类型的限定是前后都有,前面是返回类型,后面是输入参数. 利用typedef 可以简化上面的表达方式. typedef return_type (*FunctionPointer) (parameter_list); FunctionPointer func_pointer; 这样是不是容易

随机推荐