- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我们有一个未排序的 N 数序列(1, 2, 3, 4, ... N)。我们可以通过按特定顺序交换相邻元素来对整个序列进行排序。给定一个序列,我如何计算对序列进行排序所需的最小可能交换。
例如,考虑序列 {4, 2, 5, 3, 1}。
最好的排序方法是按以下顺序使用 7 个交换
贪心算法并没有被证明是富有成果的。一个反例很容易构造。接近解决方案的下一个明显选择是动态规划。
假设我们有一个未排序的序列:{A1, A2, ...Ai, A(i+1), ..., An}。我们知道对序列 {Ai, A(i+1), ..., An} 进行排序所需的最小交换次数是 Min[Ai, A(i+1), ..., An}。问题是找到 Min[A(i-1), Ai, ..., An]。
好吧,我脑海中浮现的第一个想法就是添加将 A(i-1) 放入已排序序列 {Ai, ..., An} 中正确位置所需的步数。这行得通:问题中给出的示例已使用完全相同的方法解决。
但我无法证明这个解决方案的有效性。我经常是这种情况。当我认为我已经解决了问题时,我能做的最好的事情就是获得一个“直观”的证明。我在读高中,没有接受过算法方面的正规培训。我这样做纯粹是出于兴趣。
是否有严密的数学符号可以将这个问题转化为形式化证明?这个符号可以扩展到其他问题吗?如何?如果它能以高中生可以理解的形式呈现,我将不胜感激。
最佳答案
这是一个经典的算法问题。最小交换次数等于数组中的反转次数。如果我们有索引 i
和索引j
这样 ai> aj 和 i
< j
那么这就是所谓的反转。让我们来证明这个说法!途中我需要一些引理:
引理 1:如果两个相邻元素没有反转,则数组被排序。
证明: 假设没有两个相邻元素形成反转。这意味着对于区间 [0, n-1] 中的所有 i,ai <= ai+1。作为<=
是可传递的,这意味着数组已排序。
引理 2:两个相邻元素的单次交换将使数组中的反转总数最多减少 1。
证明:当我们交换两个相邻元素 ai 和 ai+1 时,它们相对于数组中所有其他元素的相对位置将保持不变。那就是对于所有在 ai+1 之后的元素,它们仍然在 ai+1 之后,对于 ai 之前的所有元素, 它们仍然在 ai 之前。这也意味着,如果 ai 或 ai+1 与元素 aj 形成反演,那么它们仍将与元素形成反演交换后的它。因此,如果我们交换 ai 和 ai+1,我们只会影响这两个元素曾经形成的反转。由于两个元素最多只能参与一次反演,我们也证明了引理。
引理 3:我们需要至少对相邻元素执行 NI 次交换才能对数组进行排序,其中 NI 是数组中的反转次数
证明: 在有序数组中没有反转。同样根据引理 2,单次交换最多可以将反转次数减少一次。因此,我们至少需要执行与反转次数一样多的交换。
引理 4:我们总是可以对数组进行 NI 交换相邻元素的排序,其中 NI 是数组中的反转次数。
证明: 如果我们假设在我们的数组中没有两个相邻元素的反转,那么根据引理 1,数组将被排序,我们就完成了。
否则至少有一对相邻元素形成反转。我们可以交换它们,从而将反转的总数减少一次。我们可以继续执行此操作恰好 NI 次。
现在我已经从答案的开头证明了我的说法。
剩下的唯一问题是如何计算给定数组中的反转次数。您可以使用合并排序的轻微修改来做到这一点,您可以在合并阶段累积反转。你可以看看this answer有关如何实现它的详细信息。算法的整体复杂度为O(n*log(n))
.
关于algorithm - 通过使用最小交换交换相邻元素对序列进行排序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20990127/
我想在数组中找到连接(相邻)的元素。 例如,在数组中: [1,2,3,4,5] 要访问所有 2 个连通元素,输出将为: 1,2 2,3 3,4 4,5 要访问所有 3 个连通元素,输出将为 1,2,3
我有三个 Sprite ,彼此堆叠在一起。 我修改了他们的 transform.matrix 以给出他们一致增长的外观。 但是,根据比例因子,瓷砖之间有时会出现小裂缝。 cracks between
我正在阅读有关 Margin Collapsing 的文章,我遇到了这个:margin Adjacent siblings The margins of adjacent siblings are c
float div 的框阴影被其右侧的邻居截断,但左侧未截断。 我玩过 z-index 和 overflow: visible,但没有用。 HTML: CSS: .doc-page {
我有多个元素说卡片。这些卡片需要水平堆叠并且高度需要相同。这正在发生在我身上。 每张卡片都有图像、文本和按钮。每张卡片的图像和文本应采用任何卡片中的最大高度,以便它们正确对齐。这不会发生在我身上。 如
我有这个 GUI 我使用了 GridBagLayout,但我不知道为什么 Plain Bread 复选框与其相应的标签之间有很大的间距。 而且,我尝试仅增加按钮沿 x 轴的间距,但尽管重置了插图,但沿
在过去,我已经为自定义元素符号使用了数百次列表项背景图像,但不知何故从未遇到过这个问题。 基本上,我有一个 IMG float 在无序列表的左侧。元素符号背景图像设置在每个 LI 的左上角。但是, f
我正在使用 Bootstrap 框架并使用 2 列网格。 html 内容有标题、链接、副标题和文本。这增加了该列的高度。我希望它旁边的列与其高度匹配(以便图像显示)没有设置高度图像不显
我有一个 php 代码可以生成数百个 和 标签。我的问题如下,我有以下内容: X X 我想要第二个的边框颜色更重要,以便共享边框显示为灰色而不是黑色。我可以在第二个 td 中使用重要性或继承标签吗?
Place holder for Radio1 Place holder for Radio2 在此,我只想要 与相应的单选按钮相关联是可见的,但是...... * { visibilit
我正在尝试在 html 中实现以下布局。更大的 div 1。然后是它旁边的另一个 div,顶部有一个边距。如果我给 float: left 给第一个 div,给第二个 div margin-top 也
假设我有 2 个名为 IN 和 MASK 的二进制输入。实际字段大小可能是 32 到 256 位,具体取决于用于完成任务的指令集。两个输入都会改变每次调用。 Inputs: IN = ...110
我想知道是否有一种简洁/一行的方法来执行以下操作: pack :: [a] -> [(a, a)] pack [] = [] pack [_] = [] pack (x:y:xs
下面的代码分为两部分,一部分处理头部的管理,另一部分处理“主体”,当我执行代码时引发下面的异常,我该如何解决该错误?我不知道下面的错误是什么原因造成的,错误是在react的解析上 错误: Line
http://imgur.com/a/DA5i4 在上面的两张图片中你可以看到我有一个主容器,里面装满了 3 个较小的 div,一个大的在中间,两个瘦的在两边,但是右边直到中间的 div 下面才开始。
正如我在标题中解释的所有内容,然后我将只为你们提供我现在拥有的代码,我一直在努力实现我想要的东西很长时间但没有运气......它表现得像响应式的,但即使调整大小我也想将其保持在原位...截图
我编写了一个 jquery 插件,它使用 Flot 将 HTML 表格转换为图表。 HTML 是从 XSLT 生成的,在 XSLT 中我有以下代码来调用我的插件。此代码尝试在 blah1 和 blah
我正在尝试实现这样的布局。 aaa xxxxxx oooo aaa xxxxxx oooo xxxxxx xxxxxx bbb xxxxxx cccc bbb xxxxxx cc
在包含网格的 2 个 div 上使用内联 css 显示不起作用 最佳答案 为您的 div 指定宽度并根据您的要求使用“float: left”或“right”。不要对 div 使用“内联” 例如 CS
我将 MVC 项目中的一些代码返回到网页。我无法用撇号解决问题,当我的电话看起来像这样时,我如何忽略它 document.getElementById('some').insertAdjacentHT
我是一名优秀的程序员,十分优秀!