当前位置: 首页 > news >正文

AcWing113.特殊排序

题目

N N N 个元素,编号 1 , 2.. N 1,2..N 1,2..N,每一对元素之间的大小关系是确定的,关系具有反对称性,但不具有传递性。

注意:不存在两个元素大小相等的情况。

也就是说,元素的大小关系是 N N N 个点与 N ∗ ( N − 1 ) / 2 N*(N−1)/2 N(N1)/2 条有向边构成的任意有向图。

然而,这是一道交互式试题,这些关系不能一次性得知,你必须通过不超过 10000 次提问来获取信息,每次提问只能了解某两个元素之间的关系。

现在请你把这 N N N 个元素排成一行,使得每个元素都小于右边与它相邻的元素。

你可以通过我们预设的 bool 函数 compare 来获得两个元素之间的大小关系。

例如,编号为 a a a b b b 的两个元素,如果元素 a a a 小于元素 b b b,则 compare(a,b) 返回 true,否则返回 false

N N N 个元素排好序后,把他们的编号以数组的形式输出,如果答案不唯一,则输出任意一个均可。

数据范围

1 ≤ N ≤ 1000 1≤N≤1000 1N1000

输入样例

[[0, 1, 0], [0, 0, 0], [1, 1, 0]]

输出样例

[3, 1, 2]

分析

根据数学归纳法,假设前 k − 1 k-1 k1 个元素已经按要求排成一行,如果能确定第 k k k 个元素应该放在哪一个前面,即可解决此问题。

可以通过这样一种二分法确定第 k k k 个元素的位置:若第 k k k 个元素比第 m i d mid mid 个元素小,令 r = m i d r = mid r=mid,否则令 l = m i d + 1 l = mid + 1 l=mid+1。二分的初始区间可设为 [ 1 , k ] [1,k] [1,k],区间中的 k k k 这个值表示放在所有 k − 1 k-1 k1 个元素之后。

可以证明二分一定可以找到一个合法的位置插入,证明如下。

  1. 如果第 k k k 个元素比第 m i d mid mid 个元素小。
    继续比较 k k k m i d − 1 mid-1 mid1 这两个元素,若第 k k k 个元素比第 m i d − 1 mid - 1 mid1 个元素大,则 k k k 可以插在 m i d − 1 mid - 1 mid1 m i d mid mid 之间;否则说明元素 k k k 比元素 m i d − 1 mid - 1 mid1 小,那就再比较 k k k m i d − 2 mid - 2 mid2 这两个元素,依此类推,直到发现第 k k k 个元素比第 1 个元素小,那就放在最前边。
  2. 如果第 k k k 个元素比第 m i d mid mid 个元素大,同理。

以上只是一个证明,当然不会真的去依次比较 k k k 与每个元素。实际上通过二分,每询问一次 ( k k k m i d mid mid),就可以把区间 [ l , r ] [l,r] [l,r] 缩小一半,因此至多 l o g k logk logk 次询问就可以确定 k k k 应该放在哪里。把 N N N 个元素排成一行的总询问次数不超过 N l o g N NlogN NlogN

代码

// Forward declaration of compare API.
// bool compare(int a, int b);
// return bool means whether a is less than b.class Solution {
private:vector<int> arr;//第一个比arr[k]大的数的位置int binarySearch(int k) {int left = 0;int right = k;while (left < right) {int mid = (left + right) >> 1;if (compare(arr[k], arr[mid])) { //arr[k] < arr[mid]right = mid;} else {left = mid + 1;}}return left;}
public:vector<int> specialSort(int N) {for (int i = 0; i < N; i++) {arr.emplace_back(i + 1);}for (int k = 0; k < N; k++) {int cur = arr[k];int pos = binarySearch(k); //arr[k]应该放到pos位置for (int j = k - 1; j >= pos; j--) { //pos到k-1位置的数全部向后移动一个位置arr[j + 1] = arr[j];}arr[pos] = cur;}return arr;}
};
http://www.lryc.cn/news/240405.html

相关文章:

  • 数据仓库岗面试
  • 企业建数仓的第一步是选择一个好用的ETL工具
  • 行情分析 - - 加密货币市场大盘走势(11.23)
  • 穿山甲SDK 集成·android接入广告·app流量变现
  • 深度学习模型训练计算量的估算
  • 【实验笔记】C语言实验——降价提醒机器人
  • YOLOv5分割训练,从数据集标注到训练一条龙解决
  • 再添千万级罚单,某银行年内罚款过亿!金融行业合规问题亟待解决
  • 配置Nginx服务器用于Web应用代理和SSL{仅配置文件}
  • 【广州华锐互动】VR溺水预防教育:在虚拟世界中学会自救!
  • Si(111)衬底上脉冲激光沉积AlN外延薄膜的界面反应控制及其机理
  • 基于Cortex®-M4F的TM4C123GH6NMRT7R 32位MCU,LM74900QRGERQ1、LM74930QRGERQ1汽车类理想二极管
  • 苹果企业签名失败常见的问题
  • Jtti:Android alertdialog嵌套出错怎么解决
  • 解锁word密码,忘记密码怎么办?
  • 同为科技(TOWE)桌面PDU插排:一款可以DIY定制的“超级插座”
  • 使用Java Servlet生成动态二维码
  • 【已解决】微信小程序腾讯地图的map清除markers,setData将marker置空后,安卓和ios还会显示上次的内容的问题所在以及解决办法
  • 弄懂Rust编程中的Trait
  • 关于登山扣亚马逊\SHEIN出口合规标准ASTM F1774 指南解析
  • 浅析ChatGPT中涉及到的几种技术点
  • Web实战:基于Django与Bootstrap的在线计算器
  • 曲率半径的推导
  • 0时区格林威治时间转换手机当地时间-Android
  • git-3
  • 【python爬虫】scrapy在pycharm 调试
  • yoloV5模型中,x,s,n,m,l之间区别
  • RabbitMQ快速入门(简单收发消息)
  • java面试-zookeeper
  • VBA技术资料MF85:将工作簿批量另存为PDF文件