题解 | 识别有效的IP地址和掩码并进行分类统计

识别有效的IP地址和掩码并进行分类统计

https://www.nowcoder.com/practice/de538edd6f7e4bc3a5689723a7435682

改了好多次,说一下需要注意的坑。

1.

IP地址和子网掩码都应该严格遵循"{数字}.{数字}.{数字}.{数字}"的格式,需要先检测是否是这个格式。

比如"19..0."就不是。

2.

IP地址可以同时是【五类IP地址(ABCDE)】和【私有IP地址】,需要分别独立检查。

3.

对于一条地址信息,它的【IP地址】和【子网掩码】只要有一个不合法,地址信息就标为错误。

题目要求输出的第6项"错误IP或错误子网掩码的数量",实际上指的就是错误的【地址信息】的数量。

同一条地址信息如果【IP地址】和【子网掩码】都错误,不要重复计数。

4.

IP地址如果是"0.*.*.*"和"127.*.*.*",就直接跳过这条地址信息,不要用来计数。

5.

子网掩码的二进制表示需要补足到8位。

比如子网掩码"255.255.32.0",二进制表示应为"11111111.11111111.00100000.00000000",所以它是非法的。

import sys

# 检查五类IP地址和私有IP地址
def checkIP(s: str):
    ls = s.split('.')
    # 格式错误
    if (not all(item.isdigit() for item in ls)) or len(ls) != 4:
        return False
    ls = list(map(int, ls))
    # 非法IP地址
    if not all(0 <= item <= 255 for item in ls):
        return False
    # 0和127开头的IP地址直接忽略
    if ls[0] in [0, 127]:
        return -1
    # 判断ABCDE五类
    result = 0
    if 1 <= ls[0] <= 126:
        result = 2
    elif 128 <= ls[0] <= 191:
        result = 4
    elif 192 <= ls[0] <= 223:
        result = 6
    elif 224 <= ls[0] <= 239:
        result = 8
    elif 240 <= ls[0] <= 255:
        result = 10
    # 判断私有。1私有,0非私有。
    if ls[0] == 10:
        result += 1
    elif (ls[0] == 172) and (16 <= ls[1] <= 31):
        result += 1
    elif (ls[0] == 192) and (ls[1] == 168):
        result += 1
    return result


# 检查子网掩码是否合法
def checkMask(s: str):
    ls = s.split('.')
    if (not all(item.isdigit() for item in ls)) or len(ls) != 4:
        return False
    ls = list(map(int, ls))
    check = ""
    # 每一个二进制值都需要补到8位
    for item in ls:
        check += bin(item)[2:].rjust(8, '0')
    check_num = int(check[0])
    if check_num != 1:
        return False
    cnt = 0
    for char in check:
        cnt += check_num ^ int(char)
        check_num = int(char)
    if cnt != 1:
        return False
    else:
        return True


A_count, B_count, C_count, D_count, E_count, ErrorIPMask_count, privateIP_count = (
    0,
    0,
    0,
    0,
    0,
    0,
    0,
)


def output(s: str):
    global A_count, B_count, C_count, D_count, E_count, ErrorIPMask_count, privateIP_count
    ss = s.strip('\n').split('~')
    if len(ss) != 2:
        e = False
    else:
        e = True
    c_0 = checkIP(ss[0])
    c_1 = checkMask(ss[1])
    if c_0 == -1:
        pass
    elif (c_0 == False) or (c_1 == False) or (e == False):
        ErrorIPMask_count += 1
    elif c_0 > 0:
        if c_0 & 1 == 1:
            privateIP_count += 1
        if c_0 >> 1 == 1:
            A_count += 1
        elif c_0 >> 1 == 2:
            B_count += 1
        elif c_0 >> 1 == 3:
            C_count += 1
        elif c_0 >> 1 == 4:
            D_count += 1
        elif c_0 >> 1 == 5:
            E_count += 1


for line in sys.stdin:
    output(line)

print(f'{A_count} {B_count} {C_count} {D_count} {E_count} {ErrorIPMask_count} {privateIP_count}')

全部评论

相关推荐

10-10 01:10
已编辑
深圳大学 测试开发
面了100年面试不知...:六月到九月,四个项目一个实习,是魔丸吗
投了多少份简历才上岸
点赞 评论 收藏
分享
浩浩没烦恼:一二面加起来才一个小时? 我一面就一个小时多了
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务