Leetcode刷题指南之Python求两数之和【多种思路详解】,判断是否为回文数【将整数转为字符串】【不转为字符串】

举报
南蓬幽 发表于 2022/05/15 21:10:55 2022/05/15
【摘要】 💎一、Leetcode刷题之两数之和 🏆1.两数之和 🏆2.原题链接 💎二、解题报告 思路1 🏆1.思路分析 🏆2.代码详解 🏆3.按步骤分析 思路2 🏆1.思路分析 🏆2.代码详解 🏆3.按步骤分析 回文数 做题思路 解题步骤 代码整合 进阶 做题思路 解题步骤 代码整合 💎一、Leetcode刷题之两数之和 🏆1.两数之和给定一个整数数组 nums 和一个整数目...

💎一、Leetcode刷题之两数之和

🏆1.两数之和

给定一个整数数组 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.原题链接

来源:力扣(LeetCode) 链接:https://leetcode.cn/problems/two-sum

💎二、解题报告

思路1

循环枚举

🏆1.思路分析

🔑思路:
​ 提示:首先想到的是​ 双重循环遍历数组中的每一个数,枚举出两个数可组成的所有组合,判断组合中两个数相加是否等于target,如果等于就输出下标。
例如[0,1,2,3]
0,1
0,2
0,3
1,2
1,3
2,3
所有组合

🏆2.代码详解

class Solution(object):
    def twoSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        for i in range(1,len(nums)):
            for n in range(i):
                if nums[n] +nums[i] == target:
                    return [n,i]

不过有些不清晰,简单修饰后

class Solution(object):
    def twoSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        l = len(nums)
        for i in range(l):
            for n in range(i+1,l):
                if nums[i] +nums[n] == target:
                    return [n,i]

🏆3.按步骤分析

提示:
l = len(nums)
求出数组长度
for i in range(l):
第一个数从下标为0开始,遍历所有数
for n in range(i+1,l):
第一个数从下标为i+1开始,就是读取下标为i+1后面的数
if nums[i] +nums[n] == target:
判断两个数是否相加为target
return n,i
返回下标

思路2

字典模拟哈希表

🏆1.思路分析

这次我们不找两个数,找一个数,如果第一个数为x,那么y就等于target-x,我们只需找到y是否在列表中,如果在就输出下标。

class Solution(object):
    def twoSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        l = len(nums)
        for i in range(l):
            y = target - nums[i]
            for n in range(i+1,l):
                if nums[n] == y:
                    return [n,i]

但是这样还是需要双重循环,没有太大区别!
接下来就用到了字典。
enumerate() 函数用于将一个可遍历的数据对象(如列表、元组或字符串)组合为一个索引序列,同时列出数据和数据下标.
例如

nums = [10,20,30]
print(enumerate(nums))
结果:
(0 ,10)
(1 ,20)
(2 ,30)

🏆2.代码详解

class Solution(object):
    def twoSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        dict = {}
        for v, k in enumerate(nums):
            n = target - k
            if n in dict:
                return [dict[n], v]
            else:
                dict[k] = v

🏆3.按步骤分析

dict = {}
建一个空字典
for v, k in enumerate(nums):
循环遍历,为v和k赋值,v为下标,k为数值
n = target - k
求数n
if n in dict:
return [dict[n], v]
判断数n是否在字典中,如果在就返回下标。
else:
dict[k] = v
这一步是将遍历过的数和其下标,添加到字典里。因为刚开始字典为空,所以会将两个数中的第一个数添加字典中k代表的字典的key,v代表字典的value,当遍历第二个数时,查询第一个数在字典中,返回这两个数的>下标!

回文数

给你一个整数 x ,如果 x 是一个回文整数,返回 true ;否则,返回 false 。

回文数是指正序(从左向右)和倒序(从右向左)读都是一样的整数。

例如,121 是回文,而 123 不是。

示例 1:

输入:x = 121
输出:true
示例 2:

输入:x = -121
输出:false
解释:从左向右读, 为 -121 。 从右向左读, 为 121- 。
因此它不是一个回文数。
示例3:

输入:x = 10
输出:false
解释:从右向左读, 为 01 。
因此它不是一个回文数。

提示:

-231 <= x <= 231 - 1

