#include<cstdio>
#include<vector>
using namespace std;
const int SZ = 1000010;
int min(int a, int b) { return a < b ? a : b; }
int n;
const bool ifDebug = false;
int cycleNum;
vector<int>l[SZ];
typedef struct Graph {
int st[4], ts[3];
int p[SZ], sz[SZ], ln[SZ], c[SZ];
bool is3E[SZ];
int x;
void init() {
x = n;
st[0] = ts[0] = n;
st[1] = st[2] = st[3] = ts[1] = ts[2] = 0;
for (int i = 0; i < n; i++) {
p[i] = i;
sz[i] = 1;
ln[i] = c[i] = 0;
is3E[i] = false;
}
}
void init(int X) {
if (ifDebug)printf("init(%d)\n", X);
init();
x = X;
PR();
for (int i = 0; i < n; i++) {
for (int e : l[i]) {
if (e < i)Link(e, i);
}
}
}
int f(int a) {
return a == p[a] ? a : p[a] = f(p[a]);
}
void u(int A, int B) {
int a = f(A), b = f(B);
if (a != b) {
p[b] = a;
ts[gT(a)]--;
ts[gT(b)]--;
ln[a] += ln[b];
sz[a] += sz[b];
is3E[a] |= is3E[b];
}
else {
ts[gT(a)]--;
}
}
void Con(int A, int B) {
int a = f(A);
st[min(c[A], 3)]--;
if (x == n)l[A].push_back(B);
c[A]++;
st[min(c[A], 3)]++;
if (c[A] >= 3)is3E[a] = true;
}
int gT(int a) {
if (is3E[a]) {
return 2;
}
else if (sz[a] == ln[a]) {
return 1;
}
else {
return 0;
}
}
void Link(int A, int B) {
if (A == x || B == x)return;
if (ifDebug)printf("%d:%d-%d\n", x, A, B);
u(A, B);
Con(A, B);
Con(B, A);
int a = f(A);
ln[a]++;
ts[gT(a)]++;
}
bool isChain() {
return (ts[1] + ts[2] == 0);
}
void PR() {
if (!ifDebug)return;
printf("-------%d-------\n", x);
for (int a = 0; a < n; a++) {
printf("(%d)", c[a]);
if (a == p[a]) {
printf("[%d(%d,%d,%d)]", a, sz[a], gT(a), ln[a]);
printf("\n");
}
else {
printf("%d->%d\n", a, f(a));
}
}
printf("-----------------\n");
}
}Graph;
Graph MG, GV[4];
int GS;
//type: 0-chain/1-cycle/2-3 exists/3-4 exists
void Init(int N) {
cycleNum = n = N;
MG.init();
}
void Link(int A, int B) {
bool p = (MG.st[3] == 0);
MG.Link(A, B);
for (int i = 0; i < GS; i++) {
GV[i].Link(A, B);
}
if (MG.gT(MG.f(A)) == 1)cycleNum = MG.f(A);
if (MG.gT(MG.f(B)) == 1)cycleNum = MG.f(B);
if (p && MG.st[3]) {
int a = A;
if (l[A].size() < 3)a = B;
GS = l[a].size() + 1;
GV[l[a].size()].init(a);
for (int i = 0; i < l[a].size(); i++) {
GV[i].init(l[a][i]);
}
}
}
int CountCritical() {
if (MG.ts[2]) {
if (ifDebug)printf(">=3 Exists");
int r = 0;
for (int i = 0; i < GS; i++) {
if (GV[i].isChain())r++;
}
return r;
}
else if (MG.ts[1]) {
if (ifDebug)printf("cycle Exists");
if (MG.ts[1] > 1)return 0;
return MG.sz[cycleNum];
}
else {
if (ifDebug)printf("Already Chain");
return n;
}
}
int main() {
int n, l, a, b;
scanf("%d%d", &n, &l);
Init(n);
while (l--) {
scanf("%d", &a);
if (a == -1) {
printf("%d\n", CountCritical());
}
else {
scanf("%d", &b);
Link(a, b);
}
MG.PR();
for (a = 0; a < GS; a++) {
GV[a].PR();
}
if (ifDebug)printf("cycle No: %d\n", cycleNum);
}
return 0;
}
Compilation message
rings.cpp: In function 'void Link(int, int)':
rings.cpp:122:21: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
122 | for (int i = 0; i < l[a].size(); i++) {
| ~~^~~~~~~~~~~~~
rings.cpp: In function 'int main()':
rings.cpp:148:7: warning: ignoring return value of 'int scanf(const char*, ...)' declared with attribute 'warn_unused_result' [-Wunused-result]
148 | scanf("%d%d", &n, &l);
| ~~~~~^~~~~~~~~~~~~~~~
rings.cpp:151:8: warning: ignoring return value of 'int scanf(const char*, ...)' declared with attribute 'warn_unused_result' [-Wunused-result]
151 | scanf("%d", &a);
| ~~~~~^~~~~~~~~~
rings.cpp:156:9: warning: ignoring return value of 'int scanf(const char*, ...)' declared with attribute 'warn_unused_result' [-Wunused-result]
156 | scanf("%d", &b);
| ~~~~~^~~~~~~~~~
/usr/bin/ld: /tmp/ccH3CCtf.o: in function `main':
rings.cpp:(.text.startup+0x0): multiple definition of `main'; /tmp/ccmMIWRb.o:grader.cpp:(.text.startup+0x0): first defined here
collect2: error: ld returned 1 exit status