pub use __cargo_equip::prelude::*; use max_flow::MaxFlow; #[allow(unused_imports)] use proconio::{ input, marker::{Bytes, Chars, Usize1}, }; const M: usize = 500_001; fn main() { input! { h: usize, w: usize, a: [[usize; w]; h], } let mut zh = vec![vec![]; M]; let mut zw = vec![vec![]; M]; let mut b = vec![vec![]; M]; for i in 0..h { for j in 0..w { zh[a[i][j]].push(i); zw[a[i][j]].push(j); b[a[i][j]].push((i, j)); } } for i in 0..M { zh[i].sort_unstable(); zh[i].dedup(); zw[i].sort_unstable(); zw[i].dedup(); } let mut res = 0; for x in 1..M { if zh[x].is_empty() { continue; } let mut g = MaxFlow::new(zh[x].len() + zw[x].len() + 2); let s = zh[x].len() + zw[x].len(); let t = s + 1; for i in 0..zh[x].len() { g.add_edge(s, i, 1); } for i in 0..zw[x].len() { g.add_edge(i + zh[x].len(), t, 1); } for &(i, j) in &b[x] { let p = zh[x].binary_search(&i).unwrap(); let q = zw[x].binary_search(&j).unwrap(); g.add_edge(p, q + zh[x].len(), 1); } res += g.max_flow(s, t); } println!("{}", res); } // The following code was expanded by `cargo-equip`. /// # Bundled libraries /// /// - `max-flow 0.1.0 (path+███████████████████████████████████████████████████)` published in **missing** licensed under `CC0-1.0` as `crate::__cargo_equip::crates::max_flow` #[cfg_attr(any(), rustfmt::skip)] #[allow(unused)] mod __cargo_equip { pub(crate) mod crates { pub mod max_flow {use std::collections::VecDeque;pub type FlowType=i64;const INF:FlowType=std::i64::MAX/2;pub struct MaxFlow{n:usize,edges:Vec,head:Vec,next:Vec,min_cost:Vec,iter:Vec,}#[derive(Debug,Clone,Copy)]pub struct Edge{pub from:usize,pub to:usize,pub cap:FlowType,pub flow:FlowType,}impl MaxFlow{pub fn new(n:usize)->Self{Self{n,edges:vec![],head:vec![!0;n],next:vec![],min_cost:vec![],iter:vec![],}}pub fn add_edge(&mut self,from:usize,to:usize,cap:FlowType)->usize{assert!(from!=to);assert!(from=0);let m=self.edges.len();self.next.push(self.head[from]);self.head[from]=m;self.next.push(self.head[to]);self.head[to]=m+1;self.edges.push(Edge_{to,cap});self.edges.push(Edge_{to:from,cap:0});m/2}pub fn max_flow(&mut self,s:usize,t:usize)->FlowType{assert!(s!=t);let max_cap=self.edges.iter().map(|e|e.cap).max().unwrap_or(0);if max_cap==0{return 0;}let mut flow=0;for i in(0..64-max_cap.leading_zeros()).rev(){let now=1<Vec{let mut vis=vec![false;self.n];let mut q=VecDeque::new();q.push_back(s);while let Some(v)=q.pop_front(){vis[v]=true;let mut e=self.head[v];while e!=!0{let Edge_{to,cap}=self.edges[e];if cap!=0&&!vis[to]{vis[to]=true;q.push_back(to);}e=self.next[e];}}vis}pub fn edges(&self)->Vec{(0..self.edges.len()).step_by(2).map(|i|{let e=&self.edges[i];let f=&self.edges[i+1];Edge{from:f.to,to:e.to,cap:e.cap+f.cap,flow:f.cap,}}).collect()}fn build_augment_path(&mut self,s:usize,t:usize,base:FlowType)->bool{self.min_cost=vec![-1;self.n];let mut q=VecDeque::new();self.min_cost[s]=0;q.push_back(s);while q.len()>0&&self.min_cost[t]==-1{let v=q.pop_front().unwrap();let mut e=self.head[v];while e!=!0{let Edge_{to,cap}=self.edges[e];if cap>=base&&self.min_cost[to]==-1{self.min_cost[to]=self.min_cost[v]+1;q.push_back(to);}e=self.next[e];}}self.min_cost[t]!=-1}fn find_augment_path(&mut self,v:usize,t:usize,base:FlowType,flow:FlowType,)->FlowType{if v==t{return flow;}let mut sum=0;while self.iter[v]!=!0{let Edge_{to,cap}=self.edges[self.iter[v]];if cap>=base&&self.min_cost[v]0{self.edges[self.iter[v]].cap-=d;self.edges[self.iter[v]^1].cap+=d;sum+=d;if flow-sum