题目想求与给定前缀列表等价的包含IP前缀数目最少的前缀列表。

首先是怎么存储前缀列表。用一个long long存储IP地址,再存一个前缀长度,封装在一个结构体里\(<ipNum, len>\),方便后面排序等操作。IP前缀有三种输入格式,稍微分情况讨论一下。

接着以\(ipNum\)为第一关键字,\(len\)为第二关键字升序排序。

然后考虑去除匹配集被其它IP前缀包含的IP前缀。考虑之前匹配集范围的上届\(mmax\),顺序遍历一下就好了。将剩余的IP列表按之前顺序存在一个静态链表中。

最后将相邻的可合并的IP前缀合并,其实就是前缀长度最后一位0和1,之前完全相同即可。

温习一下链表的插入删除操作。

#include <bits/stdc++.h>
typedef long long LL;
const int maxn = 1000000; using namespace std; struct tIP
{
LL ipNum;
int len;
int before, next;
tIP()
{
before = next = -1;
}
bool operator < (const tIP &y) const
{
if(ipNum == y.ipNum)
return len < y.len;
return ipNum < y.ipNum;
}
void show()
{
LL num[5];
LL temp = ipNum;
for (int i = 4; i >= 1; i--)
{
num[i] = temp % 256;
temp /= 256;
}
for (int i = 1; i <=4; i++)
{
printf("%lld", num[i]);
if (i == 4)
printf("/");
else
printf(".");
}
printf("%d\n", len);
}
};
tIP ip[maxn+10]; LL getMMax(tIP iip)
{
LL temp = (1LL << (32-iip.len)) - 1;
return iip.ipNum | temp;
} int main()
{
int n;
scanf("%d", &n); char s[30];
for (int id = 1, slash, dotCnt, style; id <= n; id++)
{
slash = 0;
dotCnt = 0;
scanf("%s", s + 1);
for (int i = 1; s[i] != '\0'; i++)
{
if (s[i] == '/')
slash = 1;
if (s[i] == '.')
dotCnt ++;
} if (slash == 1 && dotCnt == 3)
style = 1;
else if (slash == 1 && dotCnt < 3)
style = 2;
else
style = 3; LL num[5];
memset(num, 0, sizeof(num));
if (style == 1 || style == 2)
{
for (int i = 1, temp = 0, numCnt = 1; ; i++)
{
if (s[i] == '.' || s[i] == '/')
{
num[numCnt++] = temp * 1LL;
temp = 0;
}
else if (s[i] == '\0')
{
ip[id].len = temp;
break;
}
else
{
temp = temp * 10 + s[i] - '0';
}
}
}
else
{
for (int i = 1, temp = 0, numCnt = 1; ; i++)
{
if (s[i] == '.')
{
num[numCnt++] = temp * 1LL;
temp = 0;
}
else if (s[i] == '\0')
{
num[numCnt++] = temp * 1LL;
ip[id].len = (numCnt-1) * 8;
break;
}
else
{
temp = temp * 10 + s[i] - '0';
}
}
}
LL ans = 0;
for (int i = 1; i <= 4; i++)
{
ans = ans * 256 + num[i];
}
ip[id].ipNum = ans;
} sort(ip + 1, ip + 1 + n); LL mmax = -1;
int st = 0, en = n + 1;
ip[st].before = -1;
ip[st].next = en;
ip[en].before = st;
ip[en].next = -1;
for (int id = 1, prev = 0; id <= n; id++)
{
if (ip[id].ipNum > mmax)
{
ip[id].before = prev;
ip[id].next = en;
ip[prev].next = ip[en].before = id;
prev = id;
mmax = getMMax(ip[id]);
}
} int pNow = ip[0].next;
while (pNow != en)
{
int p1 = pNow, p2 = ip[pNow].before;
if (p2 == 0)
pNow = ip[pNow].next;
else
{
if (ip[p1].len == ip[p2].len &&
(ip[p2].ipNum & (1LL << (32-ip[p2].len))) == 0 &&
(ip[p2].ipNum | (1LL << (32-ip[p2].len))) == ip[p1].ipNum)
{
ip[p1].before = ip[p2].before;
ip[ip[p1].before].next = p1;
ip[p1].ipNum = ip[p2].ipNum;
ip[p1].len --;
}
else
{
pNow = ip[pNow].next;
}
}
} pNow = ip[0].next;
while (pNow != en)
{
ip[pNow].show();
pNow = ip[pNow].next;
} return 0;
}

