2020-03-15 SDU week3程序設(shè)計(jì)

A - DDL 的恐懼

現(xiàn)有n個(gè)作業(yè)臭增,每個(gè)作業(yè)都有DDL拷淘,沒在DDL之前做完作業(yè)的話策吠,就沒扣掉相應(yīng)的分逛裤。

輸入

第一行輸入指的是準(zhǔn)備要輸入程序的有幾組數(shù)據(jù),接下來每組數(shù)組有兩行猴抹,第一行是每個(gè)作業(yè)的DDL带族,第二行是每個(gè)作業(yè)的分值

3
3
3 3 3
10 5 1
3
1 3 1
6 2 3
7
1 4 6 4 2 4 3
3 2 1 7 6 5 4

輸出

0
3
5

思路

考慮貪心算法。
每個(gè)作業(yè)都有分值蟀给,所以當(dāng)然要考慮把分值大的先拿下蝙砌。所以把數(shù)據(jù)給存進(jìn)結(jié)構(gòu)體了以后阳堕,就先按找分值降序,日期降序保存择克,這樣就保證先插分值高的數(shù)據(jù)恬总,分值相同的數(shù)據(jù)就先插日期比較靠后的,這樣就會對其他數(shù)據(jù)的插入產(chǎn)生最小的影響肚邢。

總結(jié)

對一道題考慮貪心算法的時(shí)候至少要自己先簡單證明一下正不正確壹堰,如果有些摸棱兩可的地方不妨多想想,或許有更好的貪心算法骡湖。

代碼

#include<iostream>
#include<algorithm>
using namespace std;
struct day {
    int a, b;//a記錄的是ddl贱纠,b記錄的是扣分
    bool operator<(const day& p) const {
        //第一關(guān)鍵字降序,第二關(guān)鍵字降序
        if (b != p.b) return b > p.b;
        return a > p.a;
    }
}days[9999];
bool vis[9999];//用來插這次訪問過的
void cal(int k)//用來計(jì)算要扣掉多少分
{
    int sum=0;
    int j;
    for (int i = 0; i < k; i++)
    {
        for (j = days[i].a; j >= 1; j--)
        {
            if (!vis[j])
            {
                vis[j] = true;
                break;
            }
        }
        if (j == 0)
            sum += days[i].b;
    }
    cout << sum<<endl;
    for (int i = 0; i < 3000; i++)
        vis[i] = false;
}
int main()
{
    int n;
    cin >> n;//輸入一共有多少組數(shù)據(jù)
    for (int i = 0; i < n; i++)
    {
        int k;
        cin >> k;//輸入這組數(shù)據(jù)中一共有多少個(gè)ddl
        for (int j = 0; j < k; j++)
        {
            cin >> days[j].a;
        }
        for (int j = 0; j < k; j++)
        {
            cin >> days[j].b;
        }
        sort(days, days + k);
        cal(k);
    }
}

B - 四個(gè)數(shù)列

現(xiàn)有4個(gè)數(shù)列响蕴,現(xiàn)要從4個(gè)數(shù)列中挑出4個(gè)數(shù)字其和為0谆焊,考慮其組合的個(gè)數(shù)。

輸入

一行是一組數(shù)據(jù)

6
-45 22 42 -16
-41 -27 56 30
-36 53 -37 77
-36 30 -75 -46
26 -38 -10 62
-32 -54 -6 45

輸出

5

思路

一開始我使用的map浦夷,之前在leetcode上做過相似的題辖试,只是那道題是用unordered_map做的,這次不支持C++11劈狐,就只能用map剃执,復(fù)雜度高了一點(diǎn),做出來的復(fù)雜度是O(n^2logn)懈息,和上課學(xué)的方法復(fù)雜度一樣肾档,但是可能常數(shù)比較大,所以沒過辫继。
map方法代碼

