剑指offer-Go版实现 第三章:高质量的代码

Golang
325
0
0
2022-08-11

章节主要目的

主要是围绕高质量代码,完成性、规范性和鲁棒性。

测试用例的编写,通常情况,我们为了完成单元测试覆盖率都是草草了事,学习这一章我们可以发现,我们之前的测试用例实用性很差,只有正常情况的用例,并且很多时候只有一个。之后我也跟着用例从三方面来运行:1、功能测试,2、负面测试,3、边界测试。负面测试可以理解为错误测试,异常情况的测试。

正式开始第三章

面试题11:数值的整数次方

leetcode-cn.com/problems/shu-zhi-d...

按照题意,很容易写出来基本的逻辑代码。

func myPow(x float64, n int) float64 {
   var result = 1.0

   if n < 0 {
       x = 1 / x
       n = -n
   }
   for i := 0; i < n; i++ {
       result *= x
   }
   return result
}

可惜,会提示超时,尴尬的很,于是乎看了一遍原文的逻辑,重新实现一遍,逻辑如下:

快速幂 算法

算法流程:

当 x = 0 时:直接返回 0 (避免后续 x = 1 / x操作报错)。

初始化 res = 1

当 n < 0 时:把问题转化至 n≥0 的范围内,即执行 x = 1/x ,n = - n ;

循环计算:当 n = 0时跳出;

当 n & 1 == 1 时:将当前 x乘入 res (即 res *= x);

执行 x = x^2 (即 x *= x)

执行 n 右移一位(即 n >>= 1)。

返回 res

func myPow(x float64, n int) float64 {
    var result = 1.0

    if n < 0 {
        x = 1 / x
        n = -n
    }
    for ; n > 0; {
        if n&1 == 1 { //又学了一招位运算,等价于 n % 2 == 1
            result *= x
        }
        x *= x
        n >>= 1 // 指数减半
    }
    return result
}

面试题12: 打印1到最大的n位数

leetcode-cn.com/problems/da-yin-co...

虽然能通过leetcode,但是不符合书本上出题的目的,因为要考虑大数的情况下才行,书上给出的思路是转换成字符串来进行输出

func printNumbers(n int) []int {
   res := make([]int,0)
   var count = 1 
   for i := 1; i < n; i++ {
       count *= 10
   }

   for i := 0; i < count; i++ {
       res = append(res,i)
   }

   return res
}

这不包括大数,不是书本出题的本意,修改如下:

考察的是代码的完整性 返回就不能是[]int 而应该是[]string,看具体代码,没办法在leetcode验证,只能自己打印日志看了。

func printNumbers(n int) []string {
    if n <= 0 {
        return nil
    }
    res := make([]int, 0)
    result := make([]string, 0)
    number := make([]int, n)
    for !increment(number) {
        for i := 0; i < len(number); i++ {
            res = append(res, number[i])
        }
    }
    //数组按n位切分成而为数组
    temp := make([]int, 0)
    for i := 0; i < len(res); i++ {
        temp = append(temp, res[i])
        if (i+1)%n == 0 { // 3余数是0 。说明是n的整数倍
            result = append(result, getString(temp))
            temp = make([]int, 0)
        }
    }
    return result
}

func getString(number []int) string {
    var res string
    isBegin := false 
    for i := 0; i < len(number); i++ {
        if number[i] != 0 {
            isBegin = true
        }
        if isBegin {
            res += fmt.Sprintf("%d", number[i])
        }
    }
    return res
}

func increment(number []int) bool {
    var isOverFlow = false //是否溢出结束 
    var nTakeOver = 0      // 进位 
    var nLength = len(number)
    for i := nLength - 1; i >= 0; i-- {
        nSum := number[i] + nTakeOver
        if nLength-1 == i {
            nSum++
        }
        if nSum >= 10 {
            if i == 0 {
                isOverFlow = true
            } else {
                nSum -= 10
                nTakeOver = 1
                number[i] = nSum
            }
        } else {
            number[i] = nSum
            break
        }
    }
    return isOverFlow
}

面试题13: 在O(1)时间删除链表节点

leetcode-cn.com/problems/shan-chu-...

leetcode要求比书本上的低,没有时间复杂度的要求

