[CCF-CSP] 202303-4 星际网络II

2024-05-25 02:36
文章标签 ccf csp 网络 ii 星际 202303

本文主要是介绍[CCF-CSP] 202303-4 星际网络II,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目描述

随着星际网络的进一步建设和规模的增大,一个新的问题出现在网络工程师面前——地址空间不够用了!原来,星际网络采用了传统的IPv6协议,虽然有 21282128 级别的可用地址数量,但面对广袤无垠的宇宙和爆炸式增长的网络用户数,如此庞大的地址空间也面临了用尽的那一天。

新的通信协议的研发工作交给了著名的网络科技圣地——西西艾弗星。最终,经过2333年的不懈努力,西西艾弗星的工程师们设计出了一种新的协议——“西西艾弗IP协议”,又称IPxxaf。

在IPxxaf协议中,一个地址由 n 位二进制位组成,其中 n 是 1616 的倍数。日常表示一个地址时,采用类似IPv6协议的十六进制表示法,每 44 位用 : 隔开。如n=32 时,地址为 2a00:0001 ,即表示一个二进制为 0010 1010 0000 0000 0000 0000 0000 0001 的地址。注意不会出现IPv6中省略每组的前导 0 或用 :: 省略一段 0 的情况。

为方便起见,记 num(s) 为地址 s 按高位在前、低位在后组成的 n 位二进制数,称一段“连续的地址“为 num(s) 成一段连续区间的一系列地址。

西西艾弗星的网络管理员负责地址的分配与管理。最开始,整个地址空间都是未分配的。用户可以随时向管理员申请一些地址:

1 id l r:表示用户id 申请地址在 l∼r 范围内(包含 l 和 r,下同)的一段连续地址块。

在地址申请操作中,管理员需要先检查地址是否可用。如果用户申请的地址全部未被分配,则检查通过;若地址中存在已经分配给其他用户的地址,则检查失败。

但有一种特殊情况:申请的地址中没有已经分配给其他用户的地址,但含有一些先前已分配给该用户本人的地址。此时可以认为检查通过,但若申请的地址先前已全部分配给该用户则检查失败。

如果上述检查通过,则管理员向用户返回 YES,并将申请的地址分配给该用户;若不通过,则向用户返回 NO,同时不改变现有的地址分配。

网络管理员要定期检查地址的分配情况,具体而言有如下两种操作:

2 s:检查地址 s 被分配给了哪个用户。若未被分配,则结果为 00。

3 l r:检查 l∼r 范围内的所有地址是否完整地分配给了某个用户。若是,回答该用户的编号;若否,回答 00。

在整个网络的运行过程中,共出现了 q 次申请地址和检查地址分配的操作。作为西西艾弗星的一名重要的网络技术顾问,你要帮网络管理员依次处理每个操作,并回答相应的结果。

输入格式

从标准输入读入数据。

第一行,22 个正整数 n,q。

接下来 q 行,每行一个操作,格式如上所述,其中的 id 为正整数,l,r,s 均为IPxxaf地址串,其中十六进制均用数字和小写字母表示。

输出格式

输出到标准输出。

输出 q 行,每行一个非负整数或字符串,表示此次操作的结果。

其中,对于操作 11 ,输出 YES 或 NO;对于操作 2,32,3,输出一个非负整数。

思路

很明显就是一个区间修改、区间查询的题目,因此可以用线段树来解决这个问题。

离散化操作:预处理所有查询中涉及到的IP地址,升序排序,用编号来表示。这里注意与上一个IP地址的连贯性,如果刚好等于上一个IP地址的下一个,就不用管;否则的话,中间跳过的IP地址,我们也要给他们留一个位置。如果没有这一步的话,后面的区间查询会少查询一段。

用线段树维护区间和,表示这个区间被占用的IP个数,用mx和mn表示占用这个区间的用户的最大id与最小id。

对于查询1:我们要查询[l,r]这个区间被占用的IP个数sum,如果sum<区间长度且这个区间没有被占用或者仅被用户id占用,查询成功,否则查询失败。

对于查询2:我们要查询[l,l]这个区间占用的用户id。

对于查询3:我们要查询[l,r]这个区间被占用的IP个数sum,如果sum=区间长度且这个区间仅被一个用户占用,输出用户编号,否则输出0。

代码

