当前位置:   article > 正文

C数据结构与算法——顺序查找和二分查找算法 应用_c语言数据结构实验报告查找算法的应用

c语言数据结构实验报告查找算法的应用

实验任务

(1) 掌握顺序查找算法的实现;
(2) 掌握二分查找算法的实现;
(3) 掌握两种查找算法的时间复杂度与适用场合。

实验内容

(1) 基于顺序查找表实现顺序查找和二分查找算法;
(2) 使用两个不同大小的查找表进行两次理论和实际性能对比;
(3) 根据实验环境调整两次查找表的大小,使得:
      i.第1次查找表的长度 = ⌊ 第2次查找表长度 / 10 ⌋;
      ii.第2次顺序查找的时间介于 [ 9000, 9999 ] ms之间。
(4) 输出实验结果,结果应与如下图1所示尽可能相似。

实验源码

注意:由于编译器优化或电脑编程环境缓存等因素,实际运行和理论值可能有差距

//
// Created by Lenovo on 2022-05-27-上午 9:14.
// 作者:小象
// 版本:1.0
//

#include <malloc.h>
#include <stdio.h>
#include <math.h>
#include <pthread_time.h>
#include <profileapi.h>

#define MAXSIZE 2147483 // int型0~21亿 顺序表可能达到的最大长度10位
#define OK 1
#define ERROR 0

// 数据元素类型定义
typedef struct {
    int key; // 关键字域
    int otherInfo; // 其他域
} ElemType;

// 顺序表定义
typedef struct {
    ElemType *elem; // 存储空间的基地址
    int length; // 当前长度
} SqSTable;

int InitTable(SqSTable *table); // 顺序表的初始化
int CreateTable(SqSTable *table, int length); // 顺序表的创建
double SearchSeq(SqSTable table, int key); // 顺序查找
double SearchSeq_Times(SqSTable table); // 顺序查找完 key 个关键字耗时
double SearchBin(SqSTable table, int key); // 二分查找
double SearchBin_Times(SqSTable table); // 二分查找完 key 个关键字耗时
void PrintResult(int data, double sequentialTimes, double binaryTimes); // 打印最终结果

/**
 * <h2>二分查找 和 顺序查找算法 实验一</h2>
 * <h3>实验要求:</h3>
 * i.第1次查找表的长度 = 第2次查找表长度 / 10; <br>
 * ii.第2次顺序查找的时间介于 [ 9000, 9999 ] ms之间。 <br>
 * @return 0
 */
int main() {

    SqSTable table;
    if (!(InitTable(&table))) {
        printf("空间申请失败!!!");
    }
    // 两种查找方式的所查找的数据量
    int test_01_data = 9666;
    int test_02_data = test_01_data * 10;
    /*
     * 第一次测试
     */
    printf("=========== 第1次测试,数据量:%6d个 ===========\n", test_01_data);
    // 创建顺序表
    CreateTable(&table, test_01_data);
    // 顺序查找
    double sequentialTimes_01 = SearchSeq_Times(table);
    // 二分查找
    double binaryTimes_01 = SearchBin_Times(table);
    // 查找结果
    PrintResult(test_01_data, sequentialTimes_01, binaryTimes_01);


    /*
     * 第二次测试
     */
    printf("=========== 第2次测试,数据量:%6d个 ===========\n", test_02_data);
    // 创建顺序表
    CreateTable(&table, test_02_data);
    // 顺序查找
    double sequentialTimes_02 = SearchSeq_Times(table);
    // 二分查找
    double binaryTimes_02 = SearchBin_Times(table);
    // 查找结果
    PrintResult(test_02_data, sequentialTimes_02, binaryTimes_02);

    // 测试环境
    printf("===== 测试环境:Windows 11,i7-9750HF@4.12Ghz =====");

    getchar();
}

// 构造一个空的顺序表
int InitTable(SqSTable *table) {
    table->elem = (ElemType *) malloc(sizeof(SqSTable) * MAXSIZE); // 动态方式需要先分配内存,而且
    // 需要用到malloc函数申请,有可能申请不到,申请到后malloc()函数会把申请到的连续数据空间首地址返回
    // 注意:这里申请到返回的地址是十六进制的地址,而elem只能存十进制的地址,所以需要强转为十进制地址后赋值
    if (table->elem == NULL) { // 如果没有申请到地址,退出
        return ERROR;
    }
    table->length = 0; // 空表长度为 0
    return OK; // 初始化成功
}

// 顺序表的创建
int CreateTable(SqSTable *table, int length) {
    for (int i = 1; i <= length; i++) { // 第一个数默认不使用
        table->elem[i].key = i;
        table->length++; // 表长加 1
    }
    return OK;
}

