状压DP小技巧

一个数有多少位1

for(int i = 1;i <= 1024;i ++) w[i] = w[i ^ i & -i] + 1;

当状态选择有条件时,不一定要考虑从开始(111111111)局面选到当前局面的最优解,可以考虑从当前局面开始选择时的最优解。例如 https://ac.nowcoder.com/acm/contest/9985/E
dp【x】为当前剩余石子为x时先手比后说多的最多解

#include <bits/stdc++.h>
using namespace std;
const int Mn = 2e6 + 5;
#define ll long long
ll dp[Mn];
ll a[100];
int e[100];
const ll inf = 0x8080808080808080;
int main()
{
    int _;scanf("%d",&_);
    while(_--){
        int n;scanf("%d",&n);
        for(int i = 0;i < n;i ++) scanf("%lld",a + i);
        int x,y;
        memset(e,0,sizeof(e));
        for(int i = 1;i < n;i ++){
            scanf("%d%d",&x,&y);
            x --;
            y --;
            e[x] ^= (1 << y);
            e[y] ^= (1 << x);
        }
        memset(dp,128,sizeof(dp));
        for(int i = 0;i < n;i ++){
            dp[(1 << i)] = a[i];
        }
        dp[0] = 0;
        for(int i = 0;i <= (1 << n) - 1;i ++){
            for(int j = 0;j < n;j ++){
                if(dp[i] == inf) continue;
                if(!(i & (1 << j)) && (e[j] & i)){
                    dp[i^(1<<j)] = max(dp[i ^ (1<<j)],a[j] - dp[i]);
                }
            }
        }
        printf("%lld\n",dp[(1 << n) - 1]);
    }
    return 0;
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值