Submission #1080843

#TimeUsernameProblemLanguageResultExecution timeMemory
1080843GrayGame (IOI13_game)C++17
63 / 100
897 ms256000 KiB
#include<bits/stdc++.h>
#include "game.h"
#define ll long long
#define ff first
#define ss second
#define ln "\n"
using namespace std;
struct SegTree1D{
	struct Node{
		Node *left, *right;
		ll gcd;
		Node(){
			left=right=nullptr; gcd=0;
		}
		void update(int tl, int tr, int i, ll x, map<int, Node*> &leaves){
			if (tl==tr){
				gcd=x;
				leaves[tl]=this;
			}else{
				int tm = (tl+tr)/2;
				if (i<=tm) {
					if (!left) left=new Node();
					left->update(tl, tm, i, x, leaves);
				}else{
					if (!right) right=new Node();
					right->update(tm+1, tr, i, x, leaves);
				}
				gcd=0;
				if(left) gcd=__gcd(left->gcd, gcd);
				if (right) gcd=__gcd(right->gcd, gcd);
			}
		}
		ll query(int tl, int tr, int l, int r){
			if (tl==l and tr==r){
				return gcd;
			}else if (l>r) return 0;
			else{
				int tm = (tl+tr)/2;
				ll ret=0;
				if (left and l<=min(r, tm)) ret=__gcd(ret, left->query(tl, tm, l, min(r, tm)));
				if (right and max(l, tm+1)<=r) ret=__gcd(ret, right->query(tm+1, tr, max(l, tm+1), r));
				return ret;
			}
		}
	};
	map<int, Node*> leaves;
	Node * root;
	int rsz;
	SegTree1D(int N){
		rsz=N;
		root=new Node();
	}
	void update(int i, ll x){
		root->update(0, rsz-1, i, x, leaves);
	}
	ll query(int l, int r){
		return root->query(0, rsz-1, l, r);
	}
	ll getval(int ind){
		if (!leaves.count(ind)) return 0;
		return leaves[ind]->gcd;
	}
};
struct SegTree2D{
	struct mNode{
		mNode *left, *right;
		SegTree1D *val;
		int csz;
		mNode(int _csz){
			csz=_csz;
			left=right=nullptr;
			val = new SegTree1D(csz);
		}
		void update(int tl, int tr, int i, int j, ll x){
			if (tl==tr){
				val->update(j, x);
			}else{
				int tm = (tl+tr)/2;
				if (i<=tm){
					if (!left) left=new mNode(csz);
					left->update(tl, tm, i, j, x);
				}else{
					if (!right) right=new mNode(csz);
					right->update(tm+1, tr, i, j, x);
				}
				val->update(j, __gcd((left?left->val->getval(j):0), (right?right->val->getval(j):0)));
			}
		}
		ll query(int li, int ri, int lj, int rj, int tl, int tr){
			if (tl==li and tr==ri){
				return val->query(lj, rj);
			}else if (li>ri) return 0;
			else{
				ll ret=0;
				int tm = (tl+tr)/2;
				if (left and li<=min(ri, tm)) ret=__gcd(ret, left->query(li, min(ri, tm), lj, rj, tl, tm));
				if (right and max(li, tm+1)<=ri) ret=__gcd(ret, right->query(max(li, tm+1), ri, lj, rj, tm+1, tr));
				return ret;
			}
		}
	};
	mNode * root;
	ll csz, rsz;
	void init(int R, int C){
		rsz=R; csz=C;
		root = new mNode(csz);
	}
	void update(int i, int j, ll x){
		root->update(0, rsz-1, i, j, x);
	}
	ll query(int li, int ri, int lj, int rj){
		return root->query(li, ri, lj, rj, 0, rsz-1);
	}
} DS;
 
 
void init(int R, int C) {
	DS.init(R, C);
}
 
void update(int P, int Q, long long K) {
	DS.update(P, Q, K);
}
 
long long calculate(int P, int Q, int U, int V) {
	/* ... */
	return DS.query(P, U, Q, V);
}
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...