This submission is migrated from previous version of oj.uz, which used different machine for grading. This submission may have different result if resubmitted.
#include<bits/stdc++.h>
using namespace std;
#define rep(i,a,b) for(int i = a; i<int(b);++i)
#define all(v) v.begin(),v.end()
#define sz(v) v.size()
#define trav(a,c) for(auto a: c)
typedef long long ll;
typedef vector<ll> vi;
typedef pair<ll,ll> pii;
vi dfsNum;
vi dfsLow;
ll currDfsNum = 0;
vector<vi> e;
set<pii> bridges;
void dfs(ll v, ll l){
if(dfsNum[v]!=-1) return;
dfsNum[v] = dfsLow[v] = currDfsNum++;
rep(i,0,e[v].size()){
if(e[v][i]==l) continue;
dfs(e[v][i],v);
dfsLow[v] = min(dfsLow[v],dfsLow[e[v][i]]);
}
}
vector<vi> cycles;
vi cycleIndices;
vector<bool> seen;
void getCycle(ll v){
if(seen[v]) return;
seen[v] = true;
cycles[cycles.size()-1].push_back(v);
cycleIndices[v] = cycles.size()-1;
rep(i,0,e[v].size()){
if(bridges.find({v,e[v][i]})==bridges.end())
getCycle(e[v][i]);
}
}
vector<vi> treeE;
vi dp;
ll numChildren(ll v,ll l){
if(dp[v]!=-1) return dp[v];
ll ans = cycles[v].size();
rep(i,0,treeE[v].size()){
if(treeE[v][i] == l) continue;
ans += numChildren(treeE[v][i],v);
}
return dp[v] = ans;
}
ll getNum(ll v,ll l,ll childrenUp){
seen[v] = true;
ll childSum = 0;
vi childrenSizes;
rep(i,0,treeE[v].size())
if(treeE[v][i] != l){
childSum += numChildren(treeE[v][i],v);
childrenSizes.push_back(numChildren(treeE[v][i],v));
}
childSum+=childrenUp;
childrenSizes.push_back(childrenUp);
//cout<<v<<" "<<l<<" "<<childrenUp<<endl;
ll ans = 0;
ll sz = cycles[v].size();
rep(i,0,childrenSizes.size()){
ans += childrenSizes[i]*(childSum - childrenSizes[i])*sz;//c here
//cout<<ans<<endl;
}
ans += childSum*(sz-1)*(sz-1)*2;//2 here
//cout<<ans<<endl;
ans += (sz)*(sz-1)*(sz-2);//3 here
//cout<<ans<<endl;
rep(i,0,treeE[v].size()){
if(treeE[v][i] == l) continue;
ans += getNum(treeE[v][i],v,childSum+sz-numChildren(treeE[v][i],v));
}
return ans;
}
ll n,m;
ll brute(){
ll ans = 0;
set<ll> bridgeNodes;
rep(i,0,n){
bool bridgeNode = false;
rep(j,0,e[i].size()){
if(dfsNum[e[i][j]]>dfsNum[i]&&dfsLow[e[i][j]]>=dfsNum[i]){
bridgeNode = true;
}
}
if(bridgeNode||e[i].size()==1) {
//cout<<i+1<<endl;
bridgeNodes.insert(i);
}
}
rep(s,0,n)
rep(f,0,n){
if(f==s)
continue;
bool fail = false;
set<ll> neededNodes;
vi p(n,-1);
queue<pii> q;
q.push({s,-2});
while(q.size()){
ll v = q.front().first;
ll parent = q.front().second;
q.pop();
if(p[v]!=-1) continue;
p[v] = parent;
rep(i,0,e[v].size())
q.push({e[v][i],v});
}
if(p[f]==-1) continue;
ll v = f;
neededNodes.insert(f);
neededNodes.insert(s);
while(p[v]!=-2){
v = p[v];
if(bridgeNodes.find(v)!=bridgeNodes.end()){
neededNodes.insert(v);
}
}
vector<bool> seen(n,false);
queue<ll> q2;
q2.push(s);
while(q2.size()){
ll v = q2.front();
q2.pop();
if(bridgeNodes.find(v)!=bridgeNodes.end()&&neededNodes.find(v)==neededNodes.end()) continue;
if(seen[v]) continue;
seen[v] = true;
rep(i,0,e[v].size()) q2.push(e[v][i]);
}
ans -= 2;
rep(i,0,n) ans += seen[i];
}
return ans;
}
int main(){
cin.sync_with_stdio(false);
cin>>n>>m;
e.resize(n);
dfsNum.resize(n,-1);
dfsLow.resize(n,-1);
vector<pii> edges(m);
rep(i,0,m){
ll a,b;
cin>>a>>b;
--a; --b;
edges[i] = {a,b};
e[a].push_back(b);
e[b].push_back(a);
}
rep(i,0,n)
dfs(i,-1);
//cout<<endl<<endl;
rep(i,0,n){
//cout<<i+1<<": "<<dfsNum[i]<<" "<<dfsLow[i]<<endl;
}
rep(i,0,m){
ll a,b;
tie(a,b) = edges[i];
if(dfsNum[a]<dfsNum[b]&&dfsLow[b]==dfsNum[b]){
bridges.insert({a,b});
bridges.insert({b,a});
}
if(dfsNum[b]<dfsNum[a]&&dfsLow[a]==dfsNum[a]){
bridges.insert({a,b});
bridges.insert({b,a});
}
}
//cout<<brute()<<endl;
//return 0;
cycleIndices.resize(n);
seen.resize(n);
rep(i,0,n){
if(seen[i]) continue;
cycles.push_back(vi(0));
getCycle(i);
//rep(j,0,cycles[cycles.size()-1].size()) cout<<cycles[cycles.size()-1][j]<<" ";
//cout<<endl;
}
treeE.resize(cycles.size());
trav(bridge,bridges){
//cout<<bridge.first+1<<" "<<bridge.second+1<<endl;
if(bridge.first<bridge.second){
ll a = cycleIndices[bridge.first];
ll b = cycleIndices[bridge.second];
treeE[a].push_back(b);
treeE[b].push_back(a);
}
}
ll ans = 0;
/*
rep(i,0,cycles.size()){
ll sz = cycles[i].size();
ans += (sz)*(sz-1)*(sz-2);
}*/
dp.resize(cycles.size(),-1);
seen.assign(cycles.size(),false);
rep(i,0,cycles.size()){
if(!seen[i]) {
ans+=getNum(i,-1,0);
//rep(j,0,cycles[i].size()) cout<<cycles[i][j]<<" ";
//cout<<": "<<endl;
//cout<<ans<<endl;
}
}
cout<<ans<<endl;
}
Compilation message (stderr)
count_triplets.cpp: In function 'll getNum(ll, ll, ll)':
count_triplets.cpp:5:20: warning: this 'for' clause does not guard... [-Wmisleading-indentation]
#define rep(i,a,b) for(int i = a; i<int(b);++i)
^
count_triplets.cpp:65:5: note: in expansion of macro 'rep'
rep(i,0,treeE[v].size())
^~~
count_triplets.cpp:70:2: note: ...this statement, but the latter is misleadingly indented as if it were guarded by the 'for'
childSum+=childrenUp;
^~~~~~~~
count_triplets.cpp: In function 'll brute()':
count_triplets.cpp:114:9: warning: unused variable 'fail' [-Wunused-variable]
bool fail = false;
^~~~
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |