评论转换输出 - 华为OD统一考试

OD统一考试

分值: 200分

题解: Java / Python / C++

alt

题目描述

在一个博客网站上,每篇博客都有评论。每一条评论都是一个非空英文字母字符串。

评论具有树状结构,除了根评论外,每个评论都有一个父评论。当评论保存时,使用以下格式:

首先是评论的内容;

然后是回复当前评论的数量。

最后是当前评论的所有子评论。(子评论使用相同的格式嵌套存储)

所有元素之间都用单个逗号分隔。

例如,如果评论如下:

image-20240112161004444

第一条评论是"hello,2,ok,0,bye,0",第二条评论是"test,0",第三条评论是"one,1,two,1,a,0"。所有评论被保存成"hello,2,ok,0,bye,0,test,0,one,1,two,1,a,0"。

对于上述格式的评论,请以另外一种格式打印:

首先打印评论嵌套的最大深度。

然后是打印n行,第i(1≤i≤n)行对应于嵌套级别为i的评论(根评论的嵌套级别为1)。

对于第i行,嵌套级别为的评论按照它们出现的顺序打印,用空格分隔开。

输入描述

一行评论。由英文字母、数字和英文逗号组成。保证每个评论都是由英文字符组成的非空字符串。每个评论的数量都是整数(至少由一个数字组成)。整个字符串的长度不超过10^6。给定的评论结构保证是合法的。

输出描述

按照给定的格式打印评论。对于每一级嵌套,评论应该按照输入中的顺序打印。

示例1

输入:
hello,2,ok,0,bye,0,test,0,one,1,two,1,a,0

输出:
3
hello test one
ok bye two
a

示例2

输入:
A,5,A,0,a,0,A,0,a,0,A,0

输出:
2
A
A a A a A

image-20240112161355044

示例3

输入:
A,3,B,2,C,0,D,1,E,0,F,1,G,0,H,1,I,1,J,0,K,1,L,0,M,2,N,0,O,1,P,0

输出:
4
A K M
B F H L N O
C D G I P
E J

image-20240112162348095

题解

题目类型:树状结构的遍历,可以使用深度优先搜索(DFS)。

解题思路

  1. 定义一个评论类,包含评论内容、回复数量以及回复的评论列表。
  2. 使用递归解析评论数据,构建评论树。
  3. 使用深度优先搜索(DFS)遍历评论树,按照嵌套级别将评论内容存储到对应的列表中。
  4. 输出最大嵌套深度和每个嵌套级别的评论内容。

代码描述

  1. 定义Comment类,包含评论内容、回复数量、回复的评论列表。
  2. 解析评论数据的函数parseComment
  3. 深度优先搜索的函数dfs,将评论内容按照嵌套级别存储到result列表中。
  4. 主函数中读取输入,构建评论树,进行深度优先搜索,并输出结果。

时间复杂度:假设评论数为N,则构建评论树和深度优先搜索的时间复杂度为O(N)。

空间复杂度:需要存储评论树和结果列表,空间复杂度为O(N)。

Java

import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;

class Comment {
    /**
     * 评论类
     */
    String data;
    int replyCnt;
    List<Comment> replys;

    public Comment(String data, int replyCnt) {
        this.data = data;
        this.replyCnt = replyCnt;
        this.replys = new ArrayList<>();
    }
}

/**
 * @author code5bug
 */
public class Main {

    static int idx;
    static String[] ds;
    static List<List<String>> result;

    /**
     * 解析评论数据
     *
     * @return
     */
    public static Comment parseComment() {
        String data = ds[idx];
        int replyCnt = Integer.parseInt(ds[idx + 1]);
        idx += 2;
        Comment comment = new Comment(data, replyCnt);
        for (int i = 0; i < replyCnt; i++) {
            comment.replys.add(parseComment());
        }
        return comment;
    }

