2024-03-30:用go语言,集团里有 n 名员工,他们可以完成各种各样的工作创造利润, 第 i 种工作会产生 profit[i] 的利润,它要求 group[i] 名成员共同参与, 如果成员参与
2024-03-30:用go语言,集团里有 n 名员工,他们可以完成各种各样的工作创造利润,
第 i 种工作会产生 profit[i] 的利润,它要求 group[i] 名成员共同参与,
如果成员参与了其中一项工作,就不能参与另一项工作,
工作的任何至少产生 minProfit 利润的子集称为 盈利计划,
并且工作的成员总数最多为 n。
有多少种计划可以选择?因为答案很大,所以 返回结果模 10^9 + 7 的值。
输入:n = 5, minProfit = 3, group = [2,2], profit = [2,3]。
输出:2。
答案2024-03-30:
来自左程云。
大体步骤如下:
这三种算法都解决了一个问题,即在给定一组工作和利润以及员工的人数限制下,找出满足最低利润要求的盈利计划数量。以下是它们的大体过程:
profitableSchemes1:
1.递归函数 f1 对组合进行深度优先搜索,尝试每种工作的所有可能性,以达到满足最低利润要求的盈利计划。
2.检查每种工作是否满足人数限制,并计算利润是否达到最低要求。
3.返回满足条件的计划数量。
profitableSchemes2:
1.使用动态规划方法,创建三维数组 dp 以保存中间结果。
2.递归函数 f2 逐步填充 dp 数组,记录以当前工作和利润数为基础时的计划数量。
3.每次计算时检查数组中是否已有记录,避免重复计算。
4.返回最终计划数量。
profitableSchemes3:
1.同样采用动态规划,但只使用二维数组 dp,减少额外空间的使用。
2.从最后一个工作向前逐步计算满足条件的计划数量。
3.根据当前工作是否选择、人数是否足够、利润是否达标,更新动态规划数组中的值。
4.最终输出满足条件的计划数量。
复杂度分析:
时间复杂度:
profitableSchemes1: 由于是基于递归的深度优先搜索,时间复杂度较高,为指数级别,取决于工作数量和员工人数。
profitableSchemes2: 使用了动态规划并记录了每个可能情况,时间复杂度为 O(m * n * minProfit),m 为工作数量,n 为员工人数。
profitableSchemes3: 类似于第二种算法,但只使用了二维数组,时间复杂度也为 O(m * n * minProfit)。
空间复杂度:
profitableSchemes1: 仅有递归调用的栈空间,空间复杂度取决于递归深度。
profitableSchemes2: 使用了三维数组
dp,空间复杂度为 O(m * n * minProfit)。profitableSchemes3: 使用了二维数组
dp,空间复杂度为 O(n * minProfit)。
Go完整代码如下:
package main
import "fmt"
func profitableSchemes1(n int, minProfit int, group []int, profit []int) int {
return f1(group, profit, 0, n, minProfit)
}
func f1(g []int, p []int, i int, r int, s int) int {
if r <= 0 {
if s <= 0 {
return 1
} else {
return 0
}
}
if i == len(g) {
if s <= 0 {
return 1
} else {
return 0
}
}
p1 := f1(g, p, i+1, r, s)
p2 := 0
if g[i] <= r {
p2 = f1(g, p, i+1, r-g[i], s-p[i])
}
return p1 + p2
}
var mod = 1000000007
func profitableSchemes2(n int, minProfit int, group []int, profit []int) int {
m := len(group)
dp := make([][][]int, m)
for a := 0; a < m; a++ {
dp[a] = make([][]int, n+1)
for b := 0; b <= n; b++ {
dp[a][b] = make([]int, minProfit+1)
for c := 0; c <= minProfit; c++ {
dp[a][b][c] = -1
}
}
}
return f2(group, profit, 0, n, minProfit, dp)
}
func f2(g []int, p []int, i int, r int, s int, dp [][][]int) int {
if r <= 0 {
if s == 0 {
return 1
} else {
return 0
}
}
if i == len(g) {
if s == 0 {
return 1
} else {
return 0
}
}
if dp[i][r][s] != -1 {
return dp[i][r][s]
}
p1 := f2(g, p, i+1, r, s, dp)
p2 := 0
if g[i] <= r {
p2 = f2(g, p, i+1, r-g[i], max(0, s-p[i]), dp)
}
ans := (p1 + p2) % mod
dp[i][r][s] = ans
return ans
}
func max(x, y int) int {
if x > y {
return x
}
return y
}
func profitableSchemes3(n int, minProfit int, group []int, profit []int) int {
dp := make([][]int, n+1)
for r := 0; r <= n; r++ {
dp[r] = make([]int, minProfit+1)
dp[r][0] = 1
}
m := len(group)
for i := m - 1; i >= 0; i-- {
for r := n; r >= 0; r-- {
for s := minProfit; s >= 0; s-- {
p1 := dp[r][s]
p2 := 0
if group[i] <= r {
p2 = dp[r-group[i]][max(0, s-profit[i])]
}
dp[r][s] = (p1 + p2) % mod
}
}
}
return dp[n][minProfit]
}
func main() {
n := 5
minProfit := 3
group := []int{2, 2}
profit := []int{2, 3}
result1 := profitableSchemes1(n, minProfit, group, profit)
fmt.Println("Result using profitableSchemes1:", result1)
result2 := profitableSchemes2(n, minProfit, group, profit)
fmt.Println("Result using profitableSchemes2:", result2)
result3 := profitableSchemes3(n, minProfit, group, profit)
fmt.Println("Result using profitableSchemes3:", result3)
}

