1、咒语

(curse.pas/c/cpp)

【题目描述】

亮亮梦到自己来到了魔法城堡,但一扇巨大的石门阻拦了他通向城堡内的路。正当他沮丧之际,突然发现门上有一处机关,机关上有一张很长的纸条。
亮亮拿起纸条的一端,只见上面写着打开机关的方法:“打开机关需要念动符咒,咒语是一串长为 L 的由 0 和 1 组成的字符串。在这张长纸条上列了 n 个长为 L 的字符串,正确的咒语即是在纷繁的 2^L 种字符串中,与这些纸条上的字符串相异度之和最小,并且在满足这一条件下, 0 的个数最多的字符串。两个字符串的相异度定义为对应位置不相等的字符对的个数。如‘011’和‘001’的相异度为 1,因为它们有且只有第二个位置上的字符不相等。”
亮亮拉起纸条,只觉得纸条似乎永远也拉不完。这上面有着数以万计的字符串,而每一个字符串的长度也或百或千,以人力看来是无法得到正确的咒语。你能帮帮他,让他得以进入魔法城堡,一窥其中的奥秘吗?

【输入格式】

第一行为一个数字 N 。

接下来的 N 行,每行为一个长为 L 的 01 字符串。数据保证 N 个字符串等长。

【输出格式】

只有一行,是一个长为 L 的字符串 S,即为正确的咒语。

【样例输入】


【样例输出】

01001

【数据规模】

对于 20%的数据, N<=5;

对于 60%的数据, N<=100;

对于 100%的数据, 1<=N<=1000, 1<=L<=1000。

题解:

按位模拟每一位对应的数字,p[i][0]:第i位'0'的个数, p[i][1]:第i位'1'的个数。
p[i][0] > p[i][1]时,为保证'字符串相异度之和最小',第i位为'0';
p[i][0] < p[i][1]时,为保证'字符串相异度之和最小',第i位为'1';
p[i][0] = p[i][1]时,为保证'0 的个数最多',第i位为'0';

#include <cstdio>
#include <cstring>
using namespace std;
char ch[];
int n, len, p[][];
int main() {
scanf("%d", &n);
for(int i = ;i <= n;i ++) {
scanf("%s", ch + );
if(len == ) len = strlen(ch + );
for(int j = ;j <= len;j ++) p[j][(ch[j] - '') & ] ++;
}
for(int i = ;i <= len;i ++)
printf("%c", p[i][] >= p[i][] ? '' : '');
return ;
}

2、神光

(light.pas/c/cpp)

【题目描述】

亮亮成功地念出了咒语,石门缓缓地自动移开,一道道绚丽的神光从城堡内激射而出。亮亮好奇而又兴奋地走入了城堡中,迎面有一座极长的魔法阵。

魔法阵可以看作一条直线,它被均匀地分成了 1 000 000 000 个位置,一个位置可以看成是一个格子。有些位置上筑有法坛,一共 N 座。亮亮只有破了眼前的魔法阵,才能继续前进,而欲破法阵,必须毁掉所有的法坛。

亮亮身前有两根法杖:一根颜色血红,能发红色神光,光芒可以笼罩连续 L个位置,并摧毁这 L 个位置上所有的法坛,最多使用 R 次;另一根颜色碧绿,能发绿色神光,光芒可以笼罩连续 2L 个位置,并摧毁这 2L 个位置上所有的法坛,最多使用 G 次。

法杖的神奇之处在于, L 的值必须由亮亮事先设定好,并且一经设定,便无法更改。亮亮需要在规定的次数下摧毁所有法坛,并且使得 L 最小。

【输入格式】

第一行三个整数 N, R, G。

第 i (2<=i<=n+1) 行一个整数 Ai ,表示第 i 座法坛的位置。

【输出格式】

只有一个整数,表示 L 的最小值。

【样例输入】


【样例输出】

4 

【样例解释】

亮亮将 L 设为 4,并用红色神光笼罩 21-24 位置,用绿色神光笼罩 1-8 位置。

【数据规模】

对于 50%的数据, N <= 100;

对于 100%的数据, 1 <= N <= 2000, 1 <= R, G, Ai <= 1,000,000,000。

