void的含义
void的字面意思是“无类型”, void * 则为“无类型指针”, void * 可以指向任何类型的数据
void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量。让我们来试着来定义 void a; | error: variable or field 'a' declared void void真正发挥作用在于:
() 对函数返回的限定;
() 对函数参数的限定; 众所周知,如果指针p1和p2的类型相同,那么我们可以直接在p1和p2间相互赋值;
如果p1和p2执行不同的数据类型,则必须使用强制类型转换运算符把赋值运算符右边的指针类型转换为左边指针的类型。 float *p1;
int *p2;
//p1 = p2; //error: cannot convert 'int*' to 'float*' in assignment|
p1 = (float *)p2; 必须改为 p1 = (float *)p2
而void * 则不同,任何类型的指针都可以直接赋值给它,无需进行强制类型转换:
void *p1;
int *p2;
p1 = p2;
但这并不意味,void *也可以无需强制类型转换地赋值给其他类型的指针。
因为“无类型”可以包容“有类型”, 而“有类型”则不能包容“无类型”
void *p1;
int *p2;
p2 = p1; //error: invalid conversion from 'void*' to 'int*' [-fpermissive] void的使用 下面给出void关键字的使用规则:
,如果函数没有返回值,那么应声明为void类型
main.c
#include <stdio.h>
#include <stdlib.h>
add(int a, int b) {
return a + b;
} // 此方法无返回值类型,编译没问题
// 说明不加返回值说明的函数的确为int函数 int main()
{
printf("2+3 = %d", add(, ));
return ;
}
在编写c/c++程序时,对于任何函数都必须一个不漏的指定其类型。void也有“自注释”的作用。 , 如果函数无参数,那么应该声明其参数为void
#include <stdio.h> fun() {
return ;
} int main()
{
printf("%d", fun());
return ;
}
编译正确说明,在c语言中,可以给无参数的函数传送任意类型的参数,但是c++编译器中编译同样的代码会出错。
在c++中,不能向无参数的函数传送任何参数。function does not take parameters
所以在c/c++中,若函数不接受任何参数,一定要指明参数为void。 , 小心使用void指针类型
ANSI:不能对void指针进行算法操作,即一下逻辑不合法:
void* pvoid;
pvoid++;
pvoid += ;
因为其坚持:进行算法操作的指针必须是确定知道其指向数据类型大小的。
eg:
int *pint;
pint ++; //ANSI 正确 [pint++的结果是使其增大sizeof(int)] 但是大名鼎鼎的GNU(GNU's Not UNIX)则不这么认为,它指定void *的算法操作与char * 一致。 void *pvoid;
pvoid++; //GNU:正确
pvoid += ; //GNU:正确
pvoid++的执行结果是其增大了1 在实际的程序设计中,为迎合ANSI标准,并提高程序的可移植性,我们可以这样实现同样的功能:
void *pvoid;
(char *)pvoid++; //ANSI:正确; GNU:正确
(char *)pvoid += ; //ANSI:错误; GNU:正确 , 如果函数的参数可以是任意类型的指针,那么应声明为void *
典型的如内存操作函数memcpy和memset的函数原型分别为:
void * memcpy(void *dest, const void *src, size_t len);
void * memset(void *buff, int c, size_t num);
这样,任何类型的指针都可以传入memcpy和memset中,这也真实地体现了“内存操作函数”的一样。
因为其操作的仅仅为一片内存,而不论这片内存是什么类型.
如果memcpy memset的参数类型不是void* 而是char* 那才叫奇怪了。
这样的memcpy和memset明显不是一个“纯粹的,脱离低级趣味的”函数! 示例:memset接受任意类型指针
int intarray[];
memset(intarray, , *sizeof(int)); //将intarray清0 示例:memcpy接受任意类型指针
int intarray1[], intarray2[];
memcpy(intarray1, intarray2, *sizeof(int)); //将intarray2拷贝给intarray1 有趣的是,memcpy memset函数返回的也是void* ,标准库的编写者是多么富有学问。[注:我们就是不带任何类型,俺们就是操作内存] , void不能代表一个真实的变量。
下面的代码企图让void代表一个真实的变量,因此都是错误的代码
void a; //wrong
function(void a); //wrong void体现了一种抽象,这个世界上的变量都是“有类型的”,譬如一个人不是男人,就是女人。(还有人妖)
void的出现只是为了一种抽象的需要,如果你正确理解了面向对象中“抽象基类”的概念,也很容易理解void数据类型。
正如不能给抽象基类定义一个实例,我们也不能定义一个void(让我们类比的称void为“抽象数据类型”)变量。 设计哲学!

