// 任务目的
// 解析串口收到的54个字节。这54个字节包含了8个车道的5大信息以及校验信息。
// 实现了查询每条车道包含了哪些信息。

#include <stdio.h>
#include <malloc.h>
#include <assert.h>

typedef unsigned char mybool;
typedef unsigned char u8;
typedef unsigned short u16;
typedef unsigned int u32;

mybool SumCheck(const u8* pScrData);

typedef enum{
RoadID1=1, // 距离雷达最远的那条车道, 一个绿化带也算一条车道。
RoadID2,
RoadID3,
RoadID4,
RoadID5,
RoadID6,
RoadID7,
RoadID8,
RoadIDNull
}RoadID_;

typedef enum{
T_LongCarCnt=1,
T_CarCnt,
T_SmallCarCnt,
T_Occupancy,
T_AveSpeed,
T_AskNull
}T_ask_;

struct CertainRoad_; // 下面定义函数指针时,形参有这个结构体的指针, 所以,这里的结构体一定要先声明。

typedef struct CertainRoad_{
RoadID_ RoadID;

// 成员函数的使用方法示例: pRoad->pGetUpdated_LongCarCnt(pRoad, FakeData);
// 第一个pRoad是CertainRoad类的一个对象的指针,作用不言而喻。
// 第二个pRoad起到类似this指针的作用。目的是可供调用对象的其他成员(对象的其他属性或者行为。)
// C++的类内的成员函数的定义内可以直接使用this指针,但是C语言不支持这样,所以要通过形参再次传入对象指针。

//u8(*pGetUpdated_LongCarCnt) ( struct CertainRoad_ *pRoad,  const u8 *pScrData);         // 所以说,形参命名为pRoad,还不能体现是和C++映衬的最佳的理解。

u8(*pGetUpdated_LongCarCnt) ( struct CertainRoad_ *this,                  const u8 *pScrData); // 第二个参数是54字节的数据哦
u8(*pGetUpdated_CarCnt) ( struct CertainRoad_ *this,                          const u8 *pScrData);
u8(*pGetUpdated_SmallCarCnt)( struct CertainRoad_ *this,                  const u8 *pScrData);
u8(*pGetUpdated_Occupancy) ( struct CertainRoad_ *this,                   const u8 *pScrData);
u8(*pGetUpdated_AveSpeed) ( struct CertainRoad_ *this,                    const u8 *pScrData);

void(*pGetUpdatedAllresults)( struct CertainRoad_ *this,      const u8 *pScrData, u8 *result);

}CertainRoad; // Certain是某的意思。CertainRoad:某条车道。

// 根据车道号获取该时间段内的长车的流量
// 规律:长车的流量 = pScrData[pRoad->RoadID+2];
u8 GetUpdated_LongCarCnt(CertainRoad* pRoad, const u8* pScrData)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);
return pScrData[pRoad->RoadID+2];
}

u8 GetUpdated_CarCnt(CertainRoad* pRoad, const u8* pScrData)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);
return pScrData[pRoad->RoadID+15];
}

u8 GetUpdated_SmallCarCnt(CertainRoad* pRoad, const u8* pScrData)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);
return GetUpdated_CarCnt(pRoad,pScrData) - GetUpdated_LongCarCnt(pRoad, pScrData);
}

u8 GetUpdated_Occupancy(CertainRoad* pRoad, const u8* pScrData)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);
return pScrData[pRoad->RoadID+28];
}

u8 GetUpdated_AveSpeed(CertainRoad* pRoad, const u8* pScrData)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);
return pScrData[pRoad->RoadID+41];
}

void GetUpdatedAllresults(CertainRoad* pRoad, const u8* pScrData, u8* result)
{
if( !(pRoad->RoadID >= RoadID1)&&(pRoad->RoadID <= RoadID8) )
assert(0);

result[0] = pRoad->pGetUpdated_LongCarCnt (pRoad, pScrData);
result[1] = pRoad->pGetUpdated_CarCnt (pRoad, pScrData);
result[2] = pRoad->pGetUpdated_SmallCarCnt(pRoad, pScrData);
result[3] = pRoad->pGetUpdated_Occupancy (pRoad, pScrData);
result[4] = pRoad->pGetUpdated_AveSpeed (pRoad, pScrData);
}

