Golang每日一练(leetDay0115) 重新安排行程、递增的三元子序列

简介: Golang每日一练(leetDay0115) 重新安排行程、递增的三元子序列

332. 重新安排行程 Reconstruct Itinerary

给你一份航线列表 tickets ,其中 tickets[i] = [fromi, toi] 表示飞机出发和降落的机场地点。请你对该行程进行重新规划排序。

所有这些机票都属于一个从 JFK(肯尼迪国际机场)出发的先生,所以该行程必须从 JFK 开始。如果存在多种有效的行程,请你按字典排序返回最小的行程组合。

  • 例如,行程 ["JFK", "LGA"]["JFK", "LGB"] 相比就更小,排序更靠前。

假定所有机票至少存在一种合理的行程。且所有的机票 必须都用一次 且 只能用一次。

示例 1:

输入:tickets = [["MUC","LHR"],["JFK","MUC"],["SFO","SJC"],["LHR","SFO"]]

输出:["JFK","MUC","LHR","SFO","SJC"]


示例 2:

输入:tickets = [["JFK","SFO"],["JFK","ATL"],["SFO","ATL"],["ATL","JFK"],["ATL","SFO"]]

输出:["JFK","ATL","JFK","SFO","ATL","SFO"]

解释:另一种有效的行程是 ["JFK","SFO","ATL","JFK","ATL","SFO"] ,但是它字典排序更大更靠后。


提示:

  • 1 <= tickets.length <= 300
  • tickets[i].length == 2
  • fromi.length == 3
  • toi.length == 3
  • fromitoi 由大写英文字母组成
  • fromi != toi

代码:

package main
import (
  "fmt"
  "sort"
)
func findItinerary(tickets [][]string) []string {
  // 构建图的邻接表
  graph := make(map[string][]string)
  for _, ticket := range tickets {
    from, to := ticket[0], ticket[1]
    graph[from] = append(graph[from], to)
  }
  // 对邻接表中的目的地进行字典排序
  for _, destinations := range graph {
    sort.Strings(destinations)
  }
  // 深度优先遍历,获取行程
  var itinerary []string
  var dfs func(from string)
  dfs = func(from string) {
    for len(graph[from]) > 0 {
      to := graph[from][0]
      graph[from] = graph[from][1:]
      dfs(to)
    }
    itinerary = append(itinerary, from)
  }
  dfs("JFK")
  // 将行程逆序,得到正确顺序
  for i, j := 0, len(itinerary)-1; i < j; i, j = i+1, j-1 {
    itinerary[i], itinerary[j] = itinerary[j], itinerary[i]
  }
  return itinerary
}
func main() {
  tickets := [][]string{{"MUC", "LHR"}, {"JFK", "MUC"}, {"SFO", "SJC"}, {"LHR", "SFO"}}
  fmt.Println(findItinerary(tickets))
  tickets = [][]string{{"JFK", "SFO"}, {"JFK", "ATL"}, {"SFO", "ATL"}, {"ATL", "JFK"}, {"ATL", "SFO"}}
  fmt.Println(findItinerary(tickets))
}

输出:

[JFK MUC LHR SFO SJC]

[JFK ATL JFK SFO ATL SFO]


334. 递增的三元子序列 Increasing Triplet Subsequence

给你一个整数数组 nums ,判断这个数组中是否存在长度为 3 的递增子序列。

如果存在这样的三元组下标 (i, j, k) 且满足 i < j < k ,使得 nums[i] < nums[j] < nums[k] ,返回 true ;否则,返回 false

示例 1:

输入:nums = [1,2,3,4,5]

输出:true

解释:任何 i < j < k 的三元组都满足题意


示例 2:

输入:nums = [5,4,3,2,1]

输出:false

解释:不存在满足题意的三元组

示例 3:

输入:nums = [2,1,5,0,4,6]

输出:true

解释:三元组 (3, 4, 5) 满足题意,因为 nums[3] == 0 < nums[4] == 4 < nums[5] == 6


提示:

  • 1 <= nums.length <= 5 * 10^5
  • -2^31 <= nums[i] <= 2^31 - 1

进阶:你能实现时间复杂度为 O(n) ,空间复杂度为 O(1) 的解决方案吗?

代码1:动态规划

