C语言宏定义结合全局变量的方法实现单片机串口透传模式

何谓透传?

根据百度百科给出的定义如下:

透传,即透明传输(pass-through),指的是在通讯中不管传输的业务内容如何,只负责将传输的内容由源地址传输到目的地址,而不对业务数据内容做任何改变。

在现实单片机产品开发过程中,如果存在多个串口,在调试打印某个模块信息的时候,大多数人的做法是将所有模块的TX、RX、GND引出来,分别接到不同的调试口去,通过PC终端去将这些信息分别打印出来。这样子做难免会弄错,甚至非常繁琐,万一不小心还会接错导致模块烧坏。

于是,透传模式的出现就是为了解决这样的问题,也就是在程序中指定使用一个串口,将其它模块的串口做一个连接,于是这个指定的串口就可以输出这些模块的串口信息,甚至,我们可以在PC上通过串口模块的TX线路将命令发送到下位机,根据命令来指定哪一个模块串口输出,这样方便快捷,也有利于分析问题,避免现实开发中的繁琐和误操作。

接下来来看实现逻辑:

//条件标志位,根据条件标志位决定打印的内容
int debug_condition = 0 ;
#define GENERAL_DEBUG 0
#define MODULE1    1
#define MODULE2    2
#define DEBUG(cond,...) \
 if(cond == GENERAL_DEBUG) \
  printf(__VA_ARGS__)
#define Module1_Debug(cond,...) \
 if(cond == GENERAL_DEBUG || cond == MODULE1) \
  printf(__VA_ARGS__)
#define Module2_Debug(cond,...) \
 if(cond == GENERAL_DEBUG || cond == MODULE2) \
  printf(__VA_ARGS__) 

GENERAL_DEBUG作为普通输出的一个标志,当全局变量debug_condition为该标志时,DEBUG宏有效。

MODULE1作为模块1输出的一个标志,当不需要其它调试信息的干扰,只需要模块1输出的时候,Module1_Debug宏有效,其余无效。MODULE2也是同样的道理。

用标准C语言模拟这个过程:

#include <stdio.h>
//条件标志位,根据条件标志位决定打印的内容
int debug_condition = 0 ;
#define GENERAL_DEBUG 0
#define MODULE1    1
#define MODULE2    2
#define DEBUG(cond,...) \
 if(cond == GENERAL_DEBUG) \
  printf(__VA_ARGS__)
#define Module1_Debug(cond,...) \
 if(cond == GENERAL_DEBUG || cond == MODULE1) \
  printf(__VA_ARGS__)
#define Module2_Debug(cond,...) \
 if(cond == GENERAL_DEBUG || cond == MODULE2) \
  printf(__VA_ARGS__)
int main(void)
{
 int cmd = 0 ;
 start:
 printf("请输入命令:\n");
 scanf("%d",&cmd);
 //将输入的命令转发给全局变量
 debug_condition = cmd ;
 switch(debug_condition)
 {
 case GENERAL_DEBUG:
  DEBUG(debug_condition,"jeadlaksl;dkl;ak\n");
  Module1_Debug(debug_condition,"Hello world %d %d %d %d\n",1,2,3,4);
  Module2_Debug(debug_condition,"Hello%d\n",1);
  break ;
 case MODULE1:
  DEBUG(debug_condition,"jeadlaksl;dkl;ak\n");
  Module1_Debug(debug_condition,"Hello world %d %d %d %d\n",1,2,3,4);
  Module2_Debug(debug_condition,"Hello%d\n",1);
  break ;
   case MODULE2:
  DEBUG(debug_condition,"jeadlaksl;dkl;ak\n");
  Module1_Debug(debug_condition,"Hello world %d %d %d %d\n",1,2,3,4);
  Module2_Debug(debug_condition,"Hello%d\n",1);
  break ;
 default:
  printf("命令输入有误!\n");
  goto start;
  break;
 }
 return 0 ;
} 

运行结果:

当输入除0,1,2以外的数,则无效,当输入为1时则打印模块1的调试信息

当输入为2时,只打印模块2的调试信息

当输入为0时,打印全部调试信息

有了实现方法,当然在单片机上去实现也就不难了,甚至是其它的平台,也一样可以用这样的方式去实现。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。如果你想了解更多相关内容请查看下面相关链接

(0)

