【文件系统—散列结构文件】

news2025/5/10 11:10:12

文章目录

    • 一、实验目的
      • 实验内容
      • 设计思路
    • 三、实验代码实现
    • 四、总结

一、实验目的

理解linux文件系统的内部技术,掌握linux与文件有关的系统调用命令,并在此基础上建立面向随机检索的散列结构文件;## 二、实验内容与设计思想

实验内容

1.设计一组散列文件函数,包括散列文件的创建,打开,关闭,读,写等;
2.编写一个测试程序,通过记录保存,查找,删除等操作,检查上述散列文件是否实现相关功能;

设计思路

  1. 设计散列文件函数
    我设计了一组散列文件函数,涵盖了散列文件的创建、打开、关闭、读和写等操作。这些函数是构建散列结构文件的基础,通过合理的设计和实现,确保了文件操作的高效性和正确性。
  2. 编写测试程序
    为了验证散列文件的功能,我编写了一个测试程序,通过记录的保存、查找和删除等操作,检查散列文件是否能正常工作。这个测试程序就像是一个 “质检员”,帮助我发现并解决代码中可能存在的问题。

三、实验代码实现

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define HASH_TABLE_SIZE 10  // 散列表大小
#define MAX_RECORD_LENGTH 100 // 记录最大长度

typedef struct Record {
    char key[20];             // 记录的关键字
    char data[MAX_RECORD_LENGTH]; // 记录的数据
    struct Record* next;      // 链接到下一个记录(用于处理冲突)
} Record;

typedef struct HashTable {
    Record* table[HASH_TABLE_SIZE]; // 散列表
} HashTable;

// 哈希函数
unsigned int hash(const char* key) {
    unsigned int hashValue = 0;
    while (*key) {
        hashValue = (hashValue << 5) + *key; // 左移 5 位并加上当前字符的 ASCII 值
        key++;
    }
    return hashValue % HASH_TABLE_SIZE;
}

// 创建散列文件
HashTable* createHashTable() {
    HashTable* ht = (HashTable*)malloc(sizeof(HashTable));
    memset(ht, 0, sizeof(HashTable)); // 初始化散列表
    return ht;
}

// 插入记录
void insertRecord(HashTable* ht, const char* key, const char* data) {
    unsigned int index = hash(key);
    Record* newRecord = (Record*)malloc(sizeof(Record));
    strcpy(newRecord->key, key);
    strncpy(newRecord->data, data, MAX_RECORD_LENGTH);
    newRecord->next = ht->table[index]; // 插入链表的头部
    ht->table[index] = newRecord;
}

// 查找记录
Record* findRecord(HashTable* ht, const char* key) {
    unsigned int index = hash(key);
    Record* record = ht->table[index];
    while (record != NULL) {
        if (strcmp(record->key, key) == 0) {
            return record; // 找到记录
        }
        record = record->next;
    }
    return NULL; // 未找到记录
}

// 删除记录
void deleteRecord(HashTable* ht, const char* key) {
    unsigned int index = hash(key);
    Record* record = ht->table[index];
    Record* prev = NULL;
    while (record != NULL) {
        if (strcmp(record->key, key) == 0) {
            if (prev == NULL) {
                ht->table[index] = record->next; // 删除链表头部元素
            } else {
                prev->next = record->next; // 删除中间或尾部元素
            }
            free(record); // 释放内存
            return;
        }
        prev = record;
        record = record->next;
    }
}

// 关闭散列文件(释放内存)
void closeHashTable(HashTable* ht) {
    for (int i = 0; i < HASH_TABLE_SIZE; i++) {
        Record* record = ht->table[i];
        while (record != NULL) {
            Record* temp = record;
            record = record->next;
            free(temp); // 释放每个记录
        }
    }
    free(ht); // 释放哈希表
}

// 测试程序
int main() {
    HashTable* ht = createHashTable();
    
    // 插入记录
    insertRecord(ht, "key1", "data1");
    insertRecord(ht, "key2", "data2");
    insertRecord(ht, "key3", "data3");
    
    // 查找记录
    Record* found = findRecord(ht, "key2");
    if (found) {
        printf("找到记录: %s -> %s\n", found->key, found->data);
    } else {
        printf("未找到记录\n");
    }
    
    // 删除记录
    deleteRecord(ht, "key2");
    found = findRecord(ht, "key2");
    if (found) {
        printf("找到记录: %s -> %s\n", found->key, found->data);
    } else {
        printf("未找到记录\n");
    }
    
    // 关闭哈希表
    closeHashTable(ht);
    
    return 0;
}

结果:
在这里插入图片描述

结果分析:

