最近做了usaco2014 open的金组,果然美帝的题还是没有太简单啊QAQ,被每年的月赛骗了QAQ

不过话说官方题解真心棒(虽然英文的啃得好艰难,我英语渣你们别鄙视我= =),标程超级优美QAQ

按照标程打,学到了好多STL的用法= =(没办法,我c++底子弱)

这道题嘛,可以发现对于每个区间,只要左边界确定,可能的集合就一共只有8种了

考虑前缀和,发现若L~R为可行解,则对于所有种类的牛,有S[R]-S[L]=K或0

如何防止枚举K,可以发现在该集合中B的s[L][bi]减去s[L][b0]就行了

那么就hash,枚举集合B,求出hash值,直接做就行了

CODE:(直接贴标程了,Map真的用的是神出鬼没啊QAQ)

#include <iostream>
#include <vector>
#include <algorithm>
#include <cstring>
#include <cstdio>
#include <cassert>
#include <map> using namespace std; #define MAXN 100010
#define GROUPS 8 int MB[MAXN][GROUPS];
int MF[MAXN][GROUPS];
int PS[MAXN][GROUPS]; int main() {
freopen("fairphoto.in", "r", stdin);
freopen("fairphoto.out", "w", stdout); int N, K; cin >> N >> K;
vector<pair<int, int> > A(N);
for(int i = 0; i < N; i++) {
cin >> A[i].first >> A[i].second;
A[i].second--;
}
sort(A.begin(), A.end()); /* Construct backstep masks */
for(int i = 0; i < GROUPS; i++) {
MB[0][i] = 1 << A[0].second;
}
for(int i = 1; i < N; i++) {
int bt = 1 << A[i].second; MB[i][0] = bt;
for(int j = 1; j < GROUPS; j++) {
if(MB[i - 1][j] & bt) {
MB[i][j] = MB[i - 1][j];
} else {
MB[i][j] = bt | MB[i - 1][j - 1];
}
}
} /* Construct forward step masks */
for(int i = 0; i < GROUPS; i++) {
MF[N - 1][i] = 1 << A[N - 1].second;
}
for(int i = N - 2; i >= 0; i--) {
int bt = 1 << A[i].second; MF[i][0] = bt;
for(int j = 1; j < GROUPS; j++) {
if(MF[i + 1][j] & bt) {
MF[i][j] = MF[i + 1][j];
} else {
MF[i][j] = bt | MF[i + 1][j - 1];
}
}
} /* Construct partial sums */
for(int i = 0; i < N; i++) {
memcpy(PS[i + 1], PS[i], sizeof(PS[i]));
++PS[i + 1][A[i].second];
} int result = -1;
for(int j = K - 1; j < GROUPS; j++) {
vector<int> V(1 + GROUPS);
map<vector<int>, int> cost_map; /* Compute the earliest starts for given masks
* and normalized partial sums. */
for(int i = N - 1; i >= 0; i--) {
int base = -1;
int m = V[GROUPS] = MF[i][j];
if(__builtin_popcount(m) <= j) continue;
for(int k = 0; k < GROUPS; k++) {
if(m & 1 << k) {
if(base == -1) {
base = PS[i][k];
}
V[k] = PS[i][k] - base;
} else {
V[k] = PS[i][k];
}
}
cost_map[V] = A[i].first;
} /* Find best start points for each ending position. */
for(int i = 0; i < N; i++) {
int base = -1;
int m = V[GROUPS] = MB[i][j];
if(__builtin_popcount(m) <= j) continue;
for(int k = 0; k < GROUPS; k++) {
if(m & 1 << k) {
if(base == -1) {
base = PS[i + 1][k];
}
V[k] = PS[i + 1][k] - base;
} else {
V[k] = PS[i + 1][k];
}
} map<vector<int>, int>::iterator it = cost_map.find(V);
if(it != cost_map.end() && it->second < A[i].first) {
result = max(result, A[i].first - it->second);
}
}
} cout << result << endl;
return 0;
}

