使用B树实现员工(人事)管理系统

1. 前言
 

使用B树来表示人事管理系统,其中每个节点代表一个人员,树的根节点为董事长,每个节点可以有多个子节点,表示下属。每一层代表一个等级分布。

  • addPerson: 添加人员功能通过查找指定上司节点,然后将新的人员作为其子节点添加。
  • deletePerson: 删除人员功能首先查找要删除人员的直接上司节点,然后从其子节点中删除该人员。
  • updatePersonInfo: 更新人员信息功能通过查找指定人员节点,然后更新其信息。
  • traversePersons: 遍历人员信息功能采用前序遍历方式,递归地遍历整个人员树。
  • levelOrderTraversal: 按照职位进行层序遍历,以便展示不同职位的人员信息。
  • searchPerson: 按照姓名进行搜索,采用递归方式在整个树中搜索指定姓名的人员节点。
  • searchPosition: 按照职位进行搜索,返回指定职位的所有人员节点列表。
  • main方法:在主函数中实现了用户交互功能,用户可以选择不同的功能编号来执行相应的操作。

2. 功能演示 
 

研究以二叉树为例(人员个数及权值自定义),实现以下任务:

  1. 采用树的孩子兄弟链表等存储结构;
  2. 实现在人事关系树中查找、添加、删除人员功能;
  3. 创建树结构;
  4. 通过深度优先遍历搜索;
  5. 通过层次优先遍历搜索;
  6. 采用Java语言编写代码实现,并撰写实验报告。

【实验思路】  

使用B树来表示人事管理系统,其中每个节点代表一个人员,树的根节点为董事长,每个节点可以有多个子节点,表示下属。每一层代表一个等级分布。

  • addPerson: 添加人员功能通过查找指定上司节点,然后将新的人员作为其子节点添加。
  • deletePerson: 删除人员功能首先查找要删除人员的直接上司节点,然后从其子节点中删除该人员。
  • updatePersonInfo: 更新人员信息功能通过查找指定人员节点,然后更新其信息。
  • traversePersons: 遍历人员信息功能采用前序遍历方式,递归地遍历整个人员树。
  • levelOrderTraversal: 按照职位进行层序遍历,以便展示不同职位的人员信息。
  • searchPerson: 按照姓名进行搜索,采用递归方式在整个树中搜索指定姓名的人员节点。
  • searchPosition: 按照职位进行搜索,返回指定职位的所有人员节点列表。
  • main方法:在主函数中实现了用户交互功能,用户可以选择不同的功能编号来执行相应的操作。

【实验步骤与实验结果】

  1. 默认初始化自动添加7个员工信息

代码

运行结果

  1. 查找员工的直属上司
  2. 按照名称查找员工信息


运行结果:

2.添加员工信息
代码


运行结果:


测试:

3.修改人员信息
代码:



运行结果:

4. 删除员工信息
代码:



运行结果:

5. 按照职位查询员工信息
代码


运行结果:

6.查询指定人员的下属
代码:


运行结果:

【实验小结】

        本次实验通过实现人事管理系统,加深了对二叉树的理解,掌握了树的存储方法和遍历算法和树寻找父类的算法加深了我对递归算法的理解。通过实践,更加熟练地运用了Java语言编程,提高了问题解决能力和编码能力。在未来的学习和工作中,将继续加强对数据结构的学习,提升编程水平。

代码总结

人的信息的javabean

package test;

