zoukankan      html  css  js  c++  java
  • AtCoder Regular Contest 100 E:Or Plus Max(DP+位运算)解题报告

    这题对于我来说是真的难。。。

    点击这里进入题目

    题目大意:言简意赅。2的N次方个数,存为Ai,令1≤K≤2的N次方−1,请你求出(i,j),使Ai+Aj最大,并且0<=i< j<=2的N次方-1且(i or j)≤K。输出Ai+Aj的最大值。


    思路:这一题长得是真的不像DP。
    这道题虽然言简意赅,但大部分都是为了严谨,我们最要注意的就是i|j<=k这句话。用膝盖想一下便知若i|j<=k,则i|j必定<=k+1。那么我们就可以设dp[k]为k下Ai+Aj的最大值,那么就有dp[k]=max(dp[k-1],不知道什么东西)。我们主要要思考的就是这个“不知道什么东西”。
    那么这个“不知道什么东西”肯定是区别于k-1下的那些(i,j)的,也就是说是新的几对(i,j),根据样例或者自己证明等等方式可知若x|y=z,那么x和y都必定小于等于z。


    口胡证明: 因为或运算在都是0的情况下才会是0,不可能把1变成0,只会把0变成1,所以数字只会变大不会变小


    接上文思路:那么,也就是说新的这几对(i,j)里,i或j等于k(因为小于等于k的前面都有了嘛),因为不管顺序,我们就直接让i等于k。式子就变成了k|j=k,根据证明可知j小于等于k,我们可以通过更改1的位数为0来得到所有小于k的数。但在这里我选择了反过来思考:我们循环这几位和k,然后更改k中的这一位的1为0,得到小于k的j,运用两个数组更新与这时的k相对应的j的最大值就行了,我们得到的就是两个一一对应的数组,每一个k对应其最大的j。
    我不怎么会写dp方程,大概就是dp[k]=max(dp[k-1],a[k]+maxa[j])
    (萌新瑟瑟发抖)


    AC程序

    //库省略
    using namespace std;
    const int maxn=300005;
    int n,a[maxn];
    int sum1[maxn],sum2[maxn];
    int ans[maxn];
    void upday(int val,int pos)
    {
        if(val>sum1[pos])
        {
            sum2[pos]=sum1[pos];
            sum1[pos]=val;
        }
        else
        {
            if(val>sum2[pos])
            {
                sum2[pos]=val;
            }
        }
    }
    int main()
    {
        cin>>n;
        for(int i=0;i<(1<<n);i++)
        {
            cin>>a[i];
            sum1[i]=a[i];
        }
        for(int i=0;i<n;i++)
        {
            for(int j=0;j<(1<<n);j++)
            {
                int t=1<<i;
                if(!(j&t))
                {
                    int nowp=(j|t);
                    int old=sum1[j];
                    upday(old,nowp);
                    old=sum2[j];
                    upday(old,nowp);
                }
            }
        }
        for(int i=0;i<(1<<n);i++)
        {
            if(i!=0)
            ans[i]=max(ans[i-1],sum1[i]+sum2[i]);
            else
            ans[i]=sum1[i]+sum2[i];
        }
        for(int i=1;i<(1<<n);i++)
        {
            cout<<ans[i]<<endl;
        }
        return 0;
    }
    
  • 相关阅读:
    微信机器人开发SDK使用教程--清粉任务
    微信机器人开发SDK使用教程--养号任务停止
    linux和 unix 介绍
    thinkphp5.1学习笔记
    nginx的相关配置记录和总结
    php优秀框架codeigniter学习系列——CI_URI类学习
    求职过程中的灵感
    设计模式学习系列——过滤器模式
    php优秀框架codeigniter学习系列——CI_Utf8类
    php优秀框架codeigniter学习系列——hooks
  • 原文地址:https://www.cnblogs.com/NightRaven/p/9333244.html
Copyright © 2011-2022 走看看