// 先不考虑O(1),遍历是最直接的
func deleteNode(head *ListNode, val int){
   if head == nil {
       return
   }
   for head.Next != nil {
       if head.Next.Val == val {
           head.Next = head.Next.Next
       }
       head = head.Next
   }
}

链表少不了假头、新链表、双指针等辅助,目前就是使用假头的最佳实例,和书上的不一样哈,课本上的删除就完了,不用返回,力扣是要删除后返回头节点的

func deleteNode(head *ListNode, val int) *ListNode {
   //for head.Next != nil { 
   //    if val == head.Next.Val { 
   //        head = head.Next.Next//删除节点 课本上这就结束了。 
   //    } 
   //}
   dummy := &ListNode{}// 生成一个新链表
   tail := dummy
   for head != nil {
       if head.Val != val {
           //添加到新的链表中
           tail.Next = head
           tail = head
       }
       head = head.Next
   }
   tail.Next = nil //设置尾部为空 
   return dummy.Next
}

上面的例子中,空间复杂度是O(n),想要是1的话,只能按照课本上的参数传递才行,否者不可实现

func deleteNode(head *ListNode, deleteNode *ListNode) {
   if head == nil || deleteNode == nil {
       return
   }
   if deleteNode.Next != nil {//不是尾节点 
       //删除的节点下一个值,替换需要删除的节点,然后删除下一个节点,等同于删除当前节点
       deleteNode.Val = deleteNode.Next.Val
       deleteNode.Next = deleteNode.Next.Next
   }else if head == deleteNode {//删除的是头节点,只有一个节点
       head = nil
   }else{ // 多个节点,删除的是尾节点 第一个if条件难道不包括吗? 
       for head.Next != deleteNode {
           head = head.Next
       }
       head.Next = nil
   }
}

面试题14: 调整数组顺序使奇数位于偶数前面

leetcode-cn.com/problems/diao-zhen...

思路:两个指针,一个指向头,一个指向尾部,两边同时向中间移动,如果头指针的值偶数,尾指针是奇数就交换

通用性:判断条件改为函数来判断,分离出来nums[left]&1 == 1和nums[right]&1 == 0换成函数就可以了,很简单,自行添加一下。

func exchange(nums []int) []int {
    if len(nums) == 0 {
        return nil
    }
    var left = 0 
    var right = len(nums) - 1

    for left < right {
        for left < right && nums[left]&1 == 1 { //奇数,右移left
            left++
        }
        for left < right && nums[right]&1 == 0 { //偶数,左移right
            right--
        }
        if left < right { //交换
            nums[left], nums[right] = nums[right], nums[left]
        }
    }
    return nums
}

面试题15: 链表中倒数第k个节点

leetcode-cn.com/problems/lian-biao...

思路:两个指针,一个指向头,一个先走k步,简称:快慢指针

func getKthFromEnd(head *ListNode, k int) *ListNode {
   if head != nil {
       return nil
   }
   var first = head
   var second = head
   for i := 0; i < k; i++ {
       second = second.Next
   }
   for second != nil {
       first = first.Next
       second = second.Next
   }
   return first
}

面试题16: 反转链表

leetcode-cn.com/problems/UHnkqh/

这题,如果没有限制返回头节点,仅仅是输出的话,实现方式是N多种,数组也是可以的。但是这里的要求是返回反转后的头节点,所以,需要辅助手段。

//需要把节点保存起来,断开之后防止找不到
func reverseList(head *ListNode) *ListNode {
   var prevHead *ListNode //保存遍历的前一个节点 
   var currentNode = head //保存遍历的当前节点 
   for currentNode != nil {
       nextNode := currentNode.Next
       currentNode.Next = prevHead
       prevHead = currentNode
       currentNode = nextNode
   }
   return prevHead
}

按照课本上的做法,我临摹过来的代码,但是,这个代码我感觉有点小问题,却又不知道问题在哪,各位朋友知道的还望指出来,并改正一下,不胜感激。

func reverseList(head *ListNode) *ListNode {
    var reversedHead *ListNode //反转后的头节点 
    var prev *ListNode
    var node = head
    for node != nil {
        next := node.Next
        if next == nil {
            reversedHead = node
        }
        node.Next = prev
        prev = node
        node = next
    }
    return reversedHead
}

