link

搬来了曾经的题解

C-Candles

题意:数轴上有一些点,从原点开始移动到达这些点中的任意\(K\)个所需要的最短总路程

\(K\)个点必然是一个区间,枚举最左边的就行了

#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
int n,k,a[100005];
int ans;
int main()
{
n=read();k=read();
register int i;
for(i=1;i<=n;++i) a[i]=read();
ans=1e9;
for(i=1;i<=n-k+1;i++)
{
if(a[i]<=0&&a[i+k-1]<=0) ans=min(ans,-a[i]);
if(a[i]<=0&&a[i+k-1]>=0) ans=min(ans,-a[i]+a[i+k-1]+min(-a[i],a[i+k-1]));
if(a[i]>=0&&a[i+k-1]>=0) ans=min(ans,a[i+k-1]);
}
return 0*printf("%d\n",ans);
return 0;
}

D-Median of Medians

题意:求一个序列中"所有子序列的中位数"所组成的序列的中位数

首先二分一个答案,这样,所有小于它的都可以看成是\(-1\),大于等于它的是\(1\)

如果一个区间的和\(≥0\),那么它的中位数就不小于二分的答案

求区间数的话,对前缀和求逆序对就行了

#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
#define swap(x,y) (x^=y^=x^=y)
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
#define MN 100005
int n,m,a[MN],b[MN],c[MN];
int t[MN<<1];
void C(int x){for(;x<MN*2;x+=(x&-x))t[x]++;}
int G(int x){int res=0;for(;x;x-=(x&-x))res+=t[x];return res;}
inline bool check(int x)
{
ll cnt=0ll;
register int i;
for(i=1;i<=n;++i) c[i]=c[i-1]+(a[i]>=x?1:-1);
memset(t,0,sizeof t);
for(i=1;i<=n;++i)
{
C(c[i-1]+MN);
cnt+=G(c[i]+MN);
}
return cnt*4>=1ll*n*(n+1);
}
int main()
{
n=read();register int i;
for(i=1;i<=n;++i) a[i]=b[i]=read();
std::sort(b+1,b+n+1);
m=std::unique(b+1,b+n+1)-b-1;
int l,r,ans;
for(l=1,r=m,ans=0;l<=r;)
{
int mid=l+r>>1;
if(check(b[mid])) ans=mid,l=mid+1;
else r=mid-1;
}
printf("%d\n",b[ans]);
}

E-Ribbons on Tree

题意:给定一棵点数为偶数的树,要求有多少种将点两两配对的方案使得每一条边至少被一对匹配点之间的最短路径覆盖。

  • 容斥

令\(F(E)\)表示\(E\)中的边不被覆盖的方案数

\[ans=\sum_{E}^{}(-1)^{|E|}F(E)
\]

  • 令\(g(n)\)表示大小为n的序列中有多少中配对方法,当n为偶数时
    \[g(n)=\frac{n!}{(\frac{n}{2})!\cdot 2^{\frac{n}{2}}}=(n-1)\cdot (n-3)\cdot ......\cdot 3\cdot 1
    \]

  • 当\(E\)确定时,原树被分成若干个联通块,且我们的点对必须同处一个块内
  • f[i][j]表示以i为根的子树内,有j个点是与i的父亲节点同处一个联通块的方案数

    然后就是树形dp了,在枚举子树的时候,虽然有三层的\(for\),但是可以发现是和子树大小相关个,我们可以当作每次是选了两个子树做匹配,节点之间两两只会配对一次,所以复杂度仍然是\(O(n^2)\)的
  • 根据容斥,在计算\(f[i][0]\)时,我们得变号
  • 答案是\(-f[1][0]\),因为我们最后多变了一次号
#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
#define swap(x,y) (x^=y^=x^=y)
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
#define MN 5005
#define mod 1000000007
ll dp[MN][MN],g[MN],t[MN],siz[MN];
int hr[MN],en;
struct edge{int to,nex;}e[MN<<1];
inline void ins(int f,int t)
{
e[++en]=(edge){t,hr[f]};hr[f]=en;
e[++en]=(edge){f,hr[t]};hr[t]=en;
}
void dfs(int x,int fa){
siz[x]=1;dp[x][1]=1;register int i,j,k;
for(i=hr[x];i;i=e[i].nex)if(e[i].to^fa)
{
dfs(e[i].to,x);
for(j=0;j<=siz[x];++j)
t[j]=dp[x][j],dp[x][j]=0;
for(j=0;j<=siz[x];++j)
for(k=0;k<=siz[e[i].to];++k)
(dp[x][j+k]+=t[j]*dp[e[i].to][k]%mod)%=mod;
siz[x]+=siz[e[i].to];
}
for(i=0;i<=siz[x];i+=2)
dp[x][0]=(dp[x][0]-dp[x][i]*g[i]%mod+mod)%mod; }
int main(){
register int i,n,x;
n=read();
for(i=1;i<n;++i) x=read(),ins(x,read());
g[0]=1;for(i=2;i<=n;g[i]=g[i-2]*(i-1)%mod,i+=2);
dfs(1,0);
printf("%lld\n",mod-dp[1][0]);
return 0;
}

