习题4-5 IP网络(IP Networks, ACM/ICPC NEERC 2005, UVa1590)

可以用一个网络地址和一个子网掩码描述一个子网(即连续的IP地址范围)。其中子网

掩码包含32个二进制位,前32-n位为1,后n位为0,网络地址的前32-n位任意,后n位为0。

所有前32-n位和网络地址相同的IP都属于此网络。

例如,网络地址为194.85.160.176(二进制为11000010|01010101|10100000|10110000),

子网掩码为255.255.255.248(二进制为11111111|11111111|11111111|11111000),则该子网

的IP地址范围是194.85.160.176~194.85.160.183。输入一些IP地址,求最小的网络(即包含IP

地址最少的网络),包含所有这些输入地址。

例如,若输入3个IP地址:194.85.160.177、194.85.160.183和194.85.160.178,包含上述3

个地址的最小网络的网络地址为194.85.160.176,子网掩码为255.255.255.248。

Sample Input

3

194.85.160.177

194.85.160.183

194.85.160.178

Sample Output

194.85.160.176

255.255.255.248

【注意:他可能有很多组输入,而每组输出之间没有空行】

https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&category=833&problem=4465&mosmsg=Submission+received+with+ID+21184770

思路:

1、先将所有ip存起来,用数组或容器什么的。

2、转换二进制

3、从第一位开始,诸位比较所有的ip在这一位上的数字一样否

4、判断出最小网络位数,即掩码为1的位数。

5、转换十进制

(我用来存二进制的ip用的是string)

(用了下vector,当然也可以用数组存,一个32*1000的数组)


/*
110000100101010110100000_10110001
110000100101010110100000_10110111
110000100101010110100000_10110010 11000010010101011010000010110000
11111111111111111111111111111000
*/
//特殊情况:只输入一个IP地址,这时掩码应该32位1
#include<iostream>
#include<stack>
#include<string>
#include<vector>
#include<cmath>
#include<cstdio>
using namespace std;
string binary(int dec)
{
string str="00000000";
stack<int> s;
int bin=0;
for(int i=7;i>=0;i--)
{
//s.push(dec%2);
str[i]=dec%2+'0';
dec /= 2;
}
// while(!s.empty())
// {
// bin = bin*10 + s.top();
// s.pop();
// }
// return bin;
return str;
}
int decimal(string bin)
{
int dec =0;
for(int i=0;i<8;i++) //从高位开始
{
dec += (int)pow(2,8-i-1)* (bin[i]-'0');
}
return dec;
}
int main()
{
int T;
while(cin>>T)
{
vector<string>ip; // 存储输入的所有ip
int num=0;
if(T==1) //防止只输入一个地址的特殊情况
num=32;
while(T--) //输入
{
int a[4];
string str;
scanf("%d.%d.%d.%d",&a[0],&a[1],&a[2],&a[3]);
for(int i=0;i<4;i++)
{
str += binary(a[i]);
}
ip.push_back(str);
}
for(int j=0;j<32;j++) //判断相等的位数
{
int iff=0;
for(int i=1;i<ip.size();i++)
{
//cout<<ip.at(i)<<endl;
if(ip.at(0)[j] == ip.at(i)[j]) //这一位相等
{
iff=1;
}
else
{
iff=0;
break;
}
}
if(iff) //这一位相等,掩码位数加一
num++;
else
break;
}
string zero="00000000"; //备用0
string mask,minip;
int score[4];//存储最小网络的四个十进制ip地址段
int score2[4];//存储mask的四个十进制的ip地址段 for(int i=1;i<=32-num;i++) //先从最后开始补0
{
minip='0'+minip;
mask='0'+mask;
}
for(int i=num-1;i>=0;i--) //倒的补
{
minip=ip[0].at(i)+minip;
mask='1'+mask;
} // cout<<num<<endl;
// for(int i=0;i<ip.size();i++)
// cout<<ip[i]<<endl;
// cout<<endl;
// cout<<mask<<endl; for(int i=0;i<4;i++) //运算最小网络
{
score[i] =decimal(minip.substr(i*8,8));
score2[i] =decimal(mask.substr(i*8,8));
//以下淘汰的方法是:边变换,边比对是否位数到了最小网络位。
//比较上面的先做好一个最小网络的二进制地址,然后在直接变换。要更复杂
// if(i*8+8>num)
// {
// //cout<<i*8<<" "<<num<<endl;
// //cout<<ip[0].substr(i*8,num-i*8)<<"|"<<zero.substr(0,32-num)<<endl;
// score[i]=decimal(ip[0].substr(i*8,num-i*8)+zero.substr(0,32-num));
// }
// else
// {
// //cout<<ip[0].substr(i*8,8)<<endl;
// score[i]=decimal(ip[0].substr(i*8,8));
// }
} printf("%d.%d.%d.%d\n",score[0],score[1],score[2],score[3]);
printf("%d.%d.%d.%d\n",score2[0],score2[1],score2[2],score2[3]);
}
return 0;
}
//AC at 2018/4/22

