自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

三種管理 C 程序中標志位的方法,最后一種比較秀

開發(fā) 后端
在嵌入式開發(fā)中難免會涉及到非常多的標志位處理,特別是玩單片機、裸機開發(fā)的朋友,比如跟一些模塊配合聯(lián)調(diào)會遇到各種信號是否到位、成功等等狀態(tài),而這些信號大多都是bool類型,1個bit即可進行標識。

正文

在嵌入式開發(fā)中難免會涉及到非常多的標志位處理,特別是玩單片機、裸機開發(fā)的朋友,比如跟一些模塊配合聯(lián)調(diào)會遇到各種信號是否到位、成功等等狀態(tài),而這些信號大多都是bool類型,1個bit即可進行標識。

當然如果僅僅是幾個標志,直接拿個uint8_t的整形來進行標識也不會影響什么,但如果特別多的話似乎就比較廢RAM了。

然而為了更好的管理這些標志位等,有個如下幾種方式供大家更好的管理這些標志位 :

1.位域直接標識

采用位域是管理這些標志位比較直接且方便的方式,代碼如下所示:

這些標志位的操作無非就是置位,清零、以及讀取三種方式。

1typedef union _tag_SystemFlag
2{
3 uint16_t all;
4 struct
5 {
6 uint16_t Run :1;
7 uint16_t Alarm :1;
8 uint16_t Online :1;
9 uint16_t TimerOver :1;
10 uint16_t Reserver :12;
11 }bit;
12
13} uSystemFlag;
14
15uSystemFlag unSystemFlag;16
17int main(int argc, char *argv[]) {
18
19 unSystemFlag.all = 0x00; //系統(tǒng)標志清除
20
21 unSystemFlag.bit.Run = 1; //置位
22 unSystemFlag.bit.Alarm = 1;
23 unSystemFlag.bit.Online = 1;
24 unSystemFlag.bit.TimerOver = 1;
25
26 unSystemFlag.bit.Run = 0; //清零
27 unSystemFlag.bit.Alarm = 0;
28 unSystemFlag.bit.Online = 0;
29 unSystemFlag.bit.TimerOver = 0;
30
31 return 0;
32}

但如代碼中這樣的操作方式在語句或語義表達上還是不夠直觀。

bug菌經(jīng)常談到,代碼可以不寫注釋,不過你的每個變量、函數(shù)名稱等需要足夠的直觀,所以很多朋友習(xí)慣把這些標志封裝起來。

2.枚舉+移位

為了更好的表達一般會對標志位進行進一步的封裝,如下代碼所示:

當然封裝成函數(shù)是相對比較耗時的,不過代碼也會更加的易懂,如果確實容忍不了函數(shù)封裝帶來的時間消耗,把函數(shù)修改為宏代碼片段或者內(nèi)斂函數(shù)(當然前提是編譯器支持)也是可行的。

1typedef enum _tag_Flag {
2cEmRun = 0,
3cEmAlarm,
4cEmOnline,
5cEmTimerOver
6}emSystemFlag;
7
8uint16_t SystemFlag ;
9//置位
10void SetFlag(emSystemFlag flag)
11{
12 SystemFlag |= ((uint16_t)0x01) << flag;
13}
14//清除
15void ClrFlag(emSystemFlag flag)
16{
17 SystemFlag &= ~(((uint16_t)0x01) << flag);
18}
19//獲得狀態(tài)
20uint8_t GetFlag(emSystemFlag flag)
21{
22 return (((SystemFlag & (((uint16_t)0x01) << flag)) != 0)? true:false);
23}
24
25int main(int argc, char *argv[]) {
26
27 SetFlag(cEmAlarm);
28
29 if(GetFlag(cEmAlarm) == true)
30 {
31 printf("ClrFlag\r\n");
32 ClrFlag(cEmAlarm);
33 }
34 else
35 {
36 printf("SetFlag\r\n");
37 SetFlag(cEmAlarm);
38 }
39 return 0;
40}

