这是算法考试的最后一题,当时匆匆写了个基于 Subset Sum 的解法,也没有考虑是否可行。

问题描述如下:

给定 \(n\) 个正整数 \(a_1 \dots a_n\) ,设下标的整数集合 \(V=\{1,2,3,\dots,n\}\) , 确定是否有三个不相交的子集 \(I,J,K \sub V\) ,满足:

\[\sum_{i \in I} a_i = \sum_{j \in J}a_j = \sum_{k \in K} a_k = \frac{sum}{3}
\]

其中, \(sum\) 是所有元素之和,要求复杂度是关于 \(n\) 和 \(sum\) 的多项式时间解法。

基于 Subset Sum 的解法

Subset Sum 问题:给定一个整数数组 nums 和整数 target ,问是否存在 nums 的子集,它的和为 target ,每个元素只能使用一次。

显然 Subset Sum 问题是背包问题的特殊情况,当背包问题中所有物品的价值等于体积,那么就是 Subset Sum 问题。

思路:

  • 假设 subsetSum(nums, target) 能够在 nums 找到所有和为 target 的子集。
  • 问题等价于找到 2 个子集 \(I, J \sub V\) ,并且 \(\sum{a_i} = \sum{a_j} = sum/3\) ,那么剩下的元素必然能保证 \(\sum{a_k} = sum/3\) .
  • 通过 Subset Sum 找到所有满足目标和为 sum/3 的所有子集 \(\mathcal{I}\) ,即 subsetSum(V, target = sum/3)
  • 对于每一个的 \(I \in \mathcal{I}\) ,令 \(V' = V - I\) ,执行 subsetSum(V', target = sum/3) ,如果返回值不为空,那么说明存在这样的 \(I,J,K\) 满足 3-Partition 的条件。

Subset Sum 问题的判定形式通过动态规划是十分容易解决的,难点在于找出所有这样的子集。

建议先完成下面的「输出所有 LCS 的练习」,再进行后文的阅读。

输出所有 Subset Sum

定义 dp[i, j] 表示在 nums[0, ..., i] 中不超过 j 的最大和。

转移方程为:

\[dp[i, j] = \left\{
\begin{aligned}
& dp[i-1, j] & \text{ if } j < a_i \\
& \max(dp[i-1, j], dp[i-1, j-a_i]+a_i), & \text{ if } j \ge a_i
\end{aligned}
\right.
\]

最后结果为 dp[n, target] == target .

代码实现

vector<vector<int>> subsetSum(vector<int> &nums, int target)
{
int n = nums.size();
vector<vector<int>> dp(n + 1, vector<int>(target + 1, 0));
for (int i = 1; i <= n; i++)
{
for (int j = 0; j <= target; j++)
{
int x = nums[i - 1];
if (j >= x) dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - x] + x);
else dp[i][j] = dp[i - 1][j];
}
} // print all subsets
vector<vector<int>> result;
function<bool(int, int, vector<int>)> getSubsets = [&](int i, int j, vector<int> subset)
{
while (i >= 1 && j >= 0)
{
int x = nums[i - 1];
if (j >= x)
{
int t = dp[i - 1][j - x] + x;
if (dp[i - 1][j] > t) i--;
else if (dp[i - 1][j] < t) j -= x, i--, subset.emplace_back(x);
else
{
getSubsets(i - 1, j, subset);
subset.emplace_back(x), getSubsets(i - 1, j - x, subset);
return true;
}
}
else i--;
}
result.emplace_back(subset);
return true;
};
if (dp[n][target] == target)
{
getSubsets(n, target, vector<int>{});
return result;
}
return {};
}
int main()
{
vector<int> nums = {1, 3, 7, 8, 9};
int t = 16;
auto result = subsetSum(nums, t);
for (auto &v : result)
{
for (int x : v) cout << x << ' ';
cout << endl;
}
}

3-Partition

基于上述的 Subset Sum ,我们可以写出 3-Partition 的代码。

bool threePartition(vector<int> &nums)
{
int sum = accumulate(nums.begin(), nums.end(), 0);
if (sum % 3 != 0) return false;
int target = sum / 3;
auto subsets = subsetSum(nums, target);
for (auto &I : subsets)
{
vector<int> buf(nums.size() - I.size());
// buf = nums - I
auto itor = set_symmetric_difference(nums.begin(), nums.end(), I.begin(), I.end(), buf.begin());
buf.resize(itor - buf.begin());
if (subsetSum(buf, target).size() != 0) return true;
}
return false;
}
int main()
{
vector<int> nums = {1, 2, 3, 4, 4, 5, 8};
cout << threePartition(nums) << endl;
}

Subset Sum 的时间复杂度为 \(O(nt)\),而此处 t = sum/3 ,因此 3-Partition 的时间复杂度为 \(O(kn \cdot sum)\) ,\(k\) 是 集合 \(I\) 的个数,显然,\(k\) 有可能是指数级别的。

