开发者社区> wuyudong> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

C语言指针的长度和类型

简介:
+关注继续查看

如果考虑应用程序的兼容性和可移植性,指针的长度就是一个问题,在大部分现代平台上,数据指针的长度通常是一样的,与指针类型无关,尽管C标准没有规定所有类型指针的长度相同,但是通常实际情况就是这样。但是函数指针长度可能与数据指针的长度不同。

指针的长度取决于使用的机器和编译器,例如:在现代windows上,指针是32位或是64位长

测试代码:

#include<stdio.h>
#include<math.h>
#include<stdlib.h>
#include<stddef.h>
struct p{
    int n;
    float f;
};
int main()
{
    struct p *sptr;
    printf("sizeof *char: %d\n", sizeof(char*));
    printf("sizeof *int: %d\n", sizeof(int*));
    printf("sizeof *float: %d\n", sizeof(float*));
    printf("sizeof *double: %d\n", sizeof(double*));
    printf("sizeof *struct: %d\n", sizeof(sptr));
    return 0;
}

运行结果:

指针相关的预定义类型:

  • size_t:用于安全地表示长度
  • ptrdiff_t:用于处理指针算术运算
  • intptr_t:用于存储指针地址
  • uintptr_t:用于存储指针地址

size_t类型

size_t 类型是标准C库中定义的,应为unsigned int,在64位系统中为 long unsigned int。 C语言中,此类型位于头文件stddef.h中。它是一个与机器相关的unsigned类型,其大小足以保证存储内存中对象的大小,它的目的是 提供一种可移植的方法来声明与系统中可寻址的内存区域一致的长度:

因为C/C++标准只定义一最低的位数,而不是必需的固定位数。而且在内存里,对数的高位对齐存储还是低位对齐存储各系统都不一样。为了提高代码的 可移植性,就有必要定义这样的数据类型。一般这种类型都会定义到它具体占几位内存等。当然,有些是编译器或系统已经给定义好的。经测试发现,在32位系统 中size_t是4字节的,而在64位系统中,size_t是8字节的,这样利用该类型可以增强程序的可移植性。

size_t类型用作sizeof操作符的返回类型,同时也是很多函数的参数类型,包括malloc和strlen

在声明例如字符数、或者数组索引这样的长度变量时用size_t是好的做法,它经常用于循环计数器、数组索引,有时候还用在指针算术运算上

打印size_t类型的值要小心,这是无符号值,如果选错格式说明符,可能会得到不可靠的结果,推荐的格式说明符是%zu,在某些情况下可以考虑用%u或%lu替代

ptrdiff_t类型

ptrdiff_t是C99标准库中定义的一个与机器相关的数据类型,定义在stddef.h这个文件内。ptrdiff_t类型变量通常用来保存两个指针减法操作的结果。
ptrdiff_t通常被定义为long int类型,size_t 是unsigned 类型,而 ptrdiff_t 则是 signed 整型。
这两种类型的差别体现了它们各自的用途:size_t 类型用于指明数组长度,它必须是一个正数;ptrdiff_t 类型则应保证足以存放同一数组中两个指针之间的差距,它有可能是负数。
#include<stdio.h>
#include<stddef.h>
#include<string.h>
int main(void)
{
    char str[] = "Hello world!";
    char *pstart = str;
    char *pend = str + strlen(str);
    ptrdiff_t difp = pend - pstart;
    printf("%d\n", difp);
    return 0;
}

intptr_t与uintptr_t类型

intptr_t与uintptr_t类型用来存放指针地址,它们提供了一种可移植且安全的方法声明指针,而且与系统中使用的指针的长度相同,对于把指针转化为整数形式很有用。uintptr_t是intptr_t的无符号版本

关于intptr_t的类型定义如下:

/* Types for `void *' pointers.  */
#if __WORDSIZE == 64
# ifndef __intptr_t_defined
typedef long int               intptr_t;
#  define __intptr_t_defined
# endif
typedef unsigned long int    uintptr_t;
#else
# ifndef __intptr_t_defined
typedef int                    intptr_t;
#  define __intptr_t_defined
# endif
typedef unsigned int        uintptr_t;
#endif

从定义可以看出,intptr_t在不同的平台是不一样的,始终与地址位数相同,因此用来存放地址。

概念上, 尽管地址是指针, 内存管理常常使用一个无符号的整数类型更好地完成; 内核对待物理内存如同一个大数组, 并且内存地址只是一个数组索引. 进一步地, 一个指针容易解引用; 当直接处理内存存取时, 你几乎从不想以这种方式解引用. 使用一个整数类型避免了这种解引用, 因此避免了 bug. 因此, 内核中通常的内存地址常常是 unsigned long, 利用了指针和长整型一直是相同大小的这个事实, 至少在 Linux 目前支持的所有平台上.C99 标准定义了 intptr_t 和 uintptr_t 类型给一个可以持有一个指针值的整型变量

测试代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdint.h>
#include <string.h>
#include <assert.h>

#define ID_STR_LEN   12
#define NAME_STR_LEN 10

typedef struct student
{
    char id[ID_STR_LEN];
    char name[NAME_STR_LEN];
    uint8_t age;
}student;

student * create_student()
{
    student *stu = (student *)malloc(sizeof(student));
    if (stu == NULL)
    return NULL;
    memset(stu, 0, sizeof(student));
    return stu;
}

void *free_student(student *stu)
{
    if (stu)
    free(stu);
    return 0;
}

static void init_student(student * stu)
{
    assert(stu);
    const char *id = "2013112210";
    const char *name = "Anker";
    uint8_t age = 21;
    memcpy(stu->id, id, strlen(id));
    memcpy(stu->name, name, strlen(name));
    stu->age = age;
}

static int handle_student(intptr_t handle)
{
    if (handle == 0)
    {
    return -1;
    }
    student *stu = (student*)handle;
    printf("id: %s\n", stu->id);
    printf("name: %s\n", stu->name);
    printf("age: %u\n", stu->age);
    return 0;
}

int main(void)
{
    student *stu;
    stu = create_student();
    init_student(stu);
    //将指针转换为intptr_t类型
    intptr_t handle = (intptr_t)stu;
    handle_student(handle);
    free_student(stu);
    return 0;
}

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
《C语言深度剖析》第四章 指针和数组 p5(完结)( C语言从入门到入土(进阶篇)(一)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
27 0
《C语言深度剖析》第四章 指针和数组 p3 C语言从入门到入土(进阶篇)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
16 0
《C语言深度剖析》第四章 指针和数组 p1 C语言从入门到入土(进阶篇)(一)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
42 0
《C语言深度剖析》第四章 指针和数组 p4 C语言从入门到入土(进阶篇)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
38 0
《C语言深度剖析》第四章 指针和数组 p1 C语言从入门到入土(进阶篇)(二)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
52 0
《C语言深度剖析》第四章 指针和数组 p2 C语言从入门到入土(进阶篇)
本章节文章是作者通过观看《C语言深度剖析》等各种资料总结的精华,基础部分省略了不少,是为了让大家能够更加深入了解C语言的魅力!因为为了避免与之前的文章发生赘述,所以就直接讲作者认为的精华部分哈!现在正文开始!
24 0
C语言-一维指针定义与使用
C语言-一维指针定义与使用
47 0
C语言-void类型作为万能指针类型
C语言-void类型作为万能指针类型
52 0
C语言-八道笔试题由浅入深玩转指针
本文章将带你刷8道比较有意思的指针笔试题,笔者将由深入浅出解析这些题目!必要的题目,作者已经加上内存布局图!希望本文对你有所帮助!
99 0
C语言中“野指针”、“悬空指针”是什么?
C语言中“野指针”、“悬空指针”是什么?
70 0
+关注
wuyudong
我的个人博客地址:http://wuyudong.com/
203
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载