CertainRoad* CreatNewRoad(RoadID_ RoadID)
{
CertainRoad* pNewRoadObj = (CertainRoad*)malloc(sizeof(CertainRoad));

pNewRoadObj->RoadID = RoadID;

pNewRoadObj->pGetUpdated_LongCarCnt = GetUpdated_LongCarCnt;
pNewRoadObj->pGetUpdated_CarCnt = GetUpdated_CarCnt;
pNewRoadObj->pGetUpdated_SmallCarCnt= GetUpdated_SmallCarCnt;
pNewRoadObj->pGetUpdated_Occupancy = GetUpdated_Occupancy;
pNewRoadObj->pGetUpdated_AveSpeed = GetUpdated_AveSpeed;

// 对上面的5个再次封装,提供统一接口
pNewRoadObj->pGetUpdatedAllresults = GetUpdatedAllresults;

return pNewRoadObj;
}

//串口底层收到数据,先进性简单逻辑判断,
//判断依据:0xff是第一个数据,之后的是0x1b 0x10 0x11 0x12四者之一。
//不符合上述依据,丢弃数据。
//符合上述依据,开始计数(要算上之前的2个),一共计数到54字节,发送一个信号量给相应的处理任务进行解析。
//处理任务进行: 和校验 、 原始数据的格式封装与转换。
int main(void)
{
// 车道1 2 3 4 5 6 7 车道8 // 这里模拟一下原始数据
u8 FakeData[54]={0xFF, 0x1B, 0x09, 0x02, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x76, 0x94, // T_LongCarCnt:这一行都是长车流量
0xFF, 0x10, 0x09, 0xf5, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x01, 0x12, // T_CarCnt 总车流量
0xFF, 0x11, 0x09, 0x20, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x03, 0x3f, // T_Occupancy 占有率
0xFF, 0x12, 0x0B, 0x88, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x00, 0x00, 0x30, 0xD4};// T_AveSpeed 平均速度

// 感觉这样写得很丑,其实输入就应该是一个车道的ID,u8类型的数据即可,而不是这样填写枚举值。
// 改进方法:在CreatNewRoad函数内增加断言,利用switch case,将u8的ID值转为枚举值。并assert断言,防止输入参数大于8。
// 暂不改进
CertainRoad* pRoad1 = CreatNewRoad(RoadID1); // 每条车道是一个对象
CertainRoad* pRoad2 = CreatNewRoad(RoadID2);
CertainRoad* pRoad3 = CreatNewRoad(RoadID3);
CertainRoad* pRoad4 = CreatNewRoad(RoadID4);
CertainRoad* pRoad5 = CreatNewRoad(RoadID5);
CertainRoad* pRoad6 = CreatNewRoad(RoadID6);
CertainRoad* pRoad7 = CreatNewRoad(RoadID7);
CertainRoad* pRoad8 = CreatNewRoad(RoadID8);
// 注:当前VS是C89标准
printf(" Main Begin \n\n");

if(SumCheck(FakeData))
{
u8 LongCarCnt=0;
u8 CarCnt=0;
u8 SmallCarCnt=0;
u8 Occupancy=0;
u8 AveSpeed=0;

u8 Result[5] = {0};
u8 i=0, j=0;
CertainRoad*pRoad = (CertainRoad*)0; // NULL

for(i=1; i<=8; i++)
{
switch(i)
{
case 1:
pRoad = pRoad1;break;
case 2:
pRoad = pRoad2;break;
case 3:
pRoad = pRoad3;break;
case 4:
pRoad = pRoad4;break;
case 5:
pRoad = pRoad5;break;
case 6:
pRoad = pRoad6;break;
case 7:
pRoad = pRoad7;break;
case 8:
pRoad = pRoad8;break;
}

//本接口的使用方式一: 让用户一个一个调用
LongCarCnt = pRoad->pGetUpdated_LongCarCnt(pRoad, FakeData);
printf("pRoad%d: LongCarCnt = %d \n",i, LongCarCnt);

CarCnt = pRoad->pGetUpdated_CarCnt(pRoad, FakeData);
printf("pRoad%d: CarCnt = %d \n", i, CarCnt);

SmallCarCnt = pRoad->pGetUpdated_SmallCarCnt(pRoad, FakeData);
printf("pRoad%d: SmallCarCnt = %d \n", i, SmallCarCnt);

Occupancy = pRoad->pGetUpdated_Occupancy(pRoad, FakeData);
printf("pRoad%d: Occupancy = %d \n", i, Occupancy);

AveSpeed = pRoad->pGetUpdated_AveSpeed(pRoad, FakeData);
printf("pRoad%d: AveSpeed = %d \n", i, AveSpeed);

//本接口的使用方式二:让用户一个一个调用太麻烦了,再封装一层,直接返回5个有物理意义的字节作为结果。
pRoad->pGetUpdatedAllresults(pRoad, FakeData, Result);

for(j=0; j<5; j++)
{
printf(" Result[%d] = %d \n", j, Result[j]);
}
printf(" -----pRoad%d ---OVER---\n\n", i);
}
}
printf(" \n\n");
return 0;
}