    /**
     * 遍历评论内容,并将评论的内容放到result的对应的嵌套级别中
     *
     * @param comment
     * @param level
     */
    public static void dfs(Comment comment, int level) {
        if (result.size() <= level) {
            result.add(new ArrayList<>());
        }
        result.get(level).add(comment.data);

        for (Comment rep : comment.replys) {
            dfs(rep, level + 1);
        }
    }

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        ds = scanner.nextLine().split(",");
        idx = 0;
        int N = ds.length;
        List<Comment> comments = new ArrayList<>();

        while (idx < N) {
            comments.add(parseComment());
        }

        result = new ArrayList<>();
        for (Comment comment : comments) {
            dfs(comment, 0);
        }

        System.out.println(result.size());
        for (List<String> cs : result) {
            System.out.println(String.join(" ", cs));
        }
    }
}

Python

class Comment:
    """评论类"""

    def __init__(self, data, reply_cnt) -> None:
        self.data = data
        self.reply_cnt = reply_cnt
        self.replys = []


def parse_comment() -> Comment:
    """解析评论数据"""
    global idx, ds
    data, reply_cnt = ds[idx], int(ds[idx + 1])
    idx += 2
    comment = Comment(data, reply_cnt)
    for _ in range(reply_cnt):  # 评论的回复
        comment.replys.append(parse_comment())
    return comment


def dfs(comment: Comment, level: int):
    """遍历评论内容,并将评论的内容放到result的对应的嵌套级别中"""
    global result
    if len(result) <= level:
        result.append([])
    result[level].append(comment.data)

    for rep in comment.replys:
        dfs(rep, level + 1)


ds = input().split(",")
idx, N = 0, len(ds)
comments = []
while idx < N:
    comments.append(parse_comment())

result = []
for comment in comments:
    dfs(comment, 0)

print(len(result))
for cs in result:
    print(" ".join(cs))

C++

#include <iostream>
#include <vector>
#include <sstream>

using namespace std;

// 评论类
class Comment {
public:
    string data;
    int replyCnt;
    vector<Comment> replys;

    Comment(const string& data, int replyCnt) : data(data), replyCnt(replyCnt) {}
};

int idx;
vector<string> ds;
vector<vector<string>> result;

/**
 * 解析评论数据
 *
 * @return
 */
Comment parseComment() {
    string data = ds[idx];
    int replyCnt = stoi(ds[idx + 1]);
    idx += 2;
    Comment comment(data, replyCnt);
    for (int i = 0; i < replyCnt; i++) {
        comment.replys.push_back(parseComment());
    }
    return comment;
}

/**
 * 遍历评论内容,并将评论的内容放到result的对应的嵌套级别中
 *
 * @param comment
 * @param level
 */
void dfs(const Comment& comment, int level) {
    if (result.size() <= static_cast<size_t>(level)) {
        result.emplace_back();
    }
    result[level].push_back(comment.data);

    for (const Comment& rep : comment.replys) {
        dfs(rep, level + 1);
    }
}

int main() {
    string input;
    getline(cin, input);
    istringstream iss(input);
    string token;
    while (getline(iss, token, ',')) {
        ds.push_back(token);
    }

    idx = 0;
    int N = ds.size();
    vector<Comment> comments;

    while (idx < N) {
        comments.push_back(parseComment());
    }

    result.clear();
    for (const Comment& comment : comments) {
        dfs(comment, 0);
    }

    cout << result.size() << endl;
    for (const auto& cs : result) {
        cout << cs[0];
        for (size_t i = 1; i < cs.size(); ++i) {
            cout << " " << cs[i];
        }
        cout << endl;
    }

    return 0;
}

🙏整理题解不易, 如果有帮助到您,请给点个赞 ‍❤️‍ 和收藏 ⭐,让更多的人看到。🙏🙏🙏

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

重新分区扩展C盘

