Submission #1059598

#TimeUsernameProblemLanguageResultExecution timeMemory
1059598pccFountain Parks (IOI21_parks)C++17
15 / 100
3593 ms100360 KiB
#include "parks.h" #include <bits/stdc++.h> #include <unistd.h> using namespace std; #pragma GCC optimize("O3,unroll-loops") #pragma GCC target("avx2,popcnt,sse4") #define pii pair<int,int> #define fs first #define sc second const int mxn = 2e5+10; int N; int S; struct DSU{ int dsu[mxn]; DSU(){ iota(dsu,dsu+mxn,0); } int root(int k){ return k == dsu[k]?k:dsu[k] = root(dsu[k]); } void onion(int a,int b){ dsu[root(a)] = root(b); } }; DSU dsu; struct DINIC{ struct E{ int t,f,c; E(int tt = 0,int cc = 0,int ff = 0):t(tt),c(cc),f(ff){} }; vector<int> lvl,ptr; vector<vector<int>> paths; vector<E> e; DINIC(int n = 0){ lvl = vector<int>(n,0); ptr = vector<int>(n,0); paths.resize(n); e.clear(); } void add_edge(int a,int b,int c,int d = 0){ paths[a].push_back(e.size()); e.push_back(E(b,c,0)); paths[b].push_back(e.size()); e.push_back(E(a,d,0)); return; } queue<int> q; bool bfs(int s,int t){ fill(lvl.begin(),lvl.end(),-1); fill(ptr.begin(),ptr.end(),0); lvl[s] = 0; q.push(s); while(!q.empty()){ auto now = q.front(); q.pop(); for(int i = 0;i<paths[now].size();i++){ int eid = paths[now][i]; if(e[eid].f == e[eid].c||lvl[e[eid].t] != -1)continue; lvl[e[eid].t] = lvl[now]+1; q.push(e[eid].t); } } return lvl[t] != -1; } int dfs(int now,int t,int fl){ if(now == t)return fl; for(int &i = ptr[now];i<paths[now].size();i++){ int eid = paths[now][i]; if(e[eid].f == e[eid].c||lvl[e[eid].t] != lvl[now]+1)continue; int nxt = e[eid].t; int re = 0; if(re = dfs(nxt,t,min(fl,e[eid].c-e[eid].f))){ e[eid].f += re; e[eid^1].f -= re; return re; } } return 0; } int flow(int s,int t){ int ans = 0; while(bfs(s,t)){ int re = 0; while(re = dfs(s,t,INT_MAX))ans += re; } return ans; } }; vector<int> ans[4]; void addans(int s,int e,int x,int y){ dsu.onion(s,e); ans[0].push_back(s); ans[1].push_back(e); ans[2].push_back(x); ans[3].push_back(y); return; } map<pii,int> mp; vector<pii> edges; vector<pii> all; DINIC g; void check(int a,int b){ if(dsu.root(a) == dsu.root(b))return; edges.push_back(pii(a,b)); dsu.onion(a,b); return; } int construct_roads(std::vector<int> x, std::vector<int> y) { S = clock(); if (x.size() == 1) { build({}, {}, {}, {}); return 1; } N = x.size(); for(int i = 0;i<N;i++){ pii pos = pii(x[i],y[i]); if(mp.find(pii(pos.fs-2,pos.sc)) != mp.end())check(mp[pii(pos.fs-2,pos.sc)],i); if(mp.find(pii(pos.fs+2,pos.sc)) != mp.end())check(mp[pii(pos.fs+2,pos.sc)],i); if(mp.find(pii(pos.fs,pos.sc-2)) != mp.end())check(mp[pii(pos.fs,pos.sc-2)],i); if(mp.find(pii(pos.fs,pos.sc+2)) != mp.end())check(mp[pii(pos.fs,pos.sc+2)],i); mp[pos] = i; } if(edges.size() != N-1)return 0; for(int i = 0;i<edges.size();i++){ auto [s,e] = edges[i]; if(x[s]>x[e])swap(s,e); if(x[s] == x[e]&&y[s]>y[e])swap(s,e); if(x[s] == x[e]){ all.push_back(pii(x[s]-1,(y[s]+y[e])>>1)); all.push_back(pii(x[s]+1,(y[s]+y[e])>>1)); } else if(y[s] == y[e]){ all.push_back(pii((x[s]+x[e])>>1,y[s]-1)); all.push_back(pii((x[s]+x[e])>>1,y[s]+1)); } } assert(all.size()<1e6); //cerr<<"EDGES: ";for(auto [s,e]:edges)cerr<<s<<' '<<e<<endl; sort(all.begin(),all.end());all.erase(unique(all.begin(),all.end()),all.end()); g = DINIC(all.size()+edges.size()+2); assert(all.size()+edges.size()+2<=1e6); int s = all.size()+edges.size();int t = s+1; for(int i = 0;i<edges.size();i++){ g.add_edge(s,i,1); } for(int i = 0;i<all.size();i++){ g.add_edge(edges.size()+i,t,1); } for(int i = 0;i<edges.size();i++){ auto [s,e] = edges[i]; if(x[s] == x[e]){ int p = lower_bound(all.begin(),all.end(),pii(x[s]-1,(y[s]+y[e])>>1))-all.begin(); g.add_edge(i,p+edges.size(),1); p = lower_bound(all.begin(),all.end(),pii(x[s]+1,(y[s]+y[e])>>1))-all.begin(); g.add_edge(i,p+edges.size(),1); } else if(y[s] == y[e]){ int p = lower_bound(all.begin(),all.end(),pii((x[s]+x[e])>>1,y[s]-1))-all.begin(); g.add_edge(i,p+edges.size(),1); p = lower_bound(all.begin(),all.end(),pii((x[s]+x[e])>>1,y[s]+1))-all.begin(); g.add_edge(i,p+edges.size(),1); } } int re = 0; re = g.flow(s,t); if(re != N-1)return 0; for(int i = 0;i<edges.size();i++){ for(int j = 0;j<g.paths[i].size();j++){ int eid = g.paths[i][j]; int nxt = g.e[eid].t; if(g.e[eid].f == 1&&ans[0].size()<N-1){ assert(nxt>=edges.size()); addans(edges[i].fs,edges[i].sc,all[nxt-edges.size()].fs,all[nxt-edges.size()].sc); } } } build(ans[0],ans[1],ans[2],ans[3]); return 1; }

