信息学奥赛复赛复习14-CSP-J2021-03网络连接-字符串处理、数据类型溢出、数据结构Map、find函数、substr函数

news2024/11/25 8:18:28
PDF文档回复:20241007

1 P7911 [CSP-J 2021] 网络连接

[题目描述]

TCP/IP 协议是网络通信领域的一项重要协议。今天你的任务,就是尝试利用这个协议,还原一个简化后的网络连接场景。

在本问题中,计算机分为两大类:服务机(Server)和客户机(Client)。服务机负责建立连接,客户机负责加入连接

需要进行网络连接的计算机共有 n 台,编号为 1∼n,这些机器将按编号递增的顺序,依次发起一条建立连接或加入连接的操作。

每台机器在尝试建立或加入连接时需要提供一个地址串。服务机提供的地址串表示它尝试建立连接的地址,客户机提供的地址串表示它尝试加入连接的地址

一个符合规范的地址串应当具有以下特征:

必须形如 a.b.c.d:e 的格式,其中 a,b,c,d,e 均为非负整数
0≤a,b,c,d≤255,0≤e≤65535
a,b,c,d,e 均不能含有多余的前导 0

相应地,不符合规范的地址串可能具有以下特征

不是形如 a.b.c.d:e 格式的字符串,例如含有多于3 个字符 . 或多于 1 个字符 : 等情况;
整数a,b,c,d,e 中某一个或多个超出上述范围
a,b,c,d,e 中某一个或多个含有多余的前导0

例如,地址串 192.168.0.255:80 是符合规范的,但 192.168.0.999:80192.168.00.1:10192.168.0.1:088192:168:0:1.233 均是不符合规范的

如果服务机或客户机在发起操作时提供的地址串不符合规范,这条操作将被直接忽略。

在本问题中,我们假定凡是符合上述规范的地址串均可参与正常的连接,你无需考虑每个地址串的实际意义。

由于网络阻塞等原因,不允许两台服务机使用相同的地址串,如果此类现象发生,后一台尝试建立连接的服务机将会无法成功建立连接;除此之外,凡是提供符合规范的地址串的服务机均可成功建立连接。

如果某台提供符合规范的地址的客户机在尝试加入连接时,与先前某台已经成功建立连接的服务机提供的地址串相同,这台客户机就可以成功加入连接,并称其连接到这台服务机;如果找不到这样的服务机,则认为这台客户机无法成功加入连接。

请注意,尽管不允许两台不同的服务机使用相同的地址串,但多台客户机使用同样的地址串,以及同一台服务机同时被多台客户机连接的情况是被允许的。

你的任务很简单:在给出每台计算机的类型以及地址串之后,判断这台计算机的连接情况

[输入格式]

第一行,一个正整数 n

接下来 n行,每行两个字符串 op,ad,按照编号从小到大给出每台计算机的类型及地址串

其中 op 保证为字符串 ServerClient 之一,ad 为一个长度不超过 25 的,仅由数字、字符 . 和字符 : 组成的非空字符串。

每行的两个字符串之间用恰好一个空格分隔开,每行的末尾没有多余的空格

[输出格式]

输出共 n 行,每行一个正整数或字符串表示第 i台计算机的连接状态。其中:

如果第 i台计算机为服务机,则:

如果其提供符合规范的地址串且成功建立连接,输出字符串 OK。
如果其提供符合规范的地址串,但由于先前有相同地址串的服务机而无法成功建立连接,输出字符串 FAIL。
如果其提供的地址串不是符合规范的地址串,输出字符串 ERR。

如果第 i 台计算机为客户机,则:

如果其提供符合规范的地址串且成功加入连接,输出一个正整数表示这台客户机连接到的服务机的编号。
如果其提供符合规范的地址串,但无法成功加入连接时,输出字符串 FAIL。
如果其提供的地址串不是符合规范的地址串,输出字符串 ERR

[输入输出样例]

输入 #1

5
Server 192.168.1.1:8080
Server 192.168.1.1:8080
Client 192.168.1.1:8080
Client 192.168.1.1:80
Client 192.168.1.1:99999

输出 #1

