先讲下题目意思

给定一个长度为\(n\)的序列\((1 \leq n \leq 100)\),

每次合并两个元素\(i,i+1\),即将\(i,i+1\)变为一个新的元素,权值为\(a[i]-a[i+1]\)(\(a\)为权值),

求在\(n-1\)次合并后剩下的元素的权值为\(t\)\((-10000 \leq t \leq 10000)\)的步骤(保证有解,spj)

解析

这题的减法有点不好弄啊...

但是,仔细想想,

如果我们将\(j\)和\(k\)合并成\((j-k)\),

再将\(i\)和\((j-k)\)合并,

就变成了\(i-j+k\),

所以,有些减法在运算时就已经消掉了!!

因此,我们只需要在每个数字前面填上加号或减号,使结果等于\(t\)就行了.

并且,注意到,第二个元素前只能是减号,因为前面已经不能消掉它的减号了.

然后,用DP递推地求符号就行了.

那么,怎么求呢?

我们可以设\(f[i][j]\)表示加到第\(i\)个数总和为\(j\)时第第\(i\)个数的符号,\(1\)为加,\(-1\)为减,

那么,在转移时,若\(f[i-1][j]\)不为零,即前面能够加到\(j\),

那么,\(f[i][j+a[i]]\)的符号就为正,\(f[i][j-a[i]]\)的符号就为负,

于是,最后从\(t\)倒推就行了,

不过注意,由于\(t\)可能为负,所以在求之前可以先加上一个值再算.

最后,注意输出方案,

首先我们可以记录下操作了几步,

由于我们是从前往后扫,所以每次用编号减掉操作次数就行了.

如果一个数的符号为正,那么它一定是被减掉后消掉的,

因此就输出它前面的标号减操作次数.

而对于剩下的数,很明显,我们对\(1\)一直操作就行了,

因此,若符号为负,输出\(1\)即可.

(讲的不清楚的地方自己算一下吧.)

上代码吧:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std; inline int read(){
int sum=0,f=1;char ch=getchar();
while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
return f*sum;
} const int ret=10000;//加上一个数把t变成非负的
int n,m,sum;
int a[100001];
int f[101][20005];
int op[100001]; int main(){
n=read();m=read();//m就是t,习惯而已qwq
for(int i=1;i<=n;i++) a[i]=read(),sum+=a[i];
sum=ret<<1;
f[2][a[1]+ret-a[2]]=-1;f[1][a[1]+ret]=1;
for(int i=3;i<=n;i++){
for(int j=0;j<=sum;j++){
if(!f[i-1][j]) continue;
if(j+a[i]<=sum) f[i][j+a[i]]=1;
if(j-a[i]>=0) f[i][j-a[i]]=-1;
}
}//DP
sum=m+ret;
for(int i=n;i>=2;i--){
op[i]=f[i][sum];
sum-=a[i]*op[i];
}//求最后的符号
int tot=0;
for(int i=2;i<=n;i++) if(op[i]==1) printf("%d\n",i-1-tot),tot++;
for(int i=2;i<=n;i++) if(op[i]==-1) puts("1");
return 0;
}

题解 【POJ1722】 SUBTRACT的更多相关文章

  1. poj1722 SUBTRACT【线性DP】

    SUBTRACT Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 2037   Accepted: 901   Special ...

  2. poj1722 SUBTRACT

    应该是基础的dp练手题 线性dp最主要的就是关于阶段的划分,这个题中我没想到的一点就是开状态的时候使用了前i个数能合成的数来记录 我自己的想法就是类似于区间dp这样的记录方法,这种方法确实开了很多冗余 ...

  3. 常规DP专题练习

    POJ2279 Mr. Young's Picture Permutations 题意 Language:Default Mr. Young's Picture Permutations Time L ...

  4. leetcode & lintcode 题解

    刷题备忘录,for bug-free 招行面试题--求无序数组最长连续序列的长度,这里连续指的是值连续--间隔为1,并不是数值的位置连续 问题: 给出一个未排序的整数数组,找出最长的连续元素序列的长度 ...

  5. IEEE Bigger系列题解

    Bigger系列题解 Bigger Python 坑点在于要高精度以及表达式求值,用java写可以很容易避免高精度问题 然后这道题就可以AC了 代码 import java.io.*; import ...

  6. Codeforces Round #177 (Div. 2) 题解

    [前言]咦?如今怎么流行打CF了?于是当一帮大爷在执着的打div 1的时候,我偷偷的在刷div 2.至于怎么决定场次嘛.一般我报一个数字A,随便再拉一个人选一个数字B.然后開始做第A^B场.假设认为机 ...

  7. 遗传编程(GA,genetic programming)算法初探,以及用遗传编程自动生成符合题解的正则表达式的实践

    1. 遗传编程简介 0x1:什么是遗传编程算法,和传统机器学习算法有什么区别 传统上,我们接触的机器学习算法,都是被设计为解决某一个某一类问题的确定性算法.对于这些机器学习算法来说,唯一的灵活性体现在 ...

  8. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  9. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

随机推荐

  1. JWT的认识和session的区别

    1.前后端分离框架中前端和后端域名不同,不能跨域请求,加上移动端无cookie,所以无法使用session.2.基于token的认证和传统的session认证的区别: 传统的session认证: 我们 ...

  2. css多种方式实现等宽布局

    本文讲的等宽布局是在不手动设置元素宽度的情况下,使用纯css实现各个元素宽度都相当的效果. 1.使用table-cell实现(兼容ie8) <style> body,div{ margin ...

  3. ubuntu 系统升级

    1.更新软件源和软件 $ sudo apt-get update && sudo apt-get dist-upgrade 2.安装update-manager-core $ sudo ...

  4. k-means算法处理聚类标签不足的异常

    k-means算法在人群聚类场景中,是一个非常实用的工具.(该算法的原理可以参考K-Means算法的Python实现) 常见调用方式 该算法常规的调用方式如下: # 从sklearn引包 from s ...

  5. go 拼接sql

    //原文链接:https://www.jianshu.com/p/a0569157c418 golang mysql拼接子查询 使用fmt.Sprintf拼接SQL 实例代码 func Select( ...

  6. 两个链表的第一个公共结点——牛客offer

    题目描述: 输入两个链表,找出它们的第一个公共结点. 题目分析: 只是数据域相同不是公共节点.公共结点代表该节点在两个链表中的数据域和指针域都是相同的,这意味着从该公共节点开始,后面的结点都是两个链表 ...

  7. mysql 8.x 集群出现:Last_IO_Error: error connecting to master 'repl@xxx:3306' - retry-time: 60 retries: 1

    网上的经验:网络不同,账号密码不对,密码太长,密码由 # 字符:检查MASTER_HOST,MASTER_USER,MASTER_PASSWORD(不知道 MASTER_LOG_FILE 有没有影响) ...

  8. c# winfrom 界面设计

    1.在用DotnetBar的RibbonControl时,界面最大化时,会把电脑桌面的任务栏遮盖住: 解决办法:在load事件中写入: , Screen.PrimaryScreen.WorkingAr ...

  9. TVM设备添加以及代码生成

    因为要添加的设备是一种类似于GPU的加速卡,TVM中提供了对GPU编译器的各种支持,有openCl,OpenGL和CUDA等,这里我们选取比较熟悉的CUDA进行模仿生成.从总体上来看,TVM是一个多层 ...

  10. 【异常】Phoenix异常:java.lang.ArithmeticException: Rounding necessary

    1 异常sql upsert into WMBIGDATA.ODS_ES_CHARGING_STATION(id,evcosType,address,serviceTel,supportOrder,o ...