面试题17: 合并两个排序的链表

leetcode-cn.com/problems/he-bing-l...

同样利用了链表的假头和新链表作为辅助

func mergeTwoLists(l1 *ListNode, l2 *ListNode) *ListNode {
    dummy := &ListNode{}
    tail := dummy
    for l1 != nil && l2 != nil {
        if l1.Val < l2.Val {
            tail.Next = l1
            l1 = l1.Next
        } else {
            tail.Next = l2
            l2 = l2.Next
        }
        tail = tail.Next
    }
    if l1 != nil {// 如果l1还有,直接拼接到后面
        tail.Next = l1
        tail = tail.Next
        l1 = l1.Next
    }

    if l2 != nil {//如果l2还有,直接拼接后面
        tail.Next = l2
        tail = tail.Next
        l2 = l2.Next
    }
    return dummy.Next
}

面试题18: 树的子结构

leetcode-cn.com/problems/shu-de-zi...

输入两棵二叉树A和B,判断B是不是A的子结构。(约定空树不是任意一个树的子结构)

主要和树相关的,就没有简单级别的,当然,和递归相关的也都是没有简单的。不熟悉递归和树的,这题需要点时间去理解。

func isSubStructure(A *TreeNode, B *TreeNode) bool {
    var result = false 
    if A != nil && B != nil {
        //先比较根节点,再比较左右子树节点 
        if A.Val == B.Val {
            //根节点满足,判断不是子结构
            result = isStructure(A,B) //不用递归,也可写成左右子树的判断
        }
        //比较左子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Left,B)
        }
        //比较右子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Right,B)
        }
    }
    return result
}

func isStructure(a *TreeNode, b *TreeNode) bool {
    if b == nil {//说明树 B 已匹配完成(越过叶子节点),因此返回 true 
        return true
    }
    if a == nil {//说明已经越过树 A 叶子节点,即匹配失败,返回 false 
        return false
    }
    if a.Val != b.Val {//值不同,匹配失败,返回 false 
        return false
    }
    return isStructure(a.Left,b.Left) && isStructure(a.Right,b.Right)
}
``````go
func isSubStructure(A *TreeNode, B *TreeNode) bool {
    var result = false 
    if A != nil && B != nil {
        //先比较根节点,再比较左右子树节点 
        if A.Val == B.Val {
            //根节点满足,判断不是子结构
            result = isStructure(A,B) //不用递归,也可写成左右子树的判断
        }
        //比较左子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Left,B)
        }
        //比较右子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Right,B)
        }
    }
    return result
}

func isStructure(a *TreeNode, b *TreeNode) bool {
    if b == nil {//说明树 B 已匹配完成(越过叶子节点),因此返回 true 
        return true
    }
    if a == nil {//说明已经越过树 A 叶子节点,即匹配失败,返回 false 
        return false
    }
    if a.Val != b.Val {//值不同,匹配失败,返回 false 
        return false
    }
    return isStructure(a.Left,b.Left) && isStructure(a.Right,b.Right)
}
``````go
func isSubStructure(A *TreeNode, B *TreeNode) bool {
    var result = false 
    if A != nil && B != nil {
        //先比较根节点,再比较左右子树节点 
        if A.Val == B.Val {
            //根节点满足,判断不是子结构
            result = isStructure(A,B) //不用递归,也可写成左右子树的判断
        }
        //比较左子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Left,B)
        }
        //比较右子树为根节点,开始遍历比较 
        if !result {
            result = isSubStructure(A.Right,B)
        }
    }
    return result
}

func isStructure(a *TreeNode, b *TreeNode) bool {
    if b == nil {//说明树 B 已匹配完成(越过叶子节点),因此返回 true 
        return true
    }
    if a == nil {//说明已经越过树 A 叶子节点,即匹配失败,返回 false 
        return false
    }
    if a.Val != b.Val {//值不同,匹配失败,返回 false 
        return false
    }
    return isStructure(a.Left,b.Left) && isStructure(a.Right,b.Right)
}

接下来的算法都是非常难的,我也不知道多久能研究透彻,可以确定的是更新时间肯定比第二第三章要晚。