OK
FAIL
1
FAIL
ERR

输入 #2

10
Server 192.168.1.1:80
Client 192.168.1.1:80
Client 192.168.1.1:8080
Server 192.168.1.1:80
Server 192.168.1.1:8080
Server 192.168.1.999:0
Client 192.168.1.1.8080
Client 192.168.1.1:8080
Client 192.168.1.1:80
Client 192.168.1.999:0

输出 #2

OK
1
FAIL
FAIL
OK
ERR
ERR
5
1
ERR

说明/提示

计算机 1 为服务机,提供符合规范的地址串 192.168.1.1:8080,成功建立连接;

计算机 2为服务机,提供与计算机 1 相同的地址串,未能成功建立连接;

计算机 3 为客户机,提供符合规范的地址串 192.168.1.1:8080,成功加入连接,并连接到服务机 1;

计算机 4为客户机,提供符合规范的地址串 192.168.1.1:80,找不到服务机与其连接;

计算机 5 为客户机,提供的地址串 192.168.1.1:99999 不符合规范

数据规模

对于 100% 的数据,保证 1≤n≤1000

2 相关知识点

1) 字符串

find函数

s.find(str) 查找字符串str在当前字符串s中第一次出现的位置 从0开始

#include<bits/stdc++.h>
using namespace std;
/*
s.find(str) 查找字符串str在当前字符串s中第一次出现的位置 从0开始 
*/
int main(){
	string str = "The apple thinks apple is delicious";//长度34
	string key = "e";
	int pos = str.find(key);// 2
	cout<<"e在字符串第1次出现的位置:"<<pos<<endl;
	key = "apple";
	pos = str.find(key);// 4
	cout<<"apple在字符串第1次出现的位置:"<<pos<<endl;
	return 0;
}
/*
输出
e在字符串第1次出现的位置:2
apple在字符串第1次出现的位置:4 
*/ 

substr函数

在C++中可以使用 substr ,成员函数来获取子字符串

str.substr(1,5) 从第一个位置开始 取5个字符
str.substr(1) 从第一个位置开始 取到结尾的所有字符

#include<bits/stdc++.h>
using namespace std;
/*
	子串 substr 
	str.substr(1,5) 从第一个位置开始 取5个字符
	str.substr(1) 从第一个位置开始 取到结尾的所有字符 
*/ 
void test8(){
	string str="abcdefg";
	cout<<str<<endl;
	string temp=str.substr(1,5);//bcdef
	cout<<temp<<endl;
	
	temp = str.substr(1);//bcdefg
	cout<<temp<<endl;
} 
int main(){
	test8(); 
	return  0;
}
/*
输出
abcdefg
bcdef
bcdefg
*/

2) 数据结构Map

C++中的std::map是一个关联容器,它存储键值对,并根据键对元素进行排序。std::map中的键是唯一的,而值可以重复。std::map通常使用红黑树实现,因此插入、删除和查找操作的时间复杂度为O(log n)

#include<bits/stdc++.h>
using namespace std;
map<string,int> mp;
/*
如果通过数字获取对应数据,比如通过学号获取学生姓名
比如string[5]={"张三","李四"},a[0] 
如果通过字符串获取,可以使用map结构 
map<string,int> mp;
key为string value为int
*/ 
int main(){
	mp["张三"]=2;
	mp["李四"]=8;
	cout<<"张三的学号是:"<<mp["张三"]<<endl;
	cout<<"李四的学号是:"<<mp["李四"];
	return 0;
}
/*
张三的学号是:2
李四的学号是:8
*/

3) 数据溢出

在C语言中,数据溢出是指变量在进行计算或赋值操作时超出了其数据类型所能表示的范围,导致结果不正确或不可预料的行为

例如

#include<bits/stdc++.h>
using namespace std;
int main(){
	char c=129;//超出了char 表示的数值范围 char最大是127
	printf("%d",c); 
	return 0;
} 
/*
输出
-127 
*/

3 思路分析

1 判断地址是否合法
  判断.和:是否在固定位置
  使用find和substr函数截取分割出的数组,判断是否在对应范围内