// 顺序查找
double SearchSeq(SqSTable table, int key) {
    int i;
    table.elem[0].key = key; // 第一个数的作用来了 -> 设置 “哨兵”
    for (i = table.length; table.elem[i].key != key; i--); // 从后往前找
}

// 顺序查找测时
double SearchSeq_Times(SqSTable table) {
    /*
     * 时间测试方式二:
     *      所用精度:纳秒级
     */
    struct timespec start;
    struct timespec end;
    // 开始时间
    clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &start);

    for (int i = 1; i <= table.length; i++) { // 所有数都查找一遍
        SearchSeq(table, i);
    }

    // 结束时间
    clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &end);
    // 总耗时
    // 转化为 ms 为单位(但是精度可以直接到 ns 级别)
    double start_ms = start.tv_sec * 1000.0 + start.tv_nsec / 1000000.0;
    double end_ms = end.tv_sec * 1000.0 + end.tv_nsec / 1000000.0;
    return end_ms - start_ms;
}

// 在有序表table中折半查找其关键字等于key的数据元素。若找到,则函数值为该元素在表中的位置,否则为0
double SearchBin(SqSTable table, int key) {
    // 置 查找区间 初值
    int leftIndex = 1; // 第一个数默认不使用
    int rightIndex = table.length;
    while (leftIndex <= rightIndex) { // 循环限制条件:查找过程中如果没有找到的情况,会造成栈溢出
        int minIndex = (leftIndex + rightIndex) / 2; // 二分查找的精髓:先找到中间那个数
        if (key == table.elem[minIndex].key) {
            // 如果需要查找的数等于中间那个数,则直接输出中间那个数(即需要查找的数的)下标
            return minIndex; // 找到待查元素
        } else if (key < table.elem[minIndex].key) {
            // 如果需要查找的数小于中间那个数,说明需要从中间那个数的左边第一个数继续进行二分查找,其他不变
            rightIndex = minIndex - 1;
        } else {
            // 如果需要查找的数大于中间那个数,说明需要从中间那个数的右边第一个数继续进行二分查找,其他不变
            leftIndex = minIndex + 1;
        }
    }
    return -1; // 如果没有找到返回-1
}

// 二分查找测时
double SearchBin_Times(SqSTable table) {
    /*
     * 时间测试方式二:
     *      所用精度:微秒级
     */
    union _LARGE_INTEGER time_start; // 开始时间
    union _LARGE_INTEGER time_over; // 结束时间
    LARGE_INTEGER f; // 计时器频率
    QueryPerformanceFrequency(&f);
    double dqFreq = (double) f.QuadPart; // 计时器频率
    QueryPerformanceCounter(&time_start); // 计时开始

    for (int i = 1; i <= table.length; i++) { // 所有数都查找一遍
        SearchBin(table, i);
    }

    QueryPerformanceCounter(&time_over); // 计时结束
    // 乘以1000000把单位由秒化为微秒,精度为1000 000/(cpu主频)微秒
    double run_time = 1000000.0 * (time_over.QuadPart - time_start.QuadPart) / dqFreq;
    return run_time / 1000.0;
}

void PrintResult(int data, double sequentialTimes, double binaryTimes) {

    // 理论ASL
    double sequentialASL = (data + 1) / 2.0;
    double binaryASL = log2(data + 1) - 1;

    printf(".--------.\n");
    printf("|理论性能'----------------------------------------.\n");
    printf("|ALS成功 : 顺序查找 = %-7.1f   二分查找 = %-4.1f |\n", sequentialASL, binaryASL);
    printf("|次数之比:   顺序查找  :  二分查找  = %7.2f : 1 |\n", sequentialASL / binaryASL);
    printf("'-------------------------------------------------'\n");

    printf(".--------.\n");
    printf("|实际性能'----------------------------------------.\n");
    printf("|ALS成功 : 顺序查找 %-7.2f ms  二分查找%-4.2f ms |\n", sequentialTimes, binaryTimes);
    printf("|次数之比:   顺序查找  :  二分查找  = %7.2f : 1 |\n", sequentialTimes / binaryTimes);
    printf("'-------------------------------------------------'\n");

}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 174
  • 175
  • 176
  • 177
  • 178
  • 179
  • 180
  • 181
  • 182
  • 183
  • 184
  • 185
  • 186
  • 187
  • 188
  • 189
  • 190
  • 191
  • 192
  • 193
  • 194
  • 195
  • 196
  • 197
  • 198
  • 199
  • 200

实验结果

在这里插入图片描述

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/酷酷是懒虫/article/detail/807590
推荐阅读
相关标签
  

闽ICP备14008679号