正题

题目链接:https://uoj.ac/problem/454


题目大意

\(Alice\)有一个长度为\(2n\)的\(01\)串,\(Bob\)有\(n\)个在\([1,2n]\)位置的下标表示它想要得到\(01\)串中这些位置的值,现在两个人可以向对方传输不超过\(m\)个\(0/1\)字符,要求使得\(Bob\)可以得到答案。

\(1\leq n\leq 1000,m=1350\)


解题思路

两种方法,都是平均两边的传输信息。

第一种方法是从左到右传输\(01\)串,由\(Bob\)考虑若一个位置需要传输,那么传输\(1\),然后\(Alice\)传输这个位置给他并考虑下一个位置。否则传输\(0\),然后\(Alice\)跳过这个位置传输下一个位置给她然后再考虑下一个位置。

不难发现这样对于\(01\)隔开的情况可以省略很多次数,所以我们直接随机打乱整个序列然后这样做即可。

第二种方法是将序列分为三块,\(Bob\)用二进制告诉\(Alice\)需要信息最多的那个块。然后剩下两个块由\(Bob\)告诉\(Alice\)需要传输哪些位置。

这样的次数\(Alice\)严格小于\(\frac{2}{3}n\times 2\),\(Bob\)严格小于\(\frac{4}{3}n+2\),都在\(1350\)次内。

因为第二种方法比较普遍所以代码使用的是第一种方法


code

Alice

#include <iostream>
#include <fstream>
#include <string>
#include<cstdlib>
#include<cstdio>
#include<algorithm> using namespace std; ifstream fin;
char get_bit() {
return getchar();
}
void send_bit(char ch) {
putchar(ch);
fflush(stdout);
} int n, m,c[2100],r[2100];
string s;
void init_t() {
fin.open("alice.in");
fin >> n >> m >> s;
}
int Z=17;
int randd(){
Z++;
return (Z*1931ll+Z*Z*3ll)%32767;
}
int main()
{
init_t();
for(int i=1;i<=2*n;i++)r[i]=i;
for(int i=1;i<=20*n;i++)swap(r[randd()%(2*n)+1],r[randd()%(2*n)+1]);
for(int i=1;i<=2*n;i++)c[r[i]]=s[i-1];
int i=1;
// for(int i=1;i<=2*n;i++)send_bit(c[i]);
while(i<=2*n){
char z=get_bit();
if(z==EOF)break;
if(z=='1'){send_bit(c[i]);i++;}
else{i++;if(i>2*n)break;send_bit(c[i]);i++;}
}
return 0;
}

Bob

#include <iostream>
#include <fstream>
#include<cstdlib>
#include<cstdio>
#include<cstring>
#include<algorithm> using namespace std; ifstream fin;
char get_bit() {
return getchar();
}
void send_bit(char ch) {
putchar(ch);
fflush(stdout);
} const int N = 1000; int n, m, p[N + 1],r[N*2+10],w[N*2+10];
char s[N+1];
ofstream fout;
void answer() {
fout.open("bob.out");
for(int i=1;i<=n;i++)
fout<<s[i];
fout<<endl;
exit(0);
}
void init_t() {
int x;
fin.open("bob.in");
fin >> n >> m;
for (x = 1; x <= n; ++x) fin >> p[x];
}
int Z=17;
int randd(){
Z++;
return (Z*1931ll+Z*Z*3ll)%32767;
}
int main() {
init_t();
for(int i=1;i<=2*n;i++)r[i]=i;
for(int i=1;i<=20*n;i++)swap(r[randd()%(2*n)+1],r[randd()%(2*n)+1]);
for(int i=1;i<=n;i++)p[i]=r[p[i]],w[p[i]]=i;
sort(p+1,p+1+n);
int z=1,i=1;
while(i<=n){
if(p[i]==z){send_bit('1');s[w[p[i]]]=get_bit();z++;i++;}
else{
send_bit('0');z++;
if(z>2*n)break;
s[w[p[i]]]=get_bit();
if(p[i]==z)i++;z++;
}
}
answer();
return 0;
}

