2024-03-27:用go语言,多维费用背包。 给你一个二进制字符串数组 strs 和两个整数 m 和 n

2024-03-27:用go语言,多维费用背包。

给你一个二进制字符串数组 strs 和两个整数 m 和 n,

请你找出并返回 strs 的最大子集的长度,该子集中 最多 有 m 个 0 和 n 个 1。

如果 x 的所有元素也是 y 的元素,集合 x 是集合 y 的 子集。

输入:strs = [“10”, “0001”, “111001”, “1”, “0”], m = 5, n = 3。

输出:4。

答案2024-03-27:

来自左程云

灵捷3.5

大体步骤如下:

1.findMaxForm1 函数使用递归的方式实现。它遍历字符串数组 strs,将每个字符串中0和1的数量存储在一个二维数组 arr 中。然后通过递归函数 process1 进行计算,不断比较所选字符串是否符合要求,选择放入或不放入子集。该方法的时间复杂度为O(2^n),空间复杂度为O(n)。

2.findMaxForm2 函数使用记忆化搜索的方式实现。它也遍历字符串数组 strs 得到二维数组 arr,但使用三维数组 dp 进行记忆化,记录已经计算过的结果,避免重复计算。该方法的时间复杂度为O(m * n * len(strs)),空间复杂度为O(m * n * len(strs))。

3.findMaxForm3 函数使用动态规划的方式实现。它从后向前遍历字符串数组 strs,得到二维数组 dp 来保存计算结果。通过比较选择当前字符串加入子集还是不加入子集,并更新动态规划数组 dp。该方法的时间复杂度为O(m * n * len(strs)),空间复杂度为O(m * n * len(strs))。

4 findMaxForm4 函数使用动态规划的方式实现。它遍历字符串数组 strs,得到二维数组 dp 来保存计算结果。使用一维数组 dp 进行滚动更新,从后向前遍历,根据当前字符串的0和1的数量,更新动态规划数组 dp。该方法的时间复杂度为O(m * n * len(strs)),空间复杂度为O(m * n)。

总时间复杂度:O(m * n * len(strs))

总额外空间复杂度:O(m * n * len(strs))

Go完整代码如下:

package main

import (
    "fmt"
)

var zeros, ones int

func findMaxForm1(strs []string, m int, n int) int {
    len := len(strs)
    arr := make([][]int, len)
    for i := 0; i < len; i++ {
        zeroAndOne(strs[i])
        arr[i] = []int{zeros, ones}
    }
    return process1(arr, 0, m, n)
}

func process1(arr [][]int, i int, z int, o int) int {
    if i == len(arr) {
        return 0
    }
    p1 := process1(arr, i+1, z, o)
    p2 := 0
    if arr[i][0] <= z && arr[i][1] <= o {
        p2 = 1 + process1(arr, i+1, z-arr[i][0], o-arr[i][1])
    }
    if p1 > p2 {
        return p1
    }
    return p2
}

func findMaxForm2(strs []string, m int, n int) int {
    len := len(strs)
    arr := make([][]int, len)
    for i := 0; i < len; i++ {
        zeroAndOne(strs[i])
        arr[i] = []int{zeros, ones}
    }
    dp := make([][][]int, len)
    for i := 0; i < len; i++ {
        dp[i] = make([][]int, m+1)
        for j := 0; j <= m; j++ {
            dp[i][j] = make([]int, n+1)
            for k := 0; k <= n; k++ {
                dp[i][j][k] = -1
            }
        }
    }
    return process2(arr, 0, m, n, dp)
}

func process2(arr [][]int, i int, z int, o int, dp [][][]int) int {
    if i == len(arr) {
        return 0
    }
    if dp[i][z][o] != -1 {
        return dp[i][z][o]
    }
    p1 := process2(arr, i+1, z, o, dp)
    p2 := 0
    if arr[i][0] <= z && arr[i][1] <= o {
        p2 = 1 + process2(arr, i+1, z-arr[i][0], o-arr[i][1], dp)
    }
    ans := p1
    if p2 > p1 {
        ans = p2
    }
    dp[i][z][o] = ans
    return ans
}
func findMaxForm3(strs []string, m int, n int) int {
    len0 := len(strs)
    dp := make([][][]int, len0+1)
    for i := len0; i >= 0; i-- {
        dp[i] = make([][]int, m+1)
        for z := 0; z <= m; z++ {
            dp[i][z] = make([]int, n+1)
        }
    }
    for i := len0 - 1; i >= 0; i-- {
        zeroAndOne(strs[i])
        for z := 0; z <= m; z++ {
            for o := 0; o <= n; o++ {
                dp[i][z][o] = dp[i+1][z][o]
                if zeros <= z && ones <= o {
                    dp[i][z][o] = max(dp[i][z][o], 1+dp[i+1][z-zeros][o-ones])
                }
            }
        }
    }
    return dp[0][m][n]
}

