This submission is migrated from previous version of oj.uz, which used different machine for grading. This submission may have different result if resubmitted.
// surely it's not too hard...?
// Subtask 5.
#if not LOCAL
#define NDEBUG 1
#endif
#include<bits/stdc++.h>
std::vector<int> positions; // it's more convenient to make it global. It's not changed very frequently anyway
struct Tree{
using Data=std::array<int64_t, 2>; // sort of parallel data tuple. Can be used for anything.
std::vector<Data> data; // "lazy"
Tree(std::size_t number): data(number*2){}
static void add1(Data& first, Data sec){
first[0]+=sec[0];
first[1]+=sec[1];
}
int offset() const{return int(data.size()/2);}
Data rawGet(int index)const{ // get at a particular position.
index+=offset();
Data result{};
for(; index; index>>=1)
add1(result, data[index]);
return result;
}
void add(int left, int right, Data const value){ // add a value to a range.
left+=offset(); right+=offset();
assert(left<=right);
while(left<right){
if(left&1) add1(data[left++], value);
if(right&1) add1(data[--right], value);
left>>=1; right>>=1;
}
}
int64_t getAtIndex(int index)const{
// again, {a, b} means a*x+b.
auto const [a, b]=(*this).rawGet(index);
return (int64_t)a*positions[index]+b;
}
int64_t getAtValue(int value)const{
// interpolate if necessary. (always work in this particular problem...)
// could be a little simpler (by explicitly store the slope on each segment), but...
// :(
auto const iterator=std::lower_bound(begin(positions), end(positions), value);
int64_t const value2=getAtIndex(int(iterator-positions.begin()));
if(*iterator==value) return value2;
int64_t const value1=getAtIndex(int(iterator-positions.begin())-1);
auto const number=value1*(*iterator-value)+value2*(value-iterator[-1]);
auto const de=*iterator-iterator[-1];
assert(number%de==0);
return number/de;
}
int minIndex() const{
// assumes some specific kind of function shape.
// currently O(log^2 n)
// could be O(log n) instead.
// :(
// the boundary binary search might be reducible to O(log n)
// too with some careful planning. I don't know.
auto const value=[&](int pos){
assert(0<=pos and pos<offset());
return (*this).rawGet(pos)[0];
};
int pos=offset();
for(int step=1<<30; step>>=1;){
if(pos-step>=0 and value(pos-step)>=0) pos-=step;
}
// now pos=first element with value >=0
std::pair<int64_t, int> result{INT64_MAX, INT_MAX};
for(int pos1=pos-1; pos1<=pos; ++pos1){ // perhaps this can be simpler? I don't know...
if((unsigned) pos1<(unsigned)offset())
result=std::min(result, std::make_pair(getAtIndex(pos1), pos1));
}
assert(([&]{ // assertions are good.
for(int pos1=0; pos1<offset(); ++pos1){
assert(getAtIndex(pos1)>=result.first);
}
return true;
}()));
return result.second;
}
};
int main(){
std::ios::sync_with_stdio(0);std::cin.tie(0);
int k, number; std::cin>>k>>number;
if(k==1) return 1;
int64_t result{};
std::vector<std::array<int, 2>> queries;
for(int _=0; _<number; ++_){
char a, b; int first, sec;
std::cin>>a>>first>>b>>sec;
if(first>sec) std::swap(first, sec);
result+=sec-first; // minimum required
if(a!=b){
result+=1; // need to cross a bridge anyway
queries.push_back({first, sec});
positions.push_back(first); positions.push_back(sec);
}
}
number=-1;
if(queries.empty()){
std::cout<<result<<'\n';
return 0;
}
std::sort(begin(positions), end(positions));
positions.erase(std::unique(begin(positions), end(positions)), end(positions));
auto minAdd=INT64_MAX;
std::sort(begin(queries), end(queries), [&](auto const& first, auto const& sec){
return first[0]+first[1]<sec[0]+sec[1];
});
Tree ffirst(positions.size()), fsec(positions.size());
// in this problem value (a, b) means y=a*x+b
auto const addSlope=[&](Tree& tree, int first, int sec, int multiplier){
auto const ifirst=int(std::lower_bound(begin(positions), end(positions), first)-begin(positions));
auto const isec=int(std::lower_bound(begin(positions), end(positions), sec)-begin(positions));
assert(positions[ifirst]==first);
assert(positions[isec]==sec);
assert(first<=sec);
assert(tree.offset()==(int)positions.size());
tree.add(0, ifirst, {-1*multiplier, first*multiplier});
tree.add(isec, tree.offset(), {1*multiplier, -sec*multiplier});
// * Data could be std::pair<int, int64_t> * save implementation time!
};
for(auto [first, sec]: queries)
addSlope(fsec, first, sec, 1);
for(int slice=0, lastSum=0; slice<=(int)queries.size(); ++slice){
int curSum=slice==(int)queries.size() ? INT_MAX: queries[slice][0]+queries[slice][1];
if(curSum>lastSum){
// find minAdd in the current slice (first<=sec, lastSum<=first+sec<=curSum)
auto const ifirst=ffirst.minIndex();
auto const isec=fsec.minIndex();
auto const sum1=positions[ifirst]+positions[isec];
auto const A=sum1<lastSum, B=sum1>curSum, C=ifirst>isec;
auto const searchFunction=[&](auto function, int left, int right){
// minAdd=min(minAdd, function(i)) for i in left..=right
// given that function is tern any (or something like that) on the segment
//
// complexity: log(value) calls to function. (which is actually pretty terrible?)
if(left>right) return; // just to be sure
int pos=left; auto value=function(pos);
auto const process=[&](int pos1){
auto const value1=function(pos1);
if(value1<value){
pos=pos1; value=value1;
}
};
for(int step=1<<30; step>>=1;){
if(pos-step>=left) process(pos-step);
if(pos+step<=right) process(pos+step);
}
minAdd=std::min(minAdd, value);
};
auto const searchSlope=[&](int sum){
assert(sum>=0);
// diff -> (sum-diff)/2, (sum+diff)/2
// diff=sum&1+2*x
// => ((sum-sum&1-2x)/2, (sum+sum&1+2x)/2)
// => ( (sum>>1)-x, ((sum+1)>>1)+x )
// obviously first<=sec
// need positions[0]<=first, sec<=positions.back()
// (to clarify: it's not "necessary", the result remains correct without that,
// but the current implementation of Tree::getAtValue can only handle those values)
searchFunction([&](int x){
return ffirst.getAtValue((sum>>1)-x)+fsec.getAtValue(((sum+1)>>1)+x);
}, 0, std::min(positions.back()-((sum+1)>>1), (sum>>1)-positions[0]));
};
if(A){
assert(not B);
if(C){
// some special case handler can be added here to be a little faster, but just in this case
// it's okay to leave it empty
}
searchSlope(lastSum);
}else if(B){
if(C){
// some special case handler can be added here to be a little faster, but just in this case
// it's okay to leave it empty
}
searchSlope(curSum);
}else if(C){
// not sure if this case is actually necessary.
searchFunction([&](int value){
return ffirst.getAtValue(value)+fsec.getAtValue(value);
}, std::max(positions[0], (lastSum+1)>>1), std::min(positions.back(), curSum>>1));
}else{
minAdd=std::min(minAdd, ffirst.getAtIndex(ifirst)+fsec.getAtIndex(isec));
}
lastSum=curSum;
}else assert(curSum==lastSum);
if(slice!=(int)queries.size()){
// apply change from queries[slice]
auto const [first, sec]=queries[slice];
addSlope(fsec, first, sec, -1);
addSlope(ffirst, first, sec, 1);
}
}
std::cout<<result+minAdd*2<<'\n';
}
# | 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... |