显然,基于上述操作,我们同样能找到所有满足条件的 \(I,J,K\) .

动态规划

上面是比较容易想到的思路,但这里的 3-Partition 是一个判定问题,我们只需要给出 YES or NO,而不需要给出具体的 \(I,J,K\) ,因此用动态规划可以使问题变得简单。

2-Partition

先考虑 Subset Sum 的一种特殊情况:给定 nums ,问是否存在一个子集 I , 使得 sum(I) = sum(nums) / 2 .

其实换汤不换药。

int twoPartition(vector<int> &nums)
{
int sum = accumulate(nums.begin(), nums.end(), 0);
int n = nums.size();
if (sum % 2 != 0) return false;
// dp[i, j] 表示前 j 个数字中,是否存在一个和为 i 的子集(允许为空集)
vector<vector<int>> dp(sum / 2 + 1, vector<int>(n + 1, false));
for (int i = 0; i <= n; i++) dp[0][i] = true;
for (int i = 1; i <= sum / 2; i++) dp[i][0] = false;
for (int i = 1; i <= sum / 2; i++)
{
for (int j = 1; j <= n; j++)
{
int x = nums[j - 1];
if (i >= x) dp[i][j] = dp[i][j - 1] || dp[i - x][j - 1];
else dp[i][j] = dp[i][j - 1];
}
}
return dp[sum / 2][n];
}

3-Partition

定义 dp[j, k] 表示:nums[1, ..., n],是否存在一个子集,使得它的和为 j ;同时存在另外一个不相交子集,它的和为 k .

注意这里的前提条件是,在 [1, ..., n] 这个范围,而且 dp[j, k] = true 当且仅当 2 个子集同时存在。

那么最后的答案是 dp[sum / 3, sum / 3] .

转移方程为:

\[dp[j, k] = dp[j-a_i][k] \text{ or } dp[j][k-a_i], \text{ for any } a_i
\]

类似于自顶向下的填表顺序,转移方程可以改写为:

\[dp[j, k] = \text{true} \quad \Rightarrow \quad dp[j+a_i, k] = dp[j, k+a_i] = \text{true}, \quad \text{for any } a_i
\]

代码实现

int threePartition(vector<int> &A)
{
int sum = accumulate(A.begin(), A.end(), 0);
int size = A.size();
if (sum % 3 != 0) return false;
vector<vector<int>> dp(sum + 1, vector<int>(sum + 1, 0));
dp[0][0] = true;
// process the numbers one by one
for (int i = 0; i < size; i++)
{
for (int j = sum; j >= 0; j--)
{
for (int k = sum; k >= 0; k--)
{
if (dp[j][k])
{
dp[j + A[i]][k] = true;
dp[j][k + A[i]] = true;
}
}
}
}
return dp[sum / 3][sum / 3];
}

输出所有 LCS

输出一个 LCS 可以参考:https://www.cnblogs.com/sinkinben/p/14536604.html

思路很简单,在填 dp 表的过程中,转移路径实际上就是记录了 LCS 的结果,我们只需要通过回溯法,找到所有 (alen, blen) => (1, 1) 的路径即可。

代码实现

int lcs(const string &a, const string &b)
{
int alen = a.length(), blen = b.length();
vector<vector<int>> dp(alen + 1, vector<int>(blen + 1, 0));
for (int i = 1; i <= alen; i++)
{
for (int j = 1; j <= blen; j++)
{
if (a[i - 1] == b[j - 1]) dp[i][j] = dp[i - 1][j - 1] + 1;
else dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
}
}
// print all lcs
function<void(int, int, string)> printlcs = [&](int i, int j, string str)
{
while (i >= 1 && j >= 1)
{
if (a[i - 1] == b[j - 1])
str.push_back(a[i - 1]), i--, j--;
else
{
if (dp[i - 1][j] > dp[i][j - 1]) i--;
else if (dp[i - 1][j] < dp[i][j - 1]) j--;
else
{
printlcs(i - 1, j, str);
printlcs(i, j - 1, str);
return;
}
}
}
reverse(str.begin(), str.end());
result.insert(str);
};
printlcs(alen, blen, "");
for (auto &x : result) cout << x << endl;
return dp[alen][blen];
}
int main()
{
// string a = "cnblog", b = "belong";
string a = "xyxxzxyzxy", b = "zxzyyzxxyxxz";
// string a = "ABCBDAB", b = "BDCABA";
cout << lcs(a, b) << endl;
}

