#include <bits/stdc++.h>
#include <ext/pb_ds/assoc_container.hpp>
#include <ext/pb_ds/tree_policy.hpp>
#include <ext/rope>
#define ll long long
#define pb push_back
#define sz(x) (int)(x).size()
#define mp make_pair
#define f first
#define s second
#define all(x) x.begin(), x.end()
#define D(x) cerr << #x << " is " << (x) << "\n";
using namespace std;
using namespace __gnu_pbds;
using namespace __gnu_cxx;
mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
template<class T> using ordered_set = tree<T, null_type, less<T>, rb_tree_tag,tree_order_statistics_node_update>; ///find_by_order(),order_of_key()
template<class T1, class T2> ostream& operator<<(ostream& os, const pair<T1,T2>& a) { os << '{' << a.f << ", " << a.s << '}'; return os; }
template<class T> ostream& operator<<(ostream& os, const vector<T>& a){os << '{';for(int i=0;i<sz(a);i++){if(i>0&&i<sz(a))os << ", ";os << a[i];}os<<'}';return os;}
template<class T> ostream& operator<<(ostream& os, const set<T>& a) {os << '{';int i=0;for(auto p:a){if(i>0&&i<sz(a))os << ", ";os << p;i++;}os << '}';return os;}
template<class T> ostream& operator<<(ostream& os, const multiset<T>& a) {os << '{';int i=0;for(auto p:a){if(i>0&&i<sz(a))os << ", ";os << p;i++;}os << '}';return os;}
template<class T1,class T2> ostream& operator<<(ostream& os, const map<T1,T2>& a) {os << '{';int i=0;for(auto p:a){if(i>0&&i<sz(a))os << ", ";os << p;i++;}os << '}';return os;}
gp_hash_table<ll,int> cnt;
const int m=1e6+1,N=1e6+1;
const ll mod=8473968574039;
vector<int> par(N),p,q;
vector<ll> pwr,hsh(N);
vector<multiset<int> > imam(N),treba(N);
int n,qq;
ll ans4;
int ans3;
ll mul(ll a,ll b)
{
return a*b%mod;
}
ll add(ll a,ll b)
{
a+=b;
if(a>=mod)
a-=mod;
return a;
}
ll sub(ll a,ll b)
{
a-=b;
if(a<0)
a+=mod;
return a;
}
int find(int tr)
{
if(par[tr]==tr)
return tr;
return par[tr]=find(par[tr]);
}
void merge(int a,int b)
{
a=find(a);
b=find(b);
if(a==b)
return;
cnt[hsh[a]]-=imam[a].size();
if(hsh[a]!=0)
ans4-=(ll)2*imam[a].size()*cnt[sub(0,hsh[a])],ans3--;
if(hsh[b]!=0)
ans4-=(ll)2*imam[b].size()*cnt[sub(0,hsh[b])],ans3--;
cnt[hsh[b]]-=imam[b].size();
if(imam[a].size()<imam[b].size())
swap(a,b);
for(auto d:imam[b])
hsh[a]=add(hsh[a],pwr[d]),imam[a].insert(d);
for(auto d:treba[b])
hsh[a]=sub(hsh[a],pwr[d]),treba[a].insert(d);
imam[b].clear();
treba[b].clear();
par[b]=a;
if(hsh[a]!=0)
cnt[hsh[a]]+=imam[a].size(),ans4+=(ll)2*imam[a].size()*cnt[sub(0,hsh[a])],ans3++;
}
void swapuj(int a,int b)
{
int pa=p[a],pb=p[b];
swap(p[a],p[b]);
a=find(a);
b=find(b);
if(a==b)
return;
cnt[hsh[a]]-=imam[a].size();
if(hsh[a]!=0)
ans4-=(ll)2*imam[a].size()*cnt[sub(0,hsh[a])],ans3--;
if(hsh[b]!=0)
ans4-=(ll)2*imam[b].size()*cnt[sub(0,hsh[b])],ans3--;
cnt[hsh[b]]-=imam[b].size();
hsh[a]=sub(hsh[a],pwr[pa]);
hsh[a]=add(hsh[a],pwr[pb]);
hsh[b]=sub(hsh[b],pwr[pb]);
hsh[b]=add(hsh[b],pwr[pa]);
imam[a].erase(imam[a].find(pa));
imam[a].insert(pb);
imam[b].erase(imam[b].find(pb));
imam[b].insert(pa);
cnt[hsh[a]]+=imam[a].size();
if(hsh[a]!=0)
ans4+=(ll)2*imam[a].size()*cnt[sub(0,hsh[a])],ans3++;
cnt[hsh[b]]+=imam[b].size();
if(hsh[b]!=0)
ans4+=(ll)2*imam[b].size()*cnt[sub(0,hsh[b])],ans3++;
}
int main()
{
//freopen("in.txt","r",stdin);
pwr.pb(1);
for(int i=0;i<N;i++)
par[i]=i,pwr.pb(mul(pwr.back(),m));
scanf("%i %i",&n,&qq);
p.resize(n);
for(int i=0;i<n;i++)
scanf("%i",&p[i]);
q=p;
sort(all(q));
for(int i=0;i<n;i++)
imam[i].insert(p[i]),hsh[i]=add(hsh[i],pwr[p[i]]),treba[i].insert(q[i]),hsh[i]=sub(hsh[i],pwr[q[i]]),cnt[hsh[i]]++;
for(int i=0;i<n;i++)
if(hsh[i]!=0)
ans4+=cnt[sub(0,hsh[i])],ans3++;
while(qq--)
{
int t,a,b;
scanf("%i",&t);
if(t==1)
scanf("%i %i",&a,&b),swapuj(a-1,b-1);
if(t==2)
scanf("%i %i",&a,&b),merge(a-1,b-1);
if(t==3)
printf(ans3==0?"DA\n":"NE\n");
if(t==4)
printf("%lld\n",ans4/2);
}
return 0;
}
Compilation message
zamjene.cpp: In function 'int main()':
zamjene.cpp:119:10: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
scanf("%i %i",&n,&qq);
~~~~~^~~~~~~~~~~~~~~~
zamjene.cpp:122:14: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
scanf("%i",&p[i]);
~~~~~^~~~~~~~~~~~
zamjene.cpp:133:14: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
scanf("%i",&t);
~~~~~^~~~~~~~~
zamjene.cpp:135:33: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
scanf("%i %i",&a,&b),swapuj(a-1,b-1);
~~~~~~~~~~~~~~~~~~~~^~~~~~~~~~~~~~~~
zamjene.cpp:137:33: warning: ignoring return value of 'int scanf(const char*, ...)', declared with attribute warn_unused_result [-Wunused-result]
scanf("%i %i",&a,&b),merge(a-1,b-1);
~~~~~~~~~~~~~~~~~~~~^~~~~~~~~~~~~~~
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
117 ms |
114400 KB |
Output is correct |
2 |
Correct |
119 ms |
114404 KB |
Output is correct |
3 |
Correct |
118 ms |
114400 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
116 ms |
114400 KB |
Output is correct |
2 |
Correct |
117 ms |
114388 KB |
Output is correct |
3 |
Correct |
117 ms |
114272 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
116 ms |
114452 KB |
Output is correct |
2 |
Correct |
117 ms |
114340 KB |
Output is correct |
3 |
Correct |
124 ms |
114400 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
133 ms |
114292 KB |
Output is correct |
2 |
Correct |
257 ms |
114400 KB |
Output is correct |
3 |
Correct |
117 ms |
114272 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
117 ms |
114400 KB |
Output is correct |
2 |
Correct |
126 ms |
114480 KB |
Output is correct |
3 |
Correct |
120 ms |
114404 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
125 ms |
115156 KB |
Output is correct |
2 |
Correct |
127 ms |
115172 KB |
Output is correct |
3 |
Correct |
127 ms |
115368 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
235 ms |
125228 KB |
Output is correct |
2 |
Correct |
246 ms |
128084 KB |
Output is correct |
3 |
Correct |
264 ms |
136660 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
1599 ms |
192852 KB |
Output is correct |
2 |
Correct |
2039 ms |
314420 KB |
Output is correct |
3 |
Correct |
2420 ms |
461640 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
1866 ms |
298416 KB |
Output is correct |
2 |
Correct |
2053 ms |
290972 KB |
Output is correct |
3 |
Correct |
1803 ms |
297516 KB |
Output is correct |
# |
결과 |
실행 시간 |
메모리 |
Grader output |
1 |
Correct |
1602 ms |
225372 KB |
Output is correct |
2 |
Correct |
1832 ms |
297272 KB |
Output is correct |
3 |
Correct |
1816 ms |
297488 KB |
Output is correct |