算法题目合集(持续更新.......)

最近更新时间:2023.11.08 yk 上海

文章目录

  • 前言
  • 题目:老板的作息表
    • 输入格式
    • 输出格式
    • 输入样例
    • 输出样例
  • 二、解题思路
    • 1.存储结构
    • 2.读入数据
    • 3.数据的处理
        • Ⅰ.时间字符串转换为数轴上的点并标记
        • Ⅱ.拓展---sscanf函数(http://t.csdnimg.cn/wHgO0)
        • Ⅲ.读出未被标记时间段
    • 4.最终代码
  • 总结


提示:以下是本篇文章正文内容,如有转载请标识


前言

考虑到有效的整理可以提高算法学习的效率,特整理出自2023年11月8日起个人有关算法学习或比赛中出现的错题.谨以此专题纪念那些我流过的"泪水"!!!


题目:老板的作息表

在这里插入图片描述
新浪微博上有人发了某老板的作息时间表,表示其每天 4:30 就起床了。但立刻有眼尖的网友问:这时间表不完整啊,早上九点到下午一点干啥了?

本题就请你编写程序,检查任意一张时间表,找出其中没写出来的时间段。


输入格式

输入第一行给出一个正整数 N,为作息表上列出的时间段的个数。随后 N 行,每行给出一个时间段,格式为:

hh:mm:ss - hh:mm:ss

其中 hh、mm、ss 分别是两位数表示的小时、分钟、秒。第一个时间是开始时间,第二个是结束时间。题目保证所有时间都在一天之内(即从 00:00:00 到 23:59:59);每个区间间隔至少 1 秒;并且任意两个给出的时间区间最多只在一个端点有重合,没有区间重叠的情况。


输出格式

按照时间顺序列出时间表中没有出现的区间,每个区间占一行,格式与输入相同。题目保证至少存在一个区间需要输出。


输入样例

8
13:00:00 - 18:00:00
00:00:00 - 01:00:05
08:00:00 - 09:00:00
07:10:59 - 08:00:00
01:00:05 - 04:30:00
06:30:00 - 07:10:58
05:30:00 - 06:30:00
18:00:00 - 19:00:00

输出样例

04:30:00 - 05:30:00
07:10:58 - 07:10:59
09:00:00 - 13:00:00
19:00:00 - 23:59:59

二、解题思路

1.存储结构

将一天24小时化为一个时间轴,则输入的时间段将被标记,则从时间轴的起点遍历,输出未被标记的时间段即为本题所求。

在这里插入图片描述
由此观察时间轴为线性结构,因此可以采用数组存储,而此数组空间大小也应该为24x60x60-1即时间轴最小分度为一秒.

不用直接用235959的原因是其中较多数据不属于时间范围内,如235899等等

const int N = 24*60*60-1;
bool st[N];//时间轴上的点用作标记

2.读入数据

由于本题每行需要读入两个时间,而考虑到后序操作需要分别对这两个时间进行操作,因此需要将两个时间分别读入.

string a,c,b;//a、b:第一、二个时间 c: 中间的'-' 
cin >> a >> c >> b;//读入数据

3.数据的处理

本题需要处理的数据为:

①上述的两个时间字符串转换为时间轴上对应的点
②标记时间轴中处于上述两个时间段之间的所有点
③读出未被标记时间段


Ⅰ.时间字符串转换为数轴上的点并标记

手段:借助get函数:将时间a、b转换为整数型

int l = get(a),r=get(b);

实现:

int get(string s)
{
    int hour,minute,second;
    sscanf(s.c_str(),"%d:%d:%d",&hour,&minute,&second);
    return hour*3600+minute*60+second;
}
//s.c_str():将string转换为char*,否则无法读入数据

标记:

for(int i=l;i<r;i++) st[i]=true;

值得一提的是sscanf函数强大的格式化输入功能:

Ⅱ.拓展—sscanf函数(http://t.csdnimg.cn/wHgO0)
Ⅲ.读出未被标记时间段

借助format函数将时间轴上的点转换为要求格式的字符串

    for(int i=0;i<N;i++)
    {
        if(!st[i])
        {
            int j = i+1;
            while(j<N && !st[j]) j++;
            cout << format(i) << " - " << format(j) << endl;
            i=j;
        }
    }

fomat函数

string format(int x)
{
    int hour = x/3600;
    x%=3600;
    int minute = x/60;
    int second = x%60;
    char str[10];
    sprintf(str,"%02d:%02d:%02d",hour,minute,second);
    return str;
}

值得一提的是sprintf函数强大的格式化输出功能:

4.最终代码

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 24*60*60-1;
bool st[N];
int n;
int get(string s)
{
    int hour,minute,second;
    sscanf(s.c_str(),"%d:%d:%d",&hour,&minute,&second);
    return hour*3600+minute*60+second;
}
string format(int x)
{
    int hour = x/3600;
    x%=3600;
    int minute = x/60;
    int second = x%60;
    char str[10];
    sprintf(str,"%02d:%02d:%02d",hour,minute,second);
    return str;
}
int main()
{
    cin >> n;
    while(n--)
    {
        string a,c,b;
        cin >> a >> c >> b;
        int l = get(a),r=get(b);
        for(int i=l;i<r;i++) st[i]=true;
    }
    for(int i=0;i<N;i++)
    {
        if(!st[i])
        {
            int j = i+1;
            while(j<N && !st[j]) j++;
            cout << format(i) << " - " << format(j) << endl;
            i=j;
        }
    }
    return 0;
}

总结

未完待续…(2023.11.08)

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值