学习资料
-----1-----
-----2-----

P5038 [SCOI2012]奇怪的游戏

一道甚神但没用到高深模型的题

思路

没思路,看题解吧

代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <queue>
#define ll long long
#define point(x, y) ((x - 1) * m + y)

using namespace std;
const int dx[] = {0, 0, 1, -1};
const int dy[] = {1, -1, 0, 0};
const int N = 55, _ = 200005;
const ll inf = 1e16;
int n, m, s, t;
int a[N][N], color[N][N];
ll s1, s2, d1, d2, mx, sum, flow, maxflow;
struct Edge { int Nxt, v; ll flow; } e[_ << 1];
int h[_], p = 1;
void add(int u, int v, ll f) {
    e[++p].Nxt = h[u]; e[p].v = v; e[p].flow = f; h[u] = p;
    e[++p].Nxt = h[v]; e[p].v = u; e[p].flow = 0; h[v] = p;
}
int d[_];
queue<int> q;
bool bfs() {
    memset(d, 0, sizeof(d));
    while(!q.empty()) q.pop();
    q.push(s); d[s] = 1;
    while(!q.empty()) {
        int u = q.front(); q.pop();
        for(int i = h[u]; i; i = e[i].Nxt)
            if(e[i].flow && !d[e[i].v])
        {
            d[e[i].v] = d[u] + 1;
            if(e[i].v == t) return true;
            q.push(e[i].v);
        }
    }
    return false;
}
ll dinic(int u, ll flow) {
    if(u == t) return flow;
    ll rest = flow, k;
    for(int i = h[u]; i && rest; i = e[i].Nxt)
        if(e[i].flow && d[e[i].v] == d[u] + 1)
    {
        k = dinic(e[i].v, min(rest, e[i].flow));
        if(!k) d[e[i].v] = 0;
        e[i].flow -= k;
        e[i ^ 1].flow += k;
        rest -= k;
    }
    return flow - rest;
}
bool check(ll x) {
    sum = flow = maxflow = 0;
    p = 1; memset(h, 0, sizeof(h));
    for(int i = 1, tx, ty; i <= n; i++)
        for(int j = 1; j <= m; j++)
            if(color[i][j] == 0)
                for(int k = 0; k < 4; k++)
    {
        tx = i + dx[k], ty = j + dy[k];
        if(tx >= 1 && tx <= n && ty >= 1 && ty <= m)
            add(point(i, j), point(tx, ty), inf);
    }
    for(int i = 1; i <= n; i++)
        for(int j = 1; j <= m; j++)
            if(color[i][j] == 0)
                add(s, point(i, j), x - a[i][j]), sum += x - a[i][j];
            else
                add(point(i, j), t, x - a[i][j]);
    while(bfs())
        while(flow = dinic(s, inf)) maxflow += flow;
    return sum == maxflow;
}
int main()
{
    int T = 0;
    scanf("%d", &T);
    while(T--)
    {
        mx = s1 = s2 = d1 = d2 = 0;
        scanf("%d%d", &n, &m); s = 0, t = n * m + 1;
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++)
                scanf("%d", &a[i][j]), mx = max(mx, 1LL * a[i][j]);
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++)
                color[i][j] = (i + j) & 1;
        for(int i = 1; i <= n; i++)
            for(int j = 1; j <= m; j++)
                if(color[i][j] == 0) s1 += a[i][j], d1++;
                else s2 += a[i][j], d2++;
        if(d1 != d2) {
            ll x = (s1 - s2) / (d1 - d2);
            if(x >= mx && check(x)) printf("%lld\n", x * d1 - s1);
            else printf("-1\n");
        }
        else {
            ll l = mx, r = 1e14, mid = 0, ans = 0;
            while(l <= r) {
                mid = (l + r) >> 1;
                if(check(mid)) ans = mid, r = mid - 1;
                else l = mid + 1;
            }
            if(s1 != s2) printf("-1\n");
            else printf("%lld\n", ans * d1 - s1);
        }
    }
    return 0;
}