[Usaco2014 Open]Gold Fair Photography(hash)的更多相关文章

  1. [BZOJ3535][Usaco2014 Open]Fair Photography

    [BZOJ3535][Usaco2014 Open]Fair Photography 试题描述 FJ's N cows (1 <= N <= 100,000) are standing a ...

  2. BZOJ3540: [Usaco2014 Open]Fair Photography

    3540: [Usaco2014 Open]Fair Photography Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 72  Solved: 29 ...

  3. bzoj 3540: [Usaco2014 Open]Fair Photography

    3540: [Usaco2014 Open]Fair Photography Description FJ's N cows (2 <= N <= 100,000) are standin ...

  4. poj3274 Gold Balanced Lineup(HASH)

    Description Farmer John's N cows (1 ≤ N ≤ 100,000) share many similarities. In fact, FJ has been abl ...

  5. Gold Balanced Lineup(hash)

    http://poj.org/problem?id=3274 ***** #include <stdio.h> #include <iostream> #include < ...

  6. P3105 [USACO14OPEN]公平的摄影Fair Photography

    题意翻译 在数轴上有 NNN 头牛,第 iii 头牛位于 xi(0≤xi≤109)x_i\:(0\le x_i\le 10^9)xi​(0≤xi​≤109) .没有两头牛位于同一位置. 有两种牛:白牛 ...

  7. Fair Photography

    题目大意: 给出直线上N个点的位置和颜色(0或1),求最大的区间,使得区间内0的个数大于等于1的个数且0的个数减去1的个数为偶数. 解题过程: 1.先贴个lsdsjy大牛的线段树的做法:http:// ...

  8. [Usaco2014 Open Gold ]Cow Optics (树状数组+扫描线/函数式线段树)

    这道题一上手就知道怎么做了= = 直接求出原光路和从目标点出发的光路,求这些光路的交点就行了 然后用树状数组+扫描线或函数式线段树就能过了= = 大量的离散+模拟+二分什么的特别恶心,考试的时候是想到 ...

  9. 解题:USACO14OPEN Fair Photography

    题面 有点像JRY的那道序列题,大概是统计题的经典套路? 先说无修改的:将白奶牛记为$-1$,花奶牛记为$1$,然后做前缀和统计某个前缀和$sum$第一次出现的位置,之后再出现就统计答案.对于修改(将 ...

随机推荐

  1. Memcached源码分析之memcached.c

    memcached.c 由于代码太多,在此省略了部分代码,例如UPD连接,二进制协议,某些错误输出和调试输出等,建议从main函数开始看起. #include "memcached.h&qu ...

  2. C语言-数组

    C语言中使用数组来存储相同类型的大批量数据. 数组: 数组名:起名规则和变量名一样: 定义数组:数组每个元素存储的数据类型+数组名[常量(时表示数组分配存储数据类型的个数也就是长度)]={每个元素,以 ...

  3. argparse 命令含参数模块

    argparse是python的一个命令行参数模块,可以解析命令行参数,生成帮助等. 你可以这样使用它: #!/usr/bin/python from argparse import Argument ...

  4. constrain to margins

    如果你点了constrain to margins,左右会有8个点的空挡,而是从8个点后开始计算约束,而没有点时,已屏幕的0点开始计算.

  5. Core Data需求

    大家都在讨论怎么使用Core Data,但是什么时候用到Core Data,这好像是大家很少讨论的问题 我们使用Core Data ,主要用来存储两种类型的数据:固定的数据,和 可能变化的数据. 1. ...

  6. --@ui-router--登录页通过路由跳转到内页的demo

    今天还是来说一下angular中的路由模块.我们实际项目中,各个页面的切换是经常会与Auth相关的.比如我网站的后台,是需要登录过的用户才能进去,那么我们用angularJS做前端路由的时候应该怎么完 ...

  7. ZOJ-2343-Robbers

    题目链接 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1398 题意: 输入t 有t个测试用例每个测试用例第一行输入三个数n, ...

  8. 使用karma+jasmine做单元测试

    目的 使用karma和jasmine来配置自动化的js单元测试. Karma和Jasmine Karma是由Angular团队所开发的一种自动化测试工具.链接:http://karma-runner. ...

  9. HTML 表单元素、 输入类型、Input 属性

    <input> 元素 最重要的表单元素是 <input> 元素. <input> 元素根据不同的 type 属性,可以变化为多种形态. 注释:下一章讲解所有 HTM ...

  10. Java线程:线程中断

    interrupt方法可以用来请求终止线程. 当对一个线程调用interrupt方法时,线程的中断状态被置位.这时每个线程都有boolean标志.每个线程都应该不时的检查这个标志,以判断线程是否被中断 ...