func zeroAndOne(str string) {
    zeros = 0
    ones = 0
    for i := 0; i < len(str); i++ {
        if str[i] == '0' {
            zeros++
        } else {
            ones++
        }
    }
}

func findMaxForm4(strs []string, m int, n int) int {
    dp := make([][]int, m+1)
    for i := 0; i <= m; i++ {
        dp[i] = make([]int, n+1)
    }
    for _, s := range strs {
        zeroAndOne(s)
        for i := m; i >= zeros; i-- {
            for j := n; j >= ones; j-- {
                dp[i][j] = max(dp[i][j], dp[i-zeros][j-ones]+1)
            }
        }
    }
    return dp[m][n]
}

func max(a, b int) int {
    if a > b {
        return a
    }
    return b
}

func main() {
    strs := []string{"10", "0001", "111001", "1", "0"}
    m := 5
    n := 3

    res1 := findMaxForm1(strs, m, n)
    fmt.Println("findMaxForm1:", res1)

    res2 := findMaxForm2(strs, m, n)
    fmt.Println("findMaxForm2:", res2)

    res3 := findMaxForm3(strs, m, n)
    fmt.Println("findMaxForm3:", res3)

    res4 := findMaxForm4(strs, m, n)
    fmt.Println("findMaxForm4:", res4)
}

在这里插入图片描述

Python完整代码如下:

# -*-coding:utf-8-*-

def zero_and_one(string):
    zeros = 0
    ones = 0
    for char in string:
        if char == '0':
            zeros += 1
        else:
            ones += 1
    return zeros, ones

def find_max_form1(strs, m, n):
    length = len(strs)
    arr = []
    for i in range(length):
        zeros, ones = zero_and_one(strs[i])
        arr.append((zeros, ones))
    return process1(arr, 0, m, n)

def process1(arr, i, z, o):
    if i == len(arr):
        return 0
    p1 = process1(arr, i+1, z, o)
    p2 = 0
    if arr[i][0] <= z and arr[i][1] <= o:
        p2 = 1 + process1(arr, i+1, z-arr[i][0], o-arr[i][1])
    if p1 > p2:
        return p1
    return p2

def find_max_form2(strs, m, n):
    length = len(strs)
    arr = []
    for i in range(length):
        zeros, ones = zero_and_one(strs[i])
        arr.append((zeros, ones))
    dp = [[[-1] * (n+1) for _ in range(m+1)] for _ in range(length)]
    return process2(arr, 0, m, n, dp)

def process2(arr, i, z, o, dp):
    if i == len(arr):
        return 0
    if dp[i][z][o] != -1:
        return dp[i][z][o]
    p1 = process2(arr, i+1, z, o, dp)
    p2 = 0
    if arr[i][0] <= z and arr[i][1] <= o:
        p2 = 1 + process2(arr, i+1, z-arr[i][0], o-arr[i][1], dp)
    ans = p1
    if p2 > p1:
        ans = p2
    dp[i][z][o] = ans
    return ans

def find_max_form3(strs, m, n):
    length = len(strs)
    dp = [[[0] * (n+1) for _ in range(m+1)] for _ in range(length+1)]
    for i in range(length-1, -1, -1):
        zeros, ones = zero_and_one(strs[i])
        for z in range(m+1):
            for o in range(n+1):
                dp[i][z][o] = dp[i+1][z][o]
                if zeros <= z and ones <= o:
                    dp[i][z][o] = max(dp[i][z][o], 1 + dp[i+1][z-zeros][o-ones])
    return dp[0][m][n]

def find_max_form4(strs, m, n):
    dp = [[0] * (n+1) for _ in range(m+1)]
    for s in strs:
        zeros, ones = zero_and_one(s)
        for i in range(m, zeros-1, -1):
            for j in range(n, ones-1, -1):
                dp[i][j] = max(dp[i][j], dp[i-zeros][j-ones]+1)
    return dp[m][n]

strs = ["10", "0001", "111001", "1", "0"]
m = 5
n = 3

res1 = find_max_form1(strs, m, n)
print("findMaxForm1:", res1)

res2 = find_max_form2(strs, m, n)
print("findMaxForm2:", res2)

res3 = find_max_form3(strs, m, n)
print("findMaxForm3:", res3)

res4 = find_max_form4(strs, m, n)
print("findMaxForm4:", res4)

在这里插入图片描述

本作品采用《CC 协议》,转载必须注明作者和本文链接
微信公众号:福大大架构师每日一题。最新面试题,涉及golang,rust,mysql,redis,云原生,算法,分布式,网络,操作系统。
讨论数量: 0
(= ̄ω ̄=)··· 暂无内容!

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!
未填写
文章
472
粉丝
21
喜欢
37
收藏
22
排名:457
访问:1.9 万
私信
所有博文
社区赞助商