package main
import "fmt"
func increasingTriplet(nums []int) bool {
  if len(nums) < 3 {
    return false
  }
  first := nums[0]  // 记录当前最小值
  second := 1 << 31 // 初始化为一个较大的值
  for i := 1; i < len(nums); i++ {
    if nums[i] <= first {
      first = nums[i]
    } else if nums[i] <= second {
      second = nums[i]
    } else {
      // 找到了递增的三元子序列
      return true
    }
  }
  return false
}
func main() {
  nums := []int{1, 2, 3, 4, 5}
  result := increasingTriplet(nums)
  fmt.Println(result)
  nums = []int{5, 4, 3, 2, 1}
  result = increasingTriplet(nums)
  fmt.Println(result)
  nums = []int{2, 1, 5, 0, 4, 6}
  result = increasingTriplet(nums)
  fmt.Println(result)
}

代码2:二分查找

package main
import "fmt"
func increasingTriplet(nums []int) bool {
    n := len(nums)
    if n < 3 {
        return false
    }
    subSeq := make([]int, 0, 3) // 存储递增子序列
    for _, num := range nums {
        if len(subSeq) == 0 || num > subSeq[len(subSeq)-1] {
            subSeq = append(subSeq, num)
            if len(subSeq) == 3 {
                return true
            }
        } else {
            left, right := 0, len(subSeq)-1
            for left < right {
                mid := left + (right-left)/2
                if subSeq[mid] >= num {
                    right = mid
                } else {
                    left = mid + 1
                }
            }
            subSeq[right] = num
        }
    }
    return false
}
func main() {
  nums := []int{1, 2, 3, 4, 5}
  result := increasingTriplet(nums)
  fmt.Println(result)
  nums = []int{5, 4, 3, 2, 1}
  result = increasingTriplet(nums)
  fmt.Println(result)
  nums = []int{2, 1, 5, 0, 4, 6}
  result = increasingTriplet(nums)
  fmt.Println(result)
}

输出:

true

false

true

三重循环暴力枚举:

```golang
func increasingTriplet(nums []int) bool {
    for i := 0; i < len(nums); i++ {
        for j := i+1; j < len(nums); j++ {
            if nums[j] > nums[i] {
                for k := j+1; k < len(nums); k++ {
                    if nums[k] > nums[j] {
                        return true
                    }
                }
            }
        }
    }
    return false
}
```

🌟 每日一练刷题专栏 🌟

持续,努力奋斗做强刷题搬运工!

👍 点赞,你的认可是我坚持的动力!

🌟 收藏,你的青睐是我努力的方向!

评论,你的意见是我进步的财富!  

主页:https://hannyang.blog.csdn.net/

Rust每日一练 专栏

(2023.5.16~)更新中...

Golang每日一练 专栏

(2023.3.11~)更新中...

Python每日一练 专栏

(2023.2.18~2023.5.18)暂停更

C/C++每日一练 专栏

(2023.2.18~2023.5.18)暂停更

Java每日一练 专栏

(2023.3.11~2023.5.18)暂停更


目录
相关文章
|
搜索推荐 编译器 C语言
【C++核心】特殊的元素集合-数组与字符串详解
这篇文章详细讲解了C++中数组和字符串的基本概念、操作和应用,包括一维数组、二维数组的定义和使用,以及C风格字符串和C++字符串类的对比。
319 5
|
10月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
199 5
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
算法 C++
c++学习笔记04 数组
这篇文章是C++学习笔记4,主题是数组。
122 4
|
C++ 索引 运维
开发与运维数组问题之在C++中数组名和指针是等价如何解决
开发与运维数组问题之在C++中数组名和指针是等价如何解决
103 6
|
C++ 索引
C++数组、vector求最大值最小值及其下标
C++数组、vector求最大值最小值及其下标
716 0
|
C++
C++代码来计算一个点围绕另一个点旋转45度后的坐标
C++代码来计算一个点围绕另一个点旋转45度后的坐标
298 0
|
安全 编译器 C语言
C++入门-数组
C++入门-数组
|
1月前
|
存储 安全 Java
【Golang】(4)Go里面的指针如何?函数与方法怎么不一样?带你了解Go不同于其他高级语言的语法
结构体可以存储一组不同类型的数据,是一种符合类型。Go抛弃了类与继承,同时也抛弃了构造方法,刻意弱化了面向对象的功能,Go并非是一个传统OOP的语言,但是Go依旧有着OOP的影子,通过结构体和方法也可以模拟出一个类。
130 1
|
Go
Golang语言之管道channel快速入门篇
这篇文章是关于Go语言中管道(channel)的快速入门教程,涵盖了管道的基本使用、有缓冲和无缓冲管道的区别、管道的关闭、遍历、协程和管道的协同工作、单向通道的使用以及select多路复用的详细案例和解释。
599 4
Golang语言之管道channel快速入门篇