답안 #861596

# 제출 시각 아이디 문제 언어 결과 실행 시간 메모리
861596 2023-10-16T14:26:30 Z beaboss Zamjene (COCI16_zamjene) C++14
84 / 140
4111 ms 139548 KB
// Source: https://oj.uz/problem/view/COCI16_zamjene
//  I think it needs 2 MODS to AC :skull:

#include "bits/stdc++.h"

using namespace std;

#define s second
#define f first
#define pb push_back

typedef long long ll;

typedef pair<ll, ll> pii;
typedef vector<pii> vpii;

typedef vector<ll> vi;

#define FOR(i, a, b) for (ll i = (a); i<b; i++)

map<ll, ll> other;
ll bad = 0;

ll merges = 0;

const ll N = 1e6 + 10;
ll in[N];
ll out[N];

ll multi[N];
ll a[N];

ll random(ll a, ll b) {
	return a + rand() % (b - a + 1);
}

ll MOD = 1e9 + 7;

struct DSU {
	vector<ll> e;
	DSU(ll N) { e = vector<ll>(N, -1);}

	// get representive component (uses path compression)
	ll get(ll x) {
		if (e[x] < 0) return x;
		else {
			e[x] = get(e[x]);

			return e[x];
		}
	}

	bool same_set(ll a, ll b) { return get(a) == get(b); }

	ll size(ll x) { return -e[get(x)]; }

	void rem(ll val) {
		// cout << val << endl;
		val = get(val);

		if (out[val] != in[val]) bad--;

		// assert(other[(out[val] - in[val] + MOD) % MOD] >= size(val));
		other[(out[val] - in[val] + MOD) % MOD] -= size(val);


		if (out[val] != in[val]) merges -= other[(in[val] - out[val] + MOD) % MOD] * size(val);

		
		
	}

	void add(ll val) {
		val = get(val);
		if (out[val] != in[val]) bad++;


		if (out[val] != in[val]) merges += other[(in[val] - out[val] + MOD) % MOD] * size(val);

		other[(out[val] - in[val] + MOD) % MOD] += size(val);

		
		
	}

	bool unite(ll x, ll y) {  // union by size


		x = get(x), y = get(y);
		
		if (x == y) return false;
		
		rem(x);
		rem(y);
		
		if (-e[x] > -e[y]) swap(x, y);
		e[x] += e[y]; e[y] = x;

		in[x] = (in[x] + in[y]) % MOD;
		out[x] = (out[y] + out[x]) % MOD;

		add(x);
		
		return true;
	}

	void switch_(ll x, ll y) {
		int posx = multi[a[x]];
		int posy = multi[a[y]];
		int oldx = x;
		int oldy = y;
		x = get(x);
		y = get(y);

		rem(x);
		rem(y);
		

		out[x] = (out[x]+posy - posx + MOD) % MOD;
		out[y] = (out[y]+posx - posy + MOD) % MOD;


		// cout << x << y << out[x] << out[y] << endl;
		add(x);
		add(y);

		swap(a[oldx], a[oldy]);
	}
};

DSU dsu(N);


ll b[N];




int main() {
	ios::sync_with_stdio(false);
	cin.tie(nullptr);

	ll n,q;
	cin >> n >> q;


	FOR(i, 1, n + 1) {
		ll d;
		cin >> d;
		if (multi[d] == 0) multi[d] = random(1e4, 1e9);

		out[i] = multi[d];
		a[i]=d;
		b[i] = d;
	}

	sort(b + 1, b + n + 1);

	FOR(i, 1, n + 1) in[i] = multi[b[i]];

	FOR(i, 1, n+1)
		dsu.add(i);

	FOR(i, 0, q) {
		ll t;
		cin >> t;
		if (t == 1) {
			ll a, b;
			cin >> a >> b;
			dsu.switch_(a, b);
		} else if (t == 2) {
			ll a, b;
			cin >> a >> b;
			dsu.unite(a, b);
		} else if (t == 3) {
			if (bad) cout << "NE" << endl;
			else cout << "DA" << endl;
			// cout << bad > 0 ? "NE" : "DA" << endl;
		} else {
			cout << merges << endl;
		}
	}
	



}












# 결과 실행 시간 메모리 Grader output
1 Correct 3 ms 16472 KB Output is correct
2 Correct 3 ms 16476 KB Output is correct
3 Correct 3 ms 16476 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 3 ms 16476 KB Output is correct
2 Correct 5 ms 16472 KB Output is correct
3 Correct 4 ms 16476 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 3 ms 16476 KB Output is correct
2 Correct 3 ms 16476 KB Output is correct
3 Correct 4 ms 16376 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 3 ms 16476 KB Output is correct
2 Correct 3 ms 16476 KB Output is correct
3 Correct 3 ms 16476 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 3 ms 16476 KB Output is correct
2 Correct 3 ms 16476 KB Output is correct
3 Correct 4 ms 16476 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 11 ms 16472 KB Output is correct
2 Correct 14 ms 16472 KB Output is correct
3 Correct 12 ms 16728 KB Output is correct
# 결과 실행 시간 메모리 Grader output
1 Correct 98 ms 21092 KB Output is correct
2 Correct 118 ms 22688 KB Output is correct
3 Incorrect 153 ms 25812 KB Output isn't correct
4 Halted 0 ms 0 KB -
# 결과 실행 시간 메모리 Grader output
1 Correct 1318 ms 49120 KB Output is correct
2 Incorrect 4111 ms 139548 KB Output isn't correct
3 Halted 0 ms 0 KB -
# 결과 실행 시간 메모리 Grader output
1 Incorrect 2120 ms 81740 KB Output isn't correct
2 Halted 0 ms 0 KB -
# 결과 실행 시간 메모리 Grader output
1 Correct 1241 ms 47836 KB Output is correct
2 Incorrect 2400 ms 86916 KB Output isn't correct
3 Halted 0 ms 0 KB -