F-Robots and Exits

题意:有\(n\)个机器人和\(m\)个出口,每次可以将所有机器人的坐标左移一格或者右移一格,当一个机器人正好在出口上时,它就会消失,问所有机器人消失的出口序列有多少种情况。

首先,机器人只会从左边或右边的第一个出口消失(除去那些最左边最右边和已经在出口上的)

  • 一个点离左边出口距离为a,右边为i,我们可以把它看成是一个点(a,b)。

  • 那么每次操作,就是把所有的点变成 (a-1,b)或者(a,b-1)

    你可能会问,不应该是(a-1,b+1)或者(a+1,b-1)吗?

    其实上,当我们执行了一次操作后,那些正好碰到坐标轴的点就已经消失,我们不要管它

    而对于剩下的点,显然它的坐标在这个范围内变化都是不会掉下去的,所以我们不妨以它的最靠近坐标轴的坐标作为它的坐标

    或者说,这个坐标仅仅只是表示一个最大变化量而已

  • 换一种理解方式,我们假设是原点从\((0,0)\)处开始向右或向左走,最后将图分成两个部分

上半部分的点掉进了左边的出口(可以想象是y轴先碰到了它,也就是横坐标先变成了0),下半部分的点掉进了右边的出口

  • 我们要询问有多少种这样的划线方式(两条折线不同当且仅当它们分出的两个部分是不同的)
  • f[i]表示最后一个经过的点是i的方案数,最后一个的意思是,经过i之后,折线不再往上走

也就是i是位于折线下方的最高的点

\[f[i]=1+\sum_{x_j<x_i,y_j<y_i}f[j]
\]

  • 我们用树状数组就可以轻易维护
#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
#define swap(x,y) (x^=y^=x^=y)
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
#define MN 100005
#define mod 1000000007
int ans,x[MN],y[MN];
int cnt,a[MN],b[MN],B[MN],id[MN];
inline bool cmp(const int&o,const int&oo){return a[o]==a[oo]?b[o]<b[oo]:a[o]<a[oo];}
int t[MN],s[MN];
void C(int x,int v){for(;x<MN;x+=(x&-x))(t[x]+=v)%=mod;}
int G(int x){int r=0;for(;x;x-=(x&-x))(r+=t[x])%=mod;return r;}
int main()
{
register int i,j,n,m;
n=read(),m=read();
for(i=1;i<=n;++i) x[i]=read();
for(j=1;j<=m;++j) y[j]=read();
std::sort(x+1,x+n+1);std::sort(y+1,y+m+1);
for(i=1;i<=n;++i)
{
int p=std::lower_bound(y,y+m+1,x[i])-y;
if(p<=1||p>m||x[i]==y[p]) continue;
a[++cnt]=x[i]-y[p-1];B[cnt]=b[cnt]=y[p]-x[i];id[cnt]=cnt;
}
std::sort(B+1,B+cnt+1);
int Bcnt=std::unique(B+1,B+cnt+1)-B-1;
for(i=1;i<=cnt;i++) b[i]=std::lower_bound(B+1,B+Bcnt+1,b[i])-B;
std::sort(id+1,id+cnt+1,cmp);
for(i=1,j=1;i<=cnt;++i)
{
if(a[id[i]]==a[id[i-1]]&&b[id[i]]==b[id[i-1]]) continue;
for(;a[id[j]]<a[id[i]];++j) if(a[id[j]]!=a[id[j-1]]||b[id[j]]!=b[id[j-1]]) C(b[id[j]],s[j]%mod);
s[i]=1+G(b[id[i]]-1);
(ans+=(s[i]%mod))%=mod;
}
printf("%d\n",ans+1);
return 0;
}

Blog来自PaperCloud,未经允许,请勿转载,TKS!

