Codeforces Round #109 (Div. 2) D. Colliders 數學

一直覺得codeforces上的題解法都很新奇,這個也不例外,說說題意,

最大10^5的數,進行10^5次操作,每一次進行加入或者刪除一個數,加入時:只有與已加入的數互素才能加入,不能加入的,輸出原因(與某個不互素,或者,已經加入)。刪除時:輸出刪除成功,或者尚未加入。

這個題看起來不簡單,經過分析之後,我發現,對於10^5這麼大的數,只能最多有7個不同的素數的冪的積的形式,要使互素,也就是說,沒有共同的素數因子,按照這個思路,我們對每一個數的素數因子進行標記,插入的時候,就可以查詢插入的這個數的素數因子有沒有出現

這個思路的時間複雜度是O(10^5*7),這個複雜度是可以解決的。

但是,其中有一個需要輸出與第幾個互素,這個也需要標記。ok,代碼見下面

#include <iostream>
#include <stdio.h>
#include <string.h>
using namespace std;

#define N 1000010
int prime[N];
bool isPrime[N+1];
int size;

void getPrime() {
    size = 0;
    memset(isPrime,true,sizeof(isPrime));
    int i;
    for(i=2; i<=N/2; i++) {
        if(isPrime[i])  //i是素數
            for(int j=i+i; j<=N; j+=i)
                isPrime[j]=0;
    }
    for(i=2; i<=N; i++) {
        if(isPrime[i])
            prime[size++]=i;
    }
}

bool active[100010];
int data[100010][10];
int count[100010];//計數出現這個素數的j

int main()
{
    //cout << 2*3*5*7*11*13*17;
    getPrime();
    for(int i = 0;i < size;i++)
    {
        for(int j = prime[i];j < 100010;j += prime[i])
            data[j][ ++data[j][0] ] = prime[i];
    }
    int n,m;
    scanf("%d%d",&n,&m);
    memset(active,0,sizeof(active));
    memset(count,0,sizeof(count));
    char c[2];int t;
    for(int i = 0;i < m;i++)
    {
        scanf("%s%d",c,&t);
        if(c[0] == '+')
        {
            //1 success 2 already on 3衝突
            if(active[t])
                printf("Already on\n");
            else
            {
                bool b = 0;
                for(int j = 1;j <= data[t][0];j++)
                    if(count[ data[t][j] ])
                    {
                        printf("Conflict with %d\n",count[ data[t][j] ]);
                        b = 1;break;
                    }
                if(!b)
                {
                    for(int j = 1;j <= data[t][0];j++)
                        count[ data[t][j] ] = t;
                    printf("Success\n");
                    active[t] = 1;
                }
            }
        }else //--
        {
            if(!active[t])
                printf("Already off\n");
            else
            {
                for(int j = 1;j <= data[t][0];j++)
                    count[ data[t][j] ] = 0;
                printf("Success\n");
                active[t] = 0;
            }
        }
    }
    return 0;
}



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章