• LeetCode[636]函数的独占时间


    难度:重点

    题目:

    有一个 单线程 CPU 正在运行一个含有 n 道函数的程序。每道函数都有一个位于 0 和 n-1 之间的唯一标识符。

    函数调用 存储在一个 调用栈 上 :当一个函数调用开始时,它的标识符将会推入栈中。而当一个函数调用结束时,它的标识符将会从栈中弹出。标识符位于栈顶的函数是 当前正在执行的函数 。每当一个函数开始或者结束时,将会记录一条日志,包括函数标识符、是开始还是结束、以及相应的时间戳。

     描述:

    给你一个由日志组成的列表 logs ,其中 logs[i] 表示第 i 条日志消息,该消息是一个按 "{function_id}:{"start" | "end"}:{timestamp}" 进行格式化的字符串。例如,"0:start:3" 意味着标识符为 0 的函数调用在时间戳 3 的 起始开始执行 ;而 "1:end:2" 意味着标识符为 1 的函数调用在时间戳 2 的 末尾结束执行。注意,函数可以 调用多次,可能存在递归调用 

    函数的 独占时间 定义是在这个函数在程序所有函数调用中执行时间的总和,调用其他函数花费的时间不算该函数的独占时间。例如,如果一个函数被调用两次,一次调用执行 2 单位时间,另一次调用执行 1 单位时间,那么该函数的 独占时间 为 2 + 1 = 3 。

    以数组形式返回每个函数的 独占时间 ,其中第 i 个下标对应的值表示标识符 i 的函数的独占时间。

    示例 1:

    输入:n = 2, logs = ["0:start:0","1:start:2","1:end:5","0:end:6"]
    输出:[3,4]
    解释:
    函数 0 在时间戳 0 的起始开始执行,执行 2 个单位时间,于时间戳 1 的末尾结束执行。 
    函数 1 在时间戳 2 的起始开始执行,执行 4 个单位时间,于时间戳 5 的末尾结束执行。 
    函数 0 在时间戳 6 的开始恢复执行,执行 1 个单位时间。 
    所以函数 0 总共执行 2 + 1 = 3 个单位时间,函数 1 总共执行 4 个单位时间。 
    

     示例 2:

    输入:n = 1, logs = ["0:start:0","0:start:2","0:end:5","0:start:6","0:end:6","0:end:7"]
    输出:[8]
    解释:
    函数 0 在时间戳 0 的起始开始执行,执行 2 个单位时间,并递归调用它自身。
    函数 0(递归调用)在时间戳 2 的起始开始执行,执行 4 个单位时间。
    函数 0(初始调用)恢复执行,并立刻再次调用它自身。
    函数 0(第二次递归调用)在时间戳 6 的起始开始执行,执行 1 个单位时间。
    函数 0(初始调用)在时间戳 7 的起始恢复执行,执行 1 个单位时间。
    所以函数 0 总共执行 2 + 4 + 1 + 1 = 8 个单位时间。
    

     示例 3:

    输入:n = 2, logs = ["0:start:0","0:start:2","0:end:5","1:start:6","1:end:6","0:end:7"]
    输出:[7,1]
    解释:
    函数 0 在时间戳 0 的起始开始执行,执行 2 个单位时间,并递归调用它自身。
    函数 0(递归调用)在时间戳 2 的起始开始执行,执行 4 个单位时间。
    函数 0(初始调用)恢复执行,并立刻调用函数 1 。
    函数 1在时间戳 6 的起始开始执行,执行 1 个单位时间,于时间戳 6 的末尾结束执行。
    函数 0(初始调用)在时间戳 7 的起始恢复执行,执行 1 个单位时间,于时间戳 7 的末尾结束执行。
    所以函数 0 总共执行 2 + 4 + 1 = 7 个单位时间,函数 1 总共执行 1 个单位时间。 

     示例 4:

    输入:n = 2, logs = ["0:start:0","0:start:2","0:end:5","1:start:7","1:end:7","0:end:8"]
    输出:[8,1]
    

     示例 5:

    输入:n = 1, logs = ["0:start:0","0:end:0"]
    输出:[1]
    

    提示:

    • 1 <= n <= 100
    • 1 <= logs.length <= 500
    • 0 <= function_id < n
    • 0 <= timestamp <= 109
    • 两个开始事件不会在同一时间戳发生
    • 两个结束事件不会在同一时间戳发生
    • 每道函数都有一个对应 "start" 日志的 "end" 日志

     Related Topics

    • 数组

    重点!!!解题思路 

    第一步:

    看例子就可以知道这道题是使用栈来实现的!!!

    给的是一个集合,我们遍历之后是一个字符串,但是这个字符串中是用 :来分隔的

    所以我们分隔之后肯定是一个数组

    数组的第一个下标是函数的id,第二个下标是函数开始结束的标记,第三个下标是调用时间

    第二步:

     只要能想明白上述结论,这道题你就差不多能做出来了

    源码+讲解:

    1. class Solution {
    2. //自定义一个函数用来存储每个字符串
    3. class Task{
    4. int id;
    5. int time;
    6. boolean isStart;
    7. public Task(String[] split){
    8. this.id=Integer.parseInt(split[0]);
    9. this.time=Integer.parseInt(split[2]);
    10. this.isStart=split[1].equals("start");
    11. }
    12. }
    13. public int[] exclusiveTime(int n, List logs) {
    14. Stack stack = new Stack<>();
    15. int[] ans=new int[n];
    16. for (String log:logs){
    17. Task task = new Task(log.split(":"));
    18. if (task.isStart){
    19. stack.push(task);
    20. }else {
    21. Task last = stack.pop();
    22. int time=task.time-last.time+1;//这里时间不要忘记+1
    23. ans[task.id]+=time; //这里一定是+=,因为如果你函数是穿插调用的,你必须加到之前那里去
    24. if (!stack.isEmpty()){ //把重叠的时间减去
    25. ans[stack.peek().id]-=time;
    26. }
    27. }
    28. }
    29. return ans;
    30. }
    31. }

     运行结果:

    如果您还有什么疑问或解答有问题,可在下方评论,我会及时回复。 

    系列持续更新中,点个订阅吧

  • 相关阅读:
    ArduinoUNO实战-第五章-有源蜂鸣器实验
    猿创征文 |【算法面试入门必刷】动态规划-线性dp(三)
    5_spring-cloud-zuul-网关
    Flume监听端口数据
    图神经网络笔记
    网络编程套接字
    解决ERROR: No query specified的错误以及\G 和 \g 的区别
    Pycharm中配置服务器中的Jupyter并使用
    react-生命周期与组件传参
    【PostgreSQL高可用之Repmgr和Patroni部分场景对比】
  • 原文地址:https://blog.csdn.net/m0_57209427/article/details/127603254