POJ 3761 Bubble Sort 快速冪取模+組合數學

来源:http://www.cnblogs.com/pach/archive/2016/10/19/5978468.html
-Advertisement-
Play Games

轉載於:http://www.cnblogs.com/767355675hutaishi/p/3873770.html 題目大意:眾所周知冒泡排序演算法多數情況下不能只掃描一遍就結束排序,而是要掃描好幾遍。現在你的任務是求1~N的排列中,需要掃描K遍才能排好序的數列的個數模20100713。註意,不同 ...


轉載於:http://www.cnblogs.com/767355675hutaishi/p/3873770.html

題目大意:眾所周知冒泡排序演算法多數情況下不能只掃描一遍就結束排序,而是要掃描好幾遍。現在你的任務是求1~N的排列中,需要掃描K遍才能排好序的數列的個數模20100713。註意,不同於真正的冒泡排序演算法,只要數列有序就立刻停止,而不用再檢驗一遍。

估計多數人都是找規律吧,先看出遞推,然後求出通項……這個題只有找出通項公式才能通過,所以首先公佈答案:

K!((K + 1) ^ (N - K) - K ^ (N - K))

好吧,現在讓我們來證明一下。

首先定義函數d(x),對於1~N的一個排列,d(x)表示第x個數前面有多少個數字大於該數。

比如說對於3 2 4 1 5,有d(1) = 0,d(2) = 1,d(3) = 0,d(4) = 3,d(5) = 0。

現在我們來證明d(x)函數的兩條性質:

(一)對於一個排列,對於所有x <= N,有d(x) = 0是這個排列是有序的充要條件。

如果存在1 <= i, j <= N,使得i < j,ai > aj,那麼由於aj前面有ai大於它,故d(j) >= 1。而這與d(j) = 0矛盾,反之亦然。所以說命題成立。

(二)冒泡排序的每次掃描的結果是,對於非零的d(x)值,這個位置的d(x)會且只會減少1。

考慮某個非零的d(x)值。由於d(x) >= 1,所以必然存在整數i∈[1, x - 1],滿足ai > ax。設m為a1, a2, ..., a(x -1)中的最大值,位置為i,則必有m > ax。那麼,在掃描到a(x - 1)和ax的時候,由於前面的交換,必然有a(x - 1) = m。

原因是如果前面的交換將m交換到了a(x - 2)的位置,那麼由於m > a(x - 1),那麼ai必然能夠被交換到a(x - 1)的位置。故由數學歸納法,只要m能夠被交換到a(i + 1)即可。而在交換之前a(i - 1) < m,m與a(i - 1)不發生交換;同時必然有m > a(i + 1),m一定會被交換到a(i + 1),故該結論成立,從而掃描a(x - 1)和ax的時候a(x - 1) = m。

這時由於m > ax,m與ax之間要交換,交換的效果由於ax前面比ax小的數字減少了一個,d(x)減小了1。所以說d(x)在這個過程中必會減少。

而另一方面,完成了m與ax的這次交換之後,這一次掃描顯然就不會再交換ax的值了(這時ax位置上的值已經是m了)。所以說,d(x)也只能減少1。這就證明完畢。

證明瞭d(x)函數的這個性質之後,我們就可以得出對於1~n的一個排列,它所需要的冒泡排序的掃描次數為

K = max (d(i), 1 <= i <= N)

而這個結論很顯然,因為只有經過K次掃描,所有位置的d值才能都變為0。

到此,我們成功地將冒泡排序的次數問題轉化為d(x)值滿足條件的數列的問題。原問題也就轉化成了有多少個排列使得其中最大的d(x)值恰好為K。然而這也是複雜的,所以說我們不妨先解決有多少個排列使得其中最大的d(x)值不大於K。

首先可以確定N >= K + 1,否則不可能出現某個位置前面有K個數大於它。

然後決定原數列中1的位置。顯而易見,如果最小數的位置為x,則其d(x) = x - 1。而d(x) <= K,故x <= K + 1,也就是說1有K + 1種放置方法;而放置2的時候,我們完全可以考慮一個新的排列2~N,這時2有K + 1種放置方法,然後再把1插到位置1~K + 1,而不影響其它數的d值。所以說,前N - K個數的放置方法的種類有

(K + 1) ^ (N - K)

