括号匹配(二)
时间限制:1000 ms | 内存限制:65535 KB
难度:6
如:
[]是匹配的
([])[]是匹配的
((]是不匹配的
([)]是不匹配的
每组测试数据都只有一行,是一个字符串S,S中只包含以上所说的四种字符,S的长度不超过100
4 [] ([])[] ((] ([)]
0 0 3 2
思路:
区间 DP。设 dp [ i ] [ j ] 表示从 i 到 j 使其匹配成功的最小添加数量。所以:
1.dp [ i ] [ j ] = min ( dp [ i ] [ j ] , dp [ i ] [ i + k ] + dp [ k + 1 ] [ j ] ) ( i <= k < j);
2.若 s [ i ] 与 s [ j ] 是彼此匹配的括号时,还需要比较 dp [ i ] [ j ] = min ( dp [ i ] [ j ] ,dp [ i + 1] [ j - 1] )。
初始化的时候,当 i == j 时,即为本身时,dp [ i ] [ j ] = 1;要求最小值,则当 i < j 时,dp [ i ] [ j ] = INF;当 i > j 时,dp [ i ] [ j ] = 0,这是需要注意的。
比如当这个括号串为 “()”时,第一条 dp 公式更新到 dp [ 0 ][ 1 ] = 2,若把全部的值都初始化为 INF 的话,根据第二条 dp 更新公式,却得不到 0 的结果,因为 dp [ 1 ] [ 0 ] = INF,所以当 i > j 时,应该初始化 dp [ i ] [ j ] = 0 。
AC:
#include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int INF = 99999999; int dp[105][105]; int main() { int n; scanf("%d", &n); while (n--) { char s[105]; int len; scanf("%s", s); len = strlen(s); for (int i = 0; i < len; ++i) for (int j = 0; j < len; ++j) { if (i < j) dp[i][j] = INF; if (i > j) dp[i][j] = 0; if (i == j) dp[i][j] = 1; } for (int j = 1; j < len; ++j) { for (int i = j - 1; i >= 0; --i) { for (int k = i; k < j; ++k) { dp[i][j] = min(dp[i][j], dp[i][k] + dp[k + 1][j]); } if (s[i] == '(' && s[j] == ')') dp[i][j] = min(dp[i][j], dp[i + 1][j - 1]); if (s[i] == '[' && s[j] == ']') dp[i][j] = min(dp[i][j], dp[i + 1][j - 1]); } } printf("%d\n", dp[0][len - 1]); } return 0; }
相关推荐
区间动态规划是一种在特定区间内进行的动态规划问题,它通常涉及计算一个二维数组`dp`,其中`dp[x][y]`表示区间 `[x, y]` 的解。 在区间动态规划中,有两种主要的状态转移方式: 1. 有些问题可以通过相邻状态直接...
- **区间DP**:解决区间相关问题。 - **矩阵链乘法**:确定最佳矩阵相乘顺序。 - **游戏DP**:涉及两人或多人间的游戏策略问题。 ### DP状态转移方程实例 下面列出了一些典型的DP状态转移方程: 1. **0-1背包问题...
同时,我们还可以通过回溯`dp`数组来恢复最优的矩阵乘法顺序,即最佳括号匹配形式。 在实际编程实现时,为了存储最优分割点,我们通常还需要一个二维数组`m`,`m[i][j]`记录了将区间[i...j]分成两部分时的最优分割...
2. **栈与队列**:基础数据结构,用于实现回溯、括号匹配等。 3. **树与图**:二叉树、平衡树(AVL、红黑树)、图的邻接矩阵和邻接表,用于处理层次关系和网络连接问题。 4. **哈希表**:提供快速查找、插入和删除...
- **栈的应用**:在实际编程中,栈被广泛应用于函数调用时保存返回地址、括号匹配问题等。 #### 1.2 队列的实现与理解 - **循环队列的特点**:循环队列利用数组模拟队列,解决了传统队列的假溢出问题。题目中强调了...
3. 栈:后进先出(LIFO)结构,常用于括号匹配、回溯算法、递归等。 4. 队列:先进先出(FIFO)结构,用于BFS搜索、事件驱动模型等。 5. 树结构:二叉树、平衡树(AVL、红黑树)、B树、B+树等,常见于数据库索引、...
在编码测试中,DFS常用于解决回溯问题,如迷宫求解、八皇后问题、括号匹配等。 4. **广度优先搜索(BFS)**:BFS是另一种遍历图或树的策略,它按照层次顺序访问节点。BFS常用于找出树的最短路径,或者在网络中寻找...
11. **有效的括号**:栈的应用,检查字符串中的括号是否匹配。 12. **合并两个有序链表**:递归或迭代方式,将两个已排序的链表合并成一个有序链表。 13. **括号生成**:动态规划,生成所有有效的括号组合。 14. **...
- **栈**:"后进先出"(LIFO)的数据结构,常用于括号匹配、函数调用等。 - **队列**:"先进先出"(FIFO)的数据结构,常用于任务调度、缓冲区管理。 - **树**:层次结构,如二叉树、平衡树(AVL、红黑树)等,...
- 扩展了一维线段树的功能,用于处理二维空间中的区间查询。 - 在计算机图形学等领域有着广泛应用。 15. **稳定婚姻匹配**: - Gale-Shapley算法可以找到一个稳定的婚姻匹配。 - 在配对问题中有着重要应用。 ...
- 常出现在各种组合数学问题中,如括号序列的合法数目、二叉树的种类数量等。 6. **康拓展开** - 康拓展开是一种将非负整数映射到多项式系数的方法,常用于解决一些计数问题。 7. **负进制** - 负进制是基数...
3. **栈(Stacks)**:利用栈的特性来解决逆序输出、括号匹配等问题。 4. **队列(Queues)**:例如实现循环队列、优先级队列等。 5. **字符串(Strings)**:涉及模式匹配、反转字符串、最长公共前后缀等。 6. *...
- **应用场景**:卡特兰数在计数问题中非常有用,如括号序列的有效配对数目等。 - **实现方式**:利用组合数学中的公式直接计算。 #### 4. Stirling Number (Second Kind) 斯特林第二类数 斯特林第二类数S(n,k)表示...