【子网掩码计算器:Python + Tkinter 实现】

news2025/4/8 3:12:19

子网掩码计算器:Python + Tkinter 实现

  • 引言
    • 代码功能概述
    • 代码实现思路
      • 1. 界面设计
      • 2. 功能实现
      • 3. 事件处理
  • 子网掩码计算器实现步骤
    • 1. 导入必要的库
    • 2. 定义主窗口类 SubnetCalculatorApp
    • 3. 创建菜单栏
    • 4. 创建界面组件
    • 5. 判断 IP 地址类别
    • 6. 计算子网信息
    • 7. 其他功能函数
    • 代码运行效果
    • 总结

引言

在网络工程和网络管理领域,子网掩码的计算是一项基础且重要的工作。通过子网掩码,我们可以将一个大的网络划分为多个小的子网,从而提高网络的安全性和可管理性。今天,我们将介绍一个使用 Python 和 Tkinter 库实现的子网掩码计算器,它可以帮助我们快速计算子网信息,并且支持历史记录的查看和导出。

代码功能概述

这个子网掩码计算器具有以下主要功能:

  • 输入验证:验证用户输入的 IP 地址和子网掩码 / CIDR 是否有效。
  • 子网计算:根据输入的 IP 地址和子网掩码 / CIDR 计算网络地址、广播地址、可用 IP 数、可用 IP 范围等信息。
  • IP 地址分类:判断输入的 IP 地址属于 A、B、C、D、E 类中的哪一类,以及是公网地址还是私有地址。
  • 结果显示:将计算结果显示在界面上,并支持复制结果到剪贴板。
  • 历史记录:保存最近 5 条计算结果,并支持查看和导出历史记录。

代码实现思路

1. 界面设计

  • 使用 Tkinter 库创建一个图形用户界面(GUI),包括输入框、按钮、标签和表格等组件。
  • 通过布局管理器(如 pack 和 grid)将这些组件排列在合适的位置。

2. 功能实现

  • 输入验证:编写函数 validate_ipvalidate_mask 来验证 IP 地址和子网掩码的格式是否正确。
  • 子网计算
    • 编写函数 ip_to_intint_to_ip 来实现 IP 地址和 32 位整数之间的转换。
    • 根据子网掩码计算网络地址、广播地址等信息。
  • IP 地址分类:编写函数 get_ip_class 来判断 IP 地址的类别。
  • 历史记录管理
    • 使用列表 self.history 来存储最近 5 条计算结果。
    • 提供查看和导出历史记录的功能。

3. 事件处理

  • 为每个按钮绑定相应的事件处理函数,例如:
    • 点击 “计算” 按钮时调用 calculate 函数进行子网计算。
    • 点击 “保存当前结果” 按钮时调用 save_current_result 函数保存结果。

子网掩码计算器实现步骤

1. 导入必要的库

import tkinter as tk
from tkinter import ttk, messagebox, filedialog
import csv
from datetime import datetime

这里导入了 Tkinter 库用于创建 GUI,csv 库用于处理 CSV 文件,datetime 库用于记录时间。

2. 定义主窗口类 SubnetCalculatorApp

class SubnetCalculatorApp(tk.Tk):
    def __init__(self):
        super().__init__()
        self.title("子网掩码计算器")
        self.geometry("650x470")  # 调整窗口大小
        self.resizable(False, False)
        self.history = []  # 存储最近5条历史记录
        self.create_widgets()
        self.create_menu()

init 方法中,我们初始化了主窗口的标题、大小、是否可调整大小属性,并创建了一个空的历史记录列表。然后,我们调用了 create_widgets 和 create_menu 方法来创建界面组件和菜单栏。

3. 创建菜单栏

def create_menu(self):
    """创建菜单栏"""
    menubar = tk.Menu(self)

    # 历史记录菜单
    history_menu = tk.Menu(menubar, tearoff=0)
    history_menu.add_command(label="查看历史记录", command=self.show_history)
    history_menu.add_command(label="导出历史记录...", command=self.export_history)
    menubar.add_cascade(label="历史记录", menu=history_menu)

    self.config(menu=menubar)

在 create_menu 方法中,我们创建了一个菜单栏,并为其添加了一个名为“历史记录”的子菜单。该子菜单包含两个选项:“查看历史记录”和“导出历史记录…”。当用户点击这些选项时,将分别调用 show_history 和 export_history 方法。最后,我们将菜单栏设置到主窗口中。

4. 创建界面组件

