数据库优化设计

本文介绍了一种通过整合多个状态字段到一个整型字段的方法,利用二进制位运算减少数据库表结构变动,提高数据管理效率。并通过具体示例展示了如何使用按位与操作查询特定组合的状态,以及如何在后端程序中更新这些状态。
摘要由CSDN通过智能技术生成

  很多时候,我们在项目中,可能会随着需求的不断更改,我们需要在原有的库表结构中增加字段,从而满足我们的业务需求。

  举一个简单的例子:

    我需要一张数据库表,用来存储某个网站的用户信息。该用户信息需要包括(帐号是否是活跃帐号、帐号是否绑定邮箱、帐号是否购买过产品、帐号是否过期....等等),一般情况下,我们可能会这么设计这张库表:

  

CREATE TABLE 'ACCOUNT' (
      `ID` int(22)  NOT NULL AUTO_INCREMENT, --自增id
      `F001` TINYINT(1) NOT NULL, --是否活跃(1:是/0:否)
      `F002` TINYINT(1) NOT NULL, --是否绑定 (1:是/0:否)
      `F003`  TINYINT(1) NOT NULL, --是否购买产品 (1:是/0:否)
      `F004`  TINYINT(1) NOT NULL, --是否过期 (1:是/0:否)
      PRIMARY KEY('ID')                  
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;

  如果按照以上方法进行设计,那么当需求变更,如:增加一个字段,帐号是否有效。那么我们就需要再增加一个字段`F005`,这样会导致管理起来非常麻烦,当新增字段时,需要去更新所有或者历史数据,很容易导致数据丢失。

 

  那么,怎么去解决这个问题,让数据字段间的关联影响尽可能的降低。现在有一种办法就是,将这些字段整合全部放到一个字段中。如:`FLAG`字段 按10进制进行存储(第一位:1:活跃,0:非活跃。第二位:1:绑定,2:未绑定。第三位:1:购买过产品,0:未购买....),解释一下:如果这个字段的值是5,转换成二进制是  101,第一位是1,第二位0,第三位1.那么就表示该帐号是“活跃、未绑定、且购买过产品”。这种方式有什么好处呢,好处在于各个类型之间的关联关系很小,不会因为更新一个字段类型而影响了其他字段。那么,这样设计,该如何查询呢?  按位与,具体情况可以举一个例子,当我需要查询所有购买过产品,而且活跃的帐号,也即(第一位和第三位为1),其他位我们填0,即101=5:

  sql语句查询:

    

SELECT * FROM ACCOUNT WHERE F005&5>0;

 

  以上sql语句就能查询出购买过产品而且活跃的帐号。

 

  那么,更新修改的时候只需要在后台程序中将各二进制位更新为需求的值就好了,例如,刚才的购买过且活跃,即101,要更改为购买过,但不是活跃的话,那么就用 (5&1)=1,从而实现将第三位置0。这里提供一个PHP函数可以很好的处理该程序业务。

  

<?php
/**
 * 取按位与要写入的数据
 * @param  $param 原值
 * @param  $postData  提交的数据key=>value,key必须从1开始的数据,value必须为0或者(key-1)次方数 key代表位
 */
function getExtValue($param = 0,$postData = array()){
    if(!empty$postData){
        foreach($postData as $key=>$value){
            $tmp1 = pow(2,($key-1));
            if (empty($key) || !is_numeric($key) || !is_numeric($value) || ($value != 0 && $value !=$tmp1)){
                continue;
            }
            $param = ($param & (0xffff^$tmp1)) | $value;
        }
    }
    return $param;
}

 

转载于:https://www.cnblogs.com/cobobryan/p/3611764.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值