当前位置: 首页 > article >正文

力扣100热题:两、三、四数之和,哈希+数组+双指针+排序

目录

一、两数之和

二、两数之和 II - 输入有序数组

三、两数之和 III - 数据结构设计

四、两数之和 IV - 输入 BST(二叉搜索树)

五、三数之和

六、四数之和


一、两数之和

题目:1. 两数之和

参考力扣题解:. - 力扣(LeetCode)

官方两种解法:第一种是暴力枚举,第二种是哈希表。这两种解法都比较简单,实现起来也不复杂。

这里我自己使用golang的对象方式,写了一个,供参考

type twoSumData struct {
	nums      []int
	target    int
	hashTable map[int]int
}

func (t *twoSumData) twoSumBase() []int {
	n := len(t.nums)
	if n <= 1 {
		return nil
	}

	for i := 0; i < n; i++ {
		for j := i + 1; j < n; j++ {
			if t.nums[i]+t.nums[j] == t.target {
				return []int{i, j}
			}
		}
	}

	return nil
}

func (t *twoSumData) twoSumHashTable() []int {
	n := len(t.nums)
	if n <= 1 {
		return nil
	}

	for i := 0; i < n; i++ {
		if j, ok := t.hashTable[t.target-t.nums[i]]; ok {
			return []int{i, j}
		}
		t.hashTable[t.nums[i]] = i
	}

	return nil
}

func twoSum(nums []int, target int) []int {
	data := &twoSumData{
		nums:      nums,
		target:    target,
		hashTable: make(map[int]int),
	}
	return data.twoSumHashTable()
}

二、两数之和 II - 输入有序数组

题目:167. 两数之和 II - 输入有序数组

还是基于前面的结构体,使用双指针,从两侧往中间,找到符合条件的结果。

func (t *twoSumData) twoSumForSortNums() []int {
	nums := t.nums
	n := len(nums)

	left, right := 0, n-1
	for left < right {
		sum := nums[left] + nums[right]
		if sum == t.target {
			return []int{left + 1, right + 1}
		} else if sum < t.target {
			left++
		} else {
			right--
		}
	}
	return nil
}

func twoSum(nums []int, target int) []int {
	data := &twoSumData{
		nums:      nums,
		target:    target,
		hashTable: make(map[int]int),
	}
	return data.twoSumForSortNums()
}

三、两数之和 III - 数据结构设计

题目:170. 两数之和 III - 数据结构设计

使用双指针,查询

type TwoSum struct {
	nums   []int
	target int
}

func Constructor() TwoSum {
	return TwoSum{
		nums:   make([]int, 0),
		target: 0,
	}
}

func (this *TwoSum) Add(number int) {
	this.nums = append(this.nums, number)
}

func (this *TwoSum) Find(value int) bool {
	sort.Ints(this.nums)
	left, right := 0, len(this.nums)-1
	for left < right {
		sum := this.nums[left] + this.nums[right]
		if sum == value {
			return true
		} else if sum < value {
			left++
		} else {
			right--
		}
	}
	
	return false
}

四、两数之和 IV - 输入 BST(二叉搜索树)

题目:653. 两数之和 IV - 输入二叉搜索树

这个题目,实际上就是在前两题的基础上,将输入修改为二叉搜索树。

题目的解法有个比较简单的方法,我们将二叉搜索树给换成数组,然后调用一、二题的函数即可。

比较复杂的方法,就是利用二叉搜索树的特点,二叉搜索树必然满足root.left.val < root.val < root.right.val

可以参考下官方题解:两数之和 IV - 输入 BST - 力扣官方题解

五、三数之和

题目:15. 三数之和

参考官方题解,使用排序+双指针

这里也使用golang的对象,完成处理

type threeSumData struct {
	nums   []int
	n      int
	target int
	res    [][]int
	first  int
	second int
	third  int
}

func (t *threeSumData) threeSumWithFixC(target int) {
	// b取值得到了,取c的值,b在c的左侧,c肯定大于b的,c从后面往前去,因为排序了,所以正常情况,b+c>= target
	for t.second < t.third {
		if t.nums[t.second]+t.nums[t.third] <= target {
			break
		}
		t.third--
	}
}

// 固定A值,取B+C = target
func (t *threeSumData) threeSumWithFixB(target int) {
	t.third = t.n - 1
	for second := t.first + 1; second < t.n; second++ {
		// 取一个b的值,去掉重复的
		if second > t.first+1 && t.nums[second] == t.nums[second-1] {
			continue
		}

		t.second = second
		// fmt.Printf("second %v\n", second)
		t.threeSumWithFixC(target)

		// 如果指针重合,随着 b 后续的增加
		// 就不会有满足 a+b+c=0 并且 b<c 的 c 了,可以退出循环
		if t.second == t.third {
			return
		}

		if t.nums[t.second]+t.nums[t.third] == target {
			t.res = append(t.res, []int{t.nums[t.first], t.nums[t.second], t.nums[t.third]})
		}
	}
}