3-Partition 问题的更多相关文章

  1. Partition:增加分区

    在关系型 DB中,分区表经常使用DateKey(int 数据类型)作为Partition Column,每个月的数据填充到同一个Partition中,由于在Fore-End呈现的报表大多数是基于Mon ...

  2. Partition:Partiton Scheme是否指定Next Used?

    在SQL Server中,为Partition Scheme多次指定Next Used,不会出错,最后一次指定的FileGroup是Partition Scheme的Next Used,建议,在执行P ...

  3. Partition:分区切换(Switch)

    在SQL Server中,对超级大表做数据归档,使用select和delete命令是十分耗费CPU时间和Disk空间的,SQL Server必须记录相应数量的事务日志,而使用switch操作归档分区表 ...

  4. sql 分组取最新的数据sqlserver巧用row_number和partition by分组取top数据

    SQL Server 2005后之后,引入了row_number()函数,row_number()函数的分组排序功能使这种操作变得非常简单 分组取TOP数据是T-SQL中的常用查询, 如学生信息管理系 ...

  5. Oracle Partition Outer Join 稠化报表

    partition outer join实现将稀疏数据转为稠密数据,举例: with t as (select deptno, job, sum(sal) sum_sal from emp group ...

  6. SQLServer中Partition By 函数的使用

    今天群里看到一个问题,在这里概述下:查询出不同分类下的最新记录.一看这不是很简单的么,要分类那就用Group By;要最新记录就用Order By呗.然后在自己的表中试着做出来: 首先呢我把表中的数据 ...

  7. [LeetCode] Partition Equal Subset Sum 相同子集和分割

    Given a non-empty array containing only positive integers, find if the array can be partitioned into ...

  8. [LeetCode] Partition List 划分链表

    Given a linked list and a value x, partition it such that all nodes less than x come before nodes gr ...

  9. 快速排序中的partition函数的枢纽元选择,代码细节,以及其标准实现

    很多笔试面试都喜欢考察快排,叫你手写一个也不是啥事.我很早之前就学了这个,对快速排序的过程是很清楚的.但是最近自己尝试手写,发现之前对算法的细节把握不够精准,很多地方甚至只是大脑中的一个映像,而没有理 ...

  10. [bigdata] kafka基本命令 -- 迁移topic partition到指定的broker

    版本 0.9.2 创建topic bin/kafka-topics.sh --create --topic topic_name --partition 6 --replication-factor ...

随机推荐

  1. IOCP实现高并发以及与传统socke编程的对比

    前言 传统socket编程中服务端一般为每一个客户端创建一个线程(一对一).这样虽然可以使程序的结构简单明了并且方便对数据处理,但是这些都是建立在创建多个线程的基础上,也就是以牺牲线程为代价.一旦有大 ...

  2. MSSQL·查看DB中所有表及列的相关信息

    阅文时长 | 0.6分钟 字数统计 | 1013.6字符 主要内容 | 1.引言&背景 2.声明与参考资料 『MSSQL·查看DB中所有表及列的相关信息』 编写人 | SCscHero 编写时 ...

  3. linux最大文件打开数和swap限制

    linux最大文件打开数和swap限制   逑熙 关注 2017.07.24 15:39* 字数 388 阅读 314评论 0喜欢 0 linux 2.6+的核心会使用硬盘的一部分做为SWAP分区,用 ...

  4. 单臂路由实现不同vlan间通信

    单臂路由实现不同vlan间通信 拓扑图 PC配置 PC1 :192.168.1.1 vlan10 192.168.1.254 PC2 :192.168.2.1 vlan20 192.168.2.254 ...

  5. Docker Swarm(十一)生产环境使用的一些建议

    一.Docker Swarm上的容器选择 并非所有服务都应该部署在Swarm集群内.数据库以及其他有状态服务就不适合部署在Swarm集群内. 理论上,你可以通过使用labels将容器部署到特定节点上, ...

  6. 数据库权限grant

    数据库权限grant 创建授权grant 权限类型(priv_type) 权限类型 代表什么? ALL 所有权限 SELECT 读取内容的权限 INSERT 插入内容的权限 UPDATE 更新内容的权 ...

  7. 070.Python聚焦爬虫数据解析

    一 聚焦爬虫数据解析 1.1 基本介绍 聚焦爬虫的编码流程 指定url 基于requests模块发起请求 获取响应对象中的数据 数据解析 进行持久化存储 如何实现数据解析 三种数据解析方式 正则表达式 ...

  8. 9.11 strace:跟踪进程的系统调用 、ltrace:跟踪进程调用库函数

    strace 是Linux环境下的一款程序调试工具,用于检查一个应用程序所使用的系统调用以及它所接收的系统信息.strace会追踪程序运行时的整个生命周期,输出每一个系统调用的名字.参数.返回值和执行 ...

  9. vue 表格中的下拉框单选、多选处理

    最近在用vue做前后端分离,需要在表格中用到下拉框,由于需求变动,从最开始的单选变为多选,折腾了许久,记录一下,供后人铺路 vue 中的表格下拉框单选 collectionsColnumOptions ...

  10. systemverilog数组类型