Leecode刷题C语言之我的日程安排表②

news2025/1/6 9:28:12

执行结果:通过

执行用时和内存消耗如下:

 

 

 


typedef struct {
    int start;
    int end;
}BOOKING;

#define MAX_BOOK_NUM (1000)
typedef struct MyCalendar_ {
    BOOKING book[MAX_BOOK_NUM];
    int bnum;
    BOOKING *sorted[MAX_BOOK_NUM];
    int num;
    int conflict[MAX_BOOK_NUM];
    int cnum;
    BOOKING cbook[MAX_BOOK_NUM];
    struct MyCalendar_ *next;
} MyCalendar;

#define MAX_OVERLAP_NUM (2)
typedef struct {
    MyCalendar calendar[MAX_OVERLAP_NUM];
} MyCalendarTwo;


MyCalendarTwo* myCalendarTwoCreate() {
    MyCalendarTwo* obj = (MyCalendarTwo*)malloc(sizeof(MyCalendarTwo));
    memset(obj, 0, sizeof(MyCalendarTwo));
    
    int i;
    for (i = 0; i < MAX_OVERLAP_NUM - 1; i++) {
        obj->calendar[i].next = &obj->calendar[i+1];
    }
    obj->calendar[i].next = NULL;
    
    return obj;
}

void myCalendarInsert(MyCalendar *obj, int pos, int start, int end) {
    assert(pos <= obj->num);
    
    BOOKING *b = &obj->book[obj->bnum++];
    b->start = start;
    b->end = end;
    
    memmove(&obj->sorted[pos+1], &obj->sorted[pos], sizeof(BOOKING *)*(obj->num-pos));
    obj->sorted[pos] = b;
    obj->num++;
}

void myCalendarRemove(MyCalendar *obj, int pos, int num) {
    assert(pos >= 0);
    assert(pos + num <= obj->num);
    
    int size = obj->num - pos - num;
    memmove(&obj->sorted[pos], &obj->sorted[pos+num], sizeof(BOOKING *)*size);
    obj->num -= num;
}

bool myCalendarCheck(MyCalendar *obj, int start, int end) {
    if (!obj->num) {
        return true;
    }
    
    int left = 0;
    int right = obj->num - 1;
    if (end <= obj->sorted[left]->start) {
        return true;
    }
    
    if (start >= obj->sorted[right]->end) {
        return true;
    }
    
    while(left < right) {
        int mid = left + (right - left) / 2;
        if (start > obj->sorted[mid]->start) {
            left = mid + 1;
        }
        else {
            right = mid;
        }
    }
    
    if (end > obj->sorted[left]->start) {
        return false;
    }
    
    if (left - 1 >= 0 && obj->sorted[left-1]->end > start) {
        return false;
    }
    
    return true;
}

void myCalendarBookInternal(MyCalendar *obj, int start, int end) {
    if (!obj->num) {
        myCalendarInsert(obj, 0, start, end);
        return;
    }
    
    int left = 0;
    int right = obj->num - 1;
    if (end <= obj->sorted[left]->start) {
        myCalendarInsert(obj, 0, start, end);
        return;
    }
    
    if (start >= obj->sorted[right]->end) {
        myCalendarInsert(obj, obj->num, start, end);
        return;
    }
    
    while(left < right) {
        int mid = left + (right - left) / 2;
        if (start > obj->sorted[mid]->start) {
            left = mid + 1;
        }
        else {
            right = mid;
        }
    }
    
    myCalendarInsert(obj, left, start, end);
    return;
}