2024-03-30:用go语言,集团里有 n 名员工,他们可以完成各种各样的工作创造利润, 第 i 种工作会产生 profit[i] 的利润,它要求 group[i] 名成员共同参与, 如果成员参与的更多相关文章
- static数据成员与const数据成员的定义与初始化
三种数据类型的初始化 1.static int a 的初始化 const int a 的初始化 static const int a的初始化 三种初始化方式 在类外初始化 在构造函数中通过初始化列表初 ...
- 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成员)
[源码下载] 不可或缺 Windows Native (21) - C++: 继承, 组合, 派生类的构造函数和析构函数, 基类与派生类的转换, 子对象的实例化, 基类成员的隐藏(派生类成员覆盖基类成 ...
- 【转载】C++ 与“类”有关的注意事项总结(十二):按成员初始化 与 按成员赋值
原文:C++ 与"类"有关的注意事项总结(十二):按成员初始化 与 按成员赋值 一.按成员初始化(与构造函数和拷贝构造函数有关) 用一个类对象初始化另一个类对象,比如: Accou ...
- [C]成员运算符"."和间接成员运算符"->"浅析
成员运算符: . 成员运算符一般和结构或者联合名一起使用,指定结构或者联合中的某个成员. 举个栗子: 如果Ronz是一个结构的名称,linux是这个结构模板指定的一个成员名. struct{ //匿名 ...
- Java中变量之局部变量、本类成员变量、父类成员变量的访问方法
变量:局部变量.本类成员变量.父类成员变量 如何访问:如果变量名相同,则采用就近原则,哪个变量离所要调用的访问最近,那就么就输出,优先顺序为:局部变量 > 本类成员变量 > 父类成员变量 ...
- C++类中的常数据成员和静态数据成员的区别
刚开始学习C++的类和对象的部分,对类中的常数据成员和静态数据成员的概念和用法经常混淆,所以今天整理一下,顺便说一下,今天是我的生日,祝我生日快乐,呵呵. 常数据成员 常数据成员是指在类中定义的不能修 ...
- OJ提交题目中的语言选项里G++与C++的区别(转)
G++? 首先更正一个概念,C++是一门计算机编程语言,G++不是语言,是一款编译器中编译C++程序的命令而已. 那么他们之间的区别是什么? 在提交题目中的语言选项里,G++和C++都代表编译的方式. ...
- 类中成员函数与数据成员private/pubic/protected
类中成员函数与数据成员private/pubic/protected
- OJ提交题目中的语言选项里G++与C++的区别(转载)
原文链接:http://blog.polossk.com/201405/c-plus-plus-g-plus-plus G++? 首先更正一个概念,C++是一门计算机编程语言,G++不是语言,是一款编 ...
- C++ static成员变量与static成员函数
类中的静态成员真是个让人爱恨交加的特性.我决定好好总结一下静态类成员的知识点,以便自己在以后面试中,在此类问题上不在被动. 静态类成员包括静态数据成员和静态函数成员两部分. 一 静态数据成员: 类 ...
随机推荐
- ElasticSearch入门安装与SpringBoot集成实战
介绍 Elasticsearch 是一个实时分布式搜索和分析引擎,一般用于全文搜索.结构化搜索,分析或者三者混用. 它的底层是基于Apache Lucene(TM)的开源搜索引擎,但是lucene只是 ...
- spring boot+sqlite+mybatis实现增删改查例子
主要是更换了下sqlite的数据源而已,其他代码不变. 我都贴一下吧,这个算是比较通用的基础增删改查代码. 1.创建test.db 可以使用Idea自带的Database插件配置,也可以命令行创建,具 ...
- C++ 多线程的错误和如何避免(14)
在 C++11 中,不要将 volatile 用于线程,仅限于 MMIO(内存映射) 简单的回答, 在声明变量类型之前添加 "volatile" 关键字不会使对该变量有任何方式的原 ...
- 记一次 splice 导致 io.Copy 阻塞的排查过程
记一次 splice 导致 io.Copy 阻塞的排查过程 简而言之,net.TCPConn 的 ReadFrom 零拷贝实现 splice 在 1.21.0 - 1.21.4 删除了 SPLICE_ ...
- 详解SSL证书系列(3)如何选择SSL证书
我们知道了在网站部署 SSL 证书后,不管是对网站本身还是对网站的用户都能够带来许多好处.那么随着 HTTPS的普及,市面上也出现了各种不同的 SSL 证书.并且由于 SSL 证书的多样性,很多人对于 ...
- 【Azure 应用服务】如果发现当前使用的订阅无法在China North 3 区中创建App Service服务,如何来解决这个问题呢?
问题描述 在创建App Service服务时,突然发现无法选择China North 3区域,如何来解决这个问题呢? 问题解答 根据Azure中服务都需要在订阅中注册的原理,因为China North ...
- python爬虫 xpath入门与lxml库基本使用,我们一同学习xpath
目录 什么是XPath? xpath语法 知识点 节点 选取节点: 选取a节点下所有的href属性 ../ 选取父节点 bookstore/book 选取子元素li bookstore//book 不 ...
- 小工具 --- 百度翻译API翻译工具
引言 最近想把一些英文官方文档的资料翻译成中文,然后转化为Markdown文档,然后发现百度通用翻译的API有不错的免费额度,个人申请也能申请到高级版.这个额度足够个人的日常使用了. 如何使用 如何使 ...
- IDEA使用与多线程
IDEA缩写和快捷键 psvm全称public static void main sout 全称public static void main alt+enter 处理异常 s.out 自动打印s c ...
- vetur 和 volar 不要一起装 - vscode插件 已解决
vetur 和 volar 不要一起装 - vscode插件 会有各种稀奇古怪的问题. 解决方案 利用 vscode 工作区 新建工作区 然后全局 将 volar 禁用工作区,起一个新的vue3项目, ...