def create_widgets(self):
    """创建界面组件"""
    # 输入区域
    input_frame = ttk.Frame(self)
    input_frame.pack(padx=10, pady=10, fill=tk.X)

    ttk.Label(input_frame, text="IP地址:").grid(row=0, column=0, sticky=tk.W)
    self.ip_entry = ttk.Entry(input_frame, width=20)
    self.ip_entry.grid(row=0, column=1, padx=5)

    ttk.Label(input_frame, text="子网掩码/CIDR:").grid(row=1, column=0, sticky=tk.W)
    self.mask_entry = ttk.Entry(input_frame, width=20)
    self.mask_entry.grid(row=1, column=1, padx=5)

    ttk.Button(input_frame, text="计算", command=self.calculate).grid(row=2, column=0, columnspan=2, pady=10)

    # 结果区域
    result_frame = ttk.LabelFrame(self, text="计算结果")
    result_frame.pack(padx=10, pady=5, fill=tk.BOTH, expand=True)

    result_labels = [
        ("网络地址:", "network"),
        ("广播地址:", "broadcast"),
        ("可用IP数:", "hosts"),
        ("可用IP范围:", "range"),
        ("CIDR表示:", "cidr"),
        ("子网掩码:", "mask"),
        ("IP版本:", "version"),
        ("网络类别:", "class")  # 新增网络类别
    ]

    for i, (label, _) in enumerate(result_labels):
        ttk.Label(result_frame, text=label).grid(row=i, column=0, sticky=tk.W, padx=5, pady=2)
        setattr(self, f"result_{label.split(':')[0]}", ttk.Label(result_frame, text=""))
        getattr(self, f"result_{label.split(':')[0]}").grid(row=i, column=1, sticky=tk.W, padx=5, pady=2)

    # 按钮区域
    button_frame = ttk.Frame(self)
    button_frame.pack(pady=5)

    ttk.Button(button_frame, text="保存当前结果", command=self.save_current_result).pack(side=tk.LEFT, padx=5)
    ttk.Button(button_frame, text="复制结果", command=self.copy_results).pack(side=tk.LEFT, padx=5)
    ttk.Button(button_frame, text="清除输入", command=self.clear_inputs).pack(side=tk.LEFT, padx=5)

这段代码创建了输入区域、结果区域和按钮区域,并为每个组件设置了相应的布局和事件处理函数。

5. 判断 IP 地址类别

def get_ip_class(self, ip_str):
    """判断IP地址类别"""
    first_octet = int(ip_str.split('.')[0])
    if 1 <= first_octet <= 126:
        return "A类(公网地址)" if first_octet != 10 else "A类(私有地址)"
    elif 128 <= first_octet <= 191:
        if first_octet == 172 and 16 <= int(ip_str.split('.')[1]) <= 31:
            return "B类(私有地址)"
        return "B类(公网地址)"
    elif 192 <= first_octet <= 223:
        if first_octet == 192 and int(ip_str.split('.')[1]) == 168:
            return "C类(私有地址)"
        return "C类(公网地址)"
    elif 224 <= first_octet <= 239:
        return "D类(组播地址)"
    elif 240 <= first_octet <= 255:
        return "E类(保留地址)"
    elif first_octet == 127:
        return "环回地址"
    return "未知类别"

这个函数根据 IP 地址的第一个八位组判断其类别,并区分公网地址和私有地址。

6. 计算子网信息

def calculate(self):
    """计算子网信息"""
    try:
        ip_str = self.ip_entry.get().strip()
        mask_str = self.mask_entry.get().strip().lstrip('/')

        if not self.validate_ip(ip_str):
            raise ValueError("无效的IP地址格式")

        # 获取网络类别
        ip_class = self.get_ip_class(ip_str)

        ip_int = self.ip_to_int(ip_str)
        mask_int, cidr = self.parse_mask(mask_str, ip_int)
        subnet_mask = self.int_to_ip(mask_int)

        network_int = ip_int & mask_int
        broadcast_int = network_int | (~mask_int & 0xFFFFFFFF)
        hosts_count = max(broadcast_int - network_int - 1, 0)

        # 更新结果
        self.result_网络地址.config(text=self.int_to_ip(network_int))
        self.result_广播地址.config(text=self.int_to_ip(broadcast_int))
        self.result_可用IP数.config(text=f"{hosts_count} 个")
        self.result_CIDR表示.config(text=f"/{cidr}")
        self.result_子网掩码.config(text=subnet_mask)
        self.result_IP版本.config(text="IPv4")
        self.result_网络类别.config(text=ip_class)  # 显示网络类别

        if hosts_count > 0:
            first_ip = self.int_to_ip(network_int + 1)
            last_ip = self.int_to_ip(broadcast_int - 1)
            self.result_可用IP范围.config(text=f"{first_ip} - {last_ip}")
        else:
            self.result_可用IP范围.config(text="无可用IP")

    except Exception as e:
        messagebox.showerror("错误", str(e))