【AtCoder】 ARC 101的更多相关文章

  1. 【AtCoder】ARC 081 E - Don't Be a Subsequence

    [题意]给定长度为n(<=2*10^5)的字符串,求最短的字典序最小的非子序列字符串. http://arc081.contest.atcoder.jp/tasks/arc081_c [算法]字 ...

  2. 【Atcoder】ARC 080 E - Young Maids

    [算法]数学+堆 [题意]给定n个数的排列,每次操作可以取两个数按序排在新序列的头部,求最小字典序. [题解] 转化为每次找字典序最小的两个数按序排在尾部,则p1和p2的每次选择都必须满足:p1在当前 ...

  3. 【Atcoder】ARC 080 F - Prime Flip

    [算法]数论,二分图最大匹配 [题意]有无限张牌,给定n张面朝上的牌的坐标(N<=100),其它牌面朝下,每次操作可以选定一个>=3的素数p,并翻转连续p张牌,求最少操作次数使所有牌向下. ...

  4. 【AtCoder】 ARC 097

    link C-K-th Substring 题意:找出已知串中第\(k\)大的子串,子串相同的不算 \(k\)好小啊,要怎么做啊 不是[Tjoi2015]弦论吗 算了,直接SAM吧 #include& ...

  5. 【AtCoder】 ARC 096

    link C-Half and Half 题意:三种pizza,可以花\(A\)价钱买一个A-pizza,花\(B\)价钱买一个B-pizza,花\(C*2\)价钱买A-pizza和B-pizza各一 ...

  6. 【AtCoder】 ARC 098

    link C-Attention 题意:一个字符队列,每个位置是\(W\)或\(E\),计算最小的修改数量,使得存在一个位置,它之前的都是\(E\),之后的都是\(F\) #include<bi ...

  7. 【AtCoder】 ARC 099

    link C-Minimization 枚举覆盖\(1\)的区间,两边的次数直接算 #include<bits/stdc++.h> #define ll long long #define ...

  8. 【AtCoder】 ARC 100

    link C-Linear Approximation 给出\(N\)个数\(A_1,A_2,...,A_N\) ,求一个数\(d\),最小化\(\sum_{i=1}^N|A_i-(d+i)|\) 把 ...

  9. 【AtCoder】 ARC 102

    link C-Triangular Relationship 发现要么全部是\(K\)的倍数,要么全部是模\(K\)余\(K/2,(K=2n)\) #include<bits/stdc++.h& ...

随机推荐

  1. pands模块的妙用爬取网页中的表格

    拿我这篇为例https://www.cnblogs.com/pythonywy/p/11574340.html import pandas as pd df = pd.read_html('https ...

  2. http的GET方法参数中不能传列表,接收端的key会变

    如下 async initTable() { await getHostAttributesForUser({'username': this.username}).then(response =&g ...

  3. 原生JS实现前端动画框架

    封装了一个JS方法,可支持块元素的常规动画:高.宽.透明度.位置等,同时支持链式动画和同时运动,参照imooc整理,具体代码如下: /** * 获取HTML元素属性值 * obj是Element, a ...

  4. 编写可维护的JavaScript-随笔(七)

    将配置数据从代码中分离出来 代码中有些数据有修改的可能,如果放在函数中的话后期修改的时候会带来一些不必要的风险 需要将配置数据从代码中抽取出来,如果配置数据多的话可以放入一个对象中,然后修改抽取出来的 ...

  5. Python3 解决windows里PIP下载安装速度慢

    直接保存为xxx.py运行即可 自动在用户文件夹创建pip文件夹,并创建配置文件:pip.ini 从此告别pip install XXXX 下载模块速度超级慢的问题! # -*- coding: ut ...

  6. Java集合学习(3):HashSet

    一.概述 HashSet实现Set接口,由哈希表(实际上是一个HashMap实例)支持.它不保证set 的迭代顺序:特别是它不保证该顺序恒久不变.此类允许使用null元素. HashSet是基于Has ...

  7. 使用jmeter 设计流程发起测试

    业务场景 需要实现用户在登录后,能够持续的发起流程. 需要注意的点: 1.使用不同的用户登录. 2.登录后发起可以持续的发起流程. 实现步骤 1.先使用badboy 录制脚本. 2.使用jmeter ...

  8. python 中根据python版本(2或3)定义函数

    示意代码如下: #_*_coding:UTF-8_*_ import time import socket import os import sys if sys.version_info.major ...

  9. 2019年杭电多校第三场 1011题Squrirrel(HDU6613+树DP)

    题目链接 传送门 题意 给你一棵无根树,要你寻找一个根节点使得在将一条边权变为\(0\)后,离树根最远的点到根节点的距离最小. 思路 本题和求树的直径很像,不过要记得的东西有点多,且状态也很多. \( ...

  10. 洛谷P3810 陌上花开(CDQ分治)

    洛谷P3810 陌上花开 传送门 题解: CDQ分治模板题. 一维排序,二维归并,三维树状数组. 核心思想是分治,即计算左边区间对右边区间的影响. 代码如下: #include <bits/st ...