#include <bits/stdc++.h>
#define N 50010
#define INF 0x3f3f3f3f
using namespace std;int n,qq,cnt;
map<string,int> mp;//离散化
struct query{//查询int op,id;string l,r;
}q[N];
struct node{//线段树int l,r,sum,mx,mn,add;
}tr[N<<4];void pushup(int u){//向上更新tr[u].sum=tr[u<<1].sum+tr[u<<1|1].sum;tr[u].mx=max(tr[u<<1].mx,tr[u<<1|1].mx);tr[u].mn=min(tr[u<<1].mn,tr[u<<1|1].mn);
}
void build(int u,int l,int r){//建树tr[u]={l,r,0,-INF,INF,0};if(l==r){tr[u].mx=-INF;tr[u].mn=INF;return;}int mid=l+r>>1;build(u<<1,l,mid);build(u<<1|1,mid+1,r);pushup(u);
}
void upd(int u,int c){//单点更新tr[u].sum=tr[u].r-tr[u].l+1;tr[u].mx=max(tr[u].mx,c);tr[u].mn=min(tr[u].mn,c);tr[u].add=c; 
}
void pushdown(int u){//延迟标记if(tr[u].add){upd(u<<1,tr[u].add);upd(u<<1|1,tr[u].add);tr[u].add=0;}
}
void update(int u,int l,int r,int c){//区间更新if(l<=tr[u].l&&tr[u].r<=r) upd(u,c);else{pushdown(u);int m=tr[u].l+tr[u].r>>1;if(l<=m) update(u<<1,l,r,c);if(r>m) update(u<<1|1,l,r,c);pushup(u);} 
}
int qsum(int u,int l,int r){//查询区间和if(l>r) return 0;if(l<=tr[u].l&&tr[u].r<=r) return tr[u].sum;int ans=0;pushdown(u);int m=tr[u].l+tr[u].r>>1;if(l<=m) ans+=qsum(u<<1,l,r);if(r>m) ans+=qsum(u<<1|1,l,r);return ans;
}
int qmx(int u,int l,int r){//查询区间最大值if(l>r) return -INF;if(l<=tr[u].l&&tr[u].r<=r) return tr[u].mx;int ans=-INF;pushdown(u);int m=tr[u].l+tr[u].r>>1;if(l<=m) ans=max(ans,qmx(u<<1,l,r));if(r>m) ans=max(ans,qmx(u<<1|1,l,r));return ans;
}
int qmn(int u,int l,int r){//查询区间最小值if(l>r) return INF;if(l<=tr[u].l&&tr[u].r<=r) return tr[u].mn;int ans=INF;pushdown(u);int m=tr[u].l+tr[u].r>>1;if(l<=m) ans=min(ans,qmn(u<<1,l,r));if(r>m) ans=min(ans,qmn(u<<1|1,l,r));return ans;
}
bool judge(string now,string pre){//判断IP地址是否连贯int c=1;for(int i=pre.size()-1;i>=0;i--){if(c==0) break;if(pre[i]==':') continue;if(pre[i]=='9') pre[i]='a',c=0;else if(pre[i]=='f') pre[i]='0',c=1;else pre[i]=pre[i]+c,c=0;}if(now==pre) return true;return false;
}
int main()
{ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);cin>>n>>qq;for(int i=1;i<=qq;i++){cin>>q[i].op;if(q[i].op==1){cin>>q[i].id>>q[i].l>>q[i].r;mp[q[i].l]=1,mp[q[i].r]=1;}else if(q[i].op==2){cin>>q[i].l;mp[q[i].l]=1;}else{cin>>q[i].l>>q[i].r;mp[q[i].l]=1,mp[q[i].r]=1;}}string pre="0000";for(int i=1;i<n/16;i++) pre+=":0000";for(auto it=mp.begin();it!=mp.end();it++){//离散化if(it->first!=pre&&!judge(it->first,pre)) ++cnt;//不连贯要空一个位置it->second=++cnt;pre=it->first;}build(1,1,cnt);for(int i=1;i<=qq;i++){if(q[i].op==1){int l=mp[q[i].l],r=mp[q[i].r],id=q[i].id;int sum=qsum(1,l,r),fg;if(sum==0) fg=1;else if(sum<r-l+1){int mx=qmx(1,l,r),mn=qmn(1,l,r);if(mx==mn&&mx==id) fg=1; else fg=0;}else fg=0;if(fg){cout<<"YES"<<endl;update(1,l,r,id);}else cout<<"NO"<<endl;}else if(q[i].op==2){int l=mp[q[i].l];if(qsum(1,l,l)==0) cout<<0<<endl;else cout<<qmx(1,l,l)<<endl;}else{int l=mp[q[i].l],r=mp[q[i].r],id=q[i].id;int sum=qsum(1,l,r);if(sum==r-l+1){int mx=qmx(1,l,r),mn=qmn(1,l,r);if(mx==mn) cout<<mx<<endl; else cout<<0<<endl;}else cout<<0<<endl;}}return 0;
}