3.宏列表

或許這里才是本文的重中之重~

以前跟大家介紹過,用宏自動化的生成各種代碼片段,以使得代碼更加的緊湊。當然可讀性會相對降低一點,但對于重復(fù)性代碼就不需要太多考慮了。

如果以前有過類似代碼處理的朋友,應(yīng)該看這段代碼還是比較輕松的吧,如果有點生疏,可以一層一層展開了解。

1#include <stdio.h>
2#include <stdlib.h>
3
4typedef unsigned char uint8_t;
5typedef unsigned int uint16_t;
6typedef signed char int8_t;
7typedef int int16_t;
8
9#define true 1
10#define false 0
11
12
13//宏列表
14#define TAG_LIST(tag) \
15tag(Run)\
16tag(Alarm)\
17tag(Online)\
18tag(TimerOver)
19
20
21//枚舉處理
22#define DEFINE_TAG(_tag) _tag,
23enum Flag {
24None = 0,
25TAG_LIST(DEFINE_TAG)
26EmMAX
27};
28#undef DEFINE_TAG
29
30//位定義變量
31uint16_t SysFlag = 0x0000;
32
33
34//通用方法定義
35uint8_t GetFlags(uint16_t mask)
36{
37 return ((SysFlag & mask) != 0)? true:false;
38}
39
40void SetFlags(uint16_t mask)
41{
42 SysFlag |= mask;
43}
44
45void ClrFlags(uint16_t mask)
46{
47 SysFlag &= ~mask;
48}
49
50
51//自動生成三類函數(shù)定義
52#define FLAG_Operater(flag) \
53uint8_t get##flag() {\
54return GetFlags(1 << flag);\
55}\
56void set##flag() {\
57SetFlags(1 << flag);\
58}\
59void clr##flag() {\
60ClrFlags(1 << flag);\
61}
62
63//反向函數(shù)關(guān)聯(lián)
64TAG_LIST(FLAG_Operater)
65
66int main(int argc, char *argv[]) {
67
68 setRun();
69 setAlarm();
70
71 if(getAlarm() == true)
72 {
73 printf("set \r\n");
74 }
75 else
76 {
77 printf("clr \r\n");
78 }
79
80 return 0;
81}

其主要的功能就是通過宏替換和代碼拼接符號,自動的生成通用的代碼片段,這樣做的好處就是不再需要我們在代碼中定義一大堆setflag、clrflag、getflag等函數(shù)。

通過上面的代碼當我們向TAGLIST宏中添加一個標識符,即可生成一系列相關(guān)的操作函數(shù)等。

這樣一方面可以及簡化代碼,同時也避免一些人工編碼帶來的錯誤。

責任編輯:龐桂玉 來源: C語言與C++編程
相關(guān)推薦

2009-07-30 16:27:33

C#比較時間

2010-07-19 14:43:21

SQL Server查

2018-01-24 15:50:16

2012-03-26 12:23:25

JavaSwing

2010-06-12 11:03:02

UML應(yīng)用

2022-08-02 13:56:37

C開發(fā)段錯誤

2025-02-10 08:43:31

Java異步編程

2009-06-09 16:53:22

Java Swing處理方法比較

2010-06-28 17:43:44

SQL Server

2024-05-27 00:20:00

2024-01-02 07:56:13

ReactuseEffect數(shù)據(jù)驅(qū)動 UI

2012-08-07 10:02:06

JSP

2014-07-30 17:10:38

LVS集群負載均衡

2009-07-03 18:32:18

JSP頁面跳轉(zhuǎn)

2024-04-24 14:46:40

人工智能編碼助手

2024-08-13 08:25:16

C#外部程序方式

2009-09-08 10:37:57

C#遍歷CheckBo

2022-04-28 07:26:17

PythonDocker容器

2009-08-26 18:10:44

C# using的用法

2023-02-24 16:45:02

點贊
收藏

51CTO技術(shù)棧公眾號