1. 程式人生 > >csp 201709-2 公共鑰匙盒

csp 201709-2 公共鑰匙盒

  • 題目

問題描述

  有一個學校的老師共用N個教室,按照規定,所有的鑰匙都必須放在公共鑰匙盒裡,老師不能帶鑰匙回家。每次老師上課前,都從公共鑰匙盒裡找到自己上課的教室的鑰匙去開門,上完課後,再將鑰匙放回到鑰匙盒中。
  鑰匙盒一共有N個掛鉤,從左到右排成一排,用來掛N個教室的鑰匙。一串鑰匙沒有固定的懸掛位置,但鑰匙上有標識,所以老師們不會弄混鑰匙。
  每次取鑰匙的時候,老師們都會找到自己所需要的鑰匙將其取走,而不會移動其他鑰匙。每次還鑰匙的時候,還鑰匙的老師會找到最左邊的空的掛鉤,將鑰匙掛在這個掛鉤上。如果有多位老師還鑰匙,則他們按鑰匙編號從小到大的順序還。如果同一時刻既有老師還鑰匙又有老師取鑰匙,則老師們會先將鑰匙全還回去再取出。
  今天開始的時候鑰匙是按編號從小到大的順序放在鑰匙盒裡的。有K

位老師要上課,給出每位老師所需要的鑰匙、開始上課的時間和上課的時長,假設下課時間就是還鑰匙時間,請問最終鑰匙盒裡面鑰匙的順序是怎樣的?

輸入格式

  輸入的第一行包含兩個整數NK
  接下來K行,每行三個整數wsc,分別表示一位老師要使用的鑰匙編號、開始上課的時間和上課的時長。可能有多位老師使用同一把鑰匙,但是老師使用鑰匙的時間不會重疊。
  保證輸入資料滿足輸入格式,你不用檢查資料合法性。

輸出格式

  輸出一行,包含N個整數,相鄰整數間用一個空格分隔,依次表示每個掛鉤上掛的鑰匙編號。

樣例輸入

5 2
4 3 3
2 2 7

樣例輸出

1 4 3 2 5

樣例說明

  第一位老師從時刻3開始使用4號教室的鑰匙,使用3單位時間,所以在時刻6還鑰匙。第二位老師從時刻2開始使用鑰匙,使用7單位時間,所以在時刻9還鑰匙。
  每個關鍵時刻後的鑰匙狀態如下(X表示空):
  時刻2後為1X345;
  時刻3後為1X3X5;
  時刻6後為143X5;
  時刻9後為14325。

樣例輸入

5 7
1 1 14
3 3 12
1 15 12
2 7 20
3 18 12
4 21 19
5 30 9

樣例輸出

1 2 3 5 4

評測用例規模與約定

  對於30%的評測用例,1 ≤ NK ≤ 10, 1 ≤ w ≤ N, 1 ≤ sc ≤ 30;
  對於60%的評測用例,1 ≤ N

K ≤ 50,1 ≤ w ≤ N,1 ≤ s ≤ 300,1 ≤ c ≤ 50;
  對於所有評測用例,1 ≤ NK ≤ 1000,1 ≤ w ≤ N,1 ≤ s ≤ 10000,1 ≤ c ≤ 100。

  • 基本思路

這個題目大概的意思就是按它的要求進行借還鑰匙,時間不一樣的優先時間短的先,然後就是還鑰匙的狀態優先於借鑰匙的狀態,最後一個要求就是,前面都符合了,按鑰匙編號小的優先操作,所以我就用了一個結構體重構進行排序,n個數據後面就變成了2n個數據,因為每個資料都有借的時間和還的時間,然後全部合起來進行排序,按順序進行鑰匙的狀態進行變化,變化到最後一次就是答案了。

  • 演算法思想

初始化一個鑰匙起初的位置,然後把每次輸入的資料都弄成兩個資料,一個是借鑰匙的最開始時間,以及還鑰匙的最後時間,然後按題目的要求進行排序,時間不一樣的按時間短的優先,下一層次還鑰匙優先借鑰匙,下一層次按鑰匙的編號小的優先,然後排序完之後,對結構體數組裡成員的鑰匙狀態進行變化,然後那個狀態變到最後就是答案了。

  • 流程
  1. 初始化鑰匙的位置
  2. 輸入資料之後生成兩個資料,一個借鑰匙的時間,一個是還鑰匙的時間
  3. 進行排序
  4. 根據結構體的鑰匙狀態進行變化,鑰匙的位置一直變
  5. 輸出最終的鑰匙位置
  • 程式碼實現

#include<cstdio>

#include<vector>

#include<cstring>

#include<algorithm>

using namespace std;

const int maxn=1009;

struct node{

    int key_num;///鑰匙編號

    int time;///老師上課時長

    int mark;///鑰匙的狀態:還(1) 借(0)

    node(int a,int b,int c){

        key_num=a;

        time=b;

        mark=c;

    }

    ///重構進行排序

    bool operator <(const node b)const {

        if(time!=b.time){///時間不一樣的優先判斷,時間短的先進行

            return time<b.time;

        }else if(mark!=b.mark){///鑰匙的狀態 還是1,借是0,先還再借

            return mark>b.mark;

        }else{

            return key_num<b.key_num;///按鑰匙的編號,小的優先

        }

    }

};

int main(){

    int n,k,w,s,c;

    vector<node> vec;

    int key[maxn];

    scanf("%d%d",&n,&k);

    for(int i=1;i<=n;++i){

        key[i]=i;///初始化鑰匙的位置按順序

    }

    for(int i=0;i<k;++i){

        scanf("%d%d%d",&w,&s,&c);

        vec.push_back(node(w,s,0));///本次老師借鑰匙的起始時間

        vec.push_back(node(w,s+c,1));///本次老師還鑰匙的最終時間

    }

    sort(vec.begin(),vec.end());///進行題目要求的方式去排序

    for(int i=0;i<vec.size();++i){

        if(vec[i].mark==1){///鑰匙狀態為1的就是要還

            for(int j=1;j<=n;++j){///進行遍歷鑰匙key的陣列

                if(key[j]==0){///鑰匙有為0的,說明這個位置是空的,可以放進去

                    key[j]=vec[i].key_num;///賦值這個鑰匙的編號

                    break;///放進去了結束了就跳出

                }

            }

        }else{

            for(int j=1;j<=n;++j){

                if(key[j]==vec[i].key_num){///這個是鑰匙狀態為0,說明要借,就遍歷key要數陣列,編號一樣的就進來

                    key[j]=0;///這個位置賦值成0,說明現在這個位置是空的

                    break;///操作完,並跳出

                }

            }

        }

    }

    for(int i=1;i<=n;++i){///上面全部都操作完,就進行key陣列的全部輸出,也就是答案了

        printf("%d ",key[i]);

    }

    return 0;

}