2 判断服务器是否可以创建
  使用map数据结构,存储已创建的服务端,如果地址未创建过,可以创建,如果已经创建过,则不能创建
3 判断客户端是否可以加入
  加入对应服务器地址,如果要加入的地址创建成功,则可以加入,输出对应服务器编号
  如果加入的服务端,未创建成功,则输出FAIL

示例程序

#include<bits/stdc++.h>
using namespace std;
int n;//输入n个地址 
string op,ad;//op 计算机类型 ad计算机地址 
/*
服务器地址和编号映射
key为服务器地址 ,value为服务器编号
通过地址 key可以获取启动服务服务器编号 
*/ 
map<string,int> serverMap;
/*
  字符串转整形
  按位逐一转换 自动去除前导0 
  使用long long数据类型 是因为有些数字过大,超出了int的范围
*/
long long s2i(string s){
	long long ret=0;
	for(int i=0;i<s.length();i++){
		ret = ret*10+s[i]-'0';
	}
	return ret;
}
/*
  判断服务器地址是否合法 合法返回true 不合法返回false 
*/
bool check(string s){
	//cnt1记录字符.的个数 cnt2 记录字符:的个数 
	int cnt1=0,cnt2=0; 
	for(int i=0;i<s.length();i++){
		if(s[i]=='.') cnt1++;//遇到累加 
		if(s[i]==':') cnt2++;//遇到累加
		if(cnt2==1 && cnt1!=3){//:有1个时 .必须有3个 否则不合法 
			return false;
		}
	}
	if(cnt1!=3 || cnt2!=1){//.不是3个 :不是1个都不合法 
		return false;
	}
	//pos 第1个.或:的位置,i通过.或: 分割出来的字符下标 
	int pos,i; 
	string str[10];//分割出来字符存放str中 
	for(i=0;i<3;i++){
		pos = s.find(".");//找到第1个.的位置 
		str[i]=s.substr(0,pos);//截取.前面字符 
		s=s.substr(pos+1);//去除原字符.前面字符并去除. 
	}
	pos = s.find(":");//继续找:位置 
	str[i]=s.substr(0,pos);//截取:前面字符 
	str[++i]=s.substr(pos+1);//截取:后面字符 
	for(int i=0;i<5;i++){//上面操作截取5个字符 放入str数组中
		//截取字符为空 不合法   
		if(str[i].length()==0) return false;
		//有前导0 不合法 
		if(str[i][0]=='0' && str[i].length()>1) return false;
		//字符串转数字 可能存在不合法数字范围超出int 
		long long tmp=s2i(str[i]);
		if(i==4){//最后一个端口 范围判断 0~65535之间,超出不合法 
			if(tmp<0 || tmp>65535){
				return false;
			}
		}else{//地址每个数,范围判断0~255之间,超出不合法 
			if(tmp<0 || tmp>255){
				return false;
			}
		}
	}
	return true;//前面没有不合法退出,走到这里返回合法 
}

int main(){
	cin>>n;//输入计算机个数 
	for(int i=1;i<=n;i++){
		cin>>op>>ad;//输入操作类型和计算机地址 
		if(!check(ad)){//如果不合法 输出ERR 
			cout<<"ERR"<<endl;
			continue;//继续判断下1个 
		}
		if(op=="Server"){//如果是服务器 
			if(serverMap[ad]==0){//此服务器没建立过连接 
				cout<<"OK"<<endl;//输出OK 
				serverMap[ad]=i;//记录此服务器地址的编号 
			}else{//此服务器已建立过连接 
				cout<<"FAIL"<<endl;//输出FAIL 
			}
		}else{//客户端 
			if(serverMap[ad]){//加入的服务器已建立连接 
				cout<<serverMap[ad]<<endl;//输出连接服务器编号 
			}else{//加入服务器还没建立连接 
				cout<<"FAIL"<<endl;//无法加入 输出FAIL 
			}	
		}
	}
	return 0;
} 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2196802.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【AI知识点】反向传播(Backpropagation)

