2019年7月20日 - LeetCode0003
https://leetcode-cn.com/problems/longest-substring-without-repeating-characters/submissions/
我的解法:
class Solution {
HashSet<Character> set = new HashSet<Character>();
public int lengthOfLongestSubstring(String s) {
int len = s.length();
char[] string = new char[len];
s.getChars(0,len,string,0);
int ans = 0;
//左右边界,左闭右开
int left = 0,right = 0;
for(int i=0;i<len;++i){
if(set.add(string[i])){
//添加成功,此字符未出现过,右边界扩张1
++right;
ans = ans>(right-left)?ans:(right-left);
}else{
//添加进集合失败,即出现了重复字符
//调整左边界:从右边界现在的位置向后调整
for(int j=right-1;j>=left;--j){
if(string[j]==string[i]){
left = j+1;
break;
}
}
++right;
ans = ans>(right-left)?ans:(right-left);
}
}
return ans;
}
}
维护左右边界两个数,然后每次添加进去一个新字符就更新边界,计算长度并与ans比较更新 ans
时间复杂度O(n2),因为最坏情况下每次重复的字符都是在子串的第一个的位置,如"abcdefgabcdefg",那第二个循环必须向前找很远.
空间复杂度O(n),因为用了HashSet可能比手动维护的数组的形式占得空间大一点吧
运行错误一次: 一开始省事儿从第二个字符开始检查,结果忘记了考虑空串,傻了傻了
答案错误一次: 低级错误,不说了太丢脸了(左闭右开算长度减完了我咋还+1了,爆哭)
结果:
官方题解:
方法1:
暴力法,代码略,考察每个子串是否满足无重复字符,会超时的.
方法2:
滑动窗口法.和我这个解法的思路是一样的,仔细想了下为什么我复杂度高然后发现改成这样就会变成O(n)了
我的方法 - 受方法2启发改:
class Solution {
HashSet<Character> set = new HashSet<Character>();
public int lengthOfLongestSubstring(String s) {
int len = s.length();
char[] string = new char[len];
s.getChars(0,len,string,0);
int ans = 0;
//左右边界,左闭右开
int left = 0,right = 0;
for(int i=0;i<len;++i){
if(set.add(string[i])){
//添加成功,此字符未出现过,右边界扩张1
++right;
ans = ans>(right-left)?ans:(right-left);
}else{
//添加进集合失败,即出现了重复字符
//调整左边界:从左边界现在的位置向前调整
for(int j=left;j<right;++j){
if(string[j]==string[i]){
left = j+1;
break;
}
}
++right;
ans = ans>(right-left)?ans:(right-left);
}
}
return ans;
}
}
这样这个方法时间复杂度就是O(2n) = O(n)的了.空间复杂度O(n)
原因在于这个虽然是两重循环,但是第二重循环每次必定是从新的地方开始,不走回头路.
如: "abcdefgabcdefg",对我的原始方法来说,当检查到第二个a的时候左边界必须向左后退到第一个a,然而下一个检查b,左边界又要向左后退到第一个b,其中[第一个b,第二个a)这些后退都是上一步检查第二个a的时候做过的了,这里在走回头路做无用功,所以浪费了时间.
对改进方法来说,检查第二个a只需要看第一个a,检查第二个b的时候只需要看第一个b,不必走回头路.
即使是对改进方法而言最坏的情况如"abcdefghijklmnnop"来说,即使检查第二个n需要从第一个a一直前进左边界直到第一个n,但是只后再也不会检查这些元素是否在左边界以左了.
体会到了"滑动窗口"这个命名的妙处.比单独的两个边界的名字有整体性.
结果:
方法3:
优化的滑动窗口,利用map而不是set,从根本上省去了第二个循环
受教,学到了的方法:
class Solution {
HashMap<Character,Integer> map = new HashMap<Character,Integer>();
public int lengthOfLongestSubstring(String s) {
int len = s.length();
char[] string = new char[len];
s.getChars(0,len,string,0);
int ans = 0;
//左右边界,左闭右开
int left = 0,right = 0;
for(int i=0;i<len;++i){
if(map.containsKey(string[i])){
//找到第一次出现时的位置
//左边界不能变小,不能走回头路
//在这个前提下把左边界变为上次出现此字符的位置再加一
left = map.get(string[i]) + 1>left?map.get(string[i]) + 1:left;
}
//如果原本就有这个字符,那么这个put是更新此字符的位置,否则是加入一个新的字符并记录它的位置
map.put(string[i],i);
right = i+1;
ans = ans>(right - left)?ans:(right - left);
}
return ans;
}
}
left的更新中,left不可以变小很重要.
时间复杂度O(n),空间复杂度O(n)
答案错误一次: 忘记考虑left不可以变小了...
结果:
(诡异的是速度慢于上一个.可能containsKey的开销在数据量比较小的时候真的不能忽略吧)
方法4:
优化的滑动窗口法,不使用HashMap,转而使用一个数组,思路跟上面其实一样,更原始一点所以效率更优一点.
代码略
时间复杂度O(n),空间复杂度O(n)
2019年7月20日 - LeetCode0003的更多相关文章
- 【2019年05月20日】A股滚动市盈率PE历史新低排名
2010年01月01日 到 2019年05月20日 之间,滚动市盈率历史新低排名. 上市三年以上的公司, 2019年05月20日市盈率在300以下的公司. 1 - 阳光照明(SH600261) - 历 ...
- tmux使用——2019年11月20日16:40:15
1.tmux 命令行的典型使用方式是,打开一个终端窗口(terminal window,以下简称"窗口"),在里面输入命令.用户与计算机的这种临时的交互,称为一次"会话& ...
- 2019年7月20日针对iPhone7/7P有锁机的爆破限制
背景 2019年7月20号苹果更改了激活策略,致使卡贴机一夜回到解放前,目前只能使用tmsi或者tmsi+iccid模式激活手机,但是缺点是移动联通信号真的不稳定,漏接电话,无法开启热点等等毛病.尤其 ...
- 【搬家啦】2019年06月20日csdn难民来报道啦~
原博地址:https://blog.csdn.net/the_fool_
- 2019年7月20日 - LeetCode0002
https://leetcode-cn.com/problems/add-two-numbers/submissions/ 我的方法: /** * Definition for singly-link ...
- 易初大数据 2019年10月20日 linux死亡导图 王庆超
- 易初大数据 2019年10月20日 spss习题 王庆超
一.选择题 1.有关spss数据字典的说法,正确的是:D A.SPSS数据集的数据字典可以复制到其他数据集中 B.SPSS数据集的数据字典是不能复制的 C.SPSS的数据字典可以通过“复制”和“黏贴” ...
- 西安活动 | 4月20日「拥抱开源,又见.NET :云时代 • 新契机」
云计算日渐兴起,成为提升企业效率和生产力的最终解决方案,而云时代也为软件开发模式带来了翻天覆地的变化.可以说 .NET Core就是这个时代催生的产物.自2016年 .NET Core 1.0 发布以 ...
- 【08月20日】A股滚动市净率PB历史新低排名
2010年01月01日 到 2019年08月20日 之间,滚动市净率历史新低排名. 上市三年以上的公司,2019年08月20日市净率在30以下的公司. 来源:A股滚动市净率(PB)历史新低排名. 1 ...
随机推荐
- MVC4使用SignalR出现$.connection is undefined错误备忘
SignalR使用过程中一定要注意js的引用顺序,否则就会出现$.connection is undefined脚本错误.
- QTableWidget和 QTableView翻页效果(准确计算Scroll,然后使用setSliderPosition函数)
以QTableView举例,QTableWidget使用相同 int CQTTableViewPageSwitch::pageCount(QTableView *p)//QTableView 总页 ...
- Indy9的TIdFTPServer封装类
在Delphi 7开发下有强大的Indy控件,版本为9,要实现一个FTP服务器,参考自带的例子,发现还要写很多函数,而且不支持中文显示文件列表等等.于是,自己改进封装了下,形成一个TFTPServer ...
- 用 eric6 与 PyQt5 实现python的极速GUI编程(35篇PyQT和200多篇Python)
[题记] 我是一个菜鸟,这个系列是我的学习笔记. PyQt5 出来有一段时间了, PyQt5 较之 PyQt4 有一些变化,而网上流传的几乎都是 PyQt4 的教程,照搬的话大多会出错. eric6 ...
- SAP TABLECONTROL 自定义SEARCH HELP
项目上需要开发一个界面如下的应用程序.这是一个MB1A发料的辅助程序,限制住移动类型和在特定字段写入产品号. 这个应用程序的主要功能毫无疑问是通过BAPI实现的.但在TABLECONTROL中需要对填 ...
- Cloudera Impala需求
Cloudera Impala需求 为了达到预期的效果,Impala依赖于软件.硬件的可用性,以及下面章节描述的配置. 继续阅读: 支持的操作系统 支持的Hadoop发布 Hive Metastore ...
- Spring Boot:整合MyBatis框架
综合概述 MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以使用简单 ...
- GIS热力图制作与位置大数据分析
最近有很多朋友咨询位置数据.热力图等等东西,我一一进行了解答,但是个人精力实在有限,特写一个博客进行详细技术说明,其实这个东西位置数据.百度地图POI.高德地图POI等数据爬取.存储都较为简单,热力图 ...
- spring 5.x 系列第4篇 —— spring AOP (代码配置方式)
文章目录 一.说明 1.1 项目结构说明 1.2 依赖说明 二.spring aop 2.1 创建待切入接口及其实现类 2.2 创建自定义切面类 2.3 配置切面 2.4 测试切面 2.5 切面执行顺 ...
- 机器学习读书笔记(五)AdaBoost
一.Boosting算法 .Boosting算法是一种把若干个分类器整合为一个分类器的方法,在boosting算法产生之前,还出现过两种比较重要的将多个分类器整合为一个分类器的方法,即boostrap ...