电脑 – 管理 使用第三方工具&#xff1a;DiskGenius数据恢复及分区管理软件 要选择完成后重启 &#xff0c;如果这里忘记勾选&#xff0c;后面也会再次提醒并默认勾选重启 "调整后容量"是指图片上显示的非C盘之外的盘符的容量&#xff0c;这里指E盘大小 上面已经利…

做一个个人博客第一步该怎么做?

做一个个人博客第一步该怎么做&#xff1f; 好多零基础的同学们不知道怎么迈出第一步。 那么&#xff0c;就找一个现成的模板学一学呗&#xff0c;毕竟我们是高贵的Ctrl c v 工程师。 但是这样也有个问题&#xff0c;那就是&#xff0c;那些模板都&#xff0c;太&#xff01;…

运动模型非线性扩展卡尔曼跟踪融合滤波算法(Matlab仿真)

卡尔曼滤波的原理和理论在CSDN已有很多文章&#xff0c;这里不再赘述&#xff0c;仅分享个人的理解和Matlab仿真代码。 1 单目标跟踪 匀速转弯&#xff08;CTRV&#xff09;运动模型下&#xff0c;摄像头输出目标状态camera_state [x, y, theta, v]&#xff0c;雷达输出目标状…

【浅尝C++】引用

&#x1f388;归属专栏&#xff1a;浅尝C &#x1f697;个人主页&#xff1a;Jammingpro &#x1f41f;记录一句&#xff1a;大半夜写博客的感觉就是不一样&#xff01;&#xff01; 文章前言&#xff1a;本篇文章简要介绍C中的引用&#xff0c;每个介绍的技术点&#xff0c;在…

井盖异动传感器,守护脚下安全

随着城市化进程的加速&#xff0c;城市基础设施的安全问题日益受到关注。其中&#xff0c;井盖作为城市地下管道的重要入口&#xff0c;其安全问题不容忽视。然而&#xff0c;传统的井盖监控方式往往存在盲区&#xff0c;无法及时发现井盖的异常移动。为此&#xff0c;我们推出…

数据库与低代码:加速开发,提升效率的完美结合

随着技术的不断进步&#xff0c;数据库和低代码开发成为了现代应用程序开发中的两大关键要素。本文将探讨如何通过结合数据库和低代码开发&#xff0c;加速应用程序的开发过程&#xff0c;并提高开发效率和质量。 在过去的几十年中&#xff0c;数据库一直被视为应用程序开发中不…

【Linux进程】查看进程fork创建进程

目录 前言 1. 查看进程 2. 通过系统调用创建进程-fork初识 总结 前言 你有没有想过在使用Linux操作系统时&#xff0c;后台运行的程序是如何管理的&#xff1f;在Linux中&#xff0c;进程是一个非常重要的概念。本文将介绍如何查看当前运行的进程&#xff0c;并且讨论如何使用…

Sip - Ubuntu 配置 miniSIPServer 服务器(测试用)

客户提供的账号过期了&#xff0c;简单搭建 SIP 服务器&#xff0c;以便测试使用。个人认为这个配置起来最为简单&#xff0c;且测试功能足够。 官网miniSIPServer - 基于 Windows 以及 Linux 平台的 VoIP (SIP) 服务器软件. miniSIPServer 可能是最容易使用的 VoIP(SIP) 服务器…

获取进行逗号分隔的id值 Split的使用

