当前位置: 首页 > news >正文

【刷题之路 | Java Python】两数之和(暴力枚举哈希表)

在这里插入图片描述

🤵‍♂️ 个人主页: @计算机魔术师
👨‍💻 作者简介:CSDN内容合伙人,全栈领域优质创作者。

🌐 推荐一款找工作神器网站: 牛客网 |笔试题库|面试经验|实习招聘内推

还没账户的小伙伴 速速点击链接登录注册吧!🙋‍♂️ 刷题通关之路等你冲!!🎉🎉🎉 开始刷爆题库,速速通关面试吧🙋‍♂️\

文章目录

  • 一、说在前面
  • 二、两数之和
    • 2.1、暴力枚举
      • 2.1.1 python实现
      • 2.1.2 java实现
    • 3.1 哈希表(Hash table)
      • 3.1.1 python实现
      • 3.1.2 Java实现
  • 今日份推荐 —— 牛客网

一、说在前面

刷题是一件日积月累的事情,我们在刷题中要保持良好习惯,让每一道题发挥最大作用!以下是 某ACM🥇金牌选手所建议的刷题方式,觉得很不错,给大家参考一下

如何正确的做一道题

  1. 从简入手: 先从简单暴力(时间复杂度高)的方法入手。
  2. 优化: 思考如何在第一步的基础上,如何优化算法,降低时间复杂度。
  3. 构思代码: 有了以上两步,我们此时应该已经有了一个正确的想法,此时我们应该构思代码,有那几部分,每部分实现什么功能,代码怎么写。而不是直接闷头去写代码,没想清楚直接去写代码,会导致写了一半发现思路不对,写的代码都是错误的。
  4. 写代码: 实现第三步代码。
  5. (Debug): 如果我们的题目没有通过测试,应该检查代码是不是有bug、思路对不对等。
  6. 总结与反思: 题目通过了,我们应该总结一下这道题考察的知识点、切入的角度、同类型的题目等,同时思考有没有更优的办法。

做到以上几点,一道题学习的就很透了,遇到同类型的题目可以举一反三啦。

数组&双指针章节

二、两数之和

hello world 一样经典的刷题入门第一题 —— 两数之和

原题如下:

给定一个整 数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并>返回它们的数组下标。

你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现。

你可以按任意顺序返回答案。

示例 1:

输入:nums = [2,7,11,15], target = 9
输出:[0,1]
解释:因为 nums[0] + nums[1] == 9 ,返回 [0, 1] 。

示例 2:

输入:nums = [3,2,4], target = 6
输出:[1,2]

示例 3:

输入:nums = [3,3], target = 6
输出:[0,1]

提示:

2 <= nums.length <= 104
-109 <= nums[i] <= 109
-109 <= target <= 109
只会存在一个有效答案

进阶:你可以想出一个时间复杂度小于 O(n2) 的算法吗?

思路历程:

2.1、暴力枚举

按照解题思路,暴力枚举,这里选择快速排序法,快速筛选

2.1.1 python实现

代码:

class Solution:
    def twoSum(self, nums: List[int], target: int) -> List[int]:
        result = []
        for i in range(len(nums)):
            for j in range(i+1,len(nums)):
                if target == nums[i] + nums[j]:
                    result.append(i)
                    result.append(j)

        return result

在这里插入图片描述

2.1.2 java实现

代码:

class Solution {
    public int[] twoSum(int[] nums, int target) {
        List<Integer> result = new ArrayList<Integer>();
        for(int i=0;i<nums.length;i++){
            for(int j=i+1;j<nums.length;j++){
                if(target == nums[i]+nums[j])
                    return new int[]{i,j};
            }
        }
        return null;
    }
}

可以看出java作为编译性语言还是要比python运行速度快很多,不过使用内存消耗更多一点
在这里插入图片描述

时间复杂度为 O(n2)
空间复杂度O(1)

3.1 哈希表(Hash table)

我们适用哈希表对其优化,我们先简单讲讲哈希表的原理

  • 数组的特点是:寻址容易,插入和删除困难;
  • 而链表的特点是:寻址困难,插入和删除容易。

我们把两者结合起来,便是哈希表

哈希表的底层实际上是基于数组来存储的,当插入键值对时,并不是直接插入该数组中,而是通过对键进行Hash运算得到Hash值,然后和数组容量取模,得到在数组中的位置后再插入(不害怕多个重复数字,使用链表把多个数字都压缩在同一个值上)。取值时,先对指定的键求Hash值,再和容量取模得到底层数组中对应的位置,如果指定的键值与存贮的键相匹配,则返回该键值对,如果不匹配,则表示哈希表中没有对应的键值对。这样做的好处是在查找、插入、删除等操作可以做到O(1),最坏的情况是O( n ),当然这种是最极端的情况,极少遇到。

在这里插入图片描述

哈希表实现原理很多,不管哪门语言,实现一个HashMap的过程均可分为三大步骤:

  1. 实现一个Hash函数
  2. 合理解决Hash冲突
  3. 实现HashMap的操作方法

我们这里不深揪算法,大概了解即可,pythondict便是哈希表算法,我们直接使用即可。

3.1.1 python实现

代码:

class Solution:
    def twoSum(self, nums: List[int], target: int) -> List[int]:
        result = []
        Hashmap = dict()
        for i,num in enumerate(nums):
            if target - num  in Hashmap:
                result.append(Hashmap[target - num])
                result.append(i)
            Hashmap[nums[i]] = i # 默认会把本次数值省略

        return result

在这里插入图片描述
与之前相比执行速度快了十倍, 内存消耗多了一点

时间复杂度O(n)
空间复杂度: O(1)

这里提一点比较秒的地方,因为有一种情况是比较特殊的

输入:
[3,2,4]
6