反向传播&#xff08;Backpropagation&#xff09; 是训练神经网络的核心算法&#xff0c;它通过反向逐层计算损失函数对每个权重的梯度&#xff0c;来反向逐层更新网络的权重&#xff0c;从而最小化损失函数。 一、反向传播的基本概念 1. 前向传播&#xff08;Forward Propag…

安装DNS

在 CentOS 7 上安装并配置 BIND 以实现 DNS 的正向和反向解析可以按照以下步骤进行&#xff1a; 安装 BIND 打开终端并运行以下命令来安装 BIND 及其工具&#xff1a; yum install bind bind-utils -y配置 BIND 编辑主配置文件&#xff1a; 使用文本编辑器打开 BIND 的主配…

双十一购物清单:这五款爆款科技好物绝不能错过!买到就是赚到!

随着一年一度的双十一购物狂欢节即将拉开帷幕&#xff0c;各大电商平台纷纷推出了一系列优惠活动&#xff0c;吸引着无数消费者的目光。对于科技爱好者而言&#xff0c;这无疑是一个绝佳的机会&#xff0c;能够以优惠的价格购得心仪的电子产品和智能设备。然而&#xff0c;在琳…

HTTPS介绍 --- 超详细保姆级知识讲解

目录 一. 认识HTTPS 二. 使用HTTPS加密的重要性 三. HTTPS的工作流程 四. 常见的加密方式 4.1 对称加密 4.2 非对称加密 五. 数据摘要 && 数据指纹 5.1 数据摘要 5.2 数据签名 六. HTTPS加密过程探究 6.1 方案一&#xff1a;只使用对…

晶体规格书及匹配测试

一、晶体参数介绍 晶体的电气规格相对比较简单,如下: 我们逐一看看每个参数, FL就是晶体的振动频率,这个晶体是24.576MHz的。 CL就是负载电容,决定了晶体频率是否准确,包括外接的实际电容、芯片的等效电容以及PCB走线的寄生电容等,核心参数。 Frequency Tolerance是…

骨传导耳机哪个牌子好?五大精选抢手骨传导耳机分享!

在数字化时代背景下&#xff0c;音乐和音频内容已经成为人们日常生活不可或缺的一部分。随着技术的发展&#xff0c;骨传导耳机凭借其独特的传输方式和健康优势&#xff0c;迅速赢得了市场和消费者的青睐。不同于传统耳机通过空气传导声音&#xff0c;骨传导耳机通过骨骼直接传…

《独自骑行与群骑之旅:探索不同的自由与氛围》

在快节奏的现代生活中&#xff0c;骑行作为一种既环保又健康的出行方式&#xff0c;越来越受到人们的青睐。然而&#xff0c;选择一个人骑车还是加入一群人的行列&#xff0c;这不仅仅是一种出行方式的选择&#xff0c;更是一种生活态度和价值观的体现。本文将探讨这两种骑行方…

【读书笔记·VLSI电路设计方法解密】问题1:什么是芯片

芯片&#xff08;集成电路或IC&#xff09;是在半导体材料的薄基底表面上制造的微型电子电路。在功能上&#xff0c;芯片是一种硬件组件&#xff0c;能够执行某些特定的功能。例如&#xff0c;一个简单的芯片可能被设计用来执行逻辑NOR&#xff08;或非&#xff09;的简单功能&…

如何在VSCode上运行C/C++代码

诸神缄默不语-个人CSDN博文目录 我是Win10&#xff0c;其他系统仅供参考。 文章目录 1. 下载所需插件2. 安装编译器3. 不借助编辑器的cpp代码执行3. 建立VSCode cpp项目3.1 c_cpp_properties.json3.2 settings.json3.3 tasks.json 4. 运行C代码参考资料 1. 下载所需插件 2. 安…

记一次N5105 NAS功耗测量

记一次N5105 NAS功耗测量 一、设备说明 HA500机器&#xff0c;N5105CPU&#xff0c;32GB内存。unraid最新6.12.13系统硬盘有一根500G M2硬盘和一个512G sata接口ssd硬盘&#xff0c;用于组成zfs raid 1&#xff0c;作为cache盘位。另外有三个4T机械硬盘&#xff0c;组成21的形…