题解:

首先我们注意到,当 R 和 G 的大小超过了 N 时,L 的最小值就是 1,
因此,我们只 需要考虑 R 和 G 小于 N 的情况,于是 R,G 的规模就降到了 2000 以内。 显然要采用二分答案的方法。
那么问题转化为,判断给定的 L 能否摧毁所有法坛,我 们采用动态规划方法。首先将法坛的位置按照从小到大进行排序。
令 dp[i][j]表示,在用了 i 次红光,j 次绿光的情况下,最多从第一座法坛开始,一直摧毁到第几座法坛。
那么状态转 移方程即为 dp[i][j] = max ( P[dp[i-1][j] + 1], Q[dp[i][j-1] + 1] )。
其中 P[k]表示使用一次红光, 能从第 k 座法坛向右(正向为右)连续摧毁到第几座,Q[k]表示使用一次绿光,能从第 k 座 法坛向右连续摧毁到第几座。
P 和 Q 数组可以通过预处理得到。最终,我们只要判断 dp[R][G] 的值是否为 N 即可

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 2e3 + ;
int read() {
int x = , f = ; char ch = getchar();
while(ch < ''||ch > '') f = (ch=='-')?-:, ch = getchar();
while(ch >= ''&&ch <= '') x=(x<<)+(x<<)+ch-'', ch = getchar();
return x * f;
}
int n, r, g, a[N], f[N][N], p[N], q[N];
bool check(int x) {
memset(f, , sizeof(f));
memset(p, , sizeof(p));
memset(q, , sizeof(q));
for(int i = ;i <= n;i ++) {
for(int j = i;j <= n;j ++) {
if(a[j] - a[i] + <= x) p[i] = j;
if(a[j] - a[i] + <= x * ) q[i] = j;
}
}
p[n+] = q[n+] = n;
for(int i = ;i <= r;i ++) {
for(int j = ;j <= g;j ++) {
if(i > ) f[i][j] = max(f[i][j], p[f[i-][j]+]);
if(j > ) f[i][j] = max(f[i][j], q[f[i][j-]+]);
}
}
return f[r][g] == n;
}
int main() {
n = read(); r = read(); g = read();
if(n <= r + g) { printf("1\n"); return ; }
for(int i = ;i <= n;i ++) a[i] = read();
sort(a + , a + n + );
int l = , r = 1e9 + ;
while(l < r) {
int mid = (l + r) >> ;
if(check(mid)) r = mid;
else l = mid + ;
}
printf("%d\n", l);
return ;
}

3、迷宫

(maze.pas/c/cpp)

【题目描述】

破了魔法阵后,亮亮进入了一座迷宫。这座迷宫叫做“梦境迷宫”,亮亮只有走出这座迷宫,才能从睡梦中醒来。

梦境迷宫可以用无向图来表示。它共有 n 个点和 m 条双向道路,每条道路都有边权,表示通过这条道路所需的时间,且每条道路可以多次经过。亮亮位于一号点,而出口则是 n 号点。原本,亮亮该找到一条最短路,快速冲出迷宫,然而,梦境迷宫的特殊之处在于,如果沿着最短路到达出口,亮亮就会永远陷入梦境。因此,亮亮必须寻找一条次短路。次短路的长度须严格大于最短路(可以有多条)的长度,同时又不大于所有除最短路外的道路的长度。

你的任务,就是编写一个程序,帮助亮亮找到通向出口的次短路。

【输入格式】

第一行有两个整数 n、 m,表示迷宫内共有 n 个点, m 条边。

接下来 m 行,每行三个整数 x、 y、 z,表示结点 x 和 y 之间连有一条边权为z 的无向边。

【输出格式】

一个整数,表示次短路的长度。

【样例输入】


【样例输出】

9

【样例解释】

最短路: 1 -> 2 -> 4 (长度为 2+4=6)

次短路: 1 -> 2 -> 3 -> 4 (长度为 2+3+4=9)

【数据规模】

对于 100%的数据, 1 <= n <= 5000, 1 <= m <= 100,000。

对于 100%的数据, 1 <= z <= 5000, z 表示无向边的边长。

题解:

次短路模板题:
分别跑两遍 spfa(),
dis[i]: 点i到 1 号点的最短距离;
dis_[i]: 点i到 n 号点的最短距离;
之后, 枚举每条边(共2 * n条边),若x, y分别为这一条边的两个端点,
则dis[i]+dis_[y]+e[i].val即为经过此边的最短路,
在这些边中较 最短路 大的取最小值即可。

#include <cstdio>
#include <queue>
#include <cstring>
using namespace std;
const int N = 5e3 + , M = 1e6 + ;
int read() {
int x = , f = ; char ch = getchar();
while(ch < ''||ch > '') f = (ch=='-')?-:, ch = getchar();
while(ch >= ''&&ch <= '') x=(x<<)+(x<<)+ch-'', ch = getchar();
return x * f;
}
int n, m, ans = 2e9 + ;
struct edge { int from, nxt, to, val; } e[M];
int cnt, head[N], dis[N], b[N], dis_[N];
void add(int from, int to, int val) {
e[++ cnt].to = to;
e[cnt].from = from;
e[cnt].val = val;
e[cnt].nxt = head[from];
head[from] = cnt;
}
void spfa() {
queue <int> q; q.push(); b[] = ;
memset(dis, 0x3f, sizeof(dis)); dis[] = ;
while(!q.empty()) {
int tp = q.front(); q.pop(); b[tp] = ;
for(int i = head[tp]; i ;i = e[i].nxt) {
if(dis[e[i].to] > dis[tp] + e[i].val) {
dis[e[i].to] = dis[tp] + e[i].val;
if(! b[e[i].to]) q.push(e[i].to), b[e[i].to] = ;
}
}
}
}
void spfa_() {
memset(b, , sizeof(b));
queue <int> q; q.push(n); b[n] = ;
memset(dis_, 0x3f, sizeof(dis_)); dis_[n] = ;
while(!q.empty()) {
int tp = q.front(); q.pop(); b[tp] = ;
for(int i = head[tp]; i ;i = e[i].nxt) {
if(dis_[e[i].to] > dis_[tp] + e[i].val) {
dis_[e[i].to] = dis_[tp] + e[i].val;
if(! b[e[i].to]) q.push(e[i].to), b[e[i].to] = ;
}
}
}
}
int main() {
n = read(); m = read();
for(int i = , x, y, z;i <= m;i ++)
x = read(), y = read(), z = read(), add(x, y, z), add(y, x, z);
spfa(); spfa_();
for(int i = ;i <= cnt;i ++) {
int len = dis[e[i].from] + dis_[e[i].to] + e[i].val;
if(len > dis[n]) ans = min(ans, len);
}
printf("%d\n", ans);
return ;
}

2019.6.21 NOIP2018提高组模拟题(二)的更多相关文章

  1. NOIP2018提高组模拟题(五)

    字符串(string) Description 小林与亮亮正在做一个游戏.小林随意地写出一个字符串,字符串只由大写 字母组成,然后指定一个非负整数 m,亮亮可以进行至多 m 次操作,每次操作 为交换相 ...

  2. NOIP2018提高组模拟题(四)

    能量(energy) Description ​ 有一块能量田,它的形状是 n*m的矩形,每一个格子上都有一个能量值 a[x][y] (可正可负).一块矩形田的能量定义为它的每个格子的能量值之和. ​ ...

  3. NOIP2018提高组模拟题(六)

    购物(shop) Description 小林来到商店中进行购物.商店里一共有 n 件物品,第 i 件物品的价格为 a[i] 元.小林总共需要购买 m 件物品,他希望他所花费的钱最少,请你计算出最小 ...

  4. 10.18 NOIP2018提高组模拟题(二)

    大水题 1.咒语 (curse.pas/c/cpp) [题目描述] 亮亮梦到自己来到了魔法城堡,但一扇巨大的石门阻拦了他通向城堡内的路.正当他沮丧之际,突然发现门上有一处机关,机关上有一张很长的纸条. ...

  5. NOIP2018提高组模拟题(二)

    咒语(curse) Description 亮亮梦到自己来到了魔法城堡,但一扇巨大的石门阻拦了他通向城堡内的路. 正当他沮丧之际,突然发现门上有一处机关,机关上有一张很长的纸条. 亮亮拿起纸条的一端, ...

  6. 11.5NOIP2018提高组模拟题

    书信(letter) Description 有 n 个小朋友, 编号为 1 到 n, 他们每人写了一封信, 放到了一个信箱里, 接下来每个人从中抽取一封书信. 显然, 这样一共有 n!种拿到书信的情 ...

  7. [JZOJ 5852] [NOIP2018提高组模拟9.6] 相交 解题报告 (倍增+LCA)

    题目链接: http://172.16.0.132/senior/#main/show/5852 题目: 题目大意: 多组询问,每次询问树上两条链是否相交 题解: 两条链相交并且仅当某一条链的两个端点 ...

  8. [JZOJ 5875] [NOIP2018提高组模拟9.20] 听我说,海蜗牛 解题报告(BFS+二分)

    题目链接: http://172.16.0.132/senior/#main/show/5875 题目: 题解: 注意这题只能经过开放的港口 我们考虑用vector存下每个点不能到的点,并把并让vec ...

  9. XJOI——NOIP2015提高组模拟题19-day1——观光旅行

    http://www.hzxjhs.com:83/contest/493/problem/3 [题目大意] 给定一个有n(n<=500000)个点,m(1<=500000)条边的无向图.给 ...