这个时候如果正常遍历所有数,会有可能添加到3,因为 6 - 3 = 3 在nums里面,即自己和自己相加了。解决办法: 错开索引,在当前索引在字典创建对应值,跳过本次循环到下一个值判断。

3.1.2 Java实现

class Solution {
    public int[] twoSum(int[] nums, int target) {
        Map<Integer,Integer>  hashmap = new HashMap<Integer,Integer>();
        List<Integer>  result = new ArrayList<Integer>();
        for(int i=0;i < nums.length;i++){
            if (hashmap.containsKey(target - nums[i])){
                return new int[]{hashmap.get(target - nums[i]),i};
            }
            hashmap.put(nums[i],i);
        }
        return new int[0];
    }
}

在这里插入图片描述
与之前暴力枚举同样快速十倍,内存消耗没有变化


到这里我们已经成功踏出我们刷题的第一步啦🎉🎉🎉

今日份推荐 —— 牛客网

学习掌握一门语言的快速方法就是通过刷题实践运用该语言的语法以及与其他语言的比较也可以得到更深的领悟和收获!

如果还不知道哪里可以适用与小白 刷题掌握语法,牛客网亲测很不错!!
点击链接跳转牛客网登录注册看看,他们现在的IT题库内容很丰富,属于国内做的很好的了,而且是课程+刷题+面经+求职+讨论区分享,一站式求职学习网站,最最最重要的里面的资源全部免费!!
在这里插入图片描述

他们的Java & Python题单是从最基础的输出、字符串格式化输出开始,经过运算符、列表、循环语句、条件语句、元组、字典、函数等知识点,一步一步教你慢慢学会Java & Python那为数不多的基本语法,最后再配合上8道具有实践意义的综合实践题,可以帮你更加有效的巩固前面学会的知识。

在这里插入图片描述

在这里插入图片描述

牛客网还提供题解专区和讨论区会有大神提供题解思路,对新手玩家及其友好,有不清楚的语法,不理解的地方,看看别人的思路,别人的代码,也许就能豁然开朗。

如果现在的你按捺不住卷起来,那就赶快点击下方链接学起来吧!
链接:点击链接跳转牛客网开始刷题之路!!!

如果还没有账号的小伙伴速速点击链接登录注册吧!🙋‍♂️ 刷题通关之路等你冲!!🎉🎉🎉

相关文章:

  • [ C++ ] STL_vector -- 迭代器失效问题
  • 06-Linux用户管理
  • 十天学前端之JS篇(五)
  • python 并发、并行处理、分布式处理
  • 推荐一款国人开源的 Redis 可视化管理工具
  • 开发工程师必备————【Day22】前端开发之jQuery更多操作
  • 04【DQL查询】
  • Vscode常用插件
  • 利用MyBatisX插件自动生成代码
  • 【数据结构】——栈和链表的面试题详解
  • 如何从 apt-get 升级中排除特定软件包
  • C++/Python:罗德里格斯旋转矩阵
  • c++征途 --- STL初识
  • 学习编程的第二十三天
  • 上交所技术——2020春招应用开发工程师(Java)笔试
  • 【腾讯Bugly干货分享】从0到1打造直播 App
  • 2019.2.20 c++ 知识梳理
  • VuePress 静态网站生成
  • vue--为什么data属性必须是一个函数
  • 阿里云前端周刊 - 第 26 期
  • 好的网址,关于.net 4.0 ,vs 2010
  • 后端_ThinkPHP5
  • 基于组件的设计工作流与界面抽象
  • 马上搞懂 GeoJSON
  • 如何设计一个比特币钱包服务
  • 如何邀请好友注册您的网站(模拟百度网盘)
  • ​MPV,汽车产品里一个特殊品类的进化过程
  • #微信小程序:微信小程序常见的配置传旨
  • #我与Java虚拟机的故事#连载03:面试过的百度,滴滴,快手都问了这些问题
  • (9)YOLO-Pose:使用对象关键点相似性损失增强多人姿态估计的增强版YOLO
  • (C语言)fgets与fputs函数详解
  • (二)【Jmeter】专栏实战项目靶场drupal部署
  • (附源码)springboot金融新闻信息服务系统 毕业设计651450
  • (深入.Net平台的软件系统分层开发).第一章.上机练习.20170424
  • (算法设计与分析)第一章算法概述-习题
  • (详细版)Vary: Scaling up the Vision Vocabulary for Large Vision-Language Models
  • (原創) 博客園正式支援VHDL語法著色功能 (SOC) (VHDL)
  • (转)平衡树
  • ... fatal error LINK1120:1个无法解析的外部命令 的解决办法
  • .halo勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .net core Swagger 过滤部分Api
  • .NET Entity FrameWork 总结 ,在项目中用处个人感觉不大。适合初级用用,不涉及到与数据库通信。
  • .Net IOC框架入门之一 Unity
  • .Net 路由处理厉害了
  • .Net 转战 Android 4.4 日常笔记(4)--按钮事件和国际化
  • .NET/ASP.NETMVC 深入剖析 Model元数据、HtmlHelper、自定义模板、模板的装饰者模式(二)...
  • .net2005怎么读string形的xml,不是xml文件。
  • .NET牛人应该知道些什么(2):中级.NET开发人员
  • .NET中 MVC 工厂模式浅析
  • @entity 不限字节长度的类型_一文读懂Redis常见对象类型的底层数据结构
  • @GetMapping和@RequestMapping的区别
  • [ vulhub漏洞复现篇 ] Celery <4.0 Redis未授权访问+Pickle反序列化利用
  • [20150321]索引空块的问题.txt
  • [acwing周赛复盘] 第 69 场周赛20220917
  • [C++] Boost智能指针——boost::scoped_ptr(使用及原理分析)