NOI.ac 模拟赛20181103 排队 翘课 运气大战
题解
排队
20% 1≤n≤20,1≤x,hi≤201\le n\le 20, 1\le x,h_i\le 201≤n≤20,1≤x,hi≤20 随便暴力
50% 1≤n≤2000,1≤x,hi≤1e91\le n \le 2000, 1\le x,h_i\le 1e91≤n≤2000,1≤x,hi≤1e9 枚举把哪个定成中位数
100% 1≤n≤2e5,1≤x,hi≤1e91\le n\le 2e5,1\le x,h_i\le 1e91≤n≤2e5,1≤x,hi≤1e9
先把高度排序。贪心地修改。最好的方法是把最中间的人变成xxx。那么就把左边比他高的或者右边比他矮的都改成xxx就好啦。
#include <bits/stdc++.h>
using namespace std;
#define LL long long
const int MAXN = 200005;
int n, x, h[MAXN];
LL Ans;
int main()
{
scanf("%d%d", &n, &x);
for(int i = 1; i <= n; i++)
scanf("%d", &h[i]);
sort(h + 1, h + n + 1);
int l = n>>1, r = (n>>1)+2;
Ans = abs(h[(n>>1)+1]-x), h[(n>>1)+1] = x;
while(l >= 1 && r <= n)
{
Ans += max(h[l]-h[l+1], 0), h[l] = min(h[l], h[l+1]); l--;
Ans += max(h[r-1]-h[r], 0), h[r] = max(h[r], h[r-1]); r++;
}
printf("%lld\n", Ans);
}
翘课
10% 2≤n,m≤500,k=12\le n,m\le 500,k=12≤n,m≤500,k=1
20% 2≤n,m≤200000,k=12\le n,m\le 200000,k=12≤n,m≤200000,k=1 暴力
40% 2≤n,m≤2000,k=22\le n,m\le 2000,k=22≤n,m≤2000,k=2 找环
70% 2≤n,m≤2000,k<n2\le n,m\le 2000,k<n2≤n,m≤2000,k<n 每次跑点判断
100% 2≤n,m≤200000,1≤k<n2\le n,m\le 200000,1\le k <n2≤n,m≤200000,1≤k<n 倒着删
考虑先把所有边加进去,再倒过来删边。
对于每个点,我们考虑要不要删掉它,删掉它表示它连接着选中的点的度小于 K 了。删完它之后我们把它邻居的度都–,然后再看它的邻居要不要被删掉。这样预处理把每个点判一次要不要删。
之后删一条边也是删完就判断连接的两个点需不需要被删掉。因为每个点只会被删一次,所以整体还是O(n)的。
题目做法可能多样,但思路应该都是倒过来。
#include<bits/stdc++.h>
using namespace std;
const int MAXN = 200005;
const int MAXM = 200005;
inline void read(int &num)
{
char ch; int flag=1;
while(!isdigit(ch=getchar()))if(ch=='-')flag=-flag;
for(num=ch-'0';isdigit(ch=getchar());num=num*10+ch-'0');
num*=flag;
}
int n, m, k, d[MAXN], Ans;
int fir[MAXN], to[MAXM<<1], nxt[MAXM<<1], cnt = 1;
inline void Add(int u, int v)
{
to[++cnt] = v; nxt[cnt] = fir[u]; fir[u] = cnt;
to[++cnt] = u; nxt[cnt] = fir[v]; fir[v] = cnt;
}
bool flg[MAXN], used[MAXM<<1];
inline void del(int u)
{
if(!flg[u]) return;
flg[u] = 0; Ans--;
for(int i = fir[u]; i; i = nxt[i])
if(used[i^1] && (used[i^1] = 0, --d[to[i]] == k-1)) del(to[i]);
}
bool vis[MAXN];
inline bool dfs1(int u)
{
if(vis[u]) return flg[u];
vis[u] = 1;
flg[u] = 1; Ans++;
for(int i = fir[u]; i; i = nxt[i])
if(dfs1(to[i]))
++d[u], used[i] = 1;
if(d[u] >= k) return 1;
del(u);
return 0;
}
int U[MAXM], V[MAXM], ans[MAXM];
int main ()
{
read(n), read(m), read(k);
for(int i = 1; i <= m; i++)
read(U[i]), read(V[i]), Add(U[i], V[i]);
for(int i = 1; i <= n; i++)
if(!vis[i]) dfs1(i);
for(int i = m; i >= 1; i--)
{
ans[i] = Ans;
if(used[i<<1] && (used[i<<1] = 0, --d[U[i]] == k-1)) del(U[i]);
if(used[i<<1|1] && (used[i<<1|1] = 0, --d[V[i]] == k-1)) del(V[i]);
}
for(int i = 1; i <= m; i++) printf("%d\n", ans[i]);
}
运气大战
20% n≤10,q≤10n\le 10,q\le 10n≤10,q≤10 暴力
另30% n≤1000,q≤100n\le 1000,q\le 100n≤1000,q≤100 这个点我也不记得为什么放了,但看起来很多人只拿到了50所以可能还是有点意义的?
另20% n≤30000,q≤500n\le 30000,q\le 500n≤30000,q≤500 nq 的 dp
100% 2≤n≤30000,1≤q≤10000,1≤wi,ri≤1e6,1≤ai̸=bi≤n2\le n\le 30000,1\le q\le 10000,1\le w_i,r_i \le 1e6,1\le a_i\not=b_i \le n2≤n≤30000,1≤q≤10000,1≤wi,ri≤1e6,1≤ai̸=bi≤n
由于排序不等式,我们尽量想顺序放。两边都排序。
由于 n 个不能配的干扰,又不能完全顺序放。
有个结论,最后匹配出第 i 个人的运气值是第 j 个的话,∣i−j∣≤2|i-j|\le2∣i−j∣≤2。这个结论从最小化逆序对的个数来看,自己把附近几个线连起来画一画证明一下。
这样就可以用 dp[i]表示到 i 为止所有配好的最优答案。计算的时候需要用到前三轮的答案然后讨论一下。这个是 O(nq)的,可以过70%。
用线段树记录区间答案。区间记录这样的信息:把这个区间前0-2个和后0-2个元素去掉的答案,用3x3的矩阵维护。这样复杂度是O(qlogn)。
由于出题人良心发现将时限改为3s,常数小的nq算法也能过1.5s左右,下贴AC代码:
#include <bits/stdc++.h>
using namespace std;
#define LL long long
const int MAXN = 30005;
const LL INF = 1ll<<60;
inline void read(int &num)
{
char ch; int flag=1;
while(!isdigit(ch=getchar()))if(ch=='-')flag=-flag;
for(num=ch-'0';isdigit(ch=getchar());num=num*10+ch-'0');
num*=flag;
}
struct node
{
int val, id;
}A[MAXN], B[MAXN];
LL dp[MAXN], cost[MAXN][2];
int n, q, posa[MAXN], posb[MAXN];
inline LL Calc(int i, int j)
{
if(i < 1 || j < 1 || i > n || j > n || A[i].id == A[j].id) return -INF;
return 1ll * A[i].val * B[j].val;
}
inline void Upd(int i)
{
if(i < 1 || i > n) return;
cost[i][0] = Calc(i, i+1) + Calc(i+1, i);
cost[i][1] = max(Calc(i, i+1) + Calc(i+1, i+2) + Calc(i+2, i),
Calc(i, i+2) + Calc(i+1, i) + Calc(i+2, i+1));
}
inline bool cmp(node x, node y) { return x.val < y.val; }
int main ()
{
read(n), read(q);
for(int i = 1; i <= n; i++) read(A[i].val), A[i].id = i;
for(int i = 1; i <= n; i++) read(B[i].val), B[i].id = i;
sort(A + 1, A + n + 1, cmp);
sort(B + 1, B + n + 1, cmp);
for(int i = 1; i <= n; i++) posa[A[i].id] = posb[B[i].id] = i, Upd(i);
int x, y;
while(q--)
{
read(x), read(y), swap(B[posb[x]].id, B[posb[y]].id), swap(posb[x], posb[y]);
for(int i = -2; i <= 0; i++)
Upd(posa[x+i]), Upd(posb[x+i]), Upd(posa[y+i]), Upd(posb[y+i]);
dp[n+1] = 0;
for(int i = n; i; i--)
{
dp[i] = max(dp[i+2] + cost[i][0], dp[i+3] + cost[i][1]);//i与i+1/2不能配对的情况一定会被另外更优的情况所取代,可能是此行的另一个max,可能是下一行的单独配对时取的max
if(A[i].id != B[i].id) dp[i] = max(dp[i], dp[i+1] + 1ll * A[i].val * B[i].val);
}
printf("%lld\n", dp[1]);
}
}
NOI.ac 模拟赛20181103 排队 翘课 运气大战的更多相关文章
- NOI.ac模拟赛20181021 ball sequence color
T1 ball 可以发现每次推动球时,是将每个球的位置 −1-1−1 ,然后把最左边的球放到 P−1P-1P−1 处. 记个 −1-1−1 次数,再用set维护就好了. #include <bi ...
- [NOI P模拟赛] 传统艺能(子序列自动机、矩阵乘法,线段树)
(2:00)OID:"完了,蓝屏了!"(代码全消失) 众人欢呼 OID:开机,"原题测试--" (30min later)OID 开始传统艺能: " ...
- 【NOI P模拟赛】最短路(树形DP,树的直径)
题面 给定一棵 n n n 个结点的无根树,每条边的边权均为 1 1 1 . 树上标记有 m m m 个互不相同的关键点,小 A \tt A A 会在这 m m m 个点中等概率随机地选择 k k k ...
- noi.acNOIP模拟赛5-count
题目链接 戳我 题意简述 你有一个n+1个数的序列,都是1~n,其中只有一个有重复,求每个长度的本质不同的子序列个数.\(mod 1e9+7\). sol 说起来也很简单,设相同的数出现的位置为\(l ...
- 【XJOI】【NOI考前模拟赛7】
DP+卡常数+高精度/ 计算几何+二分+判区间交/ 凸包 首先感谢徐老师的慷慨,让蒟蒻有幸膜拜了学军的神题.祝NOI2015圆满成功 同时膜拜碾压了蒟蒻的众神QAQ 填填填 我的DP比较逗比……( ...
- 2018/3/13 noiρ[rəʊ]模拟赛 125分
T1 60分暴力,水分也不会水,打表也不会打,正解是不可能写正解的,这辈子都写不出来正解的,虽然是zz题但是也拿不到分这样子. 正解:(啥?正解是sb组合数?这都他娘的想不到,真鸡儿丢人我自杀吧.) ...
- Newnode's NOI(P?)模拟赛 第二题 dp决策单调优化
其实直接暴力O(n3)DP+O2O(n^3)DP+O_2O(n3)DP+O2优化能过- CODE O(n3)O(n^3)O(n3) 先来个O(n3)O(n^3)O(n3)暴力DP(开了O2O_2O2 ...
- Newnode's NOI(P?)模拟赛 第三题 (主席树优化建图 + tarjan)
题目/题解戳这里 这道题题目保证a,b,ca,b,ca,b,c各是一个排列-mdzz考场上想到正解但是没看到是排列,相等的情况想了半天-然后写了暴力60分走人- 由于两两间关系一定,那么就是一个竞赛图 ...
- 【NOI P模拟赛】奶油蛋糕塔(状压 DP)
题面 数据范围 1 ≤ n ≤ 5 × 1 0 5 1\leq n\leq5\times10^5 1≤n≤5×105 . 题解 n ≤ 20 n\leq 20 n≤20 的状压应该都会吧,状态记录已经 ...
随机推荐
- PAT(B) 1070 结绳(Java)
题目链接:1070 结绳 (25 point(s)) 题目描述 给定一段一段的绳子,你需要把它们串成一条绳.每次串连的时候,是把两段绳子对折,再如下图所示套接在一起.这样得到的绳子又被当成是另一段绳子 ...
- STM32F030-UART1_DMA使用提示
STM32F030-UART1_DMA使用提示 前言: 今天把STM32F030C8T6的串口DMA学习了一下,为了加快各位研发人员的开发进度,避免浪费大量的时间在硬件平台上,写出个人代码调试的经验. ...
- 登陆并访问k8s的apiserver
kubeadm安装的k8s集群默认需要用户登陆认证,无法直接使用命令curl访问.所以首先的第一步就是获取token. 先找到k8s集群中的dns组件coredns,之前的版本使用的是kube-dns ...
- Unity项目 - 坦克大战3D TankBattle
目录 游戏原型 项目演示 绘图资源 代码实现 技术探讨 参考来源 游戏原型 游戏玩法:在有界的战场上,玩家将驾驶坦克,代表绿色阵营,与你的队友一起击溃红蓝阵营的敌人,在这场三方大战中夺得胜利! 操作指 ...
- 论文笔记 XGBoost: A Scalable Tree Boosting System
XGBoost是boosting算法的其中一种.Boosting算法的思想是将许多弱分类器集成在一起形成一个强分类器,其更关注与降低基模型的偏差.XGBoost是一种提升树模型(Gradient bo ...
- 2.5_Database Interface ODBC数据源及案例
分类 用户数据源 用户创建的数据源,称为“用户数据源”.此时只有创建者才能使用,并且只能在所定义的机器上运行.任何用户都不能使用其他用户创建的用户数据源. 系统数据源 所有用户在Windows下以服务 ...
- .NET子页Main页面实例(UI页面)
<%@ Page Language="C#" MasterPageFile="~/MasterPageDefault.master" AutoEve ...
- .net语音播放,自定义播报文字
// using System.Speech.Synthesis; SpeechSynthesizer synth = new SpeechSynthesizer(); // Configure th ...
- Linux中shell字符串分隔、字符串替换、字符串拼接
1.从properties文件中读取变量 SERVER_NAME=`sed '/project.config/!d;s/.*=//' conf/dubbo.properties | tr -d '\r ...
- C# vb .net实现羽化效果
在.net中,如何简单快捷地实现Photoshop滤镜组中的羽化效果呢?答案是调用SharpImage!专业图像特效滤镜和合成类库.下面开始演示关键代码,您也可以在文末下载全部源码: 设置授权 第一步 ...