bool myCalendarBook(MyCalendar *obj, int start, int end) {
    if (!myCalendarCheck(obj->next, start, end)) {
        return false;
    }
    
    if (!obj->num) {
        myCalendarInsert(obj, obj->num, start, end);
        return true;
    }
    
    int left = 0;
    int right = obj->num - 1;
    if (end <= obj->sorted[left]->start) {
        myCalendarInsert(obj, 0, start, end);
        return true;
    }
    
    if (start >= obj->sorted[right]->end) {
        myCalendarInsert(obj, obj->num, start, end);
        return true;
    }
    
    if (start >= obj->sorted[right]->start) {
        left = obj->num;
    }
    else if (start <= obj->sorted[0]->start) {
        left = 0;
    }
    else {
        while(left < right) {
            int mid = left + (right - left) / 2;
            if (start > obj->sorted[mid]->start) {
                left = mid + 1;
            }
            else {
                right = mid;
            }
        }
    }
    
    int ustart = start;
    int uend = end;
    
    obj->cnum = 0;
    int pos = left;
    if (left - 1 >= 0 && obj->sorted[left-1]->end > start) {
        BOOKING *b = obj->sorted[left-1];
        ustart = ustart < b->start ? ustart : b->start;
        uend = uend > b->end ? uend : b->end;
        
        int nstart = start;
        int nend = b->end < end ? b->end : end;
        
        pos--;
        obj->cbook[obj->cnum].start = nstart;
        obj->cbook[obj->cnum].end = nend;
        obj->conflict[obj->cnum++] = left - 1;
    }
    
    for (int i = left; i < obj->num; i++) {
        BOOKING *b = obj->sorted[i];
        if (end <= b->start) {
            break;
        }
        
        int nstart = b->start;
        int nend = b->end < end ? b->end : end;
        ustart = ustart < b->start ? ustart : b->start;
        uend = uend > b->end ? uend : b->end;
        
        obj->cbook[obj->cnum].start = nstart;
        obj->cbook[obj->cnum].end = nend;
        obj->conflict[obj->cnum++] = i;
    }
        
    for (int i = 0; i < obj->cnum; i++) {
        BOOKING *b = &obj->cbook[i];
        myCalendarBookInternal(obj->next, b->start, b->end);
    }
    
    myCalendarRemove(obj, pos, obj->cnum);
    myCalendarInsert(obj, pos, ustart, uend);
    return true;
}

bool myCalendarTwoBook(MyCalendarTwo* obj, int start, int end) {
    MyCalendar *c = &obj->calendar[0];
    bool success = myCalendarBook(c, start, end);
    
    return success;
}

void myCalendarTwoFree(MyCalendarTwo* obj) {
    free(obj);
}

解题思路:

这段代码实现了一个二级日历预约系统,允许用户在不同的日历上预约时间段,并检查预约是否存在冲突。下面是代码的详细思路:

数据结构设计

  1. BOOKING 结构体
    • 存储单个预约的起始时间 start 和结束时间 end
  2. MyCalendar 结构体
    • book 数组:存储所有预约的详细信息。
    • bnum:当前已存储的预约数量。
    • sorted 指针数组:存储指向 book 数组中预约的指针,这些预约按起始时间排序。
    • numsorted 数组中存储的预约数量。
    • conflict 数组:用于存储冲突预约在 sorted 数组中的索引。
    • cnum:当前冲突的数量。
    • cbook 数组:用于临时存储冲突预约的合并结果。
    • next 指针:指向下一个 MyCalendar 实例,用于实现二级日历系统。
  3. MyCalendarTwo 结构体
    • calendar 数组:存储多个 MyCalendar 实例,实现二级日历系统。这里只使用了两个日历的索引空间(MAX_OVERLAP_NUM 定义为 2),但实际上可以扩展以支持更多日历。

函数实现

  1. myCalendarTwoCreate 函数
    • 创建一个 MyCalendarTwo 实例。
    • 初始化所有成员变量。
    • 将 MyCalendar 实例链接成一个单向链表(尽管这里只使用了两个实例,链表的概念仍然适用)。
  2. myCalendarInsert 函数
    • 在 sorted 数组的指定位置插入一个新的预约。
    • 使用 memmove 函数移动元素以腾出空间。
  3. myCalendarRemove 函数
    • 从 sorted 数组中删除指定数量的预约。
    • 使用 memmove 函数移动元素以覆盖被删除的元素。
  4. myCalendarCheck 函数
    • 检查一个新的预约是否与现有的预约冲突。
    • 使用二分查找提高查找效率。
  5. myCalendarBookInternal 函数
    • 在不考虑冲突检查的情况下,将一个预约插入到 MyCalendar 实例中。
    • 使用二分查找确定插入位置。
  6. myCalendarBook 函数
    • 检查并尝试在 MyCalendar 实例中预约一个时间段。
    • 如果存在冲突,则尝试在下一级日历(next 指向的日历)中预约冲突部分。
    • 更新当前日历中的预约,合并冲突预约并删除旧的冲突预约。
  7. myCalendarTwoBook 函数
    • 在二级日历系统中预约一个时间段。
    • 目前只使用了第一个 MyCalendar 实例进行预约。
  8. myCalendarTwoFree 函数
    • 释放 MyCalendarTwo 实例所占用的内存。

总结

这段代码实现了一个复杂的二级日历预约系统,具有以下特点:

  • 支持在多个级别上进行预约。
  • 使用二分查找提高查找效率。
  • 能够处理预约冲突,并尝试在下一级日历中预约冲突部分。
  • 提供了创建、预约和释放资源的接口。

