define可变参数,float数据传输

define可变参数

一般在调试打印Debug信息的时候, 需要可变参数的宏. 从C99开始可以使编译器标准支持可变参数宏(variadic macros), 另外GCC也支持可变参数宏, 但是两种在细节上可能存在区别.

1. __VA_ARGS__

/*__VA_ARGS__ 将 "..." 传递给宏 . 如*/

#define debug(format, ...) fprintf(stderr, format, __VA_ARGS__)

2. GCC的复杂宏

/*GCC使用一种不同的语法,从而可以给可变参数一个名字,如同其它参数一样.*/

#define debug(format, args...) fprintf (stderr, format, args)

/* 这和第一条的宏例子是完全一样的,但是这么写可读性更强并且更容易进行描述. */

3. ##__VA_ARGS__

/* 上面两个定义的宏,如果出现 debug("A Message")的时候,由于宏展开后有个多余的逗号,所以将导致编译错误.为了解决这个问题,CPP 使用一个特殊的"##"操作,格式如下:  */

#define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)

/* 这里,如果可变参数被忽略或为空,"##"操作将使预处理器(preprocessor)去除掉它前面的那个逗号. */

2 float类型数据的传输

/* float型数据直接按照int型类似的发送肯定是不行的,这就需要采用指针的方法:*/

void float_char(float f,unsigned char *s)

{

unsigned char *p;

p = (unsigned char *)&f;

*s = *p;

*(s+1) = *(p+1);

*(s+2) = *(p+2);

*(s+3) = *(p+3);

}

/* 通过这种方法把数组s发送出去,在接受端接受到的就是标准的IEEE754结构的原始数据,也就是float型数据在内存中存放的值,如果需要得到这个float型数据的值还需要进行一个转换。

这种方法比较简单明了,这时候的串口接收端可以用现成的,不需要自己编写。 */

/*还可以采用共用体的方法,如果采用共用体时串口的接收端软件需要自己编写。我们知道共用体可以使不同的数据类型来共享相同的地址空间,所以程序如下:*/

void float_char(float f,unsigned char *s)

{

union change  {

float d;

unsigned char dat[4];

} r1;

r1.d = f;

*s = r1.dat[0];

*(s+1) = r1.dat[1];

*(s+2) = r1.dat[2];

*(s+3) = r1.dat[3];

}

/* 接收端采用同样的程序编写就可以得到float型数据的值了,不再需要其他的转换。 */

/* 类似的,传输long型或int型时也可以采用共用体的方法: */

void long_char(unsigned long l,unsigned char *s)

{

union change {

long d;

unsigned char dat[4];

}r1;

r1.d = l;

*s = r1.dat[0];

*(s+1) = r1.dat[1];

*(s+2) = r1.dat[2];

*(s+3) = r1.dat[3];

}

/*同时还有内存操作函数/调用库函数的方法,但这种方法对编译器要求过高*/

时间: 2024-10-12 20:19:28

define可变参数,float数据传输的相关文章

volatile,可变参数,memset,内联函数,宽字符窄字符,国际化,条件编译,预处理命令,define中##和#的区别,文件缓冲,位域

 1.volatile:要求参数修改每次都从内存中的读取.这种情况要比普通运行的变量需要的时间长. #include <stdio.h> #include <stdlib.h> #include <time.h> void main() { time_t start, end; double res = 0; time(&start);  //获取时间,传递给start //volatile强制每次从内存读取 volatile int i; for (i =

C语言的可变参数

可变参数给编程带来了很大的方便,在享受它带来的方便的同时,很有必要了解一下其实现方式,在了解编程语言的同时,也可以扩展编程的思路. 可变参数需要用到3个宏函数和一个类型,他们都定义在<stdarg.h>中,分别是: va_start(vl) va_arg(vl, type) va_end(vl) 其中vl是va_list类型,type就是对象类型(如int, double或 自定义的struct之类的). va_start函数用来初始化vl va_arg(vl, type)用来取得type类型