UOJ#454-[UER #8]打雪仗【通信题】的更多相关文章

  1. UOJ#454. 【UER #8】打雪仗

    UOJ#454. [UER #8]打雪仗 http://uoj.ac/problem/454 分析: 好玩的通信题~ 把序列分成三块,\(bob\)先发出这三块中询问点最多的一块给\(alice\). ...

  2. 蕞短鹭(artskjid) (⭐通信题/模拟⭐)

    文章目录 题面(过于冗长,主要是对通信题的一些解释) 题解 1.通信题什么意思 2.此题题解 CODE 实现 题面(过于冗长,主要是对通信题的一些解释) 题解 1.通信题什么意思 并不是两个程序同时跑 ...

  3. UOJ #455 [UER #8]雪灾与外卖 (贪心、模拟费用流)

    题目链接 http://uoj.ac/contest/47/problem/455 题解 模拟费用流,一个非常神奇的东西. 本题即为WC2019 laofu的讲课中的Problem 8,经典的老鼠进洞 ...

  4. [UOJ#245][UER#7]天路(近似算法)

    允许5%的相对误差,意味着我们可以只输出$\log_{1.05} V$种取值并保证答案合法.并且注意到答案随着区间长度而单增,故取值不同的答案区间是$O(\log_{1.05} V)$的. 于是初始x ...

  5. 【bzoj4736/uoj#274】[清华集训2016]温暖会指引我们前行 语文题+LCT

    题目描述 http://uoj.ac/problem/274 题解 语文题+LCT 对于这种语文题建议还是自己读题好一些... 读懂题后发现:由于温度互不相同,最大生成树上的路径必须走(不走的话温度大 ...

  6. 【UOJ#228】基础数据结构练习题 线段树

    #228. 基础数据结构练习题 题目链接:http://uoj.ac/problem/228 Solution 这题由于有区间+操作,所以和花神还是不一样的. 花神那道题,我们可以考虑每个数最多开根几 ...

  7. NOIp2018停课刷题记录

    Preface 老叶说了高中停课但是初中不停的消息后我就为争取民主献出一份力量 其实就是和老师申请了下让我们HW的三个人听课结果真停了 那么还是珍惜这次机会好好提升下自己吧不然就\(AFO\)了 Li ...

  8. 省选/NOI刷题Day2

    bzoj2616 放一个车的时候相当于剪掉棋盘的一行,于是就可以转移了,中间状态转移dp套dp,推一下即可 bzoj2878 环套树期望dp 手推一下递推式即可 bzoj3295 树状数组套权值线段树 ...

  9. uoj#300.【CTSC2017】吉夫特

    题面:http://uoj.ac/problem/300 一道大水题,然而我并不知道$lucas$定理的推论.. $\binom{n}{m}$为奇数的充要条件是$n&m=n$.那么我们对于每个 ...

随机推荐

  1. Specification使用notin

    废话不多说直接贴代码 Specification<Employee> employeeSpecification = new Specification<Employee>() ...

  2. Qt简单的解析Json数据例子(一)

    要解析的json的格式为: { "rootpath": "001", "usernum": 111, "childdep" ...

  3. opencv入门系列教学(五)图像的基本操作(像素值、属性、ROI和边框)

    0.序言 每个图像是由一个个点组成的,而这些点可以表示为像素值的形式. 这篇博客里我们将学会: 访问像素值并修改它们 . 访问图像属性 . 设置感兴趣区域(ROI) . 分割和合并图像. 对于图像的基 ...

  4. 源码解析.Net中DependencyInjection的实现

    前言 笔者的这篇文章和上篇文章思路一样,不注重依赖注入的使用方法,更加注重源码的实现,我尽量的表达清楚内容,让读者能够真正的学到东西.如果有不太清楚依赖注入是什么或怎么在.Net项目中使用的话,请点击 ...

  5. Python如何读写Excel文件-使用xlrd/xlwt模块

    时间: 2020-08-18 整理: qiyuan 安装和导入 1.模块介绍 在 python 中使用 xlrd/xlwt 和 openpyxl 模块可以对Excel电子表格(xls.xlsx文件)进 ...

  6. java对象的引用级别

    解释 在java中也有引用的概念,其实就可以认为是变量.标题中的引用级别是指变量与对象之前的引用级别.java中分为4种,按引用强弱关系排序分别是:强引用.软引用.弱引用.虚引用. 强引用(Stron ...

  7. MySQL双主多从+Keepalived配置

    原文转自:https://www.cnblogs.com/itzgr/p/10233932.html作者:木二 目录 一 基础环境 二 实际部署 2.1 MySQL双主+Keepalived高可用 2 ...

  8. centos7 wget安装Tomcat7

    2021-07-15 1.环境介绍 操作系统:centos7 jdk版本:jdk1.8.0.211 tomcat版本:tomcat7.0.109 2. 检查系统中是否已经安装 jdk ,如未安装, 请 ...

  9. vue+Element-ui 的 el-cascader 做高德地图的省市区三级联动并且是异步加载,点击省市区跳转到对应的区(地图可以通过后端返回的点进行标点)

    // 完整版高德地图,可以复制代码直接使用 index.html <script type="text/javascript" src="https://webap ...

  10. ubantu 编译安装xl2tpd笔记

    1. 下载xl2tpd源码 推荐一个仍在维护的网站:    https://www.xelerance.com/archives/202 也可以从其他的网站下载:例如:https://pkgs.org ...