#include<iostream>
#include<map>
using namespace std;
int A[4000];//四個(gè)數(shù)組
int B[4000];
int E[4000];
int F[4000];
int main()
{
    int n;
    cin >> n;
    int k = 0;
    bool pan = true;
    map<int, int> C;
    for (int i = 0; i < n; i++)
    {
        scanf("%d %d %d %d",&A[i],&B[i],&E[i],&F[i]);
    }
    for (int i = 0; i <  n; i++)
    {
        for (int j = 0; j <  n; j++)
        {
                C[A[i] + B[j]]++;
        }
    }
    int sum = 0;
    for (int i = 0; i <  n; i++)
    {
        for (int j =0; j <  n; j++)
        {
            //if (C.count(-(E[i] + F[j])) != 0)
            sum+=C[-(E[i] + F[j])];
        }
    }
    cout << sum;

這樣的話只能乖乖用課上的方法怒见,一看四個(gè)數(shù)組,第一個(gè)想到的可能是一個(gè)四重循環(huán)枚舉所有的數(shù)得到結(jié)果姑宽,但是復(fù)雜度是O(n^4)根本不能接受好嗎遣耍。
所以可以先每兩個(gè)數(shù)組作為一個(gè)單位,枚舉其中的每兩個(gè)數(shù)的和炮车,這樣的話4個(gè)數(shù)組就變成了兩個(gè)數(shù)組A和B舵变,復(fù)雜度降到了O(n^2),這可以接受了,在得到B后瘦穆,要是把B給升序排列纪隙,然后從B中找A中對應(yīng)的數(shù)的時(shí)候,還能二分查找扛或,那么復(fù)雜度還能降低绵咱,是O(nlog_2n)更優(yōu)!

總結(jié)

這道題考慮降低復(fù)雜度以及二分查找的使用熙兔,啟示是只要有序的排列都能二分降低復(fù)雜度

AC代碼

#include<stdio.h>
#include<algorithm>
using namespace std;
int A[5000];
int B[5000];
int C[5000];
int D[5000];
int E[9999999];
int F[9999999];
int cal(int n,int k)
{
    int sum=0,ans=-1;
    n=-n;
    int l=0,r=k-1;
    int mid=(l+r)>>1;
    while(l<=r)
    {
        mid=(l+r)>>1;
        if(F[mid]>=n){
            ans=mid;
            r=mid-1;
        }
        else l=mid+1;
    }
    if(F[ans]!=n)
    return 0;
    else
    {
        for(int i=ans;i<k;i++)
        {
            if(F[i]==n)
            sum++;
            else
            break;
        }
    }
    return sum;
}
int main()
{
    int n;
    scanf("%d",&n);
    for(int i=0;i<n;i++)
    scanf("%d %d %d %d",&A[i],&B[i],&C[i],&D[i]);
    int k=0;
    for(int i=0;i<n;i++)
    {
        for(int j=0;j<n;j++)
        {
            E[k]=A[i]+B[j];
            k++;
        }
    }
    k=0;
    for(int i=0;i<n;i++)
    {
        for(int j=0;j<n;j++)
        {
            F[k]=C[i]+D[j];
            k++;
        }
    }
    sort(F,F+k);
    int sum=0;
    int bb;
    for(int i=0;i<k;i++)
        sum+=cal(E[i],k);
    printf("%d",sum);
} 
 

C - TT 的神秘禮物

現(xiàn)在給出一個(gè)數(shù)組悲伶,然后讓數(shù)組中每兩個(gè)序號不同的數(shù)相減取絕對值生成一個(gè)新數(shù)組艾恼,求這個(gè)新數(shù)組的中位數(shù)

輸入

每兩行為一組數(shù)據(jù),第一行代表這個(gè)數(shù)組有多少個(gè)數(shù)麸锉,第二行輸入這個(gè)數(shù)組

4
1 3 2 4
3
1 10 2

輸出

1
8

思路

求中位數(shù)要排序钠绍,排序之后就會有序,有序就能用二分花沉。
如果順著題目中的思路柳爽,那就是先給數(shù)組求中位數(shù)然后排序再找中位數(shù),但是這樣的結(jié)果是復(fù)雜度O(n^2)不能接受主穗。
那就改良,使用二分毙芜。
首先要去掉絕對值忽媒,那么就給目標(biāo)數(shù)組排序,這樣從大到小相減的時(shí)候就不用求絕對值了腋粥。然后開始對所有有可能的答案進(jìn)行二分搜索晦雨,已知的答案的最大值是目標(biāo)數(shù)組的最后一個(gè)數(shù)減去第0個(gè)數(shù),但是最小的未知隘冲,但一定大于等于0闹瞧,那就設(shè)置最小值是0,這樣的話對這里面的數(shù)進(jìn)行二分展辞,每次二分的結(jié)果都去數(shù)組里面找他所在的名次奥邮,如果名次是中位數(shù)的名次了,那么中位數(shù)就在它的附近罗珍。
那么名次怎么找呢洽腺,那就要枚舉,先有一個(gè)二分出來的數(shù)x覆旱,讓數(shù)組里面每個(gè)數(shù)與其后的所有數(shù)相減蘸朋,這時(shí)候用二分枚舉,得到最后一個(gè)差值<=x的數(shù)的位置扣唱,每個(gè)數(shù)組里的數(shù)都這么做一遍藕坯,就能得到這個(gè)數(shù)x的名次。
一直按上面的步驟計(jì)算x的名次噪沙,直到枚舉x的左右游標(biāo)相等時(shí)炼彪,即x是目標(biāo)中位數(shù)。

總結(jié)

這道題考察二分搜索正歼,是二分答案的典型題霹购,使用二分搜索能有效降低代碼的復(fù)雜度。

代碼

#include<iostream>
#include<algorithm>
using namespace std;
int cat[999999];
int n;
int search(int mid)
{
    int r, l;
    int midd;
    int sum = 0;
    int ans = 0;
    for (int i = 0; i < n - 1; i++)
    {
        ans = 0;
        r = n - 1;
        l = i;
        while (r >= l)
        {
            midd = (r + l) >> 1;
            if (cat[midd] - cat[i] <= mid)
            {
                ans = midd;
                l = midd + 1;
            }
            else
                r = midd - 1;
        }
        sum += (ans - i);
    }
    return sum;
}
int main()
{
    while (cin >> n)
    {
        for (int i = 0; i < n; i++)
            scanf_s("%d", &cat[i]);
        sort(cat, cat + n);
        long long r = cat[n - 1] - cat[0];
        long long l = 0;
        long long mid;
        int result = n * (n - 1) >> 1;
        if (result % 2 == 1)//他是奇數(shù)
            result = (result >> 1) + 1;
        else
            result = result >> 1;
        long long th;
        long long now;
        while (r >= l)
        {
            mid = (r + l) >> 1;
            th = search(mid);
            if (th >= result)
            {
                now = mid;
                r = mid - 1;
            }
            else
                l = mid + 1;
        }
        printf("%d\n", now);
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
禁止轉(zhuǎn)載朋腋,如需轉(zhuǎn)載請通過簡信或評論聯(lián)系作者齐疙。
  • 序言:七十年代末膜楷,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子贞奋,更是在濱河造成了極大的恐慌赌厅,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,000評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件轿塔,死亡現(xiàn)場離奇詭異特愿,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)勾缭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評論 3 399
  • 文/潘曉璐 我一進(jìn)店門揍障,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人俩由,你說我怎么就攤上這事毒嫡。” “怎么了幻梯?”我有些...
    開封第一講書人閱讀 168,561評論 0 360
  • 文/不壞的土叔 我叫張陵兜畸,是天一觀的道長。 經(jīng)常有香客問我碘梢,道長咬摇,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,782評論 1 298
  • 正文 為了忘掉前任煞躬,我火速辦了婚禮肛鹏,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘恩沛。我一直安慰自己龄坪,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評論 6 397
  • 文/花漫 我一把揭開白布复唤。 她就那樣靜靜地躺著健田,像睡著了一般。 火紅的嫁衣襯著肌膚如雪佛纫。 梳的紋絲不亂的頭發(fā)上妓局,一...
    開封第一講書人閱讀 52,394評論 1 310
  • 那天,我揣著相機(jī)與錄音呈宇,去河邊找鬼好爬。 笑死,一個(gè)胖子當(dāng)著我的面吹牛甥啄,可吹牛的內(nèi)容都是我干的存炮。 我是一名探鬼主播,決...
    沈念sama閱讀 40,952評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼穆桂!你這毒婦竟也來了宫盔?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,852評論 0 276
  • 序言:老撾萬榮一對情侶失蹤享完,失蹤者是張志新(化名)和其女友劉穎灼芭,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體般又,經(jīng)...
    沈念sama閱讀 46,409評論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡彼绷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了茴迁。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片寄悯。...
    茶點(diǎn)故事閱讀 40,615評論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖堕义,靈堂內(nèi)的尸體忽然破棺而出猜旬,到底是詐尸還是另有隱情,我是刑警寧澤胳螟,帶...
    沈念sama閱讀 36,303評論 5 350
  • 正文 年R本政府宣布昔馋,位于F島的核電站筹吐,受9級特大地震影響糖耸,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜丘薛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評論 3 334
  • 文/蒙蒙 一嘉竟、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧洋侨,春花似錦舍扰、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,470評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至裁僧,卻和暖如春个束,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背聊疲。 一陣腳步聲響...
    開封第一講書人閱讀 33,571評論 1 272
  • 我被黑心中介騙來泰國打工茬底, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人获洲。 一個(gè)月前我還...
    沈念sama閱讀 49,041評論 3 377
  • 正文 我出身青樓阱表,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子最爬,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評論 2 359

推薦閱讀更多精彩內(nèi)容