void指针

指针有两个属性:指向变量/对象的地址长度
但是指针只存储地址,长度则取决于指针的类型
编译器根据指针的类型从指针指向的地址向后寻址
指针类型不同则寻址范围也不同,比如:
int*从指定地址向后寻找4字节作为变量的存储单元
double*从指定地址向后寻找8字节作为变量的存储单元

1.void指针是一种特别的指针
   void *vp
  //说它特别是因为它没有类型
  //或者说这个类型不能判断出指向对象的长度

2.任何指针都可以赋值给void指针
  type *p;
  vp=p;
  //不需转换
  //只获得变量/对象地址而不获得大小

3.void指针赋值给其他类型的指针时都要进行转换
   type *p=(type*)vp;
   //转换类型也就是获得指向变量/对象大小
转:http://icoding.spaces.live.com/blog/cns!209684E38D520BA6!130.entry

4.void指针不能复引用
  *vp//错误
  因为void指针只知道,指向变量/对象的起始地址
  而不知道指向变量/对象的大小(占几个字节)所以无法正确引用

5.void指针不能参与指针运算,除非进行转换
   (type*)vp++;
  //vp==vp+sizeof(type)



#include<iostream>
#include<stdlib.h>
#include<string>
using namespace std;
typedef struct tag_st
{
char id[10];
float fa[2];
}ST;
//我在程序里面这样使用的
int main()
{
ST * P=(ST *)malloc(sizeof(ST));
strcpy(P->id,"hello!");
P->fa[0]=1.1;
P->fa[1]=2.1;

ST * Q=(ST *)malloc(sizeof(ST));
strcpy(Q->id,"world!");
Q->fa[0]=3.1;
Q->fa[1]=4.1;
void ** plink=(void **)P;
*((ST *)(plink)) = * Q; //plink要先强制转换一下,目的是为了让它先知道要覆盖的大小.
                         //P的内容竟然给Q的内容覆盖掉了.
cout<<P->id<<" "<<P->fa[0]<<" "<<P->fa[1]<<endl;
return 0;
}

<think>嗯,用户问的是C/C++中void指针的用法和相关问题。首先,我需要回忆一下void指针的基本概念。void指针,也就是void*,是一种通用指针,可以指向任意类型的数据。不过,使用的时候需要注意类型转换,因为不能直接解引用void指针。 然后,用户可能想知道具体的使用场景。比如内存管理函数malloc和calloc,它们返回的就是void*,这样可以分配给任何指针类型。此外,通用函数的参数也会用到void指针,比如qsort和memcpy,这些函数需要处理不同类型的数据,void指针提供了灵活性。 接下来是示例代码部分。我需要确保代码正确,比如malloc返回的void*需要强制类型转换。比如int* ptr = (int*)malloc(sizeof(int));。还有,展示如何用void指针作为函数参数,处理不同类型的数据,比如打印int和float的例子。 可能的问题点包括类型安全,因为void指针绕过了类型检查,容易出错。还有内存管理的问题,比如分配和释放的大小必须一致,否则会导致内存泄漏或崩溃。此外,解引用前必须转换类型,否则编译器会报错。 用户还可能关心如何避免常见错误,比如总是初始化指针、正确转换类型、及时释放内存等。这部分需要给出实际的建议,比如分配后检查指针是否为NULL,使用完毕后置为NULL防止悬空指针。 然后,根据引用内容,用户提供的参考资料中有关于悬空指针的例子,比如引用[1]中的free后ptr成为悬空指针,之后置为NULL。这可能和void指针的使用有关联,比如在释放内存后,void指针也需要同样的处理,避免成为悬空指针。 另外,引用[3]提到了const成员函数和mutable关键字,虽然和void指针无直接关系,但可能在类型转换时需要注意const的正确性,避免修改不应该修改的数据。 最后,生成相关问题,需要覆盖void指针的基本用法、注意事项、与普通指针的区别、类型转换的问题、在数据结构中的应用以及调试技巧。这些问题应该能帮助用户更全面地理解void指针的使用。</think>### C/C++ void指针用法详解 void指针(通用指针)是C/C++中用于存储任意类型内存地址的特殊指针类型,具有以下核心功能和使用场景: #### 一、基础特性 1. **泛型指针**:可指向任意数据类型 ```c int num = 10; float f = 3.14; void *vp = &num; // 指向int vp = &f; // 改为指向float ``` 2. **强制类型转换**:必须显式转换后才能访问数据 ```c printf("Float value: %.2f\n", *(float*)vp); ``` #### 二、典型应用场景 1. **内存分配函数**: ```c int* arr = (int*)malloc(10*sizeof(int)); // malloc返回void* ``` 2. **通用函数参数**(需配合数据类型判断): ```c void print_data(void *data, char type) { if(type == 'i') printf("%d\n", *(int*)data); else if(type == 'f') printf("%f\n", *(float*)data); } ``` 3. **硬件编程**(直接操作内存地址): ```c void *device_reg = (void*)0xFFFF0000; // 内存映射I/O地址 ``` #### 三、关键注意事项 1. **禁止直接解引用**: ```c void *vp = &num; // printf("%d", *vp); // 编译错误 ``` 2. **指针运算限制**: ```c vp++; // 无效操作,void类型大小未知[^3] ``` 3. **类型安全风险**: ```c float f = 3.14; void *vp = &f; int wrong = *(int*)vp; // 错误类型转换导致数据错误 ``` #### 四、实践建议 1. **类型追踪**:使用联合类型或结构体记录原始类型 ```c struct TypedPointer { void *ptr; enum { INT, FLOAT } type; }; ``` 2. **内存对齐**:跨平台开发时特别注意 ```c void *aligned_alloc(size_t alignment, size_t size); ``` 3. **智能指针配合**(C++): ```cpp std::unique_ptr<void, void(*)(void*)> smart_ptr(malloc(100), free); ``` #### 五、调试技巧 1. 使用gdb的`x`命令查看void指针内存: ```shell (gdb) x/4xb void_ptr # 查看前4字节的十六进制值 ``` 2. Valgrind检测非法访问: ```shell valgrind --leak-check=full ./program ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值