# | Time | Username | Problem | Language | Result | Execution time | Memory |
---|---|---|---|---|---|---|---|
651659 | TimDee | Flight to the Ford (BOI22_communication) | C++17 | 0 ms | 0 KiB |
This submission is migrated from previous version of oj.uz, which used different machine for grading. This submission may have different result if resubmitted.
/*
SAMPLE GRADER for task COMMUNICATION
USAGE:
place together with your solution and communication.h in the same directory, then open the terminal in this directory (right-click onto an empty spot in the directory, left click on "Open in terminal") and enter:
g++ <flags> sample_grader.cpp <solution_file>
e.g.:
g++ -std=c++17 sample_grader.cpp communication.cpp
This will create a file a.out in the current directory which you can execute from the terminal as ./a.out
INPUT/OUTPUT:
The sample grader first expects on standard input the integers N and X (1 <= X <= N).
Then, the grader calls encode(N, X) and writes to standard output a protocol of all
calls to send by your program. For each call to send it expects the return value on
standard input.
Afterwards the grader calls decode(N) and writes to standard output a protocol of all
calls to receive by your program. Upon termination, it writes your verdict to standard
output.
*/
#include<vector>
#include<cstdio>
#include<set>
#include<cstdlib>
#include<cstdarg>
#include<cassert>
//#include"communication.h"
using namespace std;
void __attribute__((noreturn)) __attribute__((format(printf, 1, 2))) result(const char *msg, ...)
{
va_list args;
va_start(args, msg);
vfprintf(stdout, msg, args);
fprintf(stdout, "\n");
va_end(args);
exit(0);
}
namespace
{
enum { ENCODE, DECODE } current_phase;
int N, X;
vector<int> signals;
size_t cursor = 0;
bool flipped = false;
}
#include <bits/stdc++.h>
using namespace std;
mt19937 rng2(chrono::steady_clock::now().time_since_epoch().count());
int rand2(int a, int b) {
return a+rng2()%(b-a+1);
}
int send(int s)
{
if(current_phase == DECODE or (s != 0 and s != 1))
result("Invalid send.");
//printf("send(%d) -> ", s); fflush(stdout);
int answer;
answer=rand2(1,100); //cout<<"("<<answer<<") ";
answer&=1;
//if(scanf("%d", &answer) != 1 or (answer != 0 and answer != 1))
// result("Invalid reply to send.");
bool flipped_now = (s != answer);
if(flipped and flipped_now)
{answer^=1, flipped_now^=1;}//result("Invalid reply to send");
flipped = flipped_now;
//cout<<answer<<'\n';
signals.push_back(answer);
if(signals.size() > (size_t) 250)
result("Looks (and smells) fishy.");
return signals.back();
}
int receive()
{
if(current_phase == ENCODE) result("Invalid receive.");
if(cursor >= signals.size()) result("Assistant waiting for Godot.");
int r = signals[cursor++];
//printf("receive() -> %d\n", r);
return r;
}
/*
int len=4;
vector<vector<vector<vector<int>>>> dp((1<<len),vector<vector<vector<int>>>(len,vector<vector<int>>(2)));
for (int x=0; x<(1<<len); ++x) {
dp[x][0][0].push_back(x&1);
dp[x][0][1].push_back(!(x&1));
for (int l=1; l<len; ++l) {
for (auto v:dp[x][l-1][0]) {
dp[x][l][0].push_back(v+(x&(1<<l)));
dp[x][l][1].push_back(v+((!((x>>l)&1))<<l));
}
for (auto v:dp[x][l-1][1]) {
dp[x][l][0].push_back(v+(x&(1<<l)));
}
}
}
*/
//#include"communication.h"
#include <bits/stdc++.h>
using namespace std;
#define forn(i,n) for (int i=0; i<n; ++i)
const int PAIU=15;
pair<int,int> decode(int n) {
vector<vector<int>> paiu(PAIU);
for (int bit=PAIU*2-1; bit>=0; bit-=2) {
int p=!receive(), v=!receive();
p<<=1; int y=p+v;
forn(i,4) if (i!=y) paiu[bit/2].push_back(i);
}
vector<int> a={0};
for (int bit=PAIU-1; bit>=0; --bit) {
vector<int> b;
forn(i,3) for (auto x:a) b.push_back((x<<2)+paiu[bit][i]);
a=b;
}
sort(a.begin(), a.end());
while (a.size()>3) {
int n=a.size();
int f=n/4 + !!(n%4);
int s=2*(n/4) + (min(n%4,2));
int t=3*(n/4) + n%4;
vector<int> b;
int A=!receive(),B=!receive();
A<<=1;
int v=A+B;
if (v==3) {
for (int i=0; i<t; ++i) b.push_back(a[i]);
} else if (v==1) {
for (int i=0; i<f; ++i) b.push_back(a[i]);
for (int i=s; i<n; ++i) b.push_back(a[i]);
} else if (v==2) {
for (int i=0; i<s; ++i) b.push_back(a[i]);
for (int i=t; i<n; ++i) b.push_back(a[i]);
} else {
for (int i=f; i<n; ++i) b.push_back(a[i]);
}
a=b;
}
while (a.size()<3) a.push_back(a.back());
int len=4;
vector<vector<vector<vector<int>>>> dp((1<<len),vector<vector<vector<int>>>(len,vector<vector<int>>(2)));
for (int x=0; x<(1<<len); ++x) {
dp[x][0][0].push_back(x&1);
dp[x][0][1].push_back(!(x&1));
for (int l=1; l<len; ++l) {
for (auto v:dp[x][l-1][0]) {
dp[x][l][0].push_back(v+(x&(1<<l)));
dp[x][l][1].push_back(v+((!((x>>l)&1))<<l));
}
for (auto v:dp[x][l-1][1]) {
dp[x][l][0].push_back(v+(x&(1<<l)));
}
}
}
int x=0;
forn(i,4) {
int f=receive();
x<<=1; x+=f;
}
vector<int> ans;
for (auto v:dp[0][len-1][0]) if (v==x) ans.push_back(a[0]);
for (auto v:dp[0][len-1][1]) if (v==x) ans.push_back(a[0]);
for (auto v:dp[6][len-1][0]) if (v==x) ans.push_back(a[1]);
for (auto v:dp[6][len-1][1]) if (v==x) ans.push_back(a[1]);
for (auto v:dp[9][len-1][0]) if (v==x) ans.push_back(a[2]);
for (auto v:dp[9][len-1][1]) if (v==x) ans.push_back(a[2]);
if (ans.size()==1) ans.push_back(ans[0]);
for (auto &x:ans) x=min(x,n);
return {ans[0],ans[1]};
}
vector<int> bfv;
void bf(int x,vector<vector<int>>&paiu, int i) {
if (i<0) {bfv.push_back(x); return;}
for (auto y:paiu[i]) bf((x<<2)+y,paiu,i-1);
}
void encode(int n, int x) {
vector<vector<int>> paiu(PAIU);
for (int bit=PAIU*2-1; bit>=0; bit-=2) {
int f=(x>>bit)&1;
int s=(x>>(bit-1))&1;
int p=!send(f), v=!send(s);
p<<=1; int y=p+v;
forn(i,4) if (i!=y) paiu[bit/2].push_back(i);
}
bf(0,paiu,PAIU-1);
vector<int> a=bfv;
//vector<int> a={0};
//for (int bit=PAIU-1; bit>=0; --bit) {
// vector<int> b;
// forn(i,3) for (auto x:a) b.push_back((x<<2)+paiu[bit][i]);
// a=b;
//}
sort(a.begin(), a.end());
exit(0);
while (a.size()>3) {
int n=a.size();
int f=n/4 + !!(n%4);
int s=2*(n/4) + (min(n%4,2));
int t=3*(n/4) + n%4;
int l=0, r=n-1;
while (l<r) {
int mid=(l+r+1)>>1;
if (a[mid]>x) r=mid-1;
else l=mid;
}
vector<int> b;
int A,B;
if (r<f) {
A=!send(0), B=!send(0);
} else if (r<s) {
A=!send(0), B=!send(1);
} else if (r<t) {
A=!send(1), B=!send(0);
} else {
A=!send(1), B=!send(1);
}
A<<=1;
int v=A+B;
if (v==3) {
for (int i=0; i<t; ++i) b.push_back(a[i]);
} else if (v==1) {
for (int i=0; i<f; ++i) b.push_back(a[i]);
for (int i=s; i<n; ++i) b.push_back(a[i]);
} else if (v==2) {
for (int i=0; i<s; ++i) b.push_back(a[i]);
for (int i=t; i<n; ++i) b.push_back(a[i]);
} else {
for (int i=f; i<n; ++i) b.push_back(a[i]);
}
a=b;
}
while (a.size()<3) a.push_back(a.back());
if (x==a[0]) {
send(0); send(0); send(0); send(0);
} else if (x==a[1]) {
send(0); send(1); send(1); send(0);
} else {
send(1); send(0); send(0); send(1);
}
}
int main()
{
if(scanf("%d %d", &N, &X) != 2 or X < 1 or X > N)
result("Invalid input.");
current_phase = ENCODE;
encode(N, X);
current_phase = DECODE;
auto r = decode(N); cout<<"!"<<' '<<r.first<<' '<<r.second<<'\n';
if(r.first < 1 or r.first > N or r.second < 1 or r.second > N)
result("Invalid answer.");
if(r.first == X or r.second == X)
result("Correct: %d signals sent.", (int) signals.size());
else
result("Wrong answer.");
}