获取进行逗号分隔的id值,Split的使用 后台实现对有逗号进行分割的字符串 使用这行代码就不会有一个空数组值,直接过滤调数组中的空值 var ids = key.Split(,).Where(s => !string.IsNullOrEmpty(s

进行交流负载测试的步骤和规范

交流负载测试是一种评估系统在正常或峰值负载下的性能和稳定性的测试方法。以下是进行交流负载测试的步骤和规范&#xff1a; 1. 确定测试目标&#xff1a;首先&#xff0c;需要明确测试的目标&#xff0c;例如&#xff0c;测试系统的响应时间、吞吐量、错误率等。 2. 设计测试…

Linux系统操作命令

Linux管理 在线查询Linux命令&#xff1a; https://www.runoob.com/linux/linux-install.htmlhttps://www.linuxcool.com/https://man.linuxde.net/ 1.Linux系统目录结构 Linux系统的目录结构是一个树状结构&#xff0c;每一个文件或目录都从根目录开始&#xff0c;并且根目…

双亲委派机制[人话版]

本篇文章仅作为记录学习之用,不具有参考价值. 如果您想系统学习,请移步最下方参考资料. 介绍 今天逛了一下牛客网, 看到有面试问到了双亲委派机制是什么, tomcat有没有打破双亲委派 , 瞬间懵逼, 听都没听过的名字, 听着就稀奇古怪. 然后翻了一下网上的答案,大概了解怎么回事.…

Python自动化测试数据驱动解决数据错误

数据驱动将测试数据和测试行为完全分离&#xff0c;实施数据驱动测试步骤如下&#xff1a; A、编写测试脚本&#xff0c;脚本需要支持从程序对象、文件或者数据库读入测试数据&#xff1b; B、将测试脚本使用的测试数据存入程序对象、文件或者数据库等外部介质中&#xff1b;…

知识库软件有很多,这几个最好用

时代进步的同时&#xff0c;逐渐优化的企业知识库已经成为企业优化工作效率、提升企业竞争力的重要工具。随着云计算和大数据技术的快速发展&#xff0c;知识库软件如雨后春笋般出现在人们的视野中。下面&#xff0c;我从寻宝者的角度&#xff0c;向大家稳稳地推荐三款最优秀的…

mp-html 微信原生小程序渲染富文本

引入组件 "usingComponents": {"mp-html": "/components/mp-html/index"}使用 <mp-html content"{{info.course_info.info}}" />获取组件 介绍 mp-html&#xff0c;小程序富文本解析利器 全面支持html标签 小程序大多数都是…

C++重新认知:拷贝构造函数

一、什么是拷贝构造函数 对于简单变量来说&#xff0c;可以轻松完成拷贝。 int a 10; int b a;但是对于复杂的类对象来说&#xff0c;不仅存在变量成员&#xff0c;也存在各种函数等。因此相同类型的类对象是通过拷贝构造函数来完成复制过程的。 #include<iostream>…

使用Notepad++将多行数据合并成一行

步骤 1、按CtrlF&#xff0c;弹出“替换”的窗口&#xff1b; 2、选择“替换”菜单&#xff1b; 3、“查找目标”内容输入为&#xff1a;\r\n&#xff1b; 4、“替换为”内容为空&#xff1b; 5、“查找模式”选择为正则表达式&#xff1b; 6、设置好之后&#xff0c;点击“全…

Spring Data JPA 踩过的坑实录

前言 游戏中台一直在使用spring 全家桶&#xff0c; 本文会左右使用Spring Data JPA的坑点记录总结 主要给大家总结介绍了关于使用Spring JPA注意事项及踩过的坑。 案例1&#xff1a; 为什么只调用了 org.springframework.data.repository.CrudRepository#findById(ID id) 却…

STM32入门教程-2023版【3-4】总结GPIO使用方法

三、总结GPIO使用方法 总体上来说是比较简单的 首先初始化时钟&#xff0c;然后定义结构体&#xff0c;赋值结构体 GPIO_Mode可以选择那8种输入输出模式&#xff0c;GPIO_Pin选择引脚&#xff0c;可以用按位或的方式同时选中多个引脚,GPIO_Speed选择输出速度&#xff0c;最后使…

基于springboot+vue的个人健康管理系统(有文档、Java毕业设计)

大家好&#xff0c;我是DeBug&#xff0c;很高兴你能来阅读&#xff01;作为一名热爱编程的程序员&#xff0c;我希望通过这些教学笔记与大家分享我的编程经验和知识。在这里&#xff0c;我将会结合实际项目经验&#xff0c;分享编程技巧、最佳实践以及解决问题的方法。无论你是…