(题外话:好在上学期的网络课设就是算ip地址,所以这道题我才想了半个多小时就有思路了(哭))

uva 1590 - IP Networks(IP地址)的更多相关文章

  1. IP Networks UVA - 1590

     Alex is administrator of IP networks. His clients have a bunch of individual IP addresses and he de ...

  2. UVa 1590 IP网络(简单位运算)

    Description   Alex is administrator of IP networks. His clients have a bunch of individual IP addres ...

  3. [刷题]算法竞赛入门经典(第2版) 4-5/UVa1590 - IP Networks

    书上具体所有题目:http://pan.baidu.com/s/1hssH0KO 代码:(Accepted,0 ms) //UVa1590 - IP Networks #include<iost ...

  4. Java获取本机的IP与MAC地址

    有些机器有许多虚拟的网卡,获取IP地址时会出现一些意外,所以需要一些验证: // 获取mac地址 public static String getMacAddress() { try { Enumer ...

  5. CIDR详解和ip最长地址前缀匹配

    1.CIDR是什么 无类域间路由(CIDR)编址方案 摒弃传统的基于类的地址分配方式,允许使用任意长度的地址前缀,有效提高地址空间的利用率. 就是一个ip加一个网络掩码,不过这个掩码不是之前只有3个值 ...

  6. 几种获取IP 根据IP获取地址的方法 JS,第三方 新浪 网易 腾讯

    第一种是利用纯真ip数据库,这个可以在网上找到很多,缺点是更新有点慢. 第二种是利用门户网站的接口 目前已知的有腾讯.新浪.网易.搜狐和Google提供IP地址查询API,但是找得到的只有腾讯.新浪和 ...

  7. C#实现根据IP 查找真实地址

    IPScanner.cs public class IPScanner { private byte[] data; Regex regex = new Regex(@"(((\d{1,2} ...

  8. 获得Unix/Linux系统中的IP、MAC地址等信息

    获得Unix/Linux系统中的IP.MAC地址等信息 中高级  |  2010-07-13 16:03  |  分类:①C语言. Unix/Linux. 网络编程 ②手册  |  4,471 次阅读 ...

  9. 纯真IP根据IP地址获得地址

    <?php /** * 纯真IP根据IP地址获得地址 */ class ipLocation { public $fp; public $firstip; //第一条ip索引的偏移地址 publ ...

随机推荐

  1. mybatis 一对多 id标签作用

    一对多 MyBatis的resultMap只用于配置结果如何映射,id的唯一作用就是在嵌套的映射配置时判断数据是否相同,当配置id标签时,MyBatis只需要逐条比较所有数据中id标签字段值是否相同即 ...

  2. Pig模式

    Pig中的模式可以是用户显示声明的,也可以是Pig通过用户的使用方式猜测的. Pig对模式的认知在Pig Latin脚本执行的不同阶段可能是不同的.     下面的语句,用户显示声明了模式:3个字段, ...

  3. ExpressRoute 线路预配工作流和线路状态

    本页从较高层面引导你完成服务预配和路由配置工作流. 下图和相应的步骤说明了预配端到端 ExpressRoute 线路所要执行的任务. 使用 PowerShell 配置 ExpressRoute 线路. ...

  4. Redis 处理客户端连接的一些内部实现机制

    本文主要介绍了 Redis 处理客户端连接的一些内部实现机制,包括连接处理.超时.缓冲区等一系列内容. 注:本文所述内容基于 Redis2.6 及以上版本. 连接的建立 Redis 通过监听一个 TC ...

  5. 关于easyUI分页

    首先前台会传来两个参数,分别是rows(一页数据的大小,即一页有多少条数据)和page(第几页),根据这两个参数可以计算出从数据库中从第几 条数据开始取和要取多少条数据.数据取出来后,因为easyUI ...

  6. Linux crontab命令详解

    crontab:定时任务的守护进程,精确到分,设计秒的我们一般写脚本  -->相当于闹钟        日志文件:  ll /var/log/cron*        编辑文件: vim /et ...

  7. 在 vSphere 5.x/6.0 中配置 Network Dump Collector 服务 (2002954)

    vmware KB: https://kb.vmware.com/s/article/2002954?lang=zh_CN 重点配置命令: 使用 vSphere Client 连接到 vCenter ...

  8. git遇到问题

    简介 这里记录git使用过程中所涉及的问题,记录下解决方案. git 本地项目上传远程仓库[github] 已在远程建好仓库,在本地项目根目录下 $ git init $ git add . $ gi ...

  9. 将Python打包成可执行文件exe的心路历程

    导言: 我们有时候需要将做好的Python程序打包成为一个exe , 方便我们使用,查找了资料发现 pyinstaller .py2exe,最后还是选择的pyinstaller,用的时候踩过了挺多的坑 ...

  10. 3226. [SDOI2008]校门外的区间【线段树】

    Description   受校门外的树这道经典问题的启发,A君根据基本的离散数学的知识,抽象出5种运算维护集合S(S初始为空)并最终输出S.现在,请你完成这道校门外的树之难度增强版——校门外的区间. ...