Golang算法

Golang算法,第1张

注意点⚠️【刷题】 【回顾】 【常见算法】 ACM模式:fmt.Scan(&n) 只能往读入基本数据类型
	var n int
	var s string
	num, _ := fmt.Scan(&n, &s)
	if num == 0 {
		fmt.Println("没有元素")
	}else {
		fmt.Println("输入元素:",n,s)
	}

变量声明习惯
– 无需初始化:var
– 需要初始化:{}
– 除基本变量 让其他声明都可为nil

n,m = m,n

math方法

绝对值:func Abs(x float64) float64
次方值:func Pow(x, y float64) float64
最大值:func Max(x, y float64) float64
最小值:func Min(x, y float64) float64

向下取整:func Floor(x float64) float64
向上取整:func Ceil(x float64) float64

最大整数:MaxInt64
最小整数:MinInt64
递归:存在子集同父集想到递归返回所有可能:回溯算法返回最大、最优… 最值:动态规划但注意⚠️: 返回所有可能的确切值:回溯算法;返回所有可能的数量:动态规划!注意:类似于二分法的这种指针在循环中的,要保证每次循环指针有移动,不然就有可能陷入死循环Go也支持位运算啊啊啊啊:mid := (right + left) >> 1Go引用函数只能使用匿名函数换行才用,号!!!
	res := []int{3}
	res := []int{
		3,
	}
不要只使用基本数据类型啊!!!
使用工具结构体实现高效读写:
	/ 高效读写byte流
	buf := bytes.Buffer{}
	buf.WriteByte('1')
	buf.WriteString("2")
	buf.Bytes()
	/ 高效读写字符串
	str := strings.Builder{}
	str.WriteByte('1')
	str.WriteString("2")
	str.String()
大哥 var 声明四大引用类型和指针为nil! (切片、map、channel、接口)+ 指针
会导致空指针:
	var m map[int]int
	m[0] = 0
10e310*1032311<<31奇数:n&1 == 1 ;偶数:n&1 == 0string虽然不是基本数据类型,但底层是数组啊,string和[]byte[]rune可以直接转换
a := []byte{}
b := string(a)
c := []rune(b)
数组,字符串是基本数据类型啊 赋值会拷贝当出现点与点有向关联:画图 ⇒ 拓扑排序 【map存节点+入度】
func canFinish(numCourses int, prerequisites [][]int) bool {
    / topo[节点]入度
    topo := make(map[int]int,numCourses)
    。。。
    / 入度为0的节点存入stack
    stack := []int{}
    for i:=0;i<numCourses;i++ {
        if topo[i] == 0 {
            stack = append(stack,i)
        }
    }
    count := 0 / 也可以是[]res 返回拓扑过程
    for len(stack) > 0 {
        // 取出入度为0的节点
        zeroIndegree := stack[len(stack)-1]
        stack = stack[:len(stack)-1]
        count++
        // 将0度节点所指向的节点的入度减1
        for _,v := range prerequisites {
            if v[1] == zeroIndegree {
                topo[v[0]]--
                / 注意 :直接在这里再获取入度为0的点
                if topo[v[0]] == 0 {
                    stack = append(stack,v[0])
                }
            }
        }
    }
    return count == numCourses
}
一、链表 头节点前添加一个哨兵节点双指针排序:归并快慢指针注意点
1、使用fast.Next
2、fast先传递
for fast.Next != nil {
     fast = fast.Next.Next
     if fast == nil {
         break
     }
     slow = slow.Next
 }
二、数组 排序 默认递增
import "sort"

/ int切片
func Ints(a []int)
/ float64切片
func Float64s(a []float64)
/ string切片
func Strings(a []string)
自定义排序(<升序 、 >降序)
    sss := [][]int{}
    
    / 根据二维数组第一个元素排序
    / 注意⚠️:i、j是下标
	sort.Slice(sss, func(i, j int) bool {
		return sss[i][0] < sss[j][0]
	})
	
	/ 稳定排序
	sort.SliceStable(...,...)
复习:二分法 从1开始的数。。。关联数组下标。。。双指针copy(dst, src []Type) int函数要初始化目标切片长度 不然将复制min(len(dst),len(src))
	dst := make([]int,len(src))
	copy(dst,src)
前缀和:将遍历累加和存放到map中 三、二叉树 按层遍历递归YYDS:只思考第一步和第二步的关系 再加个递归终止条件就👌!!! 1. 手动遍历二叉树:

记住:

    for root != nil || len(stack) > 0 {
		for root != nil {
			......	
            stack = append(stack,root)
            root = root.Left
        }
        ......
   }

前序遍历

func preorderTraversal(root *TreeNode) []int {
    var res []int
    stack := []*TreeNode{}
    for root != nil || len(stack) > 0 {
        for root != nil {
            res = append(res,root.Val)
            stack = append(stack,root)
            root = root.Left
        }
        root = stack[len(stack)-1].Right
        stack = stack[:len(stack)-1]
    }
    return res
}

中序遍历

func inorderTraversal(root *TreeNode) []int {
    var res []int
    stack := []*TreeNode{}
	for root != nil || len(stack) > 0 {
		for root != nil {
			stack = append(stack, root)
			root = root.Left
		}
		root = stack[len(stack)-1].Right
		res = append(res, stack[len(stack)-1].Val)
        stack = stack[:len(stack)-1]
	}
	return res
}

后续遍历(多一个变量:上一个被记录的节点)

func postorderTraversal(root *TreeNode) []int {
    var res []int
    stack := []*TreeNode{}
    var pre *TreeNode //上一个被记录的元素
    for root != nil || len(stack) > 0 {
        for root != nil {
            stack = append(stack,root)
            root = root.Left
        }
        node := stack[len(stack)-1]
        if node.Right != nil && node.Right != pre {
            root = node.Right
        }else{
            res = append(res,node.Val)
            pre = node //该节点已被记录
            stack = stack[:len(stack)-1]
        }
    }
    return res
}
2. 🌲任意节点间路径问题:

路径:在二叉树中走一条线 且每个节点只走一遍
124. 二叉树中最大路径和
543. 二叉树的直径

func diameterOfBinaryTree(root *TreeNode) int {
    // 全局遍历作为最终结果
    var res int
    // 求数深度函数
    var depth func(*TreeNode) int
    depth = func(node *TreeNode) int {
    	......
    	/ 递归求左右子树的深度
        leftDepth := depth(node.Left)
        rightDepth := depth(node.Right)
        
        ......
        将root + left + right 作为更新最终结果res的一种情况
		......
        / 而只将 root+left 和 root+right作为返回的结果
        return max(leftPath,rightPath)+1
    }
    depth(root)
    return res
}

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/995591.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-21
下一篇 2022-05-21

发表评论

登录后才能评论

评论列表(0条)

保存