CCF-CSP题解 201812-3 CIDR合并的更多相关文章

  1. CCF 201812-3 CIDR合并

    CCF 201812-3 CIDR合并 //100分 93ms #include<stdio.h>//CCF上stdio.h比cstdio快!!! #include<string.h ...

  2. CCF CSP 201604-2 俄罗斯方块

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201604-2 俄罗斯方块 问题描述 俄罗斯方块是俄罗斯人阿列克谢·帕基特诺夫发明的一款休闲游 ...

  3. CCF CSP 201703-3 Markdown

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201703-3 Markdown 问题描述 Markdown 是一种很流行的轻量级标记语言(l ...

  4. CCF CSP 201312-3 最大的矩形

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201312-3 最大的矩形 问题描述 在横轴上放了n个相邻的矩形,每个矩形的宽度是1,而第i( ...

  5. CCF CSP 201609-3 炉石传说

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201609-3 炉石传说 问题描述 <炉石传说:魔兽英雄传>(Hearthston ...

  6. CCF CSP 201403-3 命令行选项

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201403-3 命令行选项 问题描述 请你写一个命令行分析程序,用以分析给定的命令行里包含哪些 ...

  7. CCF CSP 201709-4 通信网络

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201709-4 通信网络 问题描述 某国的军队由N个部门组成,为了提高安全性,部门之间建立了M ...

  8. CCF CSP 201409-3 字符串匹配

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201409-3 字符串匹配 问题描述 给出一个字符串和多行文字,在这些文字中找到字符串出现的那 ...

  9. CCF CSP 201503-3 节日

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201503-3 节日 问题描述 有一类节日的日期并不是固定的,而是以“a月的第b个星期c”的形 ...

  10. CCF CSP 201509-2 日期计算

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201509-2 日期计算 问题描述 给定一个年份y和一个整数d,问这一年的第d天是几月几日? ...

随机推荐

  1. python_day04

    今日内容: 一.爬虫三部曲: 1.发送请求 2.解析数据 3.保存数据 4.解析详情页,获取视频地址 mport requests import re #正则模块 import uuid #uuid. ...

  2. Mac usr/bin 目录 权限问题

    Mac进行 usr/bin 目录下修改权限问题,operation not permitted 一般情况下我们在使用mac系统过程中下载一些文件.新建一些项目之后,这些文件都会默认是只读状态,这时我们 ...

  3. 树莓派4B安装netcore

    准备材料 SDFormatter.exe ---格式化SD卡,空的SD就可以不用了 2019-09-26-raspbian-buster.img ---下载好树莓派系统镜像 win32diskimag ...

  4. 扛把子组20191031-8 alpha week 1/2 Scrum立会报告+燃尽图 06

    此作业的要求参见:https://edu.cnblogs.com/campus/nenu/2019fall/homework/9916 一.小组情况 队名:扛把子 组长:孙晓宇 组员:宋晓丽 梁梦瑶 ...

  5. 附012.Kubeadm部署高可用Kubernetes

    一 kubeadm介绍 1.1 概述 参考<附003.Kubeadm部署Kubernetes>. 1.2 kubeadm功能 参考<附003.Kubeadm部署Kubernetes& ...

  6. 【RN - 基础】之Image使用简介

    Image组件是用来加载图片的.React Native项目加载图片往往有三种方式: 从React Native项目中加载图片: 从APP项目中加载图片: 从网络中加载图片. Image组件加载图片 ...

  7. Prometheus 安装

    目录 简介 安装部署 环境准备 安装 配置环境变量 配置 启动 简介 prometheus存储的是时序数据,即按相同时序(相同名称和标签),以时间维度存储连续的数据的集合. 时序(time serie ...

  8. U盘安装centos 7 提示 “Warning: /dev/root does not exist

    背景介绍:公司需要使用台式机安装Centos 7.5 系统,来部署一个测试的数据库,在安装Centos 7.5 系统的时候,使用U启安装,但有问题. 提示信息如下 如图:安装centos 7时提示 & ...

  9. gdb调试常用方法介绍

    一.概述   GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具.没有统一的界面,所有的操作都是通过命令的方式进行提供.对于习惯了图形界面方式进行调试的猿猿来说可能不是很顺手,但是如果你在 ...

  10. Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: REFERENCES command denied to user 'nali'@'localhost' for table 'dbs'

    按照教程 Install hive on Mac with Homebrew,在 mac 上安装 Hive 时, 最后执行 hive 命令后,出现错误: Exception in thread &qu ...