当前位置:  编程技术>c/c++/嵌入式

C语言指针的长度和类型深入分析

    来源: 互联网  发布时间:2014-10-28

    本文导语:  指针是C语言的精髓,本文就以实例的形式详细分析了C语言的长度和类型。对于初学者深入理解C语言程序设计有很好的参考价值。具体分析如下: 一般来说,如果考虑应用程序的兼容性和可移植性,指针的长度就是一个问题,...

指针是C语言的精髓,本文就以实例的形式详细分析了C语言的长度和类型。对于初学者深入理解C语言程序设计有很好的参考价值。具体分析如下:

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

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

测试代码如下:

#include
#include
#include
#include
struct p{
  int n;
  float f;
};
int main()
{
  struct p *sptr;
  printf("sizeof *char: %dn", sizeof(char*));
  printf("sizeof *int: %dn", sizeof(int*));
  printf("sizeof *float: %dn", sizeof(float*));
  printf("sizeof *double: %dn", sizeof(double*));
  printf("sizeof *struct: %dn", 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
#include
#include
int main(void)
{
  char str[] = "Hello world!";
  char *pstart = str;
  char *pend = str + strlen(str);
  ptrdiff_t difp = pend - pstart;
  printf("%dn", 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 
#include 
#include 
#include 
#include 
#include 

#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: %sn", stu->id);
  printf("name: %sn", stu->name);
  printf("age: %un", 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程序设计的学习有所帮助。


    
 
 

您可能感兴趣的文章:

  • C语言中有指针,因此C语言可以创建链表,那么Java语言没有指针,那Java是否可以创建链表呢?
  • C语言指针问题?
  • C语言中常量指针与指针常量区别浅析
  • C语言:指针问题求救~
  • C语言的指针类型详细解析
  • 一个c语言指针问题
  • C语言指针学习经验总结浅谈
  • 关于c语言指针的两处小tip分享
  • C语言数组指针的小例子
  • C语言入门之指针用法教程
  • C语言创建链表错误之通过指针参数申请动态内存实例分析
  • 基于C语言中野指针的深入解析
  • C语言安全之数组长度与指针实例解析
  • 新手提问c语言指针
  • c语言指针之二级指针示例
  • 解析C语言中空指针、空指针常量、NULL & 0的详解
  • 关于C语言指针赋值的问题详解
  • 深入理解C语言 static、extern与指针函数
  • 深入分析C语言中结构体指针的定义与引用详解
  • C语言中函数与指针的应用总结
  • 请教如何用c语言在linux下实现检查某一用户密码长度?
  • 各位大虾:请问UNIX环境下C语言函数memcpy拷贝的字符的长度有没有限制,若有,能不能修改,怎么修改?
  • 关于几种 c语言内部数据类型的 字节长度
  • c语言输出字符串中最大对称子串长度的3种解决方案
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • 深入理解:Java是类型安全的语言,而C++是非类型安全的语言
  • 深入解析C语言中常数的数据类型
  • 深入c语言continue和break的区别详解
  • C语言typedef与复杂函数声明问题的深入解析
  • 深入C语言把文件读入字符串以及将字符串写入文件的解决方法
  • C语言中的BYTE和char深入解析 iis7站长之家
  • 深入理解C语言的逻辑控制
  • Android笔记之:深入为从右向左语言定义复杂字串的详解
  • 基于C语言指令的深入分析
  • 深入理解c语言数组
  • 深入C语言内存区域分配(进程的各个段)详解
  • 深入解析C语言中typedef的四个用途
  • C语言中的BYTE和char深入解析
  • 深入理解C语言中编译相关的常见错误
  • C语言关键字auto与register的深入理解
  • 深入分析Linux下如何对C语言进行编程
  • 深入探讨C语言中局部变量与全局变量在内存中的存放位置
  • 对C语言中递归算法的深入解析
  • c语言B树深入理解
  • 深入Java7的一些新特性以及对脚本语言支持API的介绍
  • 2013年7月和2013年8月编程语言排行榜
  • 如何在GTK2.0下实现国际化(语言选择根据自己设置的语言,不用系统的语言)
  • 2017 年热门编程语言排行榜出炉,你的语言上榜没?
  • 求助,在linux下,c语言和汇编语言的接口是什么?
  • 苹果OS X和IOS下最新编程语言swift介绍
  • C语言中间语言 CIL
  • c语言判断某一年是否为闰年的各种实现程序代码
  • 最近学JSP,苦于HTML语言和JAVA语言太差,请教推荐几本书,thanks.
  • PHP编程语言介绍及安装测试方法
  • 动态编程语言 LIME编程语言
  • Linux下C语言strstr()查找子字符串位置函数详细介绍(strstr原型、实现及用法)


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3