/*
又是AK局...
hjc又双叒叕AK了...
Hmmm...我侥幸
*/

Problem A card

给出无序序列a[]可以选择一个数插入到合适的位置作为一次操作,至少多少次操作后可以把序列变成有序。

对于100%的数据,序列长度 $l\leq5e5$

Solution : 最长上升子序列(LIS)-n

 这个是显然的结论,最优的话一定是保证LIS情况下,把除LIS外的数依次加到有序的LIS里面刚好加了这么多个。

所以答案是n-LIS,最优性显然: LIS保证需要插入的数字数量最少,而需要插入的数字最少只需要1步移动就一定可以保证把数列LIS增加1.

O(n log2 n)复杂度求LIS是这道题的关键。记f[i]表示长度为i的上升子序列末尾元素最大是多少。

显然当i变大的时候f[i]单调不降,有单调性,可以二分优化转移。找到第1个j,$f[j] \leq a[i]$,转移就行。

注意这个需要最大化j才能保证转移最优,所以是upper_bound而不是lower_bound。

# pragma G++ optimize()
# include <iostream>
# include <cstring>
# include <cstdio>
# include <algorithm>
using namespace std;
const int N=1e6+;
int a[N],ans,n,f[N];
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
int main()
{
freopen("card.in","r",stdin);
freopen("card.out","w",stdout);
n=read();
int ans=;
for (int i=;i<=n;i++) a[i]=read();
memset(f,0x3f,sizeof(f));
f[]=a[];
for (int i=;i<=n;i++) {
int p=upper_bound(f,f++n,a[i])-f-;
if (p==-) { f[]=min(f[],a[i]); ans=max(ans,); continue;}
ans=max(ans,p+);
f[p+]=min(f[p+],a[i]);
}
cout<<n-ans<<'\n';
return ;
}

Card.cpp

Problem B pikaqiu

给出n*m的地图每个点可以是石头(1)和路(0),经过路的代价是1,经过石头的代价是5,有起点(5)和终点(9),问起点到终点最小代价是多少,

若代价大于可支付的能量T,输出-1,否则输出剩余能量。

对于100%的数据: $n,m\leq 500$

Solution : 拆点跑Dijkstra最短路,考虑一个点和上下左右四个点有有向连边,若这个点是石头那么建长为5的边否则建长为1的边

然后这个图变成是n2个点,4n2条边的有向图,然后跑Dijkstra,求出[s,t]的最短路即可。(注意一定是单向加边,由于重边会导致本来是石头的边变成路,造成冲突)

-spfa已经死了 -不他没活过

复杂度O(n2 log2 n2)

# pragma G++ optimize()
# include <iostream>
# include <cstring>
# include <algorithm>
# include <cstdio>
# include <queue>
using namespace std;
const int N=1e6+;
const int dx[]={-,,,};
const int dy[]={,,,-};
int T,n,m;
int head[N],d[N],mp[N];
int s,tot=,INF;
bool vis[N];
struct record{
int pre,to,w;
}a[N<<];
# define Num(i,j) (m*(i-)+j)
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
void adde(int u,int v,int w)
{
a[++tot].pre=head[u];
a[tot].to=v;
a[tot].w=w;
head[u]=tot;
}
struct rec{
int id,lenth;
bool operator < (const rec a)const{
if (lenth!=a.lenth) return lenth>a.lenth;
else return id>a.id;
}
};
priority_queue<rec>q;
int dijkstra(int s,int e)
{
memset(vis,false,sizeof(vis));
memset(d,,sizeof(d));INF=d[];d[s]=;
rec Node; Node.id=s; Node.lenth=; q.push(Node);
while (! q.empty()){
rec Node=q.top(); q.pop();
int u=Node.id;
if (vis[u]==true) continue;
vis[u]=true;
for (int i=head[u];i!=;i=a[i].pre){
int v=a[i].to;
if (d[v]-a[i].w>d[u]) {
d[v]=a[i].w+d[u];
rec N; N.id=v;N.lenth=d[v];
q.push(N);
}
}
}
return d[e];
}
int main()
{
freopen("pikaqiu.in","r",stdin);
freopen("pikaqiu.out","w",stdout);
T=read(); n=read(); m=read();
int Str,End;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++) {
int t=Num(i,j);mp[t]=read();
if (mp[t]==) Str=t;
else if (mp[t]==) End=t;
}
for (int i=;i<=n;i++)
for (int j=;j<=m;j++) {
int now=Num(i,j); for (int tp=;tp<;tp++) {
int x=i+dx[tp],y=j+dy[tp];
if (x<||x>n||y<||y>m) continue;
int num=Num(x,y);
if (mp[num]==) adde(now,num,);
else adde(now,num,);
}
}
int Ans=dijkstra(Str,End);
if (Ans>=T) puts("-1");
else cout<<T-Ans<<'\n';
return ;
}