鸿蒙应用示例:DevEco Testing 工具的常用功能及使用场景

DevEco Studio 是鸿蒙生态中的集成开发环境(IDE)&#xff0c;而 DevEco Testing 工具则是专门用于测试鸿蒙应用的强大工具。本文将详细介绍 DevEco Testing 中几个常用的测试功能及其使用场景&#xff0c;并给出相应的代码示例。 【1】安装应用 使用场景&#xff1a;在鸿蒙系统…

imx6q 的 header.s的理解

首先是 非设备树的。 就是这三个文件。 header 是配置文件。 .c 文件应该是对这个文件的操作。 那么 header 怎么生成呢? 它这里调整好的 应该参数 应该就是 这个 header.s 了。 但是 这个程序 是 设备树的 流程 ,不知道 对于 非设备树 是不是 适用。 然后是设备树的。 设…

基于Arduino的遥控自平衡小车

基于Arduino的遥控自平衡小车 一、项目简介二、所需材料三、理论支持四、外壳设计五、线路连接六、检查MPU6050连接七、烧录库八、PID控制设置九、设置传感器参数十、无线移动控制十一、超声波模块 一、项目简介 一个使用Arduino Nano、MPU-6050以及便宜的6伏直流齿轮电机的自…

Vue中使用富文本编辑框的实践与探索

在Web开发中&#xff0c;富文本编辑框是一个常见的功能。本文将介绍如何在Vue项目中集成和使用富文本编辑框&#xff0c;并分享一些实践经验。 一、为什么需要富文本编辑框 在开发网站、博客、论坛等应用时&#xff0c;用户往往需要编辑和发布带格式的文本内容。传统的文本输…

分布式事务seata AT和XA性能对比

1. AT模式和XA模式性能对比&#xff1a; AT的阻塞是阻塞在了业务服务层&#xff0c;全局锁。 而XA模式是阻塞在了数据库&#xff0c;对数据库的性能影响很大。 肯定是优选AT&#xff0c;可以提升数据库的性能。 &#xff08;由于AT模式数据库事务阻塞小&#xff0c;那么同一…

公司监控电脑都能监控哪些信息?深刻回答,一文详解!

在当今数字化办公环境中&#xff0c;公司监控电脑已成为许多企业保障信息安全、提升工作效率的重要手段。 然而&#xff0c;这种监控行为也引发了关于员工隐私权的广泛讨论。 本文将深入探讨公司监控电脑所能监控的信息范围&#xff0c;以及如何在保障企业安全与尊重员工隐私…

omron fins 内存区域写入(MEMORY AREA WRITE)

1. 完整的代码如下&#xff1a; import socket import binasciiclass Omron:def __init__(self, ip, port9600):self.ip ip # PLC的IP地址self.port port # PLC的端口&#xff0c;默认为9600def send_receive_fins(self):with socket.socket(socket.AF_INET, socket.SOCK_…

某个应用的CPU使用率居然达到100%,我该怎么办?

摘至https://learn.lianglianglee.com/ CPU使用率 Linux 作为一个多任务操作系统&#xff0c;将每个 CPU 的时间划分为很短的时间片&#xff0c;再通过调度器轮流分配给各个任务使用&#xff0c;因此造成多任务同时运行的错觉。 为了维护 CPU 时间&#xff0c;Linux 通过事先定…

使用标签实现MyBatis的基础操作

目录 前言 1.配置MyBatis⽇志打印 2.参数传递 2.1 #{} 和 ${}区别 2.2传递多个参数 3.增删改查 3.1增(Insert) 3.2删(Delete) 3.3改(Update) 3.4查(Select) 前言 接下来我们会使用的数据表如下&#xff1a; 对应的实体类为&#xff1a;UserInfoMapper 所有的准备工作都…

全局数据在Python包中模块间管理方法探讨

在开发大型 Python 应用程序时&#xff0c;有时需要多个模块共享和管理全局数据。如何优雅地在 Python 包内的不同模块间共享全局数据是一个常见的设计问题。我们希望避免全局变量的混乱和难以维护的代码&#xff0c;但同时能够安全、高效地管理这些共享数据。 下面我们将探讨…