一、简要介绍
1. 无类型指针有一个很重要的特点:可存放一切数据类型的地址(来者不拒,可存放任何地址!)
#include<stdio.h>
int main(){
int a=10;
char ch='a';
double dx=12.23;
int* p=&a;
char str[10]={"hello"};
void* vp=NULL; //vp可存放任何地址!
vp=&a;
vp=&ch;
vp=&dx;
vp=&p;
vp=str;
vp=&str;
//注意:str和&str值相同,但意义不同!(str:数组首元素的地址;&str:数组的地址!)
vp=&vp;
}
2. 无类型指针绝对不能解引用,不能执行加一(自增 / 算术运算)操作!!!(故:也不能sizeof(*vp); )
(1)至于为什么不能解引用:无类型指针可以存储任何地址,所以无法确定其指向哪里,所以不能解引用或自增。
(2)sizeof(vp); // 这种写法正确,指针永远是四字节!
(3)不过 在默认情况下,GUN认为void*和char*一样。(GUN中允许无类型指针进行算术运算,指针加一即加一个字节)
3. 无类型指针赋值(使用)时必须强转!!(目的:限定类型)
//int* ip=vp; 错误写法!!没有限定无类型指针指向哪里
int* ip=(int*)vp; //正确!
二、无类型指针的应用(泛型编程)
1.将任意类型数组的每个存储单元置为0
在未接触无类型指针前,我们可能需要对每个不同类型的数组操作配有对应的函数。但接触过无类型指针后,则无论对什么类型数组置零都只需要一个函数!(前提:只能置零!!)
#include<stdio.h>
#include<assert.h>
//size_t:系统中声明其为无符号整型 typedef unsigned int size_t;
void* my_memset(void* vp,unsigned char val,size_t count){
//该函数作用:把一个一个存储单元(一个字节一个字节字节)置为0!
//assert(vp!=NULL);
if(vp==NULL) return vp;
char* cp=(char*)vp;
while(count--){
*cp=val;
cp++;
}
return vp;
}
struct Student{
char s_name[20];
int _age;
};
int main(){
int ar[10];
double dx[5];
char str[10];
struct Student studx[10];
my_memset(ar,0,sizeof(ar));
my_memset(dx,0,sizeof(dx));
my_memset(str,0,sizeof(str));
my_memset(studx,0,sizeof(studx));
//调用完函数再一个一个输出测试即可
}
注意:(1)由于该函数只能一个字节一个字节置为某个值!所以它的泛型功能只能置零!!
(2)如果是字符型变量->调用该函数则可以将每个内存单元置为任意值!!(由于char类型占一个字节!) 如下:
2. 关于无类型指针的字符串比较函数(一个字节一个字节比较)
#include<stdio.h>
#include<assert.h>
int my_memcmp(const void* dest,const void* src,size_t count){
assert(dest!=NULL&&src!=NULL);
const char* dp=(const char*)dest; //不要忘记const修饰符!(只读不可改) 其实无类型指针指向本身就不可改,加const是为了约束读程序的人
const char* sp=(const char*)src;
while(count--){
if(*dp>*sp){
return 1;
}
else if(*dp<*sp){
return -1;
}
dp++;
sp++;
}
return 0;
}
int main(){
int ar[10]={1,2,3,4,5,6,7,8,9,10};
int br[10]={1,2,3,4,5,10,9,8,7,6};
int x=my_memcmp(ar,br,sizeof(int)*6); //sizeof(int)*n; 用n来控制比较范围
printf("%d\n",x);
return 0;
}