然而,代码中存在一些潜在的问题和改进点:

  • MyCalendar 实例之间的链表连接仅用于实现二级日历的概念,但实际上并未充分利用这一结构。在当前的实现中,只使用了第一个 MyCalendar 实例。
  • 内存管理需要谨慎处理,特别是在释放资源时,要确保不会泄露内存或访问已释放的内存。
  • 代码的可读性和可维护性可以通过更好的注释和重构来提高。

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

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

相关文章

【C语言的小角落】--- 深度理解取余/取模运算

Welcome to 9ilks Code World (๑•́ ₃ •̀๑) 个人主页: 9ilk (๑•́ ₃ •̀๑) 文章专栏&#xff1a; C语言的小角落 本篇博客我们来深度理解取余/取模&#xff0c;以及它们在不同语言中出现不同现象的原因。 &#x1f3e0; 关于取整 &#x1f3b5; 向0取整…

网关的主要类型和它们的特点

网关&#xff0c;作为网络通信的关键节点&#xff0c;根据其应用场景和功能特点&#xff0c;可以分为多种类型。 1.协议网关 特点&#xff1a; • 协议转换&#xff1a;协议网关的核心功能是转换不同网络之间的通信协议。例如&#xff0c;它可以将IPv4协议的数据包转换为IPv6协…

聆听音乐 1.5.9 | 畅听全网音乐,支持无损音质下载

聆听音乐手机版是面向广大音乐爱好者的移动应用程序&#xff0c;用户可以随时随地通过手机享受丰富的音乐资源。它提供了多种魅力功能&#xff0c;让用户在手机上畅享更舒适的音乐体验&#xff0c;每位用户都能享受精彩纷呈的收听体验。此外&#xff0c;软件还支持无损音质音乐…

实现一个通用的树形结构构建工具

文章目录 1. 前言2. 树结构3. 具体实现逻辑3.1 TreeNode3.2 TreeUtils3.3 例子 4. 小结 1. 前言 树结构的生成在项目中应该都比较常见&#xff0c;比如部门结构树的生成&#xff0c;目录结构树的生成&#xff0c;但是大家有没有想过&#xff0c;如果在一个项目中有多个树结构&…

“善弈者”也需妙手,Oclean欧可林:差异化不是说说而已

作者 | 曾响铃 文 | 响铃说 俗话说&#xff0c;“牙痛不是病&#xff0c;痛起来要人命”。这话意思大家都知道&#xff0c;牙痛虽不是什么大病&#xff0c;可一旦发作却是极难忍受。 前几日&#xff0c;Oclean欧可林举办了一场AirPump A10氧气啵啵冲牙器新品品鉴会&#xff…

汇编语言与接口技术--跑马灯

一、 实验要求 在单片机开发板的LED灯D1~D8上实现跑马灯。LED与单片机引脚连线电路如下图: 单片机芯片选择AT89C51&#xff0c;晶振频率设为12MHz&#xff0c;操作参考单片机开发板使用说明。跑马灯点亮的时间间隔约为1秒。分别用定时器的模式1和模式2实现。&#xff08;用P83…

基于数据融合的智能家居环境监测系统研究与设计(论文+源码)

1总体方案设计 本次基于数据融合的智能家居环境监测系统的设计&#xff0c;其系统总体架构如图2.1所示&#xff0c;整个系统在器件上包括了主控制器STM32F103单片机&#xff0c;MQ可燃气体传感器&#xff0c;光照传感器&#xff0c;DHT11温湿度传感器&#xff0c;风扇&#xff…

Mac中配置Node.js前端vscode环境(第二期)

核心组件&#xff1a;vscode、谷歌浏览器、Node.js&#xff08;重点&#xff09;、git 一、Node.js安装&#xff08;nvm安装&#xff09; 点击macos中的终端&#xff0c;保持bash&#xff0c;而不是zsh 若为zsh&#xff0c;则可在终端中使用下面命令变成bash chsh -s /bin/…

基于SpringBoot和Thymeleaf的仿小米电商系统源码下载与安装指南-幽络源

项目概述 这是一个基于 Spring Boot 2.X 和 Thymeleaf 技术栈的仿小米电商系统。该项目包括了前台商城系统和后台管理系统&#xff0c;经幽络源测试具备完整的电商功能&#xff0c;适合用作学习、参考或作为开发电商系统的基础。 前台商城系统&#xff1a; 包括首页登录、商品…