func (t *threeSumData) threeSumWithFixA() {
	// 3 <= nums.length <= 3000
	// -10^5 <= nums[i] <= 10^5
	if t.n <= 2 {
		return
	}

	// 数组排序
	sort.Ints(t.nums)

	// 取a值,
	for first := 0; first < t.n; first++ {
		// 取一个a值,如果当前值与上一个值一样,则跳过
		if first > 0 && t.nums[first] == t.nums[first-1] {
			continue
		}

		// 已经取了a值,按照题目,a + b + c = target,那么剩余 b + c = target - a
		t.first = first
		remain := t.target - t.nums[first]
		// fmt.Printf("first %v\n", first)
		// a取值固定了,取b值,b在a值的后面
		t.threeSumWithFixB(remain)
	}
}

func threeSum(nums []int) [][]int {
	data := &threeSumData{
		nums:   nums,
		n:      len(nums),
		target: 0,
		res:    make([][]int, 0),
		first:  0,
		second: 0,
		third:  0,
	}
	data.threeSumWithFixA()
	return data.res
}

六、四数之和

题目:18. 四数之和

type fourSumData struct {
	nums                         []int
	n, target                    int
	res                          [][]int
	first, second, third, fourth int
}

func (t *fourSumData) fourSumWithFixC() {
	nums := t.nums
	n := t.n
	// 双指针
	for left, right := t.second+1, n-1; left < right; {
		if sum := nums[t.first] + nums[t.second] + nums[left] + nums[right]; sum == t.target {
			t.res = append(t.res, []int{nums[t.first], nums[t.second], nums[left], nums[right]})
			for left++; left < right && nums[left] == nums[left-1]; left++ {
			}
			for right--; left < right && nums[right] == nums[right+1]; right-- {
			}
		} else if sum < t.target {
			left++
		} else {
			right--
		}
	}
}

func (t *fourSumData) fourSumWithFixB() {
	nums := t.nums
	n := t.n
	for second := t.first + 1; second < n-2; second++ {
		// 连续的四个值,和大于target时,则四元组肯定不满足条件
		if nums[t.first]+nums[second]+nums[second+1]+nums[second+2] > t.target {
			return
		}
		// a、b、c 和 d 互不相同,如果相同,或者 A+B+最大的两个值,不满足条件,则以当前值为a值,不会再有满足条件的四元组
		if second > t.first+1 && nums[second] == nums[second-1] || nums[t.first]+nums[second]+nums[n-2]+nums[n-1] < t.target {
			continue
		}
		t.second = second
		t.fourSumWithFixC()
	}
}

func (t *fourSumData) fourSumWithFixA() {
	nums := t.nums
	sort.Ints(t.nums)
	n := t.n
	for first := 0; first < n-3; first++ {
		// 连续的四个值,和大于target时,则四元组肯定不满足条件
		if nums[first]+nums[first+1]+nums[first+2]+nums[first+3] > t.target {
			return
		}

		// a、b、c 和 d 互不相同,如果相同,或者A+最大的三个值,不满足条件,则以当前值为a值,不会再有满足条件的四元组
		if first > 0 && nums[first] == nums[first-1] || nums[first]+nums[n-3]+nums[n-2]+nums[n-1] < t.target {
			continue
		}

		t.first = first
		t.fourSumWithFixB()
	}

	return
}

func fourSum(nums []int, target int) [][]int {
	data := &fourSumData{
		nums:   nums,
		n:      len(nums),
		target: target,
		res:    make([][]int, 0),
	}
	data.fourSumWithFixA()
	return data.res
}


http://www.kler.cn/a/272084.html

相关文章:

  • 计算机网络介质访问控制全攻略:从信道划分到协议详解!!!
  • MFC程序设计(二)基于对话框编程
  • Linux中关于glibc包编译升级导致服务器死机或者linux命令无法使用的情况
  • Fastapi + vue3 自动化测试平台(4)-- fastapi分页查询封装
  • 备赛蓝桥杯之第十五届职业院校组省赛第二题:分享点滴
  • PostgreSQL数据库的运行机制和架构体系
  • 【智能算法】斑鬣狗优化算法(SHO)原理及实现
  • UE4 虚幻4快捷键教程
  • Rust 如何优雅关闭 channel
  • JVM实战篇
  • HTML静态网页成品作业(HTML+CSS)——游戏战地介绍设计制作(4个页面)
  • C#设计原则
  • 基于Java+SpringMVC+vue+element实现前后端分离校园失物招领系统详细设计
  • 数据集成工具 ---- datax 3.0
  • 6.【Linux】进程间通信(管道命名管道||简易进程池||简易客户端服务端通信)
  • Rust基础知识讲解
  • 图像分割的定义
  • ChatGPT登陆提示:“Please unblock challenges.cloudflare.com to proceed…”
  • 替代imx6ull mfgtool的方法
  • Leetcode 1. 两数之和
  • 新!PCA+DBO+K-means聚类,蜣螂优化算法DBO优化K-means,适合学习,也适合发paper。
  • 编译原理 第1章:概述
  • SpringBoot Servlet容器启动解析
  • 【论文精读】DDPM:Denoising Diffusion Probabilistic Models 去噪扩散概率模型
  • 每日OJ题_简单多问题dp④_力扣LCR 091. 粉刷房子
  • ROS2+NAV2如何快捷的在docker中使用主机的CAN