#include <bits/stdc++.h>
#include <ext/pb_ds/assoc_container.hpp>
#include <ext/pb_ds/tree_policy.hpp>
#pragma GCC optimize("unroll-loops")
#pragma GCC optimize("-O3")
#pragma GCC optimize("Ofast")
//#pragma GCC optimize("fast-math")
//#pragma GCC optimize("no-stack-protector")
#define F first
#define S second
#define sz(x) int(x.size())
#define pb push_back
#define pf push_front
#define N 100050
#define M ll(1e9 + 7)
#define inf 1e9 + 1e9
using namespace std;
using namespace __gnu_pbds;
typedef long double ld;
typedef long long ll;
typedef unsigned long long ull;
typedef short int si;
typedef array <ll, 5> a5;
typedef array <ll, 4> a4;
//typedef tree <ll, null_type, less_equal<ll>, rb_tree_tag, tree_order_statistics_node_update> ordered_set;
gp_hashtable <int, gp_hashtable <int, int> > mp;
si a[2001][2001];
char mk[2001][2001];
int id = 2;
ll sum;
int ans, dp[500], pr[500];
vector <a5> edges;
bool cmp(a4 x, a4 y) {return a[x[0]][x[1]] + a[x[2]][x[3]] > a[y[0]][y[1]] + a[y[2]][y[3]];}
vector <a4 > gr;
vector <pair <int, int> > wh, bl;
bool opr(int x, int y) {return ((x + y) % 2) == 0;}
void add(pair <int, int> t) {if (mk[t.F][t.S] == 'b') return; mk[t.F][t.S] = 'b'; if (opr(t.F, t.S)) bl.pb({t.F, t.S}); else wh.pb({t.F, t.S});}
void add_edge(int from, int to, int cost)
{
edges.pb({from, to, cost, 0, 1});
edges.pb({to, from, -cost, 0, 0});
}
int main()
{
//freopen("input.txt", "r", stdin); //freopen("output4.txt", "w", stdout);
ios_base::sync_with_stdio(0); istream::sync_with_stdio(0); cin.tie(0); cout.tie(0);
int n, k;
cin >> n >> k;
for (register int i = 0; i < n; i++)
for (register int j = 0; j < n; j++)
{
cin >> a[i][j];
sum += ll(a[i][j]);
if (i != 0) gr.pb({i, j, i - 1, j});
if (j != 0) gr.pb({i, j, i, j - 1});
}
sort(gr.begin(), gr.end(), cmp);
for (int i = 0; i < min(50, sz(gr)); i++)
{
add({gr[i][0], gr[i][1]}); add({gr[i][2], gr[i][3]});
pair <int, int> pt; pt.F = gr[i][0]; pt.S = gr[i][1];
if (pt.F != 0) add({pt.F - 1, pt.S});
if (pt.S != 0) add({pt.F, pt.S - 1});
if (pt.F + 1 != n) add({pt.F + 1, pt.S});
if (pt.S + 1 != n) add({pt.F, pt.S + 1});
pt.F = gr[i][2]; pt.S = gr[i][3];
if (pt.F != 0) add({pt.F - 1, pt.S});
if (pt.S != 0) add({pt.F, pt.S - 1});
if (pt.F + 1 != n) add({pt.F + 1, pt.S});
if (pt.S + 1 != n) add({pt.F, pt.S + 1});
}
for (auto it : wh)
{
mp[it.F][it.S] = id++;
add_edge(0, mp[it.F][it.S], 0);
}
for (auto it : bl)
{
mp[it.F][it.S] = id++;
add_edge(mp[it.F][it.S], 1, 0);
}
for (auto it : bl)
for (auto itr : wh)
if (abs(itr.F - it.F) + abs(itr.S - it.S) == 1)
add_edge(mp[itr.F][itr.S], mp[it.F][it.S], -(a[itr.F][itr.S] + a[it.F][it.S]));
for (; k > 0; k--)
{
for (int i = 0; i < 500; i++) dp[i] = 2e9;
dp[0] = 0;
for (int it = 0; it < 100; it++)
for (int i = 0; i < sz(edges); i++)
{
int to = edges[i][1], from = edges[i][0], cost = edges[i][2], flow = edges[i][3], cap = edges[i][4];
if (flow < cap && dp[from] < 2e9 && dp[from] + cost < dp[to])
{
dp[to] = dp[from] + cost;
pr[to] = i;
}
}
ans += dp[1];
int v = 1;
while (v != 0)
{
int id = pr[v];
edges[id][3]++;
edges[id ^ 1][3]--;
v = edges[id][0];
}
}
cout << sum + ll(ans) << '\n';
}
Compilation message
domino.cpp:33:1: error: 'gp_hashtable' does not name a type
gp_hashtable <int, gp_hashtable <int, int> > mp;
^~~~~~~~~~~~
domino.cpp: In function 'int main()':
domino.cpp:115:9: error: 'mp' was not declared in this scope
mp[it.F][it.S] = id++;
^~
domino.cpp:115:9: note: suggested alternative: 'cmp'
mp[it.F][it.S] = id++;
^~
cmp
domino.cpp:122:9: error: 'mp' was not declared in this scope
mp[it.F][it.S] = id++;
^~
domino.cpp:122:9: note: suggested alternative: 'cmp'
mp[it.F][it.S] = id++;
^~
cmp
domino.cpp:130:20: error: 'mp' was not declared in this scope
add_edge(mp[itr.F][itr.S], mp[it.F][it.S], -(a[itr.F][itr.S] + a[it.F][it.S]));
^~
domino.cpp:130:20: note: suggested alternative: 'cmp'
add_edge(mp[itr.F][itr.S], mp[it.F][it.S], -(a[itr.F][itr.S] + a[it.F][it.S]));
^~
cmp