进阶:你能不将整数转为字符串来解决这个问题吗?

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/palindrome-number

做题思路

将整数转为字符串,然后用字符串切片逆序输出,比较两者是否相同,如果相同就返回ture,否者返回false。

解题步骤

1.将整数转为字符串

s = str(x)

2.字符串切片逆序输出

s[::-1]

3.比较两者是否相同,如果相同就返回true,否者返回false。

if s == s[::-1]:
            return True
        else:
            return False

代码整合

class Solution(object):
    def isPalindrome(self, x):
        """
        :type x: int
        :rtype: bool
        """
        s = str(x)
        if s == s[::-1]:
            return True
        else:
            return False

进阶

你能不将整数转为字符串来解决这个问题吗?

做题思路

官方题解
方法一:反转一半数字
思路

映入脑海的第一个想法是将数字转换为字符串,并检查字符串是否为回文。但是,这需要额外的非常量空间来创建问题描述中所不允许的字符串。

第二个想法是将数字本身反转,然后将反转后的数字与原始数字进行比较,如果它们是相同的,那么这个数字就是回文。
但是,如果反转后的数字大于 \text{int.MAX}int.MAX,我们将遇到整数溢出问题。

按照第二个想法,为了避免数字反转可能导致的溢出问题,为什么不考虑只反转 \text{int}int 数字的一半?毕竟,如果该数字是回文,其后半部分反转后应该与原始数字的前半部分相同。

例如,输入 1221,我们可以将数字 “1221” 的后半部分从 “21” 反转为 “12”,并将其与前半部分 “12” 进行比较,因为二者相同,我们得知数字 1221 是回文。

算法

首先,我们应该处理一些临界情况。所有负数都不可能是回文,例如:-123 不是回文,因为 - 不等于 3。所以我们可以对所有负数返回 false。除了 0 以外,所有个位是 0 的数字不可能是回文,因为最高位不等于 0。所以我们可以对所有大于 0 且个位是 0 的数字返回 false。

现在,让我们来考虑如何反转后半部分的数字。

对于数字 1221,如果执行 1221 % 10,我们将得到最后一位数字 1,要得到倒数第二位数字,我们可以先通过除以 10 把最后一位数字从 1221 中移除,1221 / 10 = 122,再求出上一步结果除以 10 的余数,122 % 10 = 2,就可以得到倒数第二位数字。如果我们把最后一位数字乘以 10,再加上倒数第二位数字,1 * 10 + 2 = 12,就得到了我们想要的反转后的数字。如果继续这个过程,我们将得到更多位数的反转数字。

现在的问题是,我们如何知道反转数字的位数已经达到原始数字位数的一半?

由于整个过程我们不断将原始数字除以 10,然后给反转后的数字乘上 10,所以,当原始数字小于或等于反转后的数字时,就意味着我们已经处理了一半位数的数字了。

在这里插入图片描述

作者:LeetCode-Solution
链接:https://leetcode.cn/problems/palindrome-number/solution/hui-wen-shu-by-leetcode-solution/
来源:力扣(LeetCode)

解题步骤

根据官方给的题解用Python实现一下
总的来说就是翻转一半的数字,即后半部分。
1.首先将小于0或者能被10整除的整数(不包括0)返回false
大于等于0并且小于10的整数(0到9都是回文数) 返回true

if x < 0 or (x % 10 == 0 and x != 0):
            return False
if x < 10 and x >= 0:
            return True
            

2.翻转后半部分数字

y = 0
while x > y:
    y = y * 10 + x % 10
    x //= 10

x % 10取余
x //= 10取整

3.判断是否相同

  if x == y:
        return True
    else:
        return False

但是当x为奇数时会出现翻转后会多一个0,例如111翻转后为1110
因此需加上多一个判断条件,x == y // 10
又因为0到9也是奇数翻转后多一个0
所以可略去判断 x < 10 and x >= 0

代码整合

class Solution(object):
    def isPalindrome(self, x):
        """
        :type x: int
        :rtype: bool
        """
        if x < 0 or (x % 10 == 0 and x != 0):
            return False
  
        y = 0
        while x > y:
            y = y * 10 + x % 10
            x //= 10
        if x == y or x == y // 10:
            return True
        else:
            return False
【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。