用一个数维护最多32个可叠加状态,比如权限的状态,既有“读”,又有“写”

有的时候我们需要用一个数字来维护多个状态,比如权限,一个角色可以有读的权限,也可以有写的权限,设计数据库的时候,很多人会选择创建多个列来维护不同的权限

其实完全是可以用一个数字来维护多个权限的

大致的思路用数字二进制形态中的0和1以及1所处的位置来表示表示状态,数是多少位,就能表示多少个状态,并且这些状态是可以叠加的,即能“读权限”,也能“写权限”

大致代码如下,将uint封装成State32,Put函数叠加一个状态,Remove移除,Contains验证

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
using System;
 
namespace ConsoleApp {
    class Program {
        static void Main(string[] args) {
            State32 right = new State32(0); //传入一个32位的无符合整型封装成一个结构体
            right.Put(1); //写
            right.Put(2); //读
            right.Put(3); //删除
            right.Remove(2); //去掉读权限
            var b1 = right.Contains(2, out uint a1); //判断是否有读取权限
            var b2 = right.Contains(3, out uint a2); //判断是否有删除权限
            Console.ReadKey();
        }
    }
}

State32

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
/// <summary>
    /// 封装了一个uint数值,可以表示最多32个并存的状态,原理:用二进制位的顺序表示状态码,值表示状态的真假,比如值9(1001),表示有状态1和4,没有状态2和3
    /// </summary>
    public struct State32 {
        /// <summary>
        /// 32无符合的整型,用来存储状态数据
        /// </summary>
        public uint Value { get; private set; }
        private static readonly object _lock = new object();
        /// <summary>
        /// 根据传入的32位的无符合整型创建一个结构对象
        /// </summary
        public State32(uint value) {
            Value = value;
        }
        /// <summary>
        /// 判断是否包含当前状态
        /// </summary>
        public bool Contains(byte state, out uint stateValue) {
            if (state == 0 || state > 32) { throw new Exception("取值范围1-23"); }
            lock (_lock) {
                stateValue = (uint)(0b1 << state - 1); //得到一个在二进制中除了指定位为1其他位都为0的整型
                if ((Value & stateValue) > 0) {  //用“与”将其他位都变成0,验证指定位是否为1
                    return true;
                }
                return false;
            }
        }
        /// <summary>
        /// 叠加状态
        /// </summary>
        public State32 Put(byte state) {
            lock (_lock) {
                if (!Contains(state, out uint stateValue)) {
                    Value = Value | stateValue;  //用“或”将指定位从0变成1,其他位不变
                }
                return this;
            }
        }
        /// <summary>
        /// 移除状态
        /// </summary>
        public State32 Remove(byte state) {
            lock (_lock) {
                if (Contains(state, out uint stateValue)) {
                    Value = Value ^ stateValue; //用“异或”将指定位从1变成0,其他位不变
                }
                return this;
            }
        }
    }
posted @   WmW  阅读(378)  评论(0编辑  收藏  举报
编辑推荐:
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 从HTTP原因短语缺失研究HTTP/2和HTTP/3的设计差异
· 三行代码完成国际化适配,妙~啊~
点击右上角即可分享
微信分享提示