[题目] Luogu P5038 [SCOI2012]奇怪的游戏的更多相关文章

  1. P5038 [SCOI2012]奇怪的游戏 二分+网络流

    $ \color{#0066ff}{ 题目描述 }$ Blinker最近喜欢上一个奇怪的游戏. 这个游戏在一个 \(N \times M\) 的棋盘上玩,每个格子有一个数.每次\(Blinker\)会 ...

  2. P5038 [SCOI2012]奇怪的游戏

    题目链接 题意分析 首先我们需要求的是统一以后的值\(x\) 并且一般的棋盘操作我们都需要黑白染色 那么对于棋盘格子是偶数的情况的话 答案是存在单调性的 因为如果统一之后 两两搭配还是可以再加一个的 ...

  3. 题解 P5038 [SCOI2012]奇怪的游戏

    题解 题目 做这题之前,做了一道叫星际战争的题,很容易想到二分 \(+\) 网络流,那么二分啥呢? 我们先推一下式子,因为是对相邻格子加数,那么可以联想到黑白染色类问题. 设有黑色格子 \(B\) 个 ...

  4. 洛谷$P5038\ [SCOI2012]$奇怪的游戏 二分+网络流

    正解:二分+网络流 解题报告: 传送门$QwQ$ 这种什么,"同时增加",长得就挺网络流的$QwQ$?然后看到问至少加多少次,于是考虑加个二分呗?于是就大体确定了做题方向,用的网络 ...

  5. BZOJ 2756: [SCOI2012]奇怪的游戏 网络流/二分

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 1594  Solved: 396[Submit][Stat ...

  6. bzoj2756: [SCOI2012]奇怪的游戏(网络流+分情况)

    2756: [SCOI2012]奇怪的游戏 题目:传送门 题解: 发现做不出来的大难题一点一个网络流 %大佬 首先黑白染色(原来是套路...)染色之后就可以保证每次操作都一定会使黑白各一个各自的值加1 ...

  7. BZOJ 2756: [SCOI2012]奇怪的游戏 [最大流 二分]

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 3352  Solved: 919[Submit][Stat ...

  8. Bzoj2756 [SCOI2012]奇怪的游戏

    2756: [SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 3220  Solved: 886 Description ...

  9. bzoj 2756 [SCOI2012]奇怪的游戏 二分+网络流

    2756:[SCOI2012]奇怪的游戏 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 4926  Solved: 1362[Submit][Stat ...

随机推荐

  1. Dubbo 源码分析系列之一环境搭建

    环境搭建的步骤有哪些 依赖外部的环境 使用的开发工具 源码的拉取 结构大致介绍 1 依赖的外部环境 安装JDK 安装Git 安装maven 这边我们就不介绍怎么安装这些外部环境了,大家自行从安装这些外 ...

  2. [LeetCode解题报告] 502. IPO

    题目描述 假设 LeetCode 即将开始其 IPO.为了以更高的价格将股票卖给风险投资公司,LeetCode希望在 IPO 之前开展一些项目以增加其资本. 由于资源有限,它只能在 IPO 之前完成最 ...

  3. [SPOJ22343] Norma

    Description 现在有一个长度为\(N(N\leq 500000)\)的序列,定义区间\([l,r]\)的价值为\([l,r]\)的最小值乘上\([l,r]\)的最大值乘上\([l,r]\)的 ...

  4. Linux进程间通信(System V) --- 消息队列

    消息队列 IPC 原理 消息队列是消息的链式队列,如下图为消息队列的模型.整个消息队列有两种类型的数据结构. 1.msqid_ds 消息队列数据结构:描述整个消息队列的属性,主要包括整个消息队列的权限 ...

  5. VS2017进程为idXXXX 无法启动解决方案

    1.对无法启动项目的 .csproj 后缀文件用记事本打开 找到<WebProjectProperties>xx</WebProjectProperties> 删掉 让后 重新 ...

  6. 本地计算机上的MySQL服务启动后停止。某些服务在未由其他服务或程序使用时将自动

    重新安装MySQL数据库,由于安装的时候马虎,一路next(事实上,某些地方需要严格的配置,我忘记注意了),导致现在出了很多麻烦. 错误信息: 本地计算机上的MySQL服务启动后停止.某些服务在未由其 ...

  7. 【操作系统】二、JVM线程与Linux内核线程的映射

    Linux从内核2.6开始使用NPTL (Native POSIX Thread Library)支持,但这时线程本质上还轻量级进程. Java里的线程是由JVM来管理的,它如何对应到操作系统的线程是 ...

  8. java - Jsoup原理

    https://blog.csdn.net/xh16319/article/details/28129845 http://www.voidcn.com/article/p-hphczsin-ru.h ...

  9. 在使用vue-cli中遇到的几个问题

    前言:框架没有好坏之分,能解决需求就可以.之前没事用vue模仿过BOSS直聘App(纯属娱乐),实际工作中开发过一个后台管理系统,遇到过不少坑,终于闲下来稍微总结几个问题分享一下! 一.所遇到的问题( ...

  10. Openlayer3之瓦片数据接入

    瓦片数据集接入实现思路: 1.构造ol.source.TileImage数据源,构造该数据源需要以下几项: 1)空间参考,通过如下代码构造 2)TileGrid,构造需要以下几项: a)原点 b)分辨 ...