c++ void,内存操作函数的更多相关文章

  1. 内存操作函数memmove,memcpy,memset

    通过字符串的学习,我们知道字符串操作函数的操作对象是字符串,并且它的结束标志是结束符\0,当然这个说的是不 受限制的字符串函数.然而当我们想要将一段内存的数据复制到另一块内存时,我们不能使用字符串操作 ...

  2. c#读写共享内存操作函数封装

    原文 c#读写共享内存操作函数封装 c#共享内存操作相对c++共享内存操作来说原理是一样,但是c#会显得有点复杂. 现把昨天封装的读写共享内存封装的函数记录下来,一方面希望给需要这块的有点帮助,另一方 ...

  3. Delphi中复制带有String的记录结构时不能使用Move之类的内存操作函数

    请看下面的代码: program TestRecord; {$APPTYPE CONSOLE} uses  SysUtils,  Math; type  TRecordA = record    Na ...

  4. C语言中内存操作函数

      一.malloc/calloc 名称: Malloc/calloc 功能: 动态内存分配函数 头文件: #include <stdlib.h> 函数原形: void *malloc(s ...

  5. 自己实现内存操作函数memset(),memcmp(),memcpy(),memmove()

    1.memset()内存设置函数(初始化) void *my_memset(void* dest, int c, size_t count) { assert(dest != NULL); char  ...

  6. 【转】C内存操作函数

    一.malloc/calloc 名称: Malloc/calloc 功能:  动态内存分配函数 头文件: #include <stdlib.h> 函数原形: void *malloc(si ...

  7. c语言学习之基础知识点介绍(十九):内存操作函数

    一.malloc函数 /* 首先需要导入头文件 #include <stdlib.h> malloc void* malloc(n); n是字节大小 开辟堆空间,开辟的字节数以n为准 返回 ...

  8. PCI设备内存操作函数总结

    1.  ExAllocatePool() 函数说明: ExAllocatePool allocates pool memory of the specified type and returns a ...

  9. Delphi 的内存操作函数(1): 给字符指针分配内存

    马上能想到的函数有: GetMem AllocMem ReallocMem FreeMem GetMemory ReallocMemory FreeMemory New Dispose NewStr ...

随机推荐

  1. SQL Server 2008复制发布订阅(数据同步)

    数据库同步问题 1.有一台主数据库服务器A和另外一台数据库服务器B,客户端首先访问数据库B,当B数据库服务器挂掉时就访问A,当对数据库B进行DML操作时,同时对A进行更新,如果A与B之间通讯失败,则将 ...

  2. 3DES 加解密,对长度不限制

    #region 3DES /// <summary> /// 3DES加密 /// </summary> /// <param name="strString& ...

  3. ARM你必须知道的事儿——为啥“PC = PC + 8”?

    为啥是“PC = PC + 8”: “PC = PC + 8”其实这样写容易让人蒙了.“PC = PC + 8”真正含义应该是: 执行处代码地址 = PC - 8: 也就是说,”PC指向的地址“领先“ ...

  4. HDU1022 Train Problem I 栈的模拟

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1042 栈的模拟,题目大意是已知元素次序, 判断出栈次序是否合理. 需要考虑到各种情况, 分类处理. 常 ...

  5. Nginx 之并发优化

    客户端/服务端 连接数 ulimit -n 100000 nginx 链接数 10240 个 worker_connections 10240;允许打开文件数worker_processes 1;wo ...

  6. C#简单的加密类

    1.加密 public class EncryptHepler { // 验值 static string saltValue = "XXXX"; // 密码值 static st ...

  7. 实现Server.UrlEncode和Server.UrlDecode功能的js代码

    <script> var EncodeURI = function(unzipStr,isCusEncode){    if(isCusEncode){        var zipArr ...

  8. 水面波浪形View--第三方开源--WaveView(电量、能量、容量指示)

    这种WaveView在一些常见的APP开发中,以水面波浪波形的形象的生动展示手机还剩余多少电量,存储容量还有多少,比较形象直观生动. WaveView在github上的项目主页是:https://gi ...

  9. mikrotik/IPSec Dynamic End points Updater.rsc

    # IPSec Peer/Policy Updater for Dynamic WAN addresses # ============================================ ...

  10. haproxy 常用acl规则与会话保持

    一.常用的acl规则 haproxy的ACL用于实现基于请求报文的首部.响应报文的内容或其它的环境状态信息来做出转发决策,这大大增强了其配置弹性.其配置法则通常分为两 步,首先去定义ACL,即定义一个 ...