luogu1415 拆分数列
题目大意
给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数。如果有多组解,则输出使得最后一个数最小的同时,字典序最大的解(即先要满足最后一个数最小;如果有多组解,则使得第一个数尽量大;如果仍有多组解,则使得第二个数尽量大,依次类推……)。
题解
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int MAX_N = 510;
int A[MAX_N];
int F1[MAX_N], F2[MAX_N], AnsF2[MAX_N];
int N; void CharToA(char *s)
{
N = strlen(s + 1);
for (int i = 1; i <= N; i++)
A[i] = s[i] - '0';
} bool RangeLt(int l1, int r1, int l2, int r2)
{
while (A[l1] == 0 && l1 <= r1)
l1++;
while (A[l2] == 0 && l2 <= r2)
l2++;
if (r1 - l1 != r2 - l2)
return r1 - l1 < r2 - l2;
else
for (int i = 0; i <= r1 - l1; i++)
if (A[l1 + i] != A[l2 + i])
return A[l1 + i] < A[l2 + i];
return false;
} void DP1()
{
for (int i = 1; i <= N; i++)
F1[i] = 1;
for (int i = 1; i <= N; i++)
for (int j = 2; j <= i; j++)
if (RangeLt(F1[j - 1], j - 1, j, i))
F1[i] = j;
} void DP2(int ed)
{
memset(F2, 0, sizeof(F2));
F2[ed] = N;
for (int i = ed - 1; i >= 1; i--)
for (int j = i; j <= ed - 1; j++)
if (RangeLt(i, j, j + 1, F2[j + 1]))
F2[i] = j;
} bool CmpF2()
{
int cur = 1;
while (cur <= N)
{
if (F2[cur] != AnsF2[cur])
return F2[cur] > AnsF2[cur];
cur = F2[cur] + 1;
}
return false;
} void Print()
{
int cur = 1;
bool tag = false;
while (cur <= N)
{
if (tag)
putchar(',');
tag = true;
for (int i = cur; i <= AnsF2[cur]; i++)
putchar('0' + A[i]);
cur = AnsF2[cur] + 1;
}
} int main()
{
char s[MAX_N];
scanf("%s", s + 1);
CharToA(s);
DP1();
DP2(F1[N]);
memcpy(AnsF2, F2, sizeof(F2));
for (int i = F1[N] - 1; A[i] == 0; i--)
{
if (!RangeLt(F1[i - 1], i - 1, i, N))
break;
DP2(i);
if (CmpF2())
memcpy(AnsF2, F2, sizeof(F2));
}
Print();
return 0;
}
luogu1415 拆分数列的更多相关文章
- 洛谷P1415 拆分数列[序列DP 状态 打印]
题目背景 [为了响应党中央勤节俭.反铺张的精神,题目背景描述故事部分略去^-^] 题目描述 给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数.如果有多组解,则输出使得最后一个数最小的同时 ...
- 洛谷 P1415 拆分数列 解题报告
拆分数列 题目背景 [为了响应党中央勤节俭.反铺张的精神,题目背景描述故事部分略去^-^] 题目描述 给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数. 如果有多组解,则输出使得最后一个 ...
- luoguP1415 拆分数列 [dp]
题目描述 给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数.如果有多组解,则输出使得最后一个数最小的同时,字典序最大的解(即先要满足最后一个数最小:如果有多组解,则使得第一个数尽量大:如 ...
- 洛谷P1415 拆分数列(dp)
题目链接:传送门 题目: 题目背景 [为了响应党中央勤节俭.反铺张的精神,题目背景描述故事部分略去^-^] 题目描述 给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数.如果有多组解,则输 ...
- 洛谷P1415 拆分数列
题目背景 [为了响应党中央勤节俭.反铺张的精神,题目背景描述故事部分略去^-^] 题目描述 给出一列数字,需要你添加任意多个逗号将其拆成若干个严格递增的数.如果有多组解,则输出使得最后一个数最小的同时 ...
- P1415 拆分数列
传送门 DP数列长度过大无法枚举,考虑DP设f1[i]储存以第i个字符为结尾时,的最后一个数最小时,这个数的开头的位置(很难想有木有)OK,状态有了,方程想一想就出来了:设$num[i][j]$为数列 ...
- [luoguP1415] 拆分数列(DP)
传送门 t(i,j)表示下标从i到j的数 d[i]表示以i结尾的最小的数的下标 d[i]=max(j) (1<=j<=i && t(d[j-1],j-1)<t(j,i ...
- P1415 拆分数列 DP
传送门: 题意: 将一个数字串分成许多不同的小串,使得这些小串代表的数字严格递增,要求最后一个数字尽可能地小. 然后满足字典序尽可能大. 思路: 由于最后一个数字要尽可能地小,所以先处理出每个数的L[ ...
- Codeforces 950 010子序列拆分 数列跳棋
A B a,b两个序列分成K段 每一段的值相等 #include <bits/stdc++.h> #define PI acos(-1.0) #define mem(a,b) memset ...
随机推荐
- Android 根据QQ号跳转到QQ聊天界面
从自己开发的应用中根据QQ号跳转到QQ应用的聊天界面,实现起来很方便: 即: startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(" ...
- Android sensor 系统框架 (二)
连载上一篇http://www.cnblogs.com/hackfun/p/7327320.html (D) 如何加载访问.so库 在前一篇博客http://www.cnblogs.com/hackf ...
- Android基础TOP4:Tost的使用
Activity: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xm ...
- [Windows Server 2008] 手工创建安全网站
★ 欢迎来到[护卫神·V课堂],网站地址:http://v.huweishen.com★ 护卫神·V课堂 是护卫神旗下专业提供服务器教学视频的网站,每周更新视频.★ 本节我们将带领大家:手工创建安全站 ...
- Android ExpandableListView的使用详解
ExpandableListView(可扩展的ListView) ExpandableListVivew是ListView的子类,它在普通ListView的基础上进行了扩展,它把应用中的列表项分为几组 ...
- js 学习笔记---BOM
window对象 1. window 对象是Global对象,在全局作用域中声明的变量和函数都可以通过window.来访问.跟直接在window上添加属性效果一样.唯一的区别就是delete时,如果是 ...
- 4星|《超级技术:改变未来社会和商业的技术趋势》:AI对人友好吗
超级技术:改变未来社会和商业的技术趋势 多位专家或经济学人编辑关于未来的预测,梅琳达·盖茨写了其中一章.在同类书中属于水平比较高的,专家只写自己熟悉的领域,分析与预测有理有据而不仅仅是畅想性质. 以下 ...
- Oreacle 语句
SELECT * FROM = and CREATETIME >to_date('2019-01-01' , 'yyyy-mm-dd hh24:mi:ss') and CREATETIME &l ...
- Python星号表达式
有时候可能想分解出某些值然后丢弃它们,可以使用诸如 _ 或者 ign(ignored)等常用来表示待丢弃值的变量名: record = ('ACME', 50, 123.45, (12, 18, 20 ...
- B+树知识点
B+树介绍 目录 B+树 B+树的插入操作 B+树的删除操作 回到顶部 B+树 B+树和二叉树.平衡二叉树一样,都是经典的数据结构.B+树由B树和索引顺序访问方法(ISAM,是不是很熟悉?对,这也 ...