之後只需要考慮N - K + 1 ~ N的排列即可。然而,由於整個數列只有K個數字,不可能出現某個d值大於K + 1。所以說排列方法有K!種。故,所有位置d值不大於K的排列的方案數有

K!((K + 1) ^ (N - K))

但是這是不大於K的排列數量,恰好為K的有怎麼辦呢?很簡單,只需要減去不大於K - 1的排列數量便可。所以最後的答案為

K!((K + 1) ^ (N - K)) - (K - 1)!(K ^ (N - K + 1))

化簡之後我們就得到

K!((K + 1) ^ (N - K) - K ^ (N - K))

這就是原來的式子,它的正確性就證明完畢。

#include <iostream>
#include <cstdio>
using namespace std;

const int Mod=20100713;
__int64 fac[1000005]= {1};

__int64 Power(__int64 a,__int64 k)
{
    __int64 ret=1;
    for(; k; k>>=1)
    {
        if(k&1) ret=ret*a%Mod;
        a=a*a%Mod;
    }
    return ret;
}
int main()
{
    for(int i=1; i<1000001; i++)
        fac[i]=fac[i-1]*i%Mod;
    int t;
    scanf("%d",&t);
    while(t--)
    {
        int n,k;
        scanf("%d%d",&n,&k);
        printf("%I64d\n",(Power(k+1,n-k)-Power(k,n-k)+Mod)%Mod*fac[k]%Mod); //Power(k+1,n-k)-Power(k,n-k)可能小於零
} return 0; }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在現實生活中,記錄日誌非常重要。銀行轉賬時會有轉賬記錄;飛機飛行過程中,會有黑盒子(飛行數據記錄器)記錄飛行過程中的一切。如果有出現什麼問題,人們可以通過日誌數據來搞清楚到底發生了什麼。對於系統開發、調試以及運行,記錄日誌都是同樣的重要。如果沒有日誌記錄,程式崩潰時你幾乎就沒辦法弄明白到底發生了什麼 ...
  • 內置對象:不需要預先聲明就可以在腳本代碼和表達式中隨意使用,有以下特點 1.由JSP規範提供,不用編寫者實例化 2.提供Web容器實現和管理 3.所有JSP頁面均可用 4.只有在腳本元素的表達式或者代碼中才可使用(<%=使用內置對象%>或<%使用內置對象%>) 輸入輸出對象:request,resp ...
  • JavaWeb_day01 HTTP協議 HTTP(HyperText Transfer Protocol)超文本傳輸協議,是TCP/IP的應用層協議,用於定義WEB瀏覽器與WEB伺服器之間交換數據的過程以及數據本身的格式. Http協議版本號 : HTTP/1.0 HTTP/1.1 交互步驟 : ...
  • JList想添加元素,可以執行將所有元素在JList初始化時加入的靜態操作,也可以利用“列表模型”DefaultListModel處理所有列表修改細節的動態操作。 ...
  • Java Labmda表達式的一個重要用法是簡化某些匿名內部類(Anonymous Classes)的寫法。實際上Lambda表達式並不僅僅是匿名內部類的語法糖,JVM內部是通過invokedynamic指令來實現Lambda表達式的。本篇我們首先感受一下使用Lambda表達式帶來的便利之處。 ...
  • 筆者在一家互聯網公司做JavaEE開發,公司開發了移動端的產品,唯獨沒有PC端的產品,於是領導將任務分配給筆者。 使用Java開發PC客戶端,我的第一反應是使用swing API。但是,產品的需求是客戶端內嵌一個瀏覽器引擎,能夠渲染網頁內容。於是,筆者通過百度無意間發現和瞭解到JavaFX。 經過編 ...
  • 一、 hibernate是什麼 (一)hibernate 是一個orm框架,orm (object relation mapping) 對象關係映射框架 o object -> 業務層(只對對象操作) r relation-> 關係資料庫 m mapping 對象關係映射文件 Hibernate有六 ...
  • 一、插入排序 1 #-*- coding:utf-8 -*- 2 ''' 3 描述 4 插入排序的基本操作就是將一個數據插入到已經排好序的有序數據中,從而得到一個新的、個數加一的有序數據,演算法適用於少量數據的排序,時間複雜度為O(n^2)。 5 是穩定的排序方法。插入演算法把要排序的數組分成兩部分:第 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...