pikaqiu.cpp

Problem C min

一个含有n项的数列a[],求出每一项前面的第m个数到它这个区间内的最小值Mini

对于100%的数据 $n\leq 1e7$

Solution : 

单调队列题目,显然给了60%的线段树的暴力分。考虑O(n)做法。

弄一个双端队列deque然后前面弹出不在范围内的数,后面弹出不优的数和插入一个新的数。

队列维护的是下标单增,值单增的数列,同时保证在范围内,取数时取队头。

luogu 滑动窗口 单调队列裸题。

# pragma G++ optimize()
# include <deque>
# include <iostream>
# include <cstdio>
using namespace std;
const int N=1e6+;
int a[N],n,m;
deque<int>q;
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
void write(int x)
{
if (x<) x=-x,putchar('-');
if (x>) write(x/);
putchar(''+x%);
}
signed main()
{
freopen("min.in","r",stdin);
freopen("min.out","w",stdout);
n=read();m=read();
int Test=;
for (int i=;i<=n;i++) a[i]=read();
for (int i=;i<=n;i++) {
while (!q.empty()&&q.front()<i-m) q.pop_front();
while (!q.empty()&&a[q.back()]>=a[i]) q.pop_back();
q.push_back(i);
write(a[q.front()]);putchar(' ');
}
putchar('\n');
return ;
}

min.cpp

Problem D smrtfun

给出n个数对$(left_i,right_i)$选出若干组,

最大化 $\sum\limits_{i=1}^k left_i + \sum\limits_{i=1}^k right_i$ 且 $ \sum\limits_{i=1}^k left_i \geqslant 0 ,  \sum\limits_{i=1}^k right_i \geqslant  0$

对于100%的数据 $n \leq 100$

Solution: 背包问题,令$f[i][j]$表示前i个数对,$left$求和等于$k$,$right$求和的最大值。

转移比较简单就是  $f[i][j]=max \{ f[i-1][j],f[i-1][j-left_{now}]+right_{now} \} $

答案就是 max{f[n][j]} (0<=j<=INF)

然后初始值比较恶心,f[0][]都是负无穷,然后f[0][0]=0

然而c++没有下标为负数的数组,需要加上基底E=1e5才行,

时间复杂度O(n*mx2)

# pragma G++ optimize()
# include <iostream>
# include <cstdio>
# include <cstring>
# define left Lift
# define right Right
using namespace std;
const int N=,M=1e5+;
const int E=1e5;
const int INF=1e5-;
int f[N][M<<];
int left[N],right[N];
int n,mx,ans;
int max(int x,int y) {return (x>y?x:y);}
inline int read()
{
int X=,w=; char c=;
while(c<''||c>'') {w|=c=='-';c=getchar();}
while(c>=''&&c<='') X=(X<<)+(X<<)+(c^),c=getchar();
return w?-X:X;
}
int main()
{
freopen("smrtfun.in","r",stdin);
freopen("smrtfun.out","w",stdout);
n=read();
for (int i=;i<=n;i++)
left[i]=read(),right[i]=read();
memset(f[],~0x3f,sizeof(f[]));
f[][E]=;
for (int i=;i<=n;i++)
for (int j=-INF;j<=INF;j++) {
f[i][j+E]=f[i-][j+E];
if (j-left[i]<-INF||j-left[i]>INF) continue;
f[i][j+E]=max(f[i-][j-left[i]+E]+right[i],f[i][j+E]);
}
for (int j=;j<=INF;j++) if (f[n][j+E]>=) ans=max(ans,f[n][j+E]+j);
cout<<ans<<'\n';
return ;
}

smrtfun.cpp

