分类总结

2018-08-23  本文已影响15人  lifesmily

1、判断链表循环

141、 Linked List Cycle

判断链表是否是循环链表。思路主要有两种,第一种是将所有的节点的next指向head,如果存在环则会出现某个节点的next是head,否则最后一个元素next为None,非环。

# Definition for singly-linked list.
class ListNode(object):
    def __init__(self, x):
        self.val = x
        self.next = None

class Solution(object):
    def hasCycle(self, head):
        """
        :type head: ListNode
        :rtype: bool
        """
        node = head
        while node is not None:
            if node.next == head:
                return True
            temp = node.next
            node.next = head
            node = temp
        return False

第二种思路是使用两个指针,一个快指针和一个慢指针,但效率并不是很高,因为不一定会在一个回合内相遇,可能需要几个遍历才能相遇。

 def hasCycle1(self, head):
        """
        :type head: ListNode
        :rtype: bool
        """
        slow, fast = head, head
        while fast and fast.next:
            slow = slow.next
            fast = fast.next.next
            if slow == fast:
                return True
        return False
142. Linked List Cycle II

Given a linked list, return the node where the cycle begins. If there is no cycle, return null.
Note: Do not modify the linked list.

由于明确要求不能修改链表,所以141中第一种方法就不能使用。由于第二种方法相遇节点并不一定是起始节点,需要特殊分析,如下:


image.png
image.png

参考来源:http://blog.csdn.net/ebowtang/article/details/50507131

class Solution(object):
    def detectCycle(self, head):
        """
        :type head: ListNode
        :rtype: ListNode
        """
        slow, fast = head, head
        while fast and fast.next:
            slow = slow.next
            fast = fast.next.next
            if slow == fast:
                break
        if fast is None or fast.next is None:
            return None
        fast = head
        while slow != fast:
            slow = slow.next
            fast = fast.next
        return slow

2、两数之和、三数之和

1、two sum

找到两数之和为target值的下标号列表。最简单的方法就是哈希表,如下:

