杭电OJ-5616 Jam‘s balance用二进制枚举模仿三进制枚举

这篇文章假定你已经懂得二进制枚举以及了解过这个题目

为了叙述方便,我假设待测物体放在天平左端

一个砝码其实有三种状态,放在左边,放在右边,不放上去

如果我有一台三进制计算机,且用-1  0  1来储存数据,那我们自然可以用三进制枚举的方法,毕竟0 -- 3^n的每一个数字都代表了砝码的一种状态,-1代表放在左边,0代表不放,1代表放在右边,可惜我们的电脑是二进制的

那么如何用二进制来模仿三进制呢,首先我建立了二进制与三进制的映射关系,即每一个三进制数可以对应一个二进制数,每一个二进制数可以通过某种规则来转换为三进制数

来看一个三进制序列:1 0 1 1 1 0 0 1    ,这个二进制数的每个1都可以任意的换为-1,这个数里有5个1,那么这个二进制数一共对应了2^5个三进制数,

比如-1 0 1 -1 1 0 0 1;  1 0 -1 -1 1 0 -1  ;  1 0 1 1 1 0 0 1 (包括自己)

对每个二进制数都如此操作就可以遍历每个三进制数,当然,这需要一点数学证明,可以跳过不看

数学证明:

在两个三进制数中,如果0的排布不同(位置,个数),那么它们一定不等

而两个不同的二进制数的0的排布一定不同,且按照上述规则转换为三进制数后0的排布不变,

故不同的二进制数所对应的三进制数一定不等

而一个二进制数,它所对应的那些三进制数由于1和-1的排布不同,显然也是不同的,所以只剩最后一个问题了:这样转换之后会得到3^n个三进制数吗,还是会更少呢

可以看出,刚刚好是3^n;

至于1转换成-1如何在代码中体现,请直接看代码

代码:

#include<iostream>
#include<vector>
using namespace std;
bool three()
{
    int n;     //砝码数
    cin >> n;
    vector<int> w(n);   //用于储存砝码质量
    vector<int> where(n);  //用于储存二进制枚举中1的位置
    int m;    //待测质量
    int sum = 0;  //砝码可以称量的质量
    int i,j,k,p;  //用于循环迭代
    for (i = 0; i < n; i++)
    {
        cin >> w[i];
    }
    cin >> m;
    for (i = 1; i < 1 << n; i++)
    {
        sum = 0;
        for (j = 0, k = 0; j < n; j++)
        {
            if (i & (1<<j))
            {    
                sum += w[j];
                where[k] = j;  //把1的位置放进了where矢量
                k++;
            }
        }
        if (m == sum)  return true;
        //下面开始模拟三进制枚举,其实是又一次的二进制枚举
        for (p = 1; p < 1 << k; p++)
        {
            int sum2 = sum;  
            for (j = 0; j < k; j++)
            {
                if (p & (1<<j))
                    sum2 -= 2 * (w[where[j]]);  //把1换为了-1
            }
            if (m == sum2) return true;
        }
    }
    return false;
}

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值