Deepseek v3 的笔记

基本概述 Deepseek v3是Deepseek发布的旗舰模型&#xff0c;属于607B的混合专家&#xff08;MoE&#xff09;模型&#xff0c;其中活跃参数为37B。在当前的模型领域&#xff0c;它堪称最佳的开源模型&#xff0c;性能超越了Llama 3.1 405b、Qwen和Mistral等知名模型。根据基准…

机器学习中的常用特征选择方法及其应用案例

工业界中&#xff0c;特征选择是机器学习模型开发的重要步骤&#xff0c;可以提高模型性能、减少过拟合和加速计算。针对类别型特征和数值型特征&#xff0c;特征选择方法有所不同。以下详细讲解方法和案例。 一、类别型特征的特征选择 1. 基于卡方检验 卡方检验是一种衡量两…

Pyecharts SVG 标记使用笔记

Pyecharts SVG 标记使用笔记 在数据可视化中&#xff0c;图表的标记点常常用于突出显示重要数据点。Pyecharts 提供了丰富的标记点配置选项&#xff0c;其中使用 SVG 路径来自定义标记点的样式是一个非常强大的功能。本文将详细介绍如何在 Pyecharts 中使用 SVG 路径来定制标记…

Spring SpEL表达式由浅入深

标题 前言概述功能使用字面值对象属性和方法变量引用#this 和 #root变量获取类的类型调用对象(类)的方法调用类构造器类型转换运算符赋值运算符条件(关系)表达式三元表达式Elvis 操作符逻辑运算instanceof 和 正则表达式的匹配操作符 安全导航操作员数组集合(Array 、List、Map…

大型模型推理加速入门

一 推理加速基础 1. 了解基本概念与术语 首先需要了解关于大模型的一些基本的历史发展&#xff0c;概念和术语&#xff0c;不需要全部看懂&#xff0c;只需要掌握大概情况&#xff0c;可以通过下面这篇文章进行&#xff1a; 大模型入门基本概念&#xff0c;术语 2. 了解Tra…

仓库叉车高科技安全辅助设备——AI防碰撞系统N2024G-2

在当今这个高效运作、安全第一的物流时代&#xff0c;仓库作为供应链的中心地带&#xff0c;其安全与效率直接关系到企业的命脉。 随着科技的飞速发展&#xff0c;传统叉车作业模式正逐步向智能化、安全化转型&#xff0c;而在这场技术革新中&#xff0c;AI防碰撞系统N2024G-2…

python-redis访问指南

Redis&#xff08;Remote Dictionary Server&#xff09;是一种开源的内存数据结构存储&#xff0c;可用作数据库、缓存和消息代理。它功能强大且灵活&#xff0c;可根据需求调整架构和配置&#xff0c;以高性能、简单易用、支持多种数据结构而闻名&#xff0c;广泛应用于各种场…

Flash Attention V3使用

Flash Attention V3 概述 Flash Attention 是一种针对 Transformer 模型中注意力机制的优化实现&#xff0c;旨在提高计算效率和内存利用率。随着大模型的普及&#xff0c;Flash Attention V3 在 H100 GPU 上实现了显著的性能提升&#xff0c;相比于前一版本&#xff0c;V3 通…

UE5失真材质

渐变材质函数&#xff1a;RadialGradientExponential&#xff08;指数径向渐变&#xff09; 函数使用 UV 通道 0 来产生径向渐变&#xff0c;同时允许用户调整半径和中心点偏移。 用于控制渐变所在的位置及其涵盖 0-1 空间的程度。 基于 0-1 的渐变中心位置偏移。 源自中心的径…

Ansys Aqwa 中 Diffraction Analysis 的疲劳结果

了解如何执行疲劳分析&#xff0c;包括由 Ansys Aqwa 计算的海浪行为。 了解疲劳分析 大多数机器故障是由于负载随时间变化&#xff0c;而不是静态负载。这种失效通常发生在应力水平明显低于材料的屈服强度时。因此&#xff0c;当存在动态载荷时&#xff0c;仅依赖静态失效理…

MT8788安卓核心板_MTK8788核心板参数_联发科模块定制开发

MT8788安卓核心板是一款尺寸为52.5mm x 38.5mm x 2.95mm的高集成度电路板&#xff0c;专为各种智能设备应用而设计。该板卡整合了处理器、图形处理单元(GPU)、LPDDR3内存、eMMC存储及电源管理模块&#xff0c;具备出色的性能与低功耗特性。 这款核心板搭载了联发科的MT8788处理…