随机推荐

  1. UwpDesktop!WPF也能开发Surface Dial

    原文:UwpDesktop!WPF也能开发Surface Dial 前段时间巨硬发布了一款新的输入设备Surface Dial,配合Surface Studio使用简直炫酷到没朋友. 本人由于公司业务 ...

  2. C#获取字符串宽度像素

    通过Graphics对象的MeasureString方法可以获取字符串的大小,如下: Graphics graphics = CreateGraphics(); SizeF sizeF = graph ...

  3. 使用Boost的DLL库管理动态链接库(类似于Qt中的QLibrary)

    Boost 1.61新增了一个DLL库,跟Qt中的QLibrary类似,提供了跨平台的动态库链接库加载.调用等功能.http://www.boost.org/users/history/version ...

  4. Auto updater for my side loaded UWP apps

    原文: Auto updater for my side loaded UWP apps As I described before, i have a few tasks to solve for ...

  5. UBUNTU 16.04 + CUDA8.0 + CUDNN6.0 + OPENCV3.2 + MKL +CAFFE + tensorflow

    首先说一下自己机子的配置 CPU:Intel(R) Core(TM) i5-5600 CUP @3.20GHz *4 GPU : GTX 1060 OS : 64bit Ubuntu16.04LTS ...

  6. Raknet是一个基于UDP网络传输协议的C++网络库(还有一些其它库,比如nanomsg,fastsocket等等)

    Raknet是一个基于UDP网络传输协议的C++网络库,允许程序员在他们自己的程序中实现高效的网络传输服务.通常情况下用于游戏,但也可以用于其它项目. Raknet有以下好处: 高性能 在同一台计算机 ...

  7. Q_DECLARE_METATYPE(继承QObject的类都已经自动注册),注册后的类型可以作为QVariant的自定义类型

    简介 这个宏用来注册一个类(含默认构造.默认析构.拷贝构造函数)为QMetaType类型 ,注册后的类型可以作为QVariant的自定义类型. 这个宏应该放在类或者结构体外面的下面,也可以放在一个非公 ...

  8. QT5的post Event解析

    大家都知道,QT的事件机制,查了好多网上的帖子,分析的不够到位,今天给大家分享下,我的分析,请高手指正:都知道post Event通过    QScopedPointer<QEvent> ...

  9. FMX+Win32,窗口无法保持原样,应该是个bug

    从FMX发布开始,一直有这问题,大家看看是不是一个bug,应该如何修复? 新建一个FMX Application,运行后,点击窗口标题栏右上角的“最大化”按钮,此时窗口是最大化的.在windows最底 ...

  10. Spring之ApplicationContext

    (1)ApplicationContext接口容器 ApplicationContext用于加载Spring的配置文件,在程序中充当“容器”的角色.其实现类有两个.通过Ctrl +T查看: A.配置文 ...