程序首先插入三条记录:
key1 -> data1
key2 -> data2
key3 -> data3
当程序尝试查找key2时,程序找到了该记录,因此输出:Found record: key2 -> data2
接下来,程序删除了key2记录。
再次查key2 时,记录已经被删除,因此输出:Record not found

四、总结

  • 遇到的问题
    第一次运行时,我输错了编译文件的名字,结果提示权限不够。这让我意识到在操作过程中一定要细心,一个小小的失误都可能导致程序无法正常运行。经过修正后,程序的运行结果符合预期。程序先插入了三条记录,然后成功查找到了 key2 对应的记录,接着删除了该记录,再次查找时就显示未找到记录,这表明散列文件的基本功能已经实现。
    在这里插入图片描述
    为了处理哈希冲突,我使用了链式法。通过链表存储多个记录,保证了散列表的完整性。这让我认识到在设计数据结构时,不仅要考虑基本功能的实现,还要考虑可能出现的异常情况,并采取有效的处理方法。
    在实现散列表的过程中,我更加深入地理解了指针和内存管理的重要性。正确地分配和释放内存是保证程序稳定运行的关键,稍有不慎就可能导致内存泄漏或悬空指针等问题。这也提醒我在今后的编程中要养成良好的内存管理习惯。

这次实验让我对 Linux 文件系统和散列结构文件有了更深入的理解,也提升了我的编程能力和问题解决能力。在今后的学习和工作中,我将继续探索数据结构和算法的奥秘,不断提升自己的技术水平。同时,我也会更加注重细节,避免因为粗心而导致的错误。我相信,通过不断的实践和学习,我能够更好地应对各种编程挑战。

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

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

相关文章

World of Warcraft [CLASSIC][80][Deluyia] [Fragment of Val‘anyr]

瓦兰奈尔的碎片 [Fragment of Valanyr] 有时候下个班打个游戏&#xff0c;没想到套路也这么多&#xff0c;唉&#xff0c;何况现实生活&#xff0c;这一个片版本末期才1000G&#xff0c;30个&#xff0c;也就30000G&#xff0c;时光徽章等同月卡15000G&#xff0c;折合一下也就…

数组和指针典型例题合集(一维数组、字符数组、二维数组)

1.一维数组 数组名的理解 数组名是数组首元素&#xff08;第一个元素&#xff09;的地址 但是有两个例外&#xff1a; 1.sizeof &#xff08;数组名&#xff09;—— 数组名表示整个数组&#xff0c;就算的是整个数组的大小&#xff0c;单位是字节。 2.&数组名 —— 数…

地级市-机器人、人工智能等未来产业水平(2009-2023年)-社科数据

地级市-机器人、人工智能等未来产业水平&#xff08;2009-2023年&#xff09;-社科数据https://download.csdn.net/download/paofuluolijiang/90623814 https://download.csdn.net/download/paofuluolijiang/90623814 此数据集统计了2009-2023年全国地级市在机器人、人工智能等…

epub格式转txt格式工具,txt批量转PDF

epub格式转txt格式工具&#xff0c;功能如图&#xff1a; txt格式批量转PDF 参考原文&#xff1a;epub格式转txt格式工具&#xff0c;txt批量转PDF 轻轻一点就关注, 好运连连挡不住&#xff0c;点个关注吧。

电赛经验分享——模块篇

1、前言 打算在这一个专栏中&#xff0c;分享一些本科控制题电赛期间的经验&#xff0c;和大家共同探讨&#xff0c;也希望能帮助刚刚参加电赛的同学&#xff0c;了解一些基本的知识。一些见解和看法可能不同或有错误&#xff0c;欢迎批评指正。 在本文中&#xff0c;主要介绍笔…

JVM之内存管理(一)

部分内容来源&#xff1a;JavaGuide二哥Java 图解JVM内存结构 内存管理快速复习 栈帧&#xff1a;局部变量表&#xff0c;动态链接&#xff08;符号引用转为真实引用&#xff09;&#xff0c;操作数栈&#xff08;存储中间结算结果&#xff09;&#xff0c;方法返回地址 运行时…

鸿蒙编译boost整合linux跨平台应用

openharmony deveco 4.1支持armeabi-v7a deveco 5.0后不支持arm32位系统 boost编译 使用deveco的写cmake集成boost boost使用1.88的最新版本&#xff0c;带cmake工具链 https://github.com/boostorg/boost.git boost的源码都在sub_module中 deveco 4.1的版本sdk最高到9&am…

rabbitMQ消息问题与解决