在 calculate 函数中,首先验证输入的 IP 地址和子网掩码 / CIDR 是否有效,然后进行子网计算,并将结果显示在界面上。

7. 其他功能函数

代码中还包含了清空输入框、复制结果、保存当前结果、查看历史记录和导出历史记录等功能函数,这些函数的实现逻辑都比较简单,这里就不再详细介绍了。

代码运行效果

运行代码后,会弹出一个子网掩码计算器的窗口,界面如下:
在这里插入图片描述

用户可以在输入框中输入 IP 地址和子网掩码 / CIDR,点击 “计算” 按钮即可得到计算结果。点击 “保存当前结果” 按钮可以将结果保存到 CSV 文件中,点击 “复制结果” 按钮可以将结果复制到剪贴板。通过菜单栏中的 “历史记录” 菜单可以查看和导出最近 5 条计算结果。

总结

通过这个子网掩码计算器的实现,我们学习了如何使用 Python 和 Tkinter 库创建一个简单的 GUI 应用程序。同时,我们也掌握了子网计算的基本原理和 IP 地址分类的方法。希望这篇博客对你有所帮助,如果你有任何问题或建议,欢迎在评论区留言。

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

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

相关文章

入门基础项目(SpringBoot+Vue)

文章目录 1. css布局相关2. JS3. Vue 脚手架搭建4. ElementUI4.1 引入ElementUI4.2 首页4.2.1 整体框架4.2.2 Aside-logo4.2.3 Aside-菜单4.2.4 Header-左侧4.2.5 Header-右侧4.2.6 iconfont 自定义图标4.2.7 完整代码 4.3 封装前后端交互工具 axios4.3.1 安装 axios4.3.2 /src…

车载以太网-基于linux的ICMP协议