相关推荐

  • android实现手机与单片机蓝牙模块通信

    本文实例为大家分享了android实现手机与单片机蓝牙模块通信的具体代码,供大家参考,具体内容如下 我是参考原博客的内容去写的,由于原博客写的不全,少了关键的几个类,然后我就凭借自己扎实的功底补出来了,现在蓝牙工作正常,能发能收!在看这边文章之前你要先了解一下蓝牙的工作状态,我的代码里面可能解释的不是很详细,但是我自己是能看懂的! package com.example.fsl.bluetooth; import android.app.Activity; import android.blue

  • 使用UART与PC通信实现msp430g2553单片机超声波测距示例

    适用于msp430g2553单片机  使用到了hc-sr04超声测距模块,使用UART与PC通信. 复制代码 代码如下: #include <msp430.h>long current_time;//最近一次测得时间/*MyPro*/#define LED_1 BIT0                      #define SW_2 BIT3                       #define TA1_1 BIT2                      //TA0.1 HC-S

  • Android与单片机通信常用数据转换方法总结

    Android与单片机通信常用数据转换方法 1.  将GB2312转化为中文,如BAFAC2DCB2B7→胡萝卜,两个字节合成一个文字 public static String stringToGbk(String string) throws Exception { byte[] bytes = new byte[string.length() / 2]; for (int j = 0; j < bytes.length; j++) { byte high = Byte.parseByte(s

  • c#实现51单片机频率计的代码分享(数字频率计设计)

    复制代码 代码如下: #include <reg51.h>#define uchar unsigned char#define uint unsigned int#define ulong unsigned long //按键sbit button=P1^7;//LEDsbit led2=P3^5;sbit led3=P3^7;//数码管位选sbit c0 = P3^0;sbit c1 = P3^1;sbit c2 = P3^2;sbit c3 = P3^3;//数码管段码uchar code

  • Android Socket 线程连接openwrt与arduino单片机串口双向通信的实例解析

    废话不多说了,直接给大家贴代码了,具体代码如下所示: package zcd.netanything; import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.PrintWriter; import java.net.Socket; import android.app.Fragment; import android.content.BroadcastReceiver; import and

  • 关于单片机按键问题性能提升总结

    最近看了一本书<8051单片机创新教程>,这本书比大学课本写得好,在大学的课本中,我们常见的按键扫描程序如下: unsinged char KeyScan(void) { unsigned char KeyValue=0; if(KEY_IO != 0xFF) //检测到有按键按下 { DelayNms(20); //延时 20 毫秒(严重影响单片机的运行效率) if(KEY_IO != 0xFF)//确认按键按下 { switch(KEY_IO) { case 0xFE: KeyValue=

  • Android单片机与蓝牙模块通信实例代码

    啦啦毕业了,毕业前要写毕业设计,需要写一个简单的蓝牙APP进行交互,通过参考网上资料,问题顺利搞定,下面小编把具体实现思路分享给大家,供大家参考. 1.Android蓝牙编程 蓝牙3.0及以下版本编程需要使用UUID,UUID是通用唯一识别码(Universally Unique Identifier),这是一个软件构建的标准,也是被开源基金会组织应用在分布式计算环境领域的一部分.在蓝牙3.0及下一版本中,UUID被用于唯一标识一个服务,比如文件传输服务,串口服务.打印机服务等,如下: #蓝牙串

  • C语言宏定义结合全局变量的方法实现单片机串口透传模式

    何谓透传? 根据百度百科给出的定义如下: 透传,即透明传输(pass-through),指的是在通讯中不管传输的业务内容如何,只负责将传输的内容由源地址传输到目的地址,而不对业务数据内容做任何改变. 在现实单片机产品开发过程中,如果存在多个串口,在调试打印某个模块信息的时候,大多数人的做法是将所有模块的TX.RX.GND引出来,分别接到不同的调试口去,通过PC终端去将这些信息分别打印出来.这样子做难免会弄错,甚至非常繁琐,万一不小心还会接错导致模块烧坏. 于是,透传模式的出现就是为了解决这样的问

  • C/C++语言宏定义使用实例详解

     C/C++语言宏定义使用实例详解 1. #ifndef 防止头文件重定义 在一个大的软件工程里面,可能会有多个文件同时包含一个头文件,当这些文件编译链接成 一个可执行文件时,就会出现大量"重定义"的错误.在头文件中实用#ifndef #define #endif能避免头文件的重定义. 方法:例如要编写头文件test.h 在头文件开头写上两行: #ifndef TEST_H #define TEST_H //一般是文件名的大写 头文件结尾写上一行: #endif 这样一个工程文件里同时

  • C语言宏定义#define的使用

    目录 无参宏定义 定义形式 带参宏定义 定义形式 #和##运算 #运算 用法: ##运算 用法: 变参宏 #ifndef 条件编译 宏定义是高级语言编译器提供的常用语法,其目的是利用某一标识符标识某个文本字符串.在编写程序时,如果程序中反复地使用某个数据或某段程序片段,就可以考虑将这个数据或程序片段定义为宏,然后每个出现该数据或程序片段的地方用宏名替代,选择宏定义来做的好处是程序简洁,可读性好,而且当需要修改这些相同的程序片段时,只要修改宏定义中的字符串即可,不需要修改多处. 宏定义命令:def

  • C语言宏定义容易认不清的盲区梳理

    目录 1.概念 3.宏不是函数 4.宏定义不是说明或语句 5.宏不是类型定义 6.与之相关的宏定义 7.总结 1.概念 #define命令是C语言中的一个宏定义命令,它用来将一个标识符定义为一个字符串,该标识符被称为宏名,被定义的字符串称为替换文本. 命令有两种格式:一种是简单的宏定义,另一种是带参数的宏定义. (1)简单的宏定义: #define<宏名> <字符串> #defineVALUE((sizeof(a))/sizeof(a[0])) (2) 带参数的宏定义 #defin

  • C语言宏定义的扩展定义讲解

    目录 1. 常量宏定义 2. 定义宏函数 3. 宏定义和#号结合 4. 宏定义和两个#结合 5. 宏定义和do…while()的结合 6. #ifdef…#else…#endif 7. #ifndef…#else…#endif 8. #if defined() … #else … #endif 1. 常量宏定义 使用c中的#define 来定义一个常量来表示一年有多少秒 #define SECONDS_PER_YEAR (606024*365)UL 求圆的周长: #define D® (r +

  • C语言宏定义使用分析

    1.如何区分宏定义中的"宏名称"和"宏字符串"?对于带参数的宏又该注意什么? 在宏定义中,"宏名称"和"宏字符串"是通过"空格"来区分的.编译器在处理时宏定义时,首先从"#define"后第一个空格开始读取字符串,直到遇见下一个空格为止,两个空格之间的字符串为"宏名称",确定好"宏名称"之后,本行的所有其他字符串都为"宏字符串"

  • C语言中宏定义的妙用方法

    最近看了intel在linux内核中的驱动,学习到了一个非常有用的小技巧,如下代码: #define IN #define OUT #define UAdress volatile unsigned int * #define Raw_buffer void * void SetHwiPortsDataReg(IN UAdress Register , IN int value) { _SetHwiPortsDataReg(Register,&value); } void _Out_Put_va

  • C语言中宏定义使用的小细节

    #pragma#pragma 预处理指令详解 在所有的预处理指令中,#Pragma 指令可能是最复杂的了,它的作用是设定编译器的状态或者是指示编译器完成一些特定的动作.#pragma指令对每个编译器给出了一个方法,在保持与C和 C++语言完全兼容的情况下,给出主机或操作系统专有的特征.依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的. 其格式一般为: #Pragma Para.............etc.. baike.baidu.com/view/1451188.htm

  • C语言详细分析宏定义的使用

    目录 一.C语言中函数的“缺陷” 二.再次理解函数 三.C语言中的宏 四.宏与函数的不同 五.编译器组成简介 六.宏使用示例 七.再论宏常量 八.小结 一.C语言中函数的“缺陷” 实参和形参之间仅仅是值传递,因此,函数中无法直接改变实参. 二.再次理解函数 函数是一种代码复用的手段 把实现某个功能的代码片段进行封装(当作一个整体) 给这个代码片段一个合适的名字(通过名字使用代码) 定义参数(定义代码片段需要处理的问题) 三.C语言中的宏 宏是C语言中代码复用的补充方式 宏定义语法:#define

  • C语言详细分析宏定义与预处理命令的应用

    目录 宏定义与预处理命令 预处理命令 - 宏定义 定义符号常量 定义傻瓜表达式 定义代码段 预定义的宏 函数 VS 宏定义 预处理命令 - 条件式编译 示例 宏定义与预处理命令 预处理阶段:处理宏定义与预处理命令: 编译期:检查代码,分析语法.语义等,最后生成.o或.obj文件: 链接期:链接所有的.o或.obj文件,生成可执行文件. 预处理命令 - 宏定义 定义符号常量 #define PI 3.1415926 #define MAX_N 10000 定义傻瓜表达式 #define MAX(a

随机推荐