class Solution(object):
    def twoSum(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: List[int]
        """
        lookup = {}
        for i, item in enumerate(nums):
            if target - item in lookup:
                return [lookup[target-item], i]
            lookup[item] = i
        return [-1, -1]

two sum还有一种方法是双指针,一头一尾,但是需要对元素进行排序。这样的话如果不考虑排序时间复杂度,可以在O(n)内解决。

15. 3Sum

找到三数之和为0的所有组合,先要对数组进行排序,确定一个数后再利用two sum,由于数组是排序,解决two sum问题可以采用双指针,从前和后两个方向查找。同时,这个题目需要注意去重问题,具体到实现上在两个地方,如下:

class Solution(object):
    def threeSum(self, nums):
        """
        :type nums: List[int]
        :rtype: List[List[int]]
        """
        res, length = [], len(nums)
        if length < 3:
            return res
        nums = sorted(nums)
        for i in xrange(length-2):
            if i > 0 and nums[i] == nums[i-1]:
                continue
            temp = 0 - nums[i]
            left, right = i + 1, length - 1
            while left < right:
                if nums[left] + nums[right] == temp:
                    res.append([nums[i], nums[left], nums[right]])
                    while left < right and nums[right] == nums[right-1]:
                        right -= 1
                    while left < right and nums[left] == nums[left+1]:
                        left += 1
                    right -= 1
                    left += 1
                elif nums[left] + nums[right] > temp:
                    right -= 1
                else:
                    left += 1
        return res
16、 3Sum Closest
class Solution(object):
    def threeSumClosest(self, nums, target):
        """
        :type nums: List[int]
        :type target: int
        :rtype: int
        """
        res, length = -1, len(nums)
        if length < 3:
            return res
        nums.sort()
        res = float("inf")
        for i in xrange(length-2):
            if i > 0 and nums[i] == nums[i-1]:
                continue
            left, right = i + 1, length - 1
            while left < right:
                temp = nums[i]+nums[left]+nums[right]
                if abs(temp - target) < abs(res - target):
                    res = temp
                if temp > target:
                    right -= 1
                elif temp < target:
                    left += 1
                else:
                    return target
        return res

找到三个数使之与指定target值最接近,思路和15题类似,先固定一个数,找另外两个数,使用双指针的方法,确定判断指针移动条件时需要注意。


3、house rober问题

题意要求不能连续抢相邻两家。

class Solution(object):
    def rob(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """
       if nums == []:
            return 0
        if len(nums) == 1:
            return nums[0]
        rob, notrob = nums[0], 0
        for item in nums[1:]:
            rob, notrob = notrob + item, max(notrob, rob)
        return max(rob, notrob)

加入约束条件,第一家和最后一家不能同时抢。

class Solution(object):
    def rob(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """
        if len(nums) < 2:
            return sum(nums)
        # situation 1 rob first
        rob, notrob = nums[0], nums[0]
        for item in nums[2:]:
            rob, notrob = notrob + item, max(rob,notrob)
        
        # situation 2 notrob first
        rob1, notrob1 = nums[1], 0
        for item in nums[2:]:
            rob1, notrob1 = notrob1 + item, max(rob1, notrob1)
        return max(notrob,rob1,notrob1)

4 path sum问题

1)存在问题,是否有路径和为某一个数

leetcode 112

class Solution:
    # @param root, a tree node
    # @param sum, an integer
    # @return a boolean
    def hasPathSum(self, root, sum):
        if root is None:
            return False

        if root.left is None and root.right is None and root.val == sum:
            return True

        return self.hasPathSum(root.left, sum - root.val) or \
               self.hasPathSum(root.right, sum - root.val)

需要注意的是一定是判断到叶子节点(左右子树均为None),然后利用树的天然递归性求解。

2)求解所有方案问题

leetcode 113

class Solution(object):
    def pathSum(self, root, sum):
        """
        :type root: TreeNode
        :type sum: int
        :rtype: List[List[int]]
        """
        if not root: return []
        if root.left is None and root.right is None:
            if sum == root.val:
                return [[root.val]]
            else:
                return []
        a = self.pathSum(root.left, sum - root.val) + \
            self.pathSum(root.right, sum - root.val)
        return [[root.val] + i for i in a]

上面的方法比较巧妙,但也比较难想得到。这种类型的问题首先想到的是遍历,类似回溯求解合适解,

class Solution:
    # 返回二维列表,内部每个列表表示找到的路径
    def FindPath(self, root, expectNumber):
        # write code here
        if root is None:
            return []
        self.res = []
        self.helper(root, [], expectNumber)
        return self.res

    def helper(self, root, subres, number):
        if number == root.val and root.left is None and root.right is None:
            subres.append(root.val)
            self.res.append(subres[:])
            subres.pop()
            return
        if number < 0:
            return
        if root.left:
            subres.append(root.val)
            self.helper(root.left, subres, number-root.val)
            subres.pop()
        if root.right:
            subres.append(root.val)
            self.helper(root.right, subres, number - root.val)
            subres.pop()

其实,leetcode 112是DFS的具体应用,遍历每一种情况,而leetcode 113中下面解法是回溯。

5、全排列问题

leetcode中46题,可以比较方便使用回溯,但有个前提,就是所有的数字都是不相同的。

class Solution(object):
    def permute(self, nums):
        """
        :type nums: List[int]
        :rtype: List[List[int]]
        """
        self.result = []
        sub = []
        self.dfs(nums,sub)
        return self.result

    def dfs(self, nums, sub):
        if len(sub) == len(nums):
            print sub
            self.result.append(sub[:])
        for m in nums:
            if m in sub:
                continue
            sub.append(m)
            self.dfs(nums, sub)
            sub.remove(m)

上面的方法比较直接,但是效率不高,有一种通用的方法:

class Solution:
    def Permutation(self, ss):
        # write code here
        self.res = []
        self.helper(list(ss), 0, len(ss))
        return self.res

    def helper(self, s, start, end):
        if start > end:
            return
        if start == end:
            self.res.append(s[:])
        else:
            for i in range(start, end):
                s[start], s[i] = s[i], s[start]
                self.helper(s, start + 1, end)
                s[start], s[i] = s[i], s[start]  # 也是回溯的思想
s = Solution()
print s.Permutation('abc')

输出为:

[['a', 'b', 'c'], ['a', 'c', 'b'], ['b', 'a', 'c'], ['b', 'c', 'a'], ['c', 'b', 'a'], ['c', 'a', 'b']]

理论基础:


image.png

而在剑指offer中的26题,情况不一样,要无重复,并且输入是字符串,如果用交换方法的化在python中不行,因为python中不能给字符串赋值。

# -*- coding:utf-8 -*-
class Solution:
    def Permutation(self, ss):
        # write code here
        if ss == '':
            return []
        self.res = set()
        self.helper(list(ss), 0, len(ss))
        return sorted(self.res)

    def helper(self, s, start, end):
        if start > end:
            return
        if start == end:
            self.res.add(''.join(s))
        else:
            for i in range(start, end):
                s[start], s[i] = s[i], s[start]
                self.helper(s, start + 1, end)
                s[start], s[i] = s[i], s[start]



import itertools
class Solution2:
    def Permutation(self, ss):
        # write code here
        result = []
        if not ss:
            return []
        else:
            res = itertools.permutations(ss)
            for i in res:
                if "".join(i) not in result:
                    result.append("".join(i))
        return result

6、组合问题

7、动态规划路径问题

给定一个 m x n 矩阵,求出左上节点到右下节点的一条最小路径。
这样的题目看起来是要找到一条路径,但还是需要将所有的中间节点的值都求出来。

class Solution(object):
    def minPathSum(self, grid):
        """
        :type grid: List[List[int]]
        :rtype: int
        """
        m, n = len(grid), len(grid[0])
        res = [[0 for _ in xrange(n)] for _ in xrange(m)]
        res[0][0] = grid[0][0]
        for i in xrange(1, m):
            res[i][0] = res[i-1][0] + grid[i][0]
        for j in xrange(1, n):
            res[0][j] = res[0][j-1] + grid[0][j]
        for i in xrange(1, m):
            for j in xrange(1, n):
                res[i][j] = min(res[i-1][j], res[i][j-1]) + grid[i][j]

        return res[m-1][n-1]

    def minpathsum2(self,grid):
        '''

        '''
        res = grid[0]
        m, n = len(grid), len(grid[0])
        for j in xrange(1, n):
            res[j] = res[j-1] + grid[0][j]
        for i in xrange(1, m):
            res[0] = res[0] + grid[i][0]
            for j in xrange(1, n):
                res[j] = min(res[j], res[j-1]) + grid[i][j]

        return res[-1]

第一种方法是简单直接的,使用一个二维数组保存中间结果,如果原数组可以直接修改的话可以不用创建一个新的数组。第二种方法只用了一个一维数组,不难理解。
一维数组优化方法详细参考 https://blog.csdn.net/happyaaaaaaaaaaa/article/details/50856112

8、排序链表删除问题

# -*- coding:utf-8 -*-
class ListNode:
    def __init__(self, x):
        self.val = x
        self.next = None
# 重复的节点不保存
class Solution:
    def deleteDuplication(self, pHead):
        # write code here
        if pHead is None or pHead.next is None:
            return pHead
        head = pHead
        while pHead and pHead.next:
            if pHead.val != pHead.next.val:
                pHead = pHead.next
            else:
                pHead.next = pHead.next.next
        return head

# 删除重复节点,包括本身节点
class Solution1:
    def deleteDuplication(self, pHead):
        # write code here
        if pHead is None or pHead.next is None:
            return pHead
        dummy = ListNode(-1)
        dummy.next = pHead
        pre, head = dummy, dummy
        while pHead and pHead.next:
            if pHead.val != pHead.next.val:
                pHead = pHead.next
                pre = pre.next
            else:
                while pHead and pHead.next and pHead.val == pHead.next.val:
                    pHead = pHead.next
                pHead = pHead.next
                pre.next = pHead
        return dummy.next

上面为两种场景,第一是保存一个重复节点,第二是不保存。需要注意的是一些细节。
Java版

public class DeleteNodeInLinklist {
    public static ListNode DeleteDupNode(ListNode node){
        if(node==null || node.next==null)
            return node;
        ListNode dummy = new ListNode(-1);
        dummy.next = node;
        ListNode ptr = dummy;
        while(node != null && node.next != null){
            if(node.val != node.next.val){
                node = node.next;
                ptr = ptr.next;
            }else {
                while((node != null) && (node.next != null) && (node.val==node.next.val)){
                    node = node.next;
                }
                node = node.next;
                ptr.next = node;
            }
        }
        return dummy.next;
    }
    // 删除重复节点但保留一个
    public static ListNode DeleteDupNodeWithOne(ListNode node){
        if(node==null || node.next==null)
            return node;
        ListNode current = node;
        while(current != null && current.next != null){
            if(current.val == current.next.val)
                current.next = current.next.next;
            else
                current = current.next;
        }
        return node;
    }
}
上一篇下一篇

猜你喜欢

热点阅读