HGOI 20190218 题解的更多相关文章

  1. HGOI 20181028 题解

    HGOI 20181028(复赛备考) /* 真是暴力的一天,最后一题MLE?由于数组开得太大了!!! 270滚粗 考场上好像智商高了很多?!(假的) */ sol:暴力求解,然后没有数据范围吐槽一下 ...

  2. HGOI 20190310 题解

    /* 又是又双叒叕WA的一天... 我太弱鸡了... 今天上午打了4道CF */ Problem 1 meaning 给出q组询问,求下列函数的值$ f(a) = \max\limits_{0 < ...

  3. HGOI 20190303 题解

    /* 记一串数字真难. 5435 今天比赛又是hjcAK的一天. 今天开题顺序是312,在搞T1之前搞了T3 昨天某谷月赛真是毒瘤. 但是讲评的同学不错,起码T4看懂了... 构造最优状态然后DP的思 ...

  4. HGOI 20180224 题解

    /* The Most Important Things: ljc chat with fyh on QQTa说期末考Ta数学74分感觉不好但是我觉得fyh是地表最强的鸭~~(of course en ...

  5. HGOI 20190217 题解

    /* for me,开训第一天 /beacuse 文化课太差被抓去补文化课了... 看一眼题 : AK局? 但是,Wa on test #10 in problem C 290! (就差那么一咪咪) ...

  6. HGOI 20181103 题解

    problem:把一个可重集分成两个互异的不为空集合,两个集合里面的数相乘的gcd为1(将集合中所有元素的质因数没有交集) solution:显然本题并不是那么容易啊!考场上想了好久.. 其实转化为上 ...

  7. HGOI 20181101题解

    /* 又是爆0的一天(不知道今年高考难不难,反正今天(信息学)真的难!) */ solution:对于两个数相加,有一个显然的结论就是要么不进位(相对于位数大的),要么(进最多一位) 然后对于整个数组 ...

  8. HGOI 20191108 题解

    Problem A 新婚快乐 一条路,被$n$个红绿灯划分成$n+1$段,从前到后一次给出每一段的长度$l_i$,每走$1$的长度需要$1$分钟. 一开始所有红绿灯都是绿色的,$g$分钟后所有红绿灯变 ...

  9. HGOI 20191107 题解

    Problem A 树状数组 给出下列$C++$代码: 设区间加操作$modify(l,r)$为调用两次$update(r,1)$和$update(l-1,-1)$ 设$f(l,r)$表示在初始$cn ...

随机推荐

  1. 通过C# WinForm控件创建的WPF WIndow窗口控件无法输入的问题

    原文:通过WinForm控件创建的WPF 控件无法输入的问题 今天把写的一个WPF程序发布到别的机器上执行,发现一个比较奇怪的问题:在那个机器上用英文输入法无法输入数字,非要切换到中文输入法才行:但在 ...

  2. C#大型电商项目优化(二)——嫌弃EF与抛弃EF

    上一篇博文中讲述了使用EF开发电商项目的代码基础篇,提到EF后,一语激起千层浪.不少园友纷纷表示:EF不适合增长速度飞快的互联网项目,EF只适合企业级应用等等. 也有部分高手提到了分布式,确实,性能优 ...

  3. c#基础系列3---深入理解ref 和out

    "大菜":源于自己刚踏入猿途混沌时起,自我感觉不是一般的菜,因而得名"大菜",于自身共勉. 扩展阅读 c#基础系列1---深入理解 值类型和引用类型 c#基础系 ...

  4. 基于 HTML5 Canvas 的 3D WebGL 机房创建

    对于 3D 机房来说,监控已经不是什么难事,不同的人有不同的做法,今天试着用 HT 写了一个基于 HTML5 的机房,发现果然 HT 简单好用.本例是将灯光.雾化以及 eye 的最大最小距离等等功能在 ...

  5. Libp2p学习(一)

    Libp2p学习 参考资料:libp2p-specifications : https://github.com/libp2p/specs 持续更新ing 1. 介绍 Libp2p的实现目标是: 支持 ...

  6. SpringMVC环境搭建——HelloWorld

    1.新建Maven Web 工程: 2.添加相关的依赖包(Spring MVC.tomcat插件等),具体的pom.xml文件如下 <project xmlns="http://mav ...

  7. Python进阶量化交易场外篇5——标记A股市场涨跌周期

    新年伊始,很荣幸笔者的<教你用 Python 进阶量化交易>专栏在慕课专栏板块上线了,欢迎大家订阅!为了能够提供给大家更轻松的学习过程,笔者在专栏内容之外会陆续推出一些手记来辅助同学们学习 ...

  8. Linux内核设计与实现 第十七章

    1. 设备类型 linux中主要由3种类型的设备,分别是: 设备类型 代表设备 特点 访问方式 块设备 硬盘,光盘 随机访问设备中的内容 一般都是把设备挂载为文件系统后再访问 字符设备 键盘,打印机 ...

  9. 20135337——实践一:Linux基础配置

    一.配置系统,权限中简单梳理遇到的问题 1.Ubuntu中root和普通用户相互切换 1.从user用户切换到root用户 执行:sudo su 2.从root用户切回user用户 执行:su use ...

  10. (Alpha)Let's-版本发布说明

    我们的Let’s APP发布了! (下载地址在“下载与安装”部分) Alpha版本功能 Alpha版本是我们发布的第一个版本,所以仅实现了活动实体和用户实体之间的基础联系功能. 基本功能 登录和注册 ...