这篇关于[CCF-CSP] 202303-4 星际网络II的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1000239

相关文章

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

ASIO网络调试助手之一:简介

多年前,写过几篇《Boost.Asio C++网络编程》的学习文章,一直没机会实践。最近项目中用到了Asio,于是抽空写了个网络调试助手。 开发环境: Win10 Qt5.12.6 + Asio(standalone) + spdlog 支持协议: UDP + TCP Client + TCP Server 独立的Asio(http://www.think-async.com)只包含了头文件,不依

poj 3181 网络流,建图。

题意: 农夫约翰为他的牛准备了F种食物和D种饮料。 每头牛都有各自喜欢的食物和饮料,而每种食物和饮料都只能分配给一头牛。 问最多能有多少头牛可以同时得到喜欢的食物和饮料。 解析: 由于要同时得到喜欢的食物和饮料,所以网络流建图的时候要把牛拆点了。 如下建图: s -> 食物 -> 牛1 -> 牛2 -> 饮料 -> t 所以分配一下点: s  =  0, 牛1= 1~

poj 3068 有流量限制的最小费用网络流

题意: m条有向边连接了n个仓库,每条边都有一定费用。 将两种危险品从0运到n-1,除了起点和终点外,危险品不能放在一起,也不能走相同的路径。 求最小的费用是多少。 解析: 抽象出一个源点s一个汇点t,源点与0相连,费用为0,容量为2。 汇点与n - 1相连,费用为0,容量为2。 每条边之间也相连,费用为每条边的费用,容量为1。 建图完毕之后,求一条流量为2的最小费用流就行了

poj 2112 网络流+二分

题意: k台挤奶机,c头牛,每台挤奶机可以挤m头牛。 现在给出每只牛到挤奶机的距离矩阵,求最小化牛的最大路程。 解析: 最大值最小化,最小值最大化,用二分来做。 先求出两点之间的最短距离。 然后二分匹配牛到挤奶机的最大路程,匹配中的判断是在这个最大路程下,是否牛的数量达到c只。 如何求牛的数量呢,用网络流来做。 从源点到牛引一条容量为1的边,然后挤奶机到汇点引一条容量为m的边

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

从0到1,AI我来了- (7)AI应用-ComfyUI-II(进阶)

上篇comfyUI 入门 ,了解了TA是个啥,这篇,我们通过ComfyUI 及其相关Lora 模型,生成一些更惊艳的图片。这篇主要了解这些内容:         1、哪里获取模型?         2、实践如何画一个美女?         3、附录:               1)相关SD(稳定扩散模型的组成部分)               2)模型放置目录(重要)

CSP 2023 提高级第一轮 CSP-S 2023初试题 完善程序第二题解析 未完

一、题目阅读 (最大值之和)给定整数序列 a0,⋯,an−1,求该序列所有非空连续子序列的最大值之和。上述参数满足 1≤n≤105 和 1≤ai≤108。 一个序列的非空连续子序列可以用两个下标 ll 和 rr(其中0≤l≤r<n0≤l≤r<n)表示,对应的序列为 al,al+1,⋯,ar​。两个非空连续子序列不同,当且仅当下标不同。 例如,当原序列为 [1,2,1,2] 时,要计算子序列 [

配置InfiniBand (IB) 和 RDMA over Converged Ethernet (RoCE) 网络

配置InfiniBand (IB) 和 RDMA over Converged Ethernet (RoCE) 网络 服务器端配置 在服务器端,你需要确保安装了必要的驱动程序和软件包,并且正确配置了网络接口。 安装 OFED 首先,安装 Open Fabrics Enterprise Distribution (OFED),它包含了 InfiniBand 所需的驱动程序和库。 sudo

【机器学习】高斯网络的基本概念和应用领域

引言 高斯网络(Gaussian Network)通常指的是一个概率图模型,其中所有的随机变量(或节点)都遵循高斯分布 文章目录 引言一、高斯网络(Gaussian Network)1.1 高斯过程(Gaussian Process)1.2 高斯混合模型(Gaussian Mixture Model)1.3 应用1.4 总结 二、高斯网络的应用2.1 机器学习2.2 统计学2.3