//和校验,这是一个独立使用的函数
mybool SumCheck(const u8* pScrData)
{
mybool ret = 0;
u8 i = 0;
u16 CheckSUM1=0, CheckSUM2=0, CheckSUM3=0, CheckSUM4=0;

for(i=3; i<=11; i++)
{
CheckSUM1 += pScrData[i];
}
CheckSUM1 = CheckSUM1%256;

for(i=16; i<=24; i++)
{
CheckSUM2 += pScrData[i];
}
CheckSUM2 = CheckSUM2%256;

for(i=29; i<=37; i++)
{
CheckSUM3 += pScrData[i];
}
CheckSUM3 = CheckSUM3%256;

for(i=42; i<=52; i++)
{
CheckSUM4 += pScrData[i];
}
CheckSUM4 = CheckSUM4%256;

if( ( (u8)CheckSUM1 == pScrData[12]) && ((u8)CheckSUM2 == pScrData[25]) \
&& ((u8)CheckSUM3 == pScrData[38]) && ((u8)CheckSUM4 == pScrData[53]) )
{
ret = 1;
printf(" 本次串口收到的54字节 和检验 OK \n\n");
}
return ret;
}

C 面向对象编程 --- 一模块的串口协议解析的更多相关文章

  1. Python面向对象编程和模块

    在面向对象编程中,你编写表示现实世界中的事物和情景的类,并基于这些类来创建对象. 编写类时,你定义一大类对象都有的通用行为.基于类创建对象时,每个对象都自动具备这种通用行为,然后根据需要赋予每个对象独 ...

  2. javascript面向对象编程的3种常见封装形式解析

    javascript如何才能脱离函数式编程,拥抱面向对象编程呢,常见的有3种形式,其它形式可以说都是这3种的变种.   1.直接定义对象直接量的形式   var Util={     getType: ...

  3. java 面向对象编程 第20章 XML技术解析

    1.  XML:extended Markup Language  可扩展标记语言,利用标签和子标签方式描述数据. 2.  声明<?xml version=”1.0”?>版本号 注释< ...

  4. python模块、面向对象编程

    目录: 模块补充 xml 面向对象 一.模块补充 shutil: 文件复制模块:进行文件copy.压缩: 使用方法: 将文件内容拷贝到另一个文件中,可以部分内容 shutil.copyfileobj( ...

  5. 洗礼灵魂,修炼python(40)--面向对象编程(10)—定制魔法方法+time模块

    定制魔法方法 1.什么是定制魔法方法 首先定制是什么意思呢?其实就是自定义了,根据我们想要的要求来自定义.而在python中,其实那些所谓的内置函数,内置方法,内置属性之类的其实也是自定义出来的,不过 ...

  6. Python 编程核心知识体系-模块|面向对象编程(三)

    模块 面向对象编程

  7. Day07:常用模块,面向对象编程(对象&类)及内置函数

    今日内容:1.常用模块2.面向对象编程(*****)    介绍面向对象编程    类    对象3.内置函数------------------------------1.面向过程编程    核心“ ...

  8. day23 xml模块、面向对象编程介绍

    今日内容: 1.xml模块 2.面向对象编程 一.xml模块 什么是xml? xml是一种可扩展的标记语言 xml语言的语法: <person name="jack"> ...

  9. Python(三)基础篇之「模块&面向对象编程」

    [笔记]Python(三)基础篇之「模块&面向对象编程」 2016-12-07 ZOE    编程之魅  Python Notes: ★ 如果你是第一次阅读,推荐先浏览:[重要公告]文章更新. ...

随机推荐

  1. Failed to start component [StandardEngine[Tomcat].StandardHost[localhost].StandardContex

    问题描述: 在idea中maven构建web项目,启动Tomcat插件时,出现Failed to start component [StandardEngine[Tomcat].StandardHos ...

  2. css面试题汇总 (持续更新)

    前言:这篇随笔是为了准备后面的面试而整理的,网上各种面试题太多了,但是我感觉很多太偏了,而且实际开发过程中并不会遇到,因此这里我整理一些比较常用的,或者是相对比较重要的知识点,每个知识点都会由浅入深, ...

  3. 据说是最好的记忆工具——Anki

    http://www.ankichina.net/ .u1s1,确实挺好用,自建题库,全程自助. 可以插入文字.图片.音频,会安排合理的复习频率,可以随时同步,电脑手机版本全.

  4. java基础语法(二)

    一.运算符 算数运算符 算数运算符用在数学表达式中,它们的作用和在数学中的作用一样. 操作符 描述 例子 + 两数相加 1+1=2 - 两数相减 2-1=1 * 两数相乘 1*1=1 / 两数相除 1 ...

  5. 深入了解Netty【七】Netty核心组件

    1.Bootstrap与ServerBootstrap bootstrap用于引导Netty的启动,Bootstrap是客户端的引导类,ServerBootstrap是服务端引导类.类继承关系: 2. ...

  6. Visual Studio Code 下载安装

    1.官网下载:https://code.visualstudio.com/ 2.下载完成后,点开如果是黑屏的话,右键勾上.

  7. MySQL 8 安装教程(个人版)+创建用户

    Mysql 8的安装教程 解压到指定目录如:D:\WinInstall\mysql-8.0.19-winx64这时候你需要在根目录下创建两个文件,分别是data文件夹和my.ini文件,然后使用编辑器 ...

  8. 本机ping不通虚拟机,但虚拟机可以ping通本机时怎么解决

    在各自网络都连接的情况下,本机ping不通虚拟机,但虚拟机可以ping通本机时解决方案: 1.linux虚拟机中连接方式选择NAT模式 2.本地启动VMnet8,然后选择VMnet8的属性,手动输入和 ...

  9. [剑指Offer]18-题目一:删除链表的节点 题目二:删除链表中重复节点

    题目一 题目 O(1)时间复杂度删除给定链表节点. 题解 用待删除节点后一个节点的值覆盖待删除节点值,更新链接关系. 注意链表只有一个节点:删除尾结点:删除头节点的处理. 代码 class ListN ...

  10. 第一次软件工程与UML作业

    这个作业属于哪个课程 https://url.cn/IMQa18Jo 这个作业要求在哪里 https://edu.cnblogs.com/campus/fzzcxy/2018SE1/homework/ ...