Compilation message (stderr)

parks.cpp: In constructor 'DINIC::E::E(int, int, int)':
parks.cpp:34:11: warning: 'DINIC::E::c' will be initialized after [-Wreorder]
   34 |   int t,f,c;
      |           ^
parks.cpp:34:9: warning:   'int DINIC::E::f' [-Wreorder]
   34 |   int t,f,c;
      |         ^
parks.cpp:35:3: warning:   when initialized here [-Wreorder]
   35 |   E(int tt = 0,int cc = 0,int ff = 0):t(tt),c(cc),f(ff){}
      |   ^
parks.cpp: In member function 'bool DINIC::bfs(int, int)':
parks.cpp:62:19: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   62 |    for(int i = 0;i<paths[now].size();i++){
      |                  ~^~~~~~~~~~~~~~~~~~
parks.cpp: In member function 'int DINIC::dfs(int, int, int)':
parks.cpp:73:26: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
   73 |   for(int &i = ptr[now];i<paths[now].size();i++){
      |                         ~^~~~~~~~~~~~~~~~~~
parks.cpp:78:10: warning: suggest parentheses around assignment used as truth value [-Wparentheses]
   78 |    if(re = dfs(nxt,t,min(fl,e[eid].c-e[eid].f))){
      |       ~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
parks.cpp: In member function 'int DINIC::flow(int, int)':
parks.cpp:90:13: warning: suggest parentheses around assignment used as truth value [-Wparentheses]
   90 |    while(re = dfs(s,t,INT_MAX))ans += re;
      |          ~~~^~~~~~~~~~~~~~~~~~
parks.cpp: In function 'int construct_roads(std::vector<int>, std::vector<int>)':
parks.cpp:135:18: warning: comparison of integer expressions of different signedness: 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} and 'int' [-Wsign-compare]
  135 |  if(edges.size() != N-1)return 0;
      |     ~~~~~~~~~~~~~^~~~~~
parks.cpp:136:17: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  136 |  for(int i = 0;i<edges.size();i++){
      |                ~^~~~~~~~~~~~~
parks.cpp:155:17: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  155 |  for(int i = 0;i<edges.size();i++){
      |                ~^~~~~~~~~~~~~
parks.cpp:158:17: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  158 |  for(int i = 0;i<all.size();i++){
      |                ~^~~~~~~~~~~
parks.cpp:161:17: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  161 |  for(int i = 0;i<edges.size();i++){
      |                ~^~~~~~~~~~~~~
parks.cpp:179:17: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  179 |  for(int i = 0;i<edges.size();i++){
      |                ~^~~~~~~~~~~~~
parks.cpp:180:18: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  180 |   for(int j = 0;j<g.paths[i].size();j++){
      |                 ~^~~~~~~~~~~~~~~~~~
parks.cpp:183:37: warning: comparison of integer expressions of different signedness: 'std::vector<int>::size_type' {aka 'long unsigned int'} and 'int' [-Wsign-compare]
  183 |    if(g.e[eid].f == 1&&ans[0].size()<N-1){
      |                        ~~~~~~~~~~~~~^~~~
In file included from /usr/include/c++/10/cassert:44,
                 from /usr/include/x86_64-linux-gnu/c++/10/bits/stdc++.h:33,
                 from parks.cpp:2:
parks.cpp:184:15: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  184 |     assert(nxt>=edges.size());
      |            ~~~^~~~~~~~~~~~~~
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...