牛客练习赛151

文章目录

  • [D 树形DP](#D 树形DP)
  • [E 区间异或和 哈希表 DP](#E 区间异或和 哈希表 DP)

D 树形DP

cpp 复制代码
const int N = 2e5+5,M=1e5;
const double PI=acos(-1);
const long long mod =998244353, inf = 1e18+10 ,up=1e9;
int dp[N][2];// dp[节点编号now][0:fa->now/1:now->fa]=now子树的out^2
vector<int>g[N];
void dfs(int now,int fa){
  /*
    从子节点向父节点递推 每个节点只有一个父亲 只关注到父亲的边方向
    怎么确定指向哪里最优?
    一开始设置所有的子节点都设置为指向当前节点now(dp[x][1]) 
    如果dp[x][0]-dp[x][1]很大 dp[x][1]+dp[x][0]-dp[x][1]=dp[x][0]那么指向x最优
  */
  vector<int>dis;// 子节点dp[0]-dp[1]
  int sm1=0;// sum of dp[x][1]
  for(auto x:g[now]){
    if(x==fa)continue;
    dfs(x,now);
    dis.push_back(dp[x][0]-dp[x][1]);
    sm1+=dp[x][1];// 0:指向本节点now 到父节点的边没有贡献 子节点x都指向now 
  }
  sort(dis.begin(),dis.end(),greater<int>());
  dp[now][0]=sm1,dp[now][1]=sm1+1;// out=1 只有now指向fa的一条边,now没有指向子节点x的
  int pre=0;
  forr(k,1,dis.size()){// 取前k条边
    pre+=dis[k-1];
    /* 之前写错了 dp[now][0]是会改变的
    dp[now][0]=max(dp[now][0],dp[now][0]+pre+k*k);
    dp[now][1]=max(dp[now][1],dp[now][0]+pre+(k+1)*(k+1));
    */
    dp[now][0]=max(dp[now][0],sm1+pre+k*k);//最大的k个边变成0------now指向孩子x 本节点的out=k
    dp[now][1]=max(dp[now][1],sm1+pre+(k+1)*(k+1));// 还有一条now指向fa out=k+1
  }
}
void solve(){
  int n;cin>>n;
  forr(i,1,n-1){
    int u,v;cin>>u>>v;
    g[u].push_back(v);
    g[v].push_back(u);
  }
  // 一条边只影响两个端点的贡献 可以枚举记录两种方向的结果取最大值
  dfs(1,0);// 因为是树结构 随便找一个点为根 这样就有了父子关系 关注指向父亲还是指向儿子
  cout<<dp[1][0]<<endl;//根节点1没有父节点 不指向父节点
}

E 区间异或和 哈希表 DP

出题人题解:

cpp 复制代码
map<int,int> hashmp[30][2];
void solve(){
  /*区间异或和
    目标:add sum>xor sum
    - 需要一个区间某位add_sum=1 xor_sum=0,那么区间全是1且长度是偶数
    - 在以上前提下,
      区间add_sum=1时 xor_sum必为0
      add_sum=0时 这一位不全为1 xor_sum=不确定
      为了满足目标 需要xor_sum=0 
    - 则 一个区间所有位的ADD_SUM最高位是H 那么XOR_SUM高于H位都为0
    - 目标就是找偶数长度区间的某一位k add_sum=1 并且所有高位xor_sum=0 prexor_i>>(k+1)=prexor_j>>(k+1)
  */
  int n;cin>>n;
  vector<int>a(n+1),prexor(n+1,0),dp(n+1,-1);// dp[i]=第i位之前最多分多少段
  
  forr(i,1,n)cin>>a[i],prexor[i]=prexor[i-1]^a[i];

  if(n&1)return cout<<-1<<endl,void();
  dp[0]=0;
  forr(i,1,n){// 遍历每个位置
    reforr(b,0,30){
      if(a[i]>>b&1){// 这一位是1 找到了目标位置
        if(dp[i-1]!=-1){// 和前面一个位置之间能划分 i-1作为上一个位置的结尾
          // 放入哈希表
          int highxor=prexor[i-1]>>(b+1);
          hashmp[b][(i-1)&1][highxor]=max(dp[i-1],hashmp[b][(i-1)&1][highxor]);
        }
        int now_highxor=prexor[i]>>(b+1);
        if(hashmp[b][i&1].count(now_highxor)){
          dp[i]=max(dp[i],hashmp[b][i&1][now_highxor]+1);
        }
      }else{// 这一位是0 肯定不能作为目标位置 如果保留历史值之后找到的也不合法 
        // 直接消除
        hashmp[b][1].clear();
        hashmp[b][0].clear();
      }
    }
  }
  cout<<dp[n]<<endl;
}
相关推荐
WL_Aurora2 分钟前
【每日一题】贪心
python·算法
aqiu11111124 分钟前
【并查集专题top】
c++·算法
叼烟扛炮39 分钟前
C++ 知识点17 友元
开发语言·c++·算法·友员
richard_yuu1 小时前
数据结构|二叉树高阶进阶-经典算法
数据结构·c++·算法
不知名的忻1 小时前
Dijkstra算法(朴素版&堆优化版)
java·数据结构·算法··dijkstra算法
星星码️1 小时前
LeetCode刷题简单篇之反转字母
c++·算法·leetcode
naturerun2 小时前
螺旋形遍历奇数阶矩阵
c++·算法·矩阵
wuweijianlove2 小时前
算法复杂度的实验估算与误差分布建模的技术7
算法
佳xuan3 小时前
简而言之c++
c++·算法
变量未定义~3 小时前
星际争霸、宝石塔的亮度差异、寻找食物储量
算法