C语言可变参数函数实现原理

一.可变参数函数实现原理 C函数调用的栈结构: 可变参数函数的实现与函数调用的栈结构密切相关,正常情况下C的函数参数入栈规则为__stdcall, 它是从右到左的,即函数中的最右边的参数最先入栈. 本文地址:http://www.cnblogs.com/archimedes/p/variable-parameter.html,转载请注明源地址. 例如,对于函数: void fun(int a, int b, int c) { int d; ... } 其栈结构为 0x1ffc-->d 0x200

Printf可变参数使用

参考文档: http://bbs.csdn.net/topics/70288067 Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu 转载请标明来源 本文的二个重点: 1. 可变参数实际上通过首个参数的地址来获取其它参数的地址,因为是顺序存储传过来的 1.      可变参数为了处理方便,所有的浮点型都是按照double型压栈. 因此,像printf采用的可变参数,它直接使用%f显示double型,不需要管float型. 关注printf的精度,需要先关

c语言中对可变参数列表的简单理解

函数原型中一般情况下参数的数目是固定的,但是如果想在不同的时候接收不定数目的参数时该怎么办呢?c语言提供了可变参数列表来实现. 可变参数列表是通过宏来实现的,这些宏定义在stdarg.h的头文件中.头文件中声明了一个va_list类型和va_start.va_arg.va_end三个宏.我们使用可变参数列表的时候需要声明一个va_list类型的变量配合这三个宏使用. va_start(va_list变量名,省略号前面最后一个有名字的参数):在提取可变参数前必须调用这个宏实现初始化. va_arg

va_list可变参数

可变参数函数实现 va_list,va_start,va_arg,va_end va可变参数意思,variable-argument. 1. 头文件及实现 linux中定义在gcc头文件中,stdarg.h中. 41 #ifndef __GNUC_VA_LIST 42 #define __GNUC_VA_LIST 43 typedef __builtin_va_list __gnuc_va_list; 44 #endif 45 46 /* Define the standard macros f

宏可变参数列表

VA_LIST 是在C语言中解决变参问题的一组宏,定义在<stdarg.h>头文件下. VA_LIST的用法:       (1)首先在函数里定义VA_LIST型的变量,这个变量是指向参数的指针      (2)然后用VA_START宏初始化变量刚定义的VA_LIST变量,这个宏的第二个参数是第一个可变参数的前一个参数,是一个固定的参数.      (3)然后用VA_ARG返回可变的参数,VA_ARG的第二个参数是你要返回的参数的类型.      (4)最后用VA_END宏结束可变参数的获取.

可变参数函数详解

可变参数函数又称参数个数可变函数(本文也简称变参函数),即函数参数数目可变.原型声明格式为: type VarArgFunc(type FixedArg1, type FixedArg2, -); 其中,参数可分为两部分:数目确定的固定参数和数目可变的可选参数.函数至少需要一个固定参数,其声明与普通函数参数相同:可选参数由于数目不定(0个或以上),声明时用"-"表示("-"用作参数占位符).固定参数和可选参数共同构成可变参数函数的参数列表. 由于参数数目不定,使用可

c语言可变参数列表处理

函数参数的传递原理 函数参数在内存中是以栈的形式存取,从右至左入栈. 参数在内存中存放格式: 在进程中,堆栈地址是从高到低分配的.当执行一个函数的时候,将参数列表入栈,压入堆栈的高地址部分,然后入栈函数的返回地址,接着入栈函数的执行代码,这个入栈过程,堆栈地址不断递减. 总之,函数在堆栈中的分布情况是,地址从高到低,依次是:函数参数列表,函数返回地址,函数执行代码段.堆栈中,各个函数的分布情况是倒叙的.即最后一个参数在列表中地址最高部分,第一个参数在列表地址的最低部分.参数在堆栈中的 分布情况如