codeforces 433C. Ryouko's Memory Note 解题报告
题目链接:http://codeforces.com/problemset/problem/433/C
题目意思:一本书有 n 页,每页的编号依次从 1 到 n 编排。如果从页 x 翻到页 y,那么|x-y|页都需要翻到(联系生活实际就很容易理解的了)。接着有m pieces 的 information,第 i piece 的information 在第 a[i] 页。为了减少翻页的页数,允许把某一页的信息,完全搬到某一页上,这个操作只可以执行一次。问应该把哪一页的信息搬到某一页上,从而使得翻页次数最少。
比赛的时候完全没有思路,赛后想了一段时间也是如此。于是看了tutorial,以下是链接:
http://codeforces.com/blog/entry/12397
实不相瞒,我看这个也有点...看来我的理解能力真心有问题!!!直接看了一个人的AC代码,再加上输出变量终于完全明白了。
如果大家已经看明白链接里的解题思路,那么以下内容可以忽略。
/*****************************************
首先保存每一个数左右两边与该数直接相邻的数都有哪些(注意:如果相邻的数与该数相等,就不需要保存)。假设初始序列是: 1 2 3 4 3 2 。那么如果需要保存数 3 的相邻数,就是2 4 4 2,这里用到vector操作。接着求出这些相邻数与该数之差,每一个数都这样求,得出一个和为sum,注意:这个sum 对整个序列的两两紧挨的数的差是求了两次的!!!所以代码中sum 要除以 2。 紧随着对于某一个数(x),它的相邻数进行从小到大的排序,这样是为了找出中位数。这个中位数表示对于x中的所有相邻数,离这个中位数的距离是最短的。然后再求多一次所有相邻数与这个中位数的差,再求和。每一个数都是这样求,再从这些和里面找出最少的一个就是答案。
****************************************/
别人的思路真是清晰,好佩服他们。以下是我看明白之后写的
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std; #define LL long long
#define pb push_back
const int maxn = 1e5 + ;
vector<int> adj[maxn]; // adj[i]: 数值为i的数中相邻的两个不等于i的数
LL sum, ans; // sum: 每个数相邻数之差的总和; ans: 最后答案
LL adjsum[maxn], adjminsum[maxn]; // adjsum[i]:数值为i的数相邻之间的差 adjminsum[i]: 数值为i的数相邻之间的差的最小值
int a[maxn]; // 原始序列 int main()
{
int n, m;
while (scanf("%d%d", &n, &m) != EOF)
{
for (int i = ; i <= m; i++)
scanf("%d", a+i);
for (int i = ; i <= m; i++)
{
if (i != && a[i-] != a[i])
adj[a[i]].pb(a[i-]);
if (i != m && a[i+] != a[i])
adj[a[i]].pb(a[i+]);
}
sum = ;
memset(adjsum, , sizeof(adjsum));
for (int i = ; i <= n; i++)
{
if (adj[i].size())
{
sort(adj[i].begin(), adj[i].end()); // 排序以便找出中位数
for (int j = ; j < adj[i].size(); j++)
adjsum[i] += (i >= adj[i][j] ? i-adj[i][j] : adj[i][j]-i);
sum += adjsum[i];
}
}
sum /= ; // 由于每个数相邻数之差的总和算了两遍,除以2刚好只算了一次
ans = sum;
memset(adjminsum, , sizeof(adjminsum));
for (int i = ; i <= n; i++)
{
if (adj[i].size())
{
int mid = adj[i].size() / ;
for (int j = ; j < adj[i].size(); j++)
adjminsum[i] += (adj[i][mid] >= adj[i][j] ? adj[i][mid]-adj[i][j] : adj[i][j]-adj[i][mid]);
ans = min(ans, sum-adjsum[i]+adjminsum[i]);
}
}
printf("%lld\n", ans);
for (int i = ; i <= n; i++) // 清空
adj[i].clear();
}
return ;
}
codeforces 433C. Ryouko's Memory Note 解题报告的更多相关文章
- CodeForces 433C Ryouko's Memory Note (中位数定理)
<题目链接> 题目大意:给你一堆数字,允许你修改所有相同的数字成为别的数字,不过只能修改一次,问你修改后序列相邻数字的距离和最小是多少. 解题分析: 首先,修改不是任意的,否则那样情况太多 ...
- CodeForces 433C Ryouko's Memory Note-暴力
Ryouko's Memory Note Time Limit:1000MS Memory Limit:262 ...
- codeforces C. Ryouko's Memory Note
题意:给你m个数,然后你选择一个数替换成别的数,使得.最小.注意选择的那个数在这m个数与它相同的数都必须替换同样的数. 思路:用vector记录每一个数与它相邻的数,如果相同不必记录,然后遍历替换成与 ...
- Codeforces Round #248 (Div. 1) A. Ryouko's Memory Note 水题
A. Ryouko's Memory Note 题目连接: http://www.codeforces.com/contest/434/problem/A Description Ryouko is ...
- codeforces 434A A. Ryouko's Memory Note(数学)
题目链接: A. Ryouko's Memory Note time limit per test 1 second memory limit per test 256 megabytes input ...
- Codeforces Round #248 (Div. 2) C. Ryouko's Memory Note
题目链接:http://codeforces.com/contest/433/problem/C 思路:可以想到,要把某一个数字变成他的相邻中的数字的其中一个,这样总和才会减少,于是我们可以把每个数的 ...
- Codeforces Round #248 (Div. 2) C. Ryouko's Memory Note (vector 替换)
题目链接 题意:给m个数字, 这些数字都不大于 n, sum的值为相邻两个数字 差的绝对值.求这n个数字里把一个数字 用 其中另一个数字代替以后, 最小的sum值. 分析:刚开始以为两个for 最坏 ...
- codeforces B. Dima and Text Messages 解题报告
题目链接:http://codeforces.com/problemset/problem/358/B 题目意思:给出n个单词(假设为word1,word2.word3...wordn)和一句test ...
- codeforces C1. The Great Julya Calendar 解题报告
题目链接:http://codeforces.com/problemset/problem/331/C1 这是第一次参加codeforces比赛(ABBYY Cup 3.0 - Finals (onl ...
随机推荐
- 关于内存 转载自http://blog.csdn.net/xluren/article/details/8150723
首先感谢下原作者,写的真的非常明白,非常详细 1.预备知识—程序的内存分配 一个由C/C++编译的程序占用的内存分为以下几个部分 1.栈区(stack)— 由编译器自动分配释放 ,存放函数的参数值,局 ...
- nyoj_90_整数划分_201403161553
整数划分 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 将正整数n表示成一系列正整数之和:n=n1+n2+…+nk, 其中n1≥n2≥…≥nk≥1,k≥1. 正整数 ...
- Codeforces 848B Rooter's Song(分类+模拟)
题目链接 Rooter's Song 题意 有n个舞者站在x轴上或y轴上,每个人有不同的出发时间.x轴上的舞者垂直x轴正方向移动,y轴上的舞者垂直y轴正方向移动. 当x轴的舞者和y轴的舞者相遇时,他 ...
- for 循环进化史
ECMAScript 6已经逐渐普及,经过二十多年的改进,很多功能也有了更成熟的语句,比如 for 循环 这篇博客将介绍一下从最初的 for 循环,到 ES6 的 for-of 等四种遍历方法 先定义 ...
- android 图片浏览器滑动切换图片
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools= ...
- python内存泄露诊断过程记录pyrasite
工具:pyrasite;包含三个命令行 pyrasite / pyrasite-shell / pyrasite-memory-viewer 安装:gdb meliae urwid 说明:Pyrasi ...
- [React] Persist Form Data in React and Formik with formik-persist
It can be incredibly frustrating to spend a few minutes filling out a form only to accidentally lose ...
- Unity3d插件]EasyTouch简单使用方法
EasyTouch使用 EasyTouch 文件夹[-] 一.效果图 二.操作步骤 1.官方文档上的步骤 2.翻译一下以上的步骤 3.依据官方的这些提示.自己来做一个属于自己的人物遥感控制 对于移动平 ...
- SolidEdge 工程图中如何给零件着色 给装配体着色
点击着色按钮,然后点击更新视图即可.
- HDU oj 开门人与关门人
题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=1234 #include<stdio.h> #include<string.h> ...