zoukankan      html  css  js  c++  java
  • APIO2007 动物园 题解

    x### 原题链接
    https://www.luogu.com.cn/problem/P3622

    题目大意

    有一圈围栏,每个围栏有一种动物,有若干个小朋友,每个小朋友能看到连续的 (5) 个动物,每个小朋友对每种的动物的喜好不一样,如果一个小朋友会高兴,当且仅当 至少有一个他害怕的动物被移走,或者是至少有一个他喜欢的动物没有被移走。问调整某些动物后,最多有多少个小朋友会高兴。

    分析

    题目的目的是移走若干动物,使得所有的小朋友中,高兴的人数最多。其中问题的关键点在于如果确定了一个位置的小朋友所能看到的 (5) 个位置的动物的移动状态确定后,那么在他旁边的小朋友(如果存在)能看到的动物中有 (4) 个的状态也就是确定了,而且两者之间只有一位的状态可以选择,而且对应的只有移动和不移动两种状态。同时每种状态只有 (5) 位组成,因此可以考虑状压DP搞一下。

    有个小细节需要注意,就是题目的图片中给出的小朋友位于他看到的 (5) 个位置的中间,而数据的输入格式给出的是他能看到的第一个位置编号x,因此我们可以统一一下,就认为能看到位置编号为 ({x,x+1,x+2,x+3,x+4}) 的小朋友位于 (x)。这样是不会影响答案的(因为小朋友所能看到的范围是不变的,只是站位不一样,当移动状态确定后,高兴的总人数是不变的)。

    定义状态,设 (f[i][j]) 表示从 (1)(i),位置 (i) 的小朋友能看到的动物的移动状态为 (j) 时,高兴人数的最大值,我们用 (1) 表示移动该动物,(0) 表示不移动。
    考虑转移方程,先看下面的图片,其中 (x) 表示 (0)(1)

    当位置 (i) 的小朋友对应的动物移动状态为 (j) 时,那么与位置 (i-1) 的小朋友看到的动物移动状态 (j') 会有 (4) 位是重合的状态,因此 (j) 可以由上一个阶段的 (2) 个状态转移过来,分别对应的是 j&15<<1j&15<<1|1,即取 (j) 的低四位作为 (j') 高四位,(j') 的最低位是 (0)(1)。两者取 max 之后,还要加上位置 (i) 开始的连续 (5) 个位置的动物移动状态为 (j) 的时候,高兴的小朋友的人数。所以转移方程为:$$f[i][j]=max(f[i-1][(j&15)<<1],f[i-1][(j&15)<<1|1])+cnt[i][j]$$
    这里注意运算符的优先级问题。

    那么问题又来了,这个 (cnt[i][j]) 怎么求?
    我们再重新强调一下它的定义:从 (i) 开始连续的 (5) 个动物的移动状态为 (j) 时,高兴的小朋友的人数。
    根据我们前面的说的小细节的修改,输入中的 (E) 表示小朋友能看到的第一个位置,那么我们就认为这个小朋友站在 (E) 这个位置。后面给了我们他对某些动物的害怕和喜欢的情况,我们可以先组织一下他对这 (5) 个动物的害怕和喜欢的状态。假设他害怕的一个动物位于 (x),由于是环形,我们找一下 (x)(E) 的相对位置:x = (x - E + n) % n,那么 fear |= (1 << x)like 也做同样的处理。根据题目中给出的是否高兴的条件,我们枚举每种移动的状态 (j)

    • 至少有一个害怕的被移走:(j & fear) != 0
    • 至少有一个喜欢的没被移走:(~j & like) != 0
      至此我们可以统计出 cnt[i][j],大致代码如下:
    // 记录每个小朋友害怕和喜欢的动物的状态
    int fear = 0, like = 0;
    for (int j = 0; j < f; ++j) {
    	qread(num); // 写的快读
    	num = (num - e + n) % n;
    	fear |= (1 << num);
    }
    // like也是一样的操作
    
    // 处理起点e的区间移动状态对应的高兴的小朋友的数量
    for (int j = 0; j < 32; ++j) { // 枚举每种移动状态
    	if ((fear & j) || (~j & like)) {
    		++cnt[e][j];
    	}
    }
    

    到这里我们基本可以补全代码了:

    for (int i = 1; i <= lan; ++i) {
    	for (int j = 0; j < 32; ++j) {
    		f[i][j] = max(f[i-1][(j&15)<<1], f[i-1][(j&15)<<1|1]) + cnt[i][j];
    	}
    }
    

    然后找到 (f[n][...]) 当中的最大值?
    提交上去会有红色的 WA

    问题在于这是一个环,那么就必须要考虑到收尾相连的情况,也就是收尾重合的部分的动物移动状态也必须是相同的才行。
    见下图:

    所有动物的移动状态确定后,我们必须保证选取的 (f[n][...]) 中的答案对应的状态 (j) 必须与 (1) 确定的状态中阴影部分是相同的,也就是说我们必须要保证最后的答案是从开始 (1) 阶段,对应的状态为阴影部分加上 (0)(1) 而来。那要怎么保证呢?

    我们可以枚举开始的状态,可以规定一个 (0) 阶段,枚举每个起始的状态 (s),初始化 (f[0][s]=0),其他值都初始化为绝对值大于总人数的负数(即负无穷),这样可以保证最终的答案必然会从我们的初始状态转移过来。那么针对这个起始状态 (s),我们的答案对应的是 (f[n][s]),只有这一个是可取的,只有这样才能保证我们的环形的客观条件。
    补全代码:

    int ans = 0;
    for (int s = 0; s < 32; ++s) { // 枚举初始状态
    	memset(f[0], 128, sizeof(f[0])); // 足够小就可以,这里的足够小是对于题目中的人数来定的
    	f[0][s] = 0;
    	for (int i = 1; i <= lan; ++i) {
    		for (int j = 0; j < 32; ++j) {
    			f[i][j] = max(f[i-1][(j&15)<<1], f[i-1][(j&15)<<1|1]) + cnt[i][j];
    		}
    	}
    	ans = max(ans, f[lan][s]);
    }
    
  • 相关阅读:
    mybatis <=或这个>=提示错误Tag name expecte问题解决
    Navicat 设置自增长初始值
    mysql 时间与字符串相互转换
    jquery 动态控制显隐
    mysql 查询常见时间段数据
    jquery 取得select选中的值
    java8 运算语法集
    springboot 单元测试
    idea 自动生成并跳转单元测试
    限制页面被pc端访问
  • 原文地址:https://www.cnblogs.com/kuangbiaopilihu/p/13195170.html
Copyright © 2011-2022 走看看