Leetcode力扣秋招刷题路-0068
从0开始的秋招刷题路,记录下所刷每道题的题解,帮助自己回顾总结
68. 文本左右对齐
给定一个单词数组 words 和一个长度 maxWidth ,重新排版单词,使其成为每行恰好有 maxWidth 个字符,且左右两端对齐的文本。
你应该使用 “贪心算法” 来放置给定的单词;也就是说,尽可能多地往每行中放置单词。必要时可用空格 ’ ’ 填充,使得每行恰好有 maxWidth 个字符。
要求尽可能均匀分配单词间的空格数量。如果某一行单词间的空格不能均匀分配,则左侧放置的空格数要多于右侧的空格数。
文本的最后一行应为左对齐,且单词之间不插入额外的空格。
注意:
单词是指由非空格字符组成的字符序列。
每个单词的长度大于 0,小于等于 maxWidth。
输入单词数组 words 至少包含一个单词。
示例 1:
输入: words = [“This”, “is”, “an”, “example”, “of”, “text”, “justification.”], maxWidth = 16
输出:
[
“This is an”,
“example of text”,
"justification. "
]
示例 2:
输入:words = [“What”,“must”,“be”,“acknowledgment”,“shall”,“be”], maxWidth = 16
输出:
[
“What must be”,
"acknowledgment ",
"shall be "
]
解释: 注意最后一行的格式应为 "shall be " 而不是 “shall be”,
因为最后一行应为左对齐,而不是左右两端对齐。
第二行同样为左对齐,这是因为这行只包含一个单词。
示例 3:
输入:words = [“Science”,“is”,“what”,“we”,“understand”,“well”,“enough”,“to”,“explain”,“to”,“a”,“computer.”,“Art”,“is”,“everything”,“else”,“we”,“do”],maxWidth = 20
输出:
[
“Science is what we”,
“understand well”,
“enough to explain to”,
“a computer. Art is”,
“everything else we”,
"do "
]
提示:
1 <= words.length <= 300
1 <= words[i].length <= 20
words[i] 由小写英文字母和符号组成
1 <= maxWidth <= 100
words[i].length <= maxWidth
一个句子,和一个长度表示一行最长的长度,然后对齐文本,有下边几个规则。
- 同一个单词只能出现在一行中,不能拆分
- 一行如果只能放下一个单词,该单词放在最左边,然后空格补齐,例如 “acknowledgement#####”,这里只是我为了直观,# 表示空格,题目并没有要求。
- 一行如果有多个单词,最左边和最右边不能有空格,每个单词间隙尽量平均,如果无法平均,把剩余的空隙从左边开始分配。例如,“enough###to###explain##to”,3 个间隙,每个 2 个空格的话,剩下 2 个空格,从左边依次添加一个空格。
- 最后一行执行左对齐,单词间一个空格,末尾用空格补齐。
解法一
这道题关键就是理解题目,然后就是一些细节的把控了,我主要是下边的想法。
一行一行计算该行可以放多少个单词,然后计算单词间的空隙是多少,然后把它添加到结果中。
public List<String> fullJustify(String[] words, int maxWidth) {List<String> ans = new ArrayList<>();//当前行单词已经占用的长度int currentLen = 0;//保存当前行的单词List<String> row = new ArrayList<>();//遍历每个单词for (int i = 0; i < words.length;) {//判断加入该单词是否超过最长长度//分了两种情况,一种情况是加入第一个单词,不需要多加 1//已经有单词的话,再加入单词,需要多加个空格,所以多加了 1if (currentLen == 0 && currentLen + words[i].length() <= maxWidth|| currentLen > 0 && currentLen + 1 + words[i].length() <= maxWidth) {row.add(words[i]);if (currentLen == 0) {currentLen = currentLen + words[i].length();} else {currentLen = currentLen + 1 + words[i].length();}i++;//超过的最长长度,对 row 里边的单词进行处理} else {//计算有多少剩余,也就是总共的空格数,因为之前计算 currentLen 多算了一个空格,这里加回来int sub = maxWidth - currentLen + row.size() - 1;//如果只有一个单词,那么就直接单词加空格就可以if (row.size() == 1) {String blank = getStringBlank(sub);ans.add(row.get(0) + blank);} else {//用来保存当前行的结果StringBuilder temp = new StringBuilder();//将第一个单词加进来temp.append(row.get(0));//计算平均空格数int averageBlank = sub / (row.size() - 1);//如果除不尽,计算剩余空格数int missing = sub - averageBlank * (row.size() - 1);//前 missing 的空格数比平均空格数多 1String blank = getStringBlank(averageBlank + 1);int k = 1;for (int j = 0; j < missing; j++) {temp.append(blank + row.get(k));k++;}//剩下的空格数就是求得的平均空格数blank = getStringBlank(averageBlank);for (; k < row.size(); k++) {temp.append(blank + row.get(k));}//将当前结果加入 ans.add(temp.toString());}//清空以及置零row = new ArrayList<>();currentLen = 0;}}//单独考虑最后一行,左对齐StringBuilder temp = new StringBuilder();temp.append(row.get(0));for (int i = 1; i < row.size(); i++) {temp.append(" " + row.get(i));}//剩余部分用空格补齐temp.append(getStringBlank(maxWidth - currentLen));//最后一行加入到结果中ans.add(temp.toString());return ans;
}
//得到 n 个空白
private String getStringBlank(int n) {StringBuilder str = new StringBuilder();for (int i = 0; i < n; i++) {str.append(" ");}return str.toString();
}
但是这个算法,在 leetcode 跑,速度只打败了 30% 多的人,1 ms。然后在 discuss 里转了一圈寻求原因,发现大家思路都是这样子,然后找了一个人的跑了下
public List<String> fullJustify(String[] words, int maxWidth) {int left = 0; List<String> result = new ArrayList<>();while (left < words.length) {int right = findRight(left, words, maxWidth);result.add(justify(left, right, words, maxWidth));left = right + 1;}return result;
}//找到当前行最右边的单词下标
private int findRight(int left, String[] words, int maxWidth) {int right = left;int sum = words[right++].length();while (right < words.length && (sum + 1 + words[right].length()) <= maxWidth)sum += 1 + words[right++].length();return right - 1;
}//根据不同的情况添加不同的空格
private String justify(int left, int right, String[] words, int maxWidth) {if (right - left == 0) return padResult(words[left], maxWidth);boolean isLastLine = right == words.length - 1;int numSpaces = right - left;int totalSpace = maxWidth - wordsLength(left, right, words);String space = isLastLine ? " " : blank(totalSpace / numSpaces);int remainder = isLastLine ? 0 : totalSpace % numSpaces;StringBuilder result = new StringBuilder();for (int i = left; i <= right; i++)result.append(words[i]).append(space).append(remainder-- > 0 ? " " : "");return padResult(result.toString().trim(), maxWidth);
}//当前单词的长度
private int wordsLength(int left, int right, String[] words) {int wordsLength = 0;for (int i = left; i <= right; i++) wordsLength += words[i].length();return wordsLength;
}private String padResult(String result, int maxWidth) {return result + blank(maxWidth - result.length());
}private String blank(int length) {return new String(new char[length]).replace('\0', ' ');
}
看了下,发现思想和自己也是一样的。但是这个速度却打败了 100% ,0 ms。考虑了下,差别应该在我的算法里使用了一个叫做 row 的 list 用来保存当前行的单词,用了很多 row.get ( index ),而上边的算法只记录了 left 和 right 下标,取单词直接用的 words 数组。然后尝试着在我之前的算法上改了一下,去掉 row,用两个变量 start 和 end 保存当前行的单词范围。主要是 ( end - start ) 代替了之前的 row.size ( ), words [ start + k ] 代替了之前的 row.get ( k )。
public List<String> fullJustify2(String[] words, int maxWidth) {List<String> ans = new ArrayList<>();int currentLen = 0;int start = 0;int end = 0;for (int i = 0; i < words.length;) {//判断加入该单词是否超过最长长度//分了两种情况,一种情况是加入第一个单词,不需要多加 1//已经有单词的话,再加入单词,需要多加个空格,所以多加了 1if (currentLen == 0 && currentLen + words[i].length() <= maxWidth|| currentLen > 0 && currentLen + 1 + words[i].length() <= maxWidth) {end++;if (currentLen == 0) {currentLen = currentLen + words[i].length();} else {currentLen = currentLen + 1 + words[i].length();}i++;} else {int sub = maxWidth - currentLen + (end - start) - 1;if (end - start == 1) {String blank = getStringBlank(sub);ans.add(words[start] + blank);} else {StringBuilder temp = new StringBuilder();temp.append(words[start]);int averageBlank = sub / ((end - start) - 1);//如果除不尽,计算剩余空格数int missing = sub - averageBlank * ((end - start) - 1);String blank = getStringBlank(averageBlank + 1);int k = 1;for (int j = 0; j < missing; j++) {temp.append(blank + words[start+k]);k++;}blank = getStringBlank(averageBlank);for (; k <(end - start); k++) {temp.append(blank + words[start+k]);}ans.add(temp.toString());}start = end;currentLen = 0;}}StringBuilder temp = new StringBuilder();temp.append(words[start]);for (int i = 1; i < (end - start); i++) {temp.append(" " + words[start+i]);}temp.append(getStringBlank(maxWidth - currentLen));ans.add(temp.toString());return ans;
}
//得到 n 个空白
private String getStringBlank(int n) {StringBuilder str = new StringBuilder();for (int i = 0; i < n; i++) {str.append(" ");}return str.toString();
}
果然,速度也到了打败 100%,0 ms。
总
充分说明 list 的读取还是没有数组的直接读取快呀,还有就是要向上边的作者学习,多封装几个函数,思路会更加清晰,代码也会简明。
相关文章:
Leetcode力扣秋招刷题路-0068
从0开始的秋招刷题路,记录下所刷每道题的题解,帮助自己回顾总结 68. 文本左右对齐 给定一个单词数组 words 和一个长度 maxWidth ,重新排版单词,使其成为每行恰好有 maxWidth 个字符,且左右两端对齐的文本。 你应该…...

Nginx介绍及安装(windows版,Linux版)
目录 一、Nginx介绍 1、Nginx优势 2、Nginx作用 3、部署静态资源 4、代理 5、负载均衡 二、Nginx安装步骤(windows版) 三、Nginx安装步骤(Linux版) 1、官网下载安装包,下载完之后上传到Linux系统上 2、在Lin…...

Camera | 4.瑞芯微平台MIPI摄像头应用程序编写
前面3篇我们讲解了camera的基础概念,MIPI协议,CSI2,常用命令等,本文带领大家入门,如何用c语言编写应用程序来操作摄像头。 Linux下摄像头驱动都是基于v4l2架构,要基于该架构编写摄像头的应用程序ÿ…...
【1250. 检查「好数组」】
来源:力扣(LeetCode) 描述: 给你一个正整数数组 nums,你需要从中任选一些子集,然后将子集中每一个数乘以一个 任意整数,并求出他们的和。 假如该和结果为 1,那么原数组就是一个「…...

Spring 如何解决循环依赖?
什么是循环依赖 ? 一个或多个对象之间存在直接或间接的依赖关系,这种依赖关系构成一个环形调用,有下面 3 种方式。 我们看一个简单的 Demo,对标“情况 2”。 Service public class Louzai1 {Autowiredprivate Louzai2 louzai2;…...
CocoaPods使用指南
前言 对于大多数软件开发团队来说,依赖管理工具必不可少,它能针对开源和私有依赖进行安装与管理,从而提升开发效率,降低维护成本。针对不同的语言与平台,其依赖管理工具也各有不同,例如 npm 管理 Javascri…...

Kafka 消息队列
目录主流的消息队列消息队列的应用场景缓存/肖锋解耦异步处理KafkaKafka的定义Kafka的底层基础架构Kafka分区如何保证Leader选举Kafka分区如何保证Leader和Follower数据的一致性Kafka 中消费者的消费方式Kafka 高效读写数据的原因(高性能吞吐的原因)&…...
华为OD机试 - 挑选字符串(Python)| 真题+思路+考点+代码+岗位
挑选字符串 题目 给定a-z,26 个英文字母小写字符串组成的字符串A和B, 其中A可能存在重复字母,B不会存在重复字母, 现从字符串A中按规则挑选一些字母可以组成字符串B 挑选规则如下: 同一个位置的字母只能挑选一次, 被挑选字母的相对先后顺序不能被改变, 求最多可以同时…...

对比Hashtable、HashMap、TreeMap有什么不同?
第9讲 | 对比Hashtable、HashMap、TreeMap有什么不同? Map 是广义 Java 集合框架中的另外一部分,HashMap 作为框架中使用频率最高的类型之一,它本身以及相关类型自然也是面试考察的热点。 今天我要问你的问题是,对比 Hashtable、…...

测试新版Android Studio的手机镜像效果
学更好的别人, 做更好的自己。 ——《微卡智享》 本文长度为669字,预计阅读2分钟 前言 春节刚上班,就开始了疯狂出差的节奏,期间发现Android Studio发布新的版本2022.1.1(Electric Eel),里面两个更新的内容蓝牙模拟器和…...

女生可以参加IT培训吗?
2023年了,就不要把性别当作选择专业的前提条件了。虽然这句话说过很多次了,作为IT行业来说,是非常欢迎女生的加入;尤其是整天都是面对一大堆男攻城狮,工作氛围一点都不活跃,反而显得压抑和杂乱,…...

刷题25-重排链表
重排链表 解题思路:通过观察链表可以发现,把链表一分为二,后半段链表反转,然后两个链表穿插连接,当链表的节点总数是奇数时,要保证链表的前半段比后半段多一个节点。 关于把链表一分为二,可以…...

VHDL-延迟模型-惯性延迟与传输延迟
目录 1,惯性延时 2,传输延时 信号通过元件都会有延迟,延迟时间的计算是逻辑仿真的重要功能。考虑延迟信息得到的仿真输出波形可以更精确地反映实际电路的情况。针对元件的延时,人们根据需要建立了一些用的延时模型,这…...

2023年美赛(MCM/ICM)简介
2023年美赛将要如期开赛,这里为了 让大家对今年的美赛有一个直接 客观的了解。对2023年美赛(MCM/ICM)进行一下简要的介绍。相关资料大家可以查看另一篇文章一、竞赛时间February 16-20, 2023开赛时间 北京时间 17号(本周五) 6:00结束时间 北京时间 21号(…...

5min完成linux环境Jenkins的安装
5min搞定linux环境Jenkins的安装安装Jenkinsstep1: 使用wget 命令下载Jenkinsstep2、创建Jenkins日志目录并运行jekinsstep3、访问jenkins并解锁jenkins,安装插件以及创建管理员用户step4、到此,就完成了Finish、以上步骤中遇到的问题1、 jenkins启动不了…...
华为OD机试 - 字母计数(Python)| 真题+思路+考点+代码+岗位
字母计数 题目 给出一个只包含字母的字符串, 不包含空格,统计字符串中各个子字母(区分大小写)出现的次数, 并按照字母出现次数从大到小的顺序输出各个字母及其出现次数 如果次数相同,按照自然顺序排序,且小写字母在大写字母之前 输入 输入一行仅包含字母的字符串 输出 按…...

DENSE 数据集 - STF 数据集(CVPR 2020)
DENSE 数据集 - STF 数据集 - Seeing Through Fog Without Seeing Fog: Deep Multimodal Sensor Fusion in Unseen Adverse Weather(CVPR 2020)摘要1. 引言2. 相关工作3. 多模式恶劣天气数据集3.1 多模态传感器设置3.2 记录4. 自适应深度融合4.1 自适应多…...
华为OD机试 - 静态扫描最优成本(Python)| 真题+思路+考点+代码+岗位
静态扫描最优成本 题目 静态扫描快速识别源代码的缺陷,静态扫描的结果以扫描报告作为输出: 文件扫描的成本和文件大小相关,如果文件大小为 N ,则扫描成本为 N 个金币扫描报告的缓存成本和文件大小无关,每缓存一个报告需要 M 个金币扫描报告缓存后,后继再碰到该文件则不…...

【ns-3】零基础安装教程
文章目录前言1. 安装虚拟机及Ubuntu2. 安装依赖库3. 下载ns-34. 构建ns-3前言 近期因工作需要开始接触ns-3。作者零基础,从零开始顺利完成了ns-3的安装。本篇为ns-3安装过程记录贴或针对小白的零基础教程。 本篇内容所使用到的软件版本信息如下:VMware…...
华为OD机试 - 新学校选址(Python)| 真题+思路+考点+代码+岗位
新学校选址 题目 为了解新学期学生暴涨的问题,小乐村要建立所新学校 考虑到学生上学安全问题,需要所有学生家到学校的距离最短. 假设学校和所有学生家都走在一条直线之上,请问学校建立在什么位置, 能使得到学校到各个学生家的距离和最短 输入 第一行: 整数 n 取值范围 [1,1…...
挑战杯推荐项目
“人工智能”创意赛 - 智能艺术创作助手:借助大模型技术,开发能根据用户输入的主题、风格等要求,生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用,帮助艺术家和创意爱好者激发创意、提高创作效率。 - 个性化梦境…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...

【JVM】- 内存结构
引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...

【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
安卓基础(aar)
重新设置java21的环境,临时设置 $env:JAVA_HOME "D:\Android Studio\jbr" 查看当前环境变量 JAVA_HOME 的值 echo $env:JAVA_HOME 构建ARR文件 ./gradlew :private-lib:assembleRelease 目录是这样的: MyApp/ ├── app/ …...
Xen Server服务器释放磁盘空间
disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...

中医有效性探讨
文章目录 西医是如何发展到以生物化学为药理基础的现代医学?传统医学奠基期(远古 - 17 世纪)近代医学转型期(17 世纪 - 19 世纪末)现代医学成熟期(20世纪至今) 中医的源远流长和一脉相承远古至…...
JS设计模式(4):观察者模式
JS设计模式(4):观察者模式 一、引入 在开发中,我们经常会遇到这样的场景:一个对象的状态变化需要自动通知其他对象,比如: 电商平台中,商品库存变化时需要通知所有订阅该商品的用户;新闻网站中࿰…...

VM虚拟机网络配置(ubuntu24桥接模式):配置静态IP
编辑-虚拟网络编辑器-更改设置 选择桥接模式,然后找到相应的网卡(可以查看自己本机的网络连接) windows连接的网络点击查看属性 编辑虚拟机设置更改网络配置,选择刚才配置的桥接模式 静态ip设置: 我用的ubuntu24桌…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...