public class Person {
    private int id;
    private String name;
    private int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Person(int id, String name, int age) {
        this.id = id;
        this.name = name;
        this.age = age;
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

 节点类
 

package test;

import java.util.ArrayList;
import java.util.List;

/**
 * @Description: 树的节点类、
 * @Author: windStop
 * @Date: 2024/6/12 16:37
 */
// 树的节点类
public class TreeNode {
    int height; // 数的层数 (记录职位)
    Person person; // 人员的个人信息
    TreeNode parent; // 记录直属上司 (父亲)
    List<TreeNode> children; // (记录下属,集合中的元素互为兄弟节点)

    public TreeNode(Person person) {
        this.person = person;
        this.children = new ArrayList<>(); // 初始化 children 列表
    }
}

核心功能代码:
 

package test;


import java.util.*;


// 人事管理系统
public class PersonManagementSystem {
    // 根节点(董事长)
    private TreeNode root;
    private String[] position = {"董事长","经理","团队负责人","员工"};

    public PersonManagementSystem(Person rootPerson) {
        this.root = new TreeNode(rootPerson);
        this.root.height = 1;
    }


    /**
     * 添加人员
     * @param parentPerson 要添加人员的直属上司(信息的javabean)
     * @param childrenPerson  添加人员的信息
     */
    public void addPerson(Person parentPerson, Person childrenPerson) {
        TreeNode parentNode = findNode(root, parentPerson);
        if (parentNode != null) {
            TreeNode childNode = new TreeNode(childrenPerson);
            parentNode.children.add(childNode);
            childNode.parent = parentNode; // 记录父类
            childNode.height = parentNode.height + 1; // 记录高度
        } else {
            System.err.println("节点没有找到,无法添加");
        }
    }

    /**
     * 递归寻找:查找人当前的节点
     * @param node 查找人的:上司(递归会变)
     * @param person 查找人的javabean
     * @return
     */
    public TreeNode findNode(TreeNode node, Person person) {
        // 查找的人就是根节点
        if (node.person.getId() == person.getId()) {
            return node;
        }
        // 往下循环查找
        for (TreeNode child : node.children) {
            TreeNode found = findNode(child, person);//子问题
            if (found != null) {//找到
                return found;
            }
        }
        return null;//找完也没有找到
    }

    /**
     * 删除人员  ->  通过查找到该人员的直属上司,从直属上司的名单(list)中进行删除
     * @param person 要删除人员的信息
     * @return
     */
    public boolean deletePerson(Person person) {
        if (root == null) {
            System.err.println("节点没有找到,无法删除。");
            return false;
        }
        TreeNode parentNode = findParent(root, person);
        if (parentNode != null) {
            List<TreeNode> children = parentNode.children;
            for (int i = 0; i < children.size(); i++) {
                TreeNode child = children.get(i);
                if (child.person.getId() == person.getId()) {
                    children.remove(i);
                    return true; // 成功删除
                }
            }
        }
        System.err.println("要删除的人员节点不存在。");
        return false;
    }

    /**
     * 递归寻找:查找人的父节点
     * @param node 查找人的:上司(递归会变)
     * @param person 查找人的javabean
     * @return
     */
    public TreeNode findParent(TreeNode node, Person person) {
        for (TreeNode child : node.children) {
            //当前节点是要找的节点的父节点
            if (child.person.getId() == person.getId()) {
                return node;
            } else {
                //当前节点不是,就递归查找 (子问题,以子节点为根)
                TreeNode parent = findParent(child, person);
                if (parent != null) {
                    return parent;
                }
            }
        }
        //找不到
        return null;
    }

    /**
     * 更新人员的信息
     * @param oldPerson 老信息
     * @param newPerson 新信息
     * @return
     */
    public boolean updatePersonInfo(Person oldPerson, Person newPerson) {
        TreeNode nodeToUpdate = findNode(root, oldPerson);
        if (nodeToUpdate != null) {
            // 更新人员信息
            nodeToUpdate.person.setName(newPerson.getName());
            nodeToUpdate.person.setAge(newPerson.getAge());
            nodeToUpdate.person.setId(newPerson.getId());
            return true; // 成功更新
        } else {
            System.err.println("要更新的人员节点不存在。");
            return false;
        }
    }

    /**
     * 遍历人员信息 -> 属于根左右 的前序遍历
     * @param node 从跟开始遍历到结束
     */
    public void traversePersons(TreeNode node) {
        if (node != null) {
            System.out.println("姓名:" + node.person.getName() + "\n工号为:" + node.person.getId());
            for (TreeNode child : node.children) {
                traversePersons(child);
            }
        }
    }

    // 层序遍历
//    public void levelOrderTraversal() {
//        if (root == null) return;
//
//        Queue<TreeNode> queue = new LinkedList<>();
//        queue.offer(root);
//
//        while (!queue.isEmpty()) {
//            int size = queue.size();
//            for (int i = 0; i < size; i++) {
//                TreeNode node = queue.poll();
//                System.out.println(position[node.height - 1] + ":\n姓名:" + node.person.getName() + "\n工号为:" + node.person.getId());
//                for (TreeNode child : node.children) {
//                    queue.offer(child);
//                }
//            }
//        }
//    }

    /**
     * 按照员工职位进行分类查询
     */
    public void levelOrderTraversal() {
        if (root == null) return;

        Queue<TreeNode> queue = new LinkedList<>();
        queue.offer(root);

        int level = 1;
        while (!queue.isEmpty()) {
            int size = queue.size();
            System.out.print(position[level - 1] + ":");
            for (int i = 0; i < size; i++) {
                TreeNode node = queue.poll();
                System.out.print( "工号:"+ node.person.getId() + ", 姓名:" + node.person.getName() + " ");
                for (TreeNode child : node.children) {
                    queue.offer(child);
                }
            }
            System.out.println();
            level++;
        }
    }



    /**
     * 根据名称进行搜索
     * @param name 人名
     * @return
     */
    public TreeNode searchPerson(String name) {
        return searchPerson(root, name);
    }

    /**
     * 递归搜索(按照人名)
     * @param node
     * @param name
     * @return
     */
    private TreeNode searchPerson(TreeNode node, String name) {
        if (node.person.getName().equals(name)) {
            return node;
        } else {
            for (TreeNode child : node.children) {
                TreeNode found = searchPerson(child, name);
                if (found != null) {
                    return found;
                }
            }
        }
        return null;
    }

    /**
     * 按照职位进行搜索
     * @param height 职位
     * @return
     */
    private List<TreeNode> searchPosition(int height) {
        if (root == null) return null;

        List<TreeNode> persons = new ArrayList<>();
        Queue<TreeNode> queue = new LinkedList<>();
        queue.offer(root);

        while (!queue.isEmpty()) {
            int size = queue.size();
            for (int i = 0; i < size; i++) {
                TreeNode node = queue.poll();
                if (node != null && node.height == height) {
                    persons.add(node);
                }
                if (node != null) {
                    for (TreeNode child : node.children) {
                        queue.offer(child);
                    }
                }
            }
        }
        return persons;
    }


    public static void main(String[] args) {
        // 默认创建的人员
        // 创建人员
        Person chairman = new Person(1, "蔡申友", 18);
        Person manager = new Person(2, "李四", 19);
        Person teamLeader1 = new Person(3, "王五", 20);
        Person teamLeader2 = new Person(4, "赵六", 31);
        Person staff1 = new Person(5, "小明", 32);
        Person staff2 = new Person(6, "小红", 33);
        Person staff3 = new Person(7, "小李", 34);

        // 创建人事管理系统
        PersonManagementSystem system = new PersonManagementSystem(chairman);

        // 添加人员
        system.addPerson(chairman, manager);
        system.addPerson(manager, teamLeader1);
        system.addPerson(manager, teamLeader2);
        system.addPerson(teamLeader1, staff1);
        system.addPerson(teamLeader1, staff2);
        system.addPerson(teamLeader2, staff3);


        Scanner scanner = new Scanner(System.in);

        while (true) {
            System.out.println("-------------------------------");
            System.out.println("------请选择功能:---------------");
            System.out.println("-----1. 添加人员----------------");
            System.out.println("-----2. 修改人员信息-------------");
            System.out.println("-----3. 删除人员----------------");
            System.out.println("-----4. 按照名称查找人员----------");
            System.out.println("-----5. 遍历人员信息-------------");
            System.out.println("-----6. 按职位遍历人员信息-------");
            System.out.println("-----7. 查询人员的管理名单-------");
            System.out.println("-----8. 退出程序----------------");
            System.out.println("-------请输入对应功能的编号:-----");
            System.out.println("-------------------------------");

            int choice = scanner.nextInt();

            switch (choice) {
                case 1:
                    System.out.println("请输入你要添加的人员职位:");
                    System.out.println("1. 董事长");
                    System.out.println("2. 经理 ");
                    System.out.println("3. 团队负责人");
                    System.out.println("4. 员工");
                    int x = scanner.nextInt();
                    if (x == 1){
                        System.err.println("因为已存在董事长,无法添加董事长");
                        return;
                    }
                    List<TreeNode> treeNodes = system.searchPosition(x - 1);
                    if (treeNodes != null){
                        System.out.println("查询到的上司信息如下图所示:");
                        System.out.println("请选择你要添加人员的上司:");
                        for (int i = 0; i < treeNodes.size(); i++) {
                            System.out.println(i + 1  + ". " + treeNodes.get(i).person.getName());
                        }
                    }else {
                        System.out.println("该上司职位没有人员,无法添加。");
                    }
                    int choose = scanner.nextInt();
                    TreeNode parent = system.searchPerson(treeNodes.get(choose - 1).person.getName());

                    System.out.println("---请输入:你要添加人员的工号:");
                    int id = scanner.nextInt();
                    System.out.println("---请输入:你要添加人员的名称:");
                    String name = scanner.next();
                    System.out.println("---请输入:你要添加人员的年龄:");
                    int age = scanner.nextInt();
                    Person child = new Person(id,name,age);
                    system.addPerson(parent.person, child);
                    break;
                case 2:
                    // 修改人员信息
                    System.out.println("请输入:你要修改人员的名称:");
                    String updateName = scanner.next();
                    TreeNode updateNode = system.searchPerson(updateName);
                    if (updateNode == null){
                        System.out.println("不存在此人无法查询!");
                        break;
                    }
                    Person oldPerson = updateNode.person;
                    System.out.println("---请输入:你要修改人员的工号:");
                    int updateId = scanner.nextInt();
                    System.out.println("---请输入:你要修改人员的名称:");
                    String updateNewName = scanner.next();
                    System.out.println("---请输入:你要修改人员的年龄:");
                    int updateage = scanner.nextInt();
                    Person newPerson = new Person(updateId,updateNewName,updateage);
                    system.updatePersonInfo(oldPerson, newPerson);
                    break;
                case 3:
                    // 删除人员
                    System.out.println("请输入:你要删除人员的名称:");
                    String deleteName = scanner.next();
                    TreeNode deleteNode = system.searchPerson(deleteName);
                    if (deleteNode == null){
                        System.out.println("不存在此人无法查询!");
                        break;
                    }
                    Person personToDelete = deleteNode.person;
                    system.deletePerson(personToDelete);
                    break;
                case 4:
                    // 查找人员
                    System.out.print("请输入要查找的人员姓名:");
                    String nameToSearch = scanner.next();
                    TreeNode foundNode = system.searchPerson(nameToSearch);
                    if (foundNode != null) {
                        System.out.println("找到了该人员信息:");
                        System.out.println("姓名:" + foundNode.person.getName() + "\n工号:" + foundNode.person.getId());
                    } else {
                        System.out.println("未找到该人员信息。");
                    }
                    break;
                case 5:
                    // 遍历人员信息
                    system.traversePersons(system.root);
                    break;
                case 6:
                    // 层序遍历人员信息
                    system.levelOrderTraversal();
                    break;
                case 7:
                    // 查询指定名称的人员的下属
                    System.out.println("请输入:你要查询人员的名称:");
                    String findName = scanner.next();
                    TreeNode treeNode = system.searchPerson(findName);
                    if (treeNode == null){
                        System.out.println("不存在此人无法查询!");
                        break;
                    }
                    System.out.print(system.position[treeNode.height - 1] + ":" +findName + "对应的管理下属信息为:");
                    for (TreeNode node : treeNode.children) {
                        System.out.print( "工号:"+ node.person.getId() + ", 姓名:" + node.person.getName() + " ");
                    }
                    System.out.println();
                    break;
                case 8:
                    // 退出程序
                    System.out.println("程序已退出。");
                    System.exit(0);
                default:
                    System.out.println("请输入有效的功能编号。");
            }
        }
    }


}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

程序员/码农创业有多少种可能?

程序员创业&#xff0c;无疑是当下科技浪潮中的一股强大力量。凭借扎实的技术功底和敏锐的市场洞察力&#xff0c;在创业道路上展现出了无限的活力和创造力。那么&#xff0c;程序员创业究竟有哪些事情可以做呢&#xff1f;可以从技术产品的研发入手。 可以利用自己的专业知识…

分析GIS在疾病传播模型和公共卫生决策中的作用

在这个全球化日益加深的时代&#xff0c;疾病的跨国界传播成为全球公共卫生面临的重大挑战。地理信息科学&#xff08;GIS&#xff09;作为一门集成了空间数据采集、处理、分析及可视化的技术体系&#xff0c;在公共健康领域展现出其不可替代的价值。本文旨在深入探讨GIS如何助…

电动两轮车——电源方案

随着城镇化的发展人们的活动半径不断变宽&#xff0c;短交通出行方式仍能覆盖主要的范围。从主要国家核心地区的出行数据看平均通勤半径不高于15km&#xff0c;摩托车、电动两轮车等两轮出行方式能更好匹配日常短交通出行需求。 应用框图 通常&#xff0c;电动两轮车由三部分…

3D gaussian-splatting项目环境配置记录

1.前景 项目论文&#xff1a;https://arxiv.org/abs/2308.04079 GitHub项目下载地址&#xff1a;https://github.com/graphdeco-inria/gaussian-splatting git clone时里面的子模块小项目会git不到&#xff0c;需要单独github下来&#xff0c;放入相应文件夹。 sibr_viewer…

C# WinForm —— 33 ContextMenuStrip介绍

1. 简介 右键某个控件/窗体时&#xff0c;弹出来的菜单&#xff0c;比如VS中右键窗体&#xff0c;弹出来的这个菜单&#xff1a; 和MenuStrip类似&#xff0c;ContextMenuStrip主菜单下面可以有子菜单&#xff0c;子菜单下面可以有下一级子菜单 2. 属性 和MenuStrip一样 …

第6章 应用层

考纲内容 &#xff08;一&#xff09;网络应用模型 客户/服务器模型&#xff1b;P2P模型 &#xff08;二&#xff09;域名系统(DNS) 层次域名空间&#xff1b;域名服务器&#xff1b;域名解析过程 &#xff08;三&#xff09;文件传输协议(FTP) …

升级和维护老旧LabVIEW程序

在升级老旧LabVIEW程序至64位环境时&#xff0c;需要解决兼容性、性能和稳定性等问题。本文从软件升级、硬件兼容性、程序优化、故障修复等多个角度详细分析。具体包括64位迁移注意事项、修复页面跳转崩溃、解决关闭程序后残留进程的问题&#xff0c;确保程序在新环境中的平稳运…

[Java基本语法] 从0到1带你精通Java基本语法

&#x1f338;个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 &#x1f3f5;️热门专栏:&#x1f355; Collection与数据结构 (92平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm1001.2014.3001.5482 &#x1f9c0;线程与…

undetected_chromedriver驱动浏览器结束报错OSError: [WinError 6] 句柄无效

undetected_chromedriver驱动浏览器结束报错OSError: [WinError 6] 句柄无效 问题背景 使用undetected_chromedriver包驱动浏览器结束后报错句柄无效 Exception ignored in: <function Chrome.del at 0x000001DD50F07A60> Traceback (most recent call last): File “D:…

ESP32 IDF ADF 加入音频

需要把mp3制作成音频bin 用ADF自带工具 果用户需要生成自己的 audio-esp.bin&#xff0c;则需要执行 mk_audio_bin.py 脚本&#xff08;位于 $ADF_PATH/tools/audio_tone/mk_audio_tone.py&#xff09;&#xff0c;并且指定相关文件的路径。 源 MP3 文件在 tone_mp3_folder …

软考-架构设计师-综合知识总结(试卷:2009~2022)(下篇)

说明 本文档对2009到2022年试卷的综合知识进行了归纳总结&#xff0c;同时对叶宏主编的《系统架构设计师教程》划分重点。 第十七章&#xff1a;通信系统架构设计 17.2 考题总结 第十八章&#xff1a;安全架构设计 18.1 重要知识点 18.2 考题总结 第十九章&#xff1a;大数据…

2080. 区间内查询数字的频率

题目&#xff1a; 请你设计一个数据结构&#xff0c;它能求出给定子数组内一个给定值的 频率 。 子数组中一个值的 频率 指的是这个子数组中这个值的出现次数。 请你实现 RangeFreqQuery 类&#xff1a; RangeFreqQuery(int[] arr) 用下标从 0 开始的整数数组 arr 构造一个…

跨国大文件传输需要哪些方面?怎么实现数据快速传输?

跨国大文件传输涉及到许多方面&#xff0c;包括网络速度、安全性、可靠性和法律合规性等。 以下是跨国大文件传输时需要考虑的一些重要方面&#xff1a; 高速稳定的网络连接&#xff1a;确保有足够的带宽和稳定的网络连接以支持大文件的快速传输。这可能需要考虑到跨国网络的延…

JVM 一些常见问题QA

GC Roots 虚拟机栈中引用的对象&#xff1b; 本地方法栈中JNI引用的对象&#xff1b; 方法区中类静态变量引用的对象&#xff1b; 方法区中常量引用的对象&#xff1b; Full GC是Minor GCMajor GC吗&#xff1f; Minor GC&#xff1a;回收年轻代&#xff1b; Major GC&…

[Cloud Networking] Layer 2 Protocol

文章目录 1. STP / RSTP / MSTP Protocol1.1 STP的作用1.2 STP 生成树算法的三个步骤1.3 STP缺点 2. ARP Protocol3. MACSEC 1. STP / RSTP / MSTP Protocol 1.1 STP的作用 消除二层环路&#xff1a;通过阻断冗余链路来消除网络中可能存在的环路链路备份&#xff1a;当活动链…

openh264 编码器源码分析:AnalyzePictureComplexity 函数

介绍 文件位置&#xff1a; openh264/codec/processing/src/complexityanalysis/ComplexityAnalysis.cpp 功能&#xff1a; 作为CWelsPreProcess类中一个方法&#xff0c;用来分析当前图像与参考图像之间的复杂度关系&#xff0c;以便编码策略。 原型&#xff1a; void CWels…

01背包问题(模板)

一、题目描述 描述 你有一个背包&#xff0c;最多能容纳的体积是V。 现在有n个物品&#xff0c;第i个物品的体积为vi,价值为wi。 &#xff08;1&#xff09;求这个背包至多能装多大价值的物品&#xff1f; &#xff08;2&#xff09;若背包恰好装满&#xff0c;求至多能装多大价…

XMind软件下载-详细安装教程视频

​简介 XMind是一款实用的思维导图软件&#xff0c;简单易用、美观、功能强大&#xff0c;拥有高效的可视化思维模式&#xff0c;具备可扩展、跨平台、稳定性和性能&#xff0c;真正帮助用户提高生产率&#xff0c;促进有效沟通及协作。中文官方网站&#xff1a;http://www.x…

Oracle最终会扼杀MySQL?(译)

原文网站&#xff1a;https://www.percona.com/blog/is-oracle-finally-killing-mysql/ 作者&#xff1a;Peter Zaitsev 自从Oracle收购了MySQL后&#xff0c;很多人怀疑Oracle对开源MySQL的善意&#xff0c;这篇percona的文章深入分析了Oracle已经和将要对MySQL采取的措施&a…

【C++】——继承(详解)

一 继承的定义和概念 1.1 继承的定义 继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段&#xff0c;它允许程序员在保 持原有类特性的基础上进行扩展&#xff0c;增加功能&#xff0c;这样产生新的类&#xff0c;称派生类&#xff0c;被继承的称为基类…