对于车载以太网-ICMP的技术要求: /** ICMP报文格式解析* -----------------* ICMP协议用于网络诊断和错误报告,常见应用包括Ping测试。* ICMP报文结构包括:IP头部、ICMP头部和ICMP数据部分。* 下面详细介绍每个部分的结构、字段的作用以及如何解析它们。* * ICMP头部结构:*…

虚拟机快照与linux的目录结构

虚拟机快照是对虚拟机某一时刻状态的完整捕获&#xff0c;包括内存、磁盘、配置及虚拟硬件状态等&#xff0c;保存为独立文件。 其作用主要有数据备份恢复、方便系统测试实验、用于灾难恢复以及数据对比分析。具有快速创建和恢复、占用空间小、可多个快照并存的特点。在管理维…

Unity小功能实现:鼠标点击移动物体

1、功能描述 当玩家点击鼠标时&#xff0c;场景中的物体会移动到鼠标点击的位置。这个功能可以用于控制角色移动、放置物体等场景。 2、实现步骤 创建Unity项目&#xff1a;首先&#xff0c;打开Unity并创建一个新的3D项目。 添加3D物体&#xff1a;在场景中创建一个3D物体&am…

算法题笔记(自用)——Python

目录 一. 进制&位运算&ASCAII 二. format格式化输出 1. 基本用法 2. 位置参数 3. 格式化数字 4. 对齐和填充 5. 格式化二进制、八进制、十六进制 6. 格式化百分比 7. 格式化科学计数法 8. 格式化字符串字面量&#xff08;f-string&#xff09; 三. 字符串 使…

爬虫系列之【数据解析之JSON】《三》

目录 前置知识 一、 json.loads()&#xff1a;JSON 转 Python 数据 二、json.dump()&#xff1a;python数据 转 json 并写入文件 三、json.loads() &#xff1a;json 转 python数据 四、json.load() &#xff1a;json 转 python数据&#xff08;在文件操作中更方便&#xf…

了解Java集合的概念和体系:Collection<T>、Collections与Stream的使用

学习目标 本文知识是对集合层级的介绍&#xff0c;应用开发中实际使用的是他们的子级&#xff0c;感兴趣的小伙伴或者想深入了解有关Java集合知识的朋友可以选择阅读&#xff01; Stream的方法使用使用部分代码块内大多有两种实现方式&#xff0c;是为了更好的理解方法底层的代…

进程优先级和进程切换 ─── linux第12课

目录 Z(zombie)-僵尸进程 僵尸进程危害 孤儿进程 ​编辑 进程优先级 查看系统进程 用top命令更改已存在进程的nice&#xff1a; 其他概念 进程切换 进程如何切换 进程的调度 进程 内核数据结构 代码和数据 创建进程时 ,先创建内核数据结构 再加载代码和数据 进程退…

光速解决phpstudy无法启动MySQL服务

问题描述 在初步安装使用phpstudy时&#xff0c;会出现mysql服务启动失败的情况&#xff0c;具体表现为点击一键启动后&#xff0c;mysql启动又立即停止 原因及解决方法&#xff1a; phpstudy数据库无法启动有以下几个原因&#xff0c;可以看看自己是第几种&#xff1a; 服务名…

txt 转 json 使用python语言

需求: 把如下的txt文档转成json输出 代码 import jsondef txt_to_json(input_file, output_file):data_list []with open(input_file, r, encodingutf-8) as f:for line in f:# 分割数据并去除换行符parts line.strip().split(,)print(f"{parts}")print(type(par…

Cargo, the Rust package manager, is not installed or is not on PATH.

今天在Windows操作系统上通过pip 安装jupyter的时候遇到这个报错&#xff0c;Cargo, the Rust package manager, is not installed or is not on PATH.。 解决办法 官网&#xff1a;https://rustup.rs/# 下载&#xff1a;https://win.rustup.rs/x86_64 安装完成之后&#xff0c…

JavaWeb个人笔记

技术栈 前端 : HTML CSS JavaScript ES6 Nodejs npm vite vue3 router pinia axios element-plus 后端&#xff1a;HTTP xml Tomcat Servlet Request Response Cookie Sesssion Filter Listener MySQL JDBC Druid Jackson lombok jwt . HTML <!DOCTYPE html> 文档声…

http报文的content-type参数和spring mvc传参问题

很早之前博主聊过HTTP的报文结构以及其中和传参相关的重要参数content-type还有spring mvc&#xff0c;以前的三篇文章&#xff1a; HTTP与HTTPS协议详解&#xff1a;基础与安全机制-CSDN博客 详解Http的Content-Type_content-type application-CSDN博客 如何在Spring Boot中…

14. LangChain项目实战1——基于公司制度RAG回答机器人

教学视频&#xff1a; 12. 基于Gradio搭建基于公司制度RAG_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV11VXRYTErZ/ 环境配置&#xff1a; python版本&#xff1a;3.10.8 服务器&#xff1a;Ubuntu 依赖包requirements.txt文件内容&#xff1a; aiofiles23.2.1 …

FPGA开发,使用Deepseek V3还是R1(6):以滤波器为例

以下都是Deepseek生成的答案 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;1&#xff09;&#xff1a;应用场景 FPGA开发&#xff0c;使用Deepseek V3还是R1&#xff08;2&#xff09;&#xff1a;V3和R1的区别 FPGA开发&#xff0c;使用Deepseek V3还是R1&#x…

JVM常用概念之垃圾回收设计与停顿

在我们应用程序运行期间&#xff0c;我们是需要尽可能避免垃圾回收。 图1&#xff1a;不同垃圾回收器的设计&#xff08;黄色代表STW&#xff0c;绿色代表并发&#xff09; 实验 计算机配置 Hardware Overview:Model Name: MacBook ProModel Identifier: MacBookPro14,2Pro…

uniapp-原生android插件开发摘要

uni-app在App侧的原生扩展插件&#xff0c;支持使用java、object-c等原生语言编写&#xff0c;从HBuilderX 3.6起&#xff0c;新增支持了使用uts来开发原生插件。 基础项目 UniPlugin-Hello-AS工程请在App离线SDK中查找 基础项目(App离线SDK)已经配置好了自定义插件所需要的…

派可数据BI接入DeepSeek,开启智能数据分析新纪元

派可数据BI产品完成接入DeepSeek&#xff0c;此次接入标志着派可数据BI在智能数据分析领域迈出了重要一步&#xff0c;将为用户带来更智能、更高效、更便捷的数据分析体验。 派可数据BI作为国内领先的商业智能解决方案提供商&#xff0c;一直致力于为用户提供高效、稳定易扩展…

M4 Mac mini运行DeepSeek-R1模型

前言 最近DeepSeek大模型很火&#xff0c;实际工作中也有使用&#xff0c;很多人觉得需要很好的显卡才能跑起来&#xff0c;至少显存需要很高&#xff0c;但实际上一般的核显机器也能跑起来&#xff0c;只不过内存要求要大&#xff0c;对于个人而言&#xff0c;实际上Mac M芯片…

MaxKB上架至阿里云轻量应用服务器镜像市场

近日&#xff0c;MaxKB开源知识库问答系统已上架至阿里云轻量应用服务器镜像市场&#xff0c;目前是阿里云此类镜像市场中唯一推荐的AI应用镜像。 ▲图1 MaxKB已经上架至阿里云轻量应用服务器镜像市场 MaxKB是飞致云旗下开源项目&#xff0c;是一款基于大语言模型和RAG&…