KM
code:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
ll w[305][305]; // cost matrix
namespace KM {
ll cal(int n, int m) {
vector<ll> u(n + 1), v(m + 1), p(m + 1), way(m + 1);
for (int i = 1; i <= n; i++) {
p[0] = i;
ll j0 = 0;
vector<ll> minv(m + 1, 1e18);
vector<char> used(m + 1, false);
do {
used[j0] = true;
ll i0 = p[j0], delta = 1e18, j1;
for (int j = 1; j <= m; ++j) {
if (!used[j]) {
ll cur = w[i0][j] - u[i0] - v[j];
if (cur < minv[j]) {
minv[j] = cur, way[j] = j0;
}
if (minv[j] < delta) {
delta = minv[j], j1 = j;
}
}
}
for (int j = 0; j <= m; ++j) {
if (used[j]) {
u[p[j]] += delta, v[j] -= delta;
}
else {
minv[j] -= delta;
}
}
j0 = j1;
} while (p[j0] != 0);
do {
ll j1 = way[j0];
p[j0] = p[j1];
j0 = j1;
} while (j0);
}
ll res = 0;
for (int i = 1; i <= m; i++) {
res += w[p[i]][i];
}
return res;
}
}
signed main() {
cin.sync_with_stdio(false), cin.tie(nullptr),cout.tie(nullptr);
int n; cin >> n;
ll x,y,z,v;
for (int i = 1; i <= n; i++) {
cin >> x >> y >> z >> v;
for (int j = 0; j <= n - 1; j++)
w[i][j+1]=x*x+y*y+(z+v*j)*(z+v*j);
}
cout << KM::cal(n, n) << '\n';
return 0;
} 最小费用流:
code:
#include <bits/stdc++.h>
using namespace std;
const int N = 607;
namespace atcoder {
template <class Cap, class Cost> struct mcf_graph {
public:
mcf_graph() {}
mcf_graph(int n) : _n(n), g(n) {}
int add_edge(int from, int to, Cap cap, Cost cost) {
assert(0 <= from && from < _n);
assert(0 <= to && to < _n);
int m = pos.size();
pos.push_back({ from, g[from].size() });
int from_id = g[from].size();
int to_id = g[to].size();
if (from == to) to_id++;
g[from].push_back(_edge{ to, to_id, cap, cost });
g[to].push_back(_edge{ from, from_id, 0, -cost });
return m;
}
struct edge {
int from, to;
Cap cap, flow;
Cost cost;
};
edge get_edge(int i) {
int m = pos.size();
assert(0 <= i && i < m);
auto _e = g[pos[i].first][pos[i].second];
auto _re = g[_e.to][_e.rev];
return edge{
pos[i].first, _e.to, _e.cap + _re.cap, _re.cap, _e.cost,
};
}
vector<edge> edges() {
int m = pos.size();
vector<edge> result(m);
for (int i = 0; i < m; i++) {
result[i] = get_edge(i);
}
return result;
}
pair<Cap, Cost> flow(int s, int t) {
return flow(s, t, numeric_limits<Cap>::max());
}
pair<Cap, Cost> flow(int s, int t, Cap flow_limit) {
return slope(s, t, flow_limit).back();
}
vector<pair<Cap, Cost>> slope(int s, int t) {
return slope(s, t, numeric_limits<Cap>::max());
}
vector<pair<Cap, Cost>> slope(int s, int t, Cap flow_limit) {
assert(0 <= s && s < _n);
assert(0 <= t && t < _n);
assert(s != t);
vector<Cost> dual(_n, 0), dist(_n);
vector<int> pv(_n), pe(_n);
vector<bool> vis(_n);
auto dual_ref = [&]() {
fill(dist.begin(), dist.end(),
numeric_limits<Cost>::max());
fill(pv.begin(), pv.end(), -1);
fill(pe.begin(), pe.end(), -1);
fill(vis.begin(), vis.end(), false);
struct Q {
Cost key;
int to;
bool operator<(Q r) const { return key > r.key; }
};
priority_queue<Q> que;
dist[s] = 0;
que.push(Q{ 0, s });
while (!que.empty()) {
int v = que.top().to;
que.pop();
if (vis[v]) continue;
vis[v] = true;
if (v == t) break;
for (int i = 0; i < g[v].size(); i++) {
auto e = g[v][i];
if (vis[e.to] || !e.cap) continue;
Cost cost = e.cost - dual[e.to] + dual[v];
if (dist[e.to] - dist[v] > cost) {
dist[e.to] = dist[v] + cost;
pv[e.to] = v;
pe[e.to] = i;
que.push(Q{ dist[e.to], e.to });
}
}
}
if (!vis[t]) {
return false;
}
for (int v = 0; v < _n; v++) {
if (!vis[v]) continue;
dual[v] -= dist[t] - dist[v];
}
return true;
};
Cap flow = 0;
Cost cost = 0, prev_cost_per_flow = -1;
vector<pair<Cap, Cost>> result;
result.push_back({ flow, cost });
while (flow < flow_limit) {
if (!dual_ref()) break;
Cap c = flow_limit - flow;
for (int v = t; v != s; v = pv[v]) {
c = min(c, g[pv[v]][pe[v]].cap);
}
for (int v = t; v != s; v = pv[v]) {
auto& e = g[pv[v]][pe[v]];
e.cap -= c;
g[v][e.rev].cap += c;
}
Cost d = -dual[s];
flow += c;
cost += c * d;
if (prev_cost_per_flow == d) {
result.pop_back();
}
result.push_back({ flow, cost });
prev_cost_per_flow = d;
}
return result;
}
private:
int _n;
struct _edge {
int to, rev;
Cap cap;
Cost cost;
};
vector<pair<int, int>> pos;
vector<vector<_edge>> g;
};
}
using namespace atcoder;
mcf_graph<int64_t, int64_t> mc(N);
int main() {
ios::sync_with_stdio(false);
cin.tie(nullptr);
int n,s,t;
long long x,y,z,v,res(0);
cin>>n;
s=0,t=2*n+1;
for(int i=1;i<=n;++i) {
cin>>x>>y>>z>>v;
for(int j=1;j<=n;++j)
mc.add_edge(j,n+i,1,z*z),z+=v;
res+=x*x+y*y;
mc.add_edge(s,i,1,0);
mc.add_edge(n+i,t,1,0);
}
cout<<res+mc.flow(s, t).second<<'\n';
} 
京公网安备 11010502036488号