rabbitMQ 消息顺序性、消息幂等性、消息不丢失、最终一致性、补偿机制、消息队列设计 1.消息顺序性 溯源&#xff1a; 消息队列中的若干消息如果是对同一个数据进行操作&#xff0c;这些操作具有前后的关系&#xff0c;必须要按前后的顺序执行&#xff0c;否则就会造成数据异常…

Java SE(10)——抽象类接口

1.抽象类 1.1 概念 在之前讲Java SE(6)——类和对象&#xff08;一&#xff09;的时候说过&#xff0c;所有的对象都可以通过类来抽象。但是反过来&#xff0c;并不是说所有的类都是用来抽象一个具体的对象。如果一个类本身没有足够的信息来描述一个具体的对象&#xff0c;而…

学习threejs,使用Physijs物理引擎

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言1.1 ☘️Physijs 物理引擎1.1.1 ☘️…

allure生成测试报告(搭配Pytest、allure-pytest)

文章目录 前言allure简介allure安装软件下载安装配置环境变量安装成功验证 allure运行流程allure装饰器函数基本说明装饰器函数使用allure.attach 命令行运行利用allure-pytest生成中间结果json 查看测试报告总览页面每个tab页的说明类别页面测试套图表页面时间刻度功能页面包 …

龙虎榜——20250509

上证指数今天缩量&#xff0c;整体跌多涨少&#xff0c;走势处于日线短期的高位~ 深证指数今天缩量小级别震荡&#xff0c;大盘股表现更好~ 2025年5月9日龙虎榜行业方向分析 一、核心行业方向 军工航天 • 代表个股&#xff1a;航天南湖、天箭科技、襄阳轴承。 • 驱动逻辑…

操作系统的初步了解

目录 引言&#xff1a;什么是操作系统&#xff1f; 一、设计操作系统的目的 二、操作系统是做什么的&#xff1a; 操作系统主要有四大核心任务&#xff1a; 1. 管理硬件 2. 运行软件 3. 存储数据 4. 提供用户界面 如何理解操作系统的管理呢&#xff1f; 1. 什么是操作…

软件工程之软件项目管理深度解析

前文基础&#xff1a; 1.软件工程学概述&#xff1a;软件工程学概述-CSDN博客 2.软件过程深度解析&#xff1a;软件过程深度解析-CSDN博客 3.软件工程之需求分析涉及的图与工具&#xff1a;软件工程之需求分析涉及的图与工具-CSDN博客 4.软件工程之形式化说明技术深度解…

基于Boost库、Jsoncpp、cppjieba、cpp-httplib等构建Boost搜索引擎

⭐️个人主页&#xff1a;小羊 ⭐️所属专栏&#xff1a;项目 很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~ 目录 项目背景技术栈和项目环境正排索引和倒排索引数据去标签与清洗下载数据源去标签 建立索引构建正排索引构建倒排索引 建立搜索引擎h…

Qt 通过控件按钮实现hello world + 命名规范(7)

文章目录 使用编辑框来完成 hello world通过编辑图形化界面方式通过纯代码方式 通过按钮的方式来创建 hello world通过编辑图形化界面方式通过纯代码方式 总结Qt Creator中的快捷键如何使用文档命名规范 简介&#xff1a;这篇文章着重点并不在于创建hello world程序&#xff0c…

关于大数据的基础知识(二)——国内大数据产业链分布结构

成长路上不孤单&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a; 【14后&#x1f60a;///计算机爱好者&#x1f60a;///持续分享所学&#x1f60a;///如有需要欢迎收藏转发///&#x1f60a;】 今日分享关于大数据的基础知识&#xff08;二&a…

Winform(11.案例讲解1)

今天写两个案例,用于更好的理解控件的使用 在写之前先写一个类 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace _1.案例讲解 { internal class Student { public string …

电机密集型工厂环境下的无线通信技术选型与优化策略

点击下面图片带您领略全新的嵌入式学习路线 &#x1f525;爆款热榜 88万阅读 1.6万收藏 在电机、变频器、电焊机等强电磁干扰源遍布的工业环境中&#xff0c;无线通信系统的可靠性面临严峻挑战。本文从抗干扰能力、传输稳定性、实时性需求三大核心维度出发&#xff0c;结合工…

一文了解氨基酸的分类、代谢和应用

氨基酸&#xff08;Amino acids&#xff09;是在分子中含有氨基和羧基的一类化合物。氨基酸是生命的基石&#xff0c;人类所有的疾病与健康状况都与氨基酸有直接或间接的关系。氨基酸失衡可引起肝硬化、神经系统感染性疾病、糖尿病、免疫性疾病、心血管疾病、肾病、肿瘤等各类疾…