OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3991|回复: 11

用u8存储8个标记位实现位带操作

[复制链接]

17

主题

97

帖子

0

精华

高级会员

Rank: 4

积分
822
金钱
822
注册时间
2020-7-21
在线时间
85 小时
发表于 2020-8-29 20:46:27 | 显示全部楼层 |阅读模式
1金钱
        一般来说我们做一个标记位会用u8 flag = 0; 然后需要置一的时候flag = 1;但如果标记位比较多,有flag1, flag2, flag3 之类的,那么定义一堆的u8 flag就很浪费存储空间,因此我希望用一个u8存储8个flag来节约空间并减少全局变量。但这样子一方面要搞移位操作,我们组里有人不熟悉。而且可读性也不好。可能搞着搞着哪一位是什么自己都忘了。
      然后看到原子哥有一个类似于PAout(1) = 1; 直接就相当于GPIOA->BSRR = 1 << 1;想向各位大佬请教一下它的原理和我这个想法的可行性。有具体操作就更好了

最佳答案

查看完整内容[请看2#楼]

C语言的位域是可以实现的位访问的,其语法类似于结构体,可以让你使用"."来访问指定变量的指定位域,比如flag.bit1,flag.bit2 这样的语法来访问某一个flag变量的第1,和第2位: 这是一段完整的C程序,你可以编译看一下输出,除了flag.bits7 以外,其他的值只可以是0或1,因为只定义了1位,但是bits7的值可以是 0-3,因为定义的时候,他占拥了2个位。其中的bit1,bit2....bits7 都是可以自定义名称的,当然flag也可以。 根据 ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

6

主题

279

帖子

0

精华

高级会员

Rank: 4

积分
750
金钱
750
注册时间
2015-3-27
在线时间
98 小时
发表于 2020-8-29 20:46:28 | 显示全部楼层
本帖最后由 弦子的弦 于 2020-8-30 16:11 编辑

C语言的位域是可以实现的位访问的,其语法类似于结构体,可以让你使用"."来访问指定变量的指定位域,比如flag.bit1,flag.bit2 这样的语法来访问某一个flag变量的第1,和第2位:
#include <stdio.h>

struct bits
{
    unsigned char bit1:1;
    unsigned char bit2:1;
    unsigned char bit3:1;
    unsigned char bit4:1;
    unsigned char bit5:1;
    unsigned char bit6:1;
    unsigned char bits7:2;
};

struct bits flag;

int main(void)
{
    flag.bit1 = 0;
    flag.bit2 = 1;
    flag.bit3 = 0;
    flag.bit4 = 1;
    flag.bit5 = 0;
    flag.bit6 = 1;
    flag.bits7 = 3;
    printf("flag.bit1 = %d\nflag.bit2 = %d\nflag.bit3 = %d\nflag.bit4 = %d\nflag.bit5 = %d\nflag.bit6 = %d\nflag.bit7 = %d\n",
               flag.bit1,flag.bit2,flag.bit3,flag.bit4,flag.bit5,flag.bit6,flag.bits7);
    printf("flag sizof is %ld byte(s)\n",sizeof(flag));

这是一段完整的C程序,你可以编译看一下输出,除了flag.bits7 以外,其他的值只可以是0或1,因为只定义了1位,但是bits7的值可以是 0-3,因为定义的时候,他占拥了2个位。其中的bit1,bit2....bits7 都是可以自定义名称的,当然flag也可以。

根据你具体所需要表示的意义来命名就可以了。类似,flag.led,flag.button 这样就可以提高可读性了,用法也很简单,他们是可以直接被赋值和读取的.

参考:https://www.cnblogs.com/WindSun/p/11283910.html
此外,位域也可以和共同体一起使用,这样做的目的是你可以单独读写flag中8个bit的值,也可以一次性设置,这样可以简化数据的处理。
回复

使用道具 举报

12

主题

3397

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
8670
金钱
8670
注册时间
2020-5-11
在线时间
4135 小时
发表于 2020-8-29 21:04:50 | 显示全部楼层
我一般这样操作:
#define BIT(x)    (1<<(x))
#define IOSET(port,x)  (port)|=(x)
#define IOCLR(port,x)  (port)&=~(x)

enum{
FLAG1=BIT(0),//标志1
FLAG2=BIT(1)//标志2
};
U8 mainflag=0;

中断()
{
IOSET(mainflag,FLAG1);//置标志
}
主函数
{
if(mainflag & FLAG1)//有标志
{
    IOCLR(mainflag,FLAG1);//清标志
}
}
专治疑难杂症
回复

使用道具 举报

51

主题

2166

帖子

2

精华

论坛元老

Rank: 8Rank: 8

积分
10653
金钱
10653
注册时间
2017-4-14
在线时间
2780 小时
发表于 2020-8-29 21:58:34 | 显示全部楼层
位域。。。。。。
回复

使用道具 举报

17

主题

97

帖子

0

精华

高级会员

Rank: 4

积分
822
金钱
822
注册时间
2020-7-21
在线时间
85 小时
 楼主| 发表于 2020-8-30 11:12:27 | 显示全部楼层
进一步解释一下,问题是定义一个u8 flag = 0x00,   其中包含八个标记位,当flag = 0xff时意思是八个标记位全部置一,希望做的是把这八个位各用一个flag1,flag2……flag8来替代,当flag1 = 1 时,flag= 0x01,flag2 = 1时flag = 0x02类似于这样
回复

使用道具 举报

10

主题

778

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
6745
金钱
6745
注册时间
2017-4-12
在线时间
1258 小时
发表于 2020-8-30 13:56:11 | 显示全部楼层
自己做个封装不就好咯,一个set_byFlag(uint8_t _bits , uint8_t status),一个get_byFlag(uint8_t _bits)
回复

使用道具 举报

11

主题

314

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1039
金钱
1039
注册时间
2020-7-21
在线时间
258 小时
发表于 2020-8-30 14:47:24 | 显示全部楼层
本帖最后由 DFY 于 2020-8-30 15:04 编辑

你可以参考库函数的配置IO口状态的函数,这个跟你想要哪个引脚输出高低电平的原理差不多:哪组IO(Flag)、第几个引脚(第几个位)、什么电平(0 or 1)例如:
void setFlag(uint8_t flag,uint8_t flag_num,uint8_t state)
{
       if(state)
               flag |= 0x01<< flag_num;
       else
               flag &= ~(0x01 << flag_num);
}

uint8_t getFlag(uint8_t flag,uint8_t flag_num)
{
      return (flag & (0x01 << flag_num));
}
回复

使用道具 举报

11

主题

2149

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
4919
金钱
4919
注册时间
2015-1-10
在线时间
617 小时
发表于 2020-8-30 15:33:32 | 显示全部楼层
给一个参考,我经常用的置位,取位,清位,取反的位操作
#define setbit(x, y)                        x|=(1<<y)       //某一位置1
#define clrbit(x, y)                        x&=~(1<<y)      //某一位清0
#define reversebit(x, y)            x^=(1<<y)       //某一位取反
#define getbit(x, y)                        ((x) >> (y)&1)  //获取某一位的值

x是定义的标志,y是位,然后用宏定义或是枚举定义0 bit~7bit或是到31bit的每位
回复

使用道具 举报

0

主题

9

帖子

0

精华

初级会员

Rank: 2

积分
145
金钱
145
注册时间
2019-4-22
在线时间
61 小时
发表于 2020-8-30 16:06:57 | 显示全部楼层
分享一個

typedef union tf_tp
{
        struct
        {
                u8        T1_00s                                :1;
                u8        T0_50s                                :1;
                u8        T0_30s                                :1;
                u8        T0_20s                                :1;
                u8        T0_10s                                :1;
                u8        T0_01s                                :1;
                u8        T0_50s_clock        :1;
                u8        T0_30s_clock        :1;
        }bits;
       
  u8 byte;
}Time_Flag_Type;

Time_Flag_Type Time_Flag;

這樣就能組合成一個8位的標記位
使用的時候只要Time_Flag.bits.T0_20s = 1;  這樣就能單獨控制一個位
也可以用一整個Byte 像這樣Time_Flag.byte = 0xFF;
8個不夠也可以依此類推擴充成16 32的
回复

使用道具 举报

17

主题

97

帖子

0

精华

高级会员

Rank: 4

积分
822
金钱
822
注册时间
2020-7-21
在线时间
85 小时
 楼主| 发表于 2020-8-30 18:22:04 | 显示全部楼层
Axian 发表于 2020-8-30 16:06
分享一個

typedef union tf_tp

感谢!!
回复

使用道具 举报

17

主题

97

帖子

0

精华

高级会员

Rank: 4

积分
822
金钱
822
注册时间
2020-7-21
在线时间
85 小时
 楼主| 发表于 2020-8-30 18:22:25 | 显示全部楼层
弦子的弦 发表于 2020-8-29 20:46
C语言的位域是可以实现的位访问的,其语法类似于结构体,可以让你使用"."来访问指定变量的指定位域,比如fl ...

感谢!!
回复

使用道具 举报

9

主题

219

帖子

1

精华

金牌会员

Rank: 6Rank: 6

积分
1434
金钱
1434
注册时间
2020-5-12
在线时间
394 小时
发表于 2020-8-31 08:27:14 | 显示全部楼层
用位域+联合体最方便。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-10 16:45

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表