icodelab 另一个转圈游戏

描述

老师让n(1<=n<=100000)个小朋友围成一个圈,第k名同学和第k+1以及第k-1名同学相邻,1号同学和n号同学相邻,老师在一个包里放了很多写有数字的纸条,可能多个个纸条上的数字是相同的,并且数字num的范围是1<=num<=1000000。接下来每一位同学i从包里抽出一张纸条Ai,每位同学都走一圈,同时拍打能整除纸条上数字的同学后背,然后走回到自己的位置上。要求你计算出每位同学总共拍打了多少同学的后背。

输入

第一行包含一个整数n,接下来的n行,每行输入一个整数Ai。

输出

输出共有n行,每行一个数字,表示该同学拍打其他同学的数量。

输入样例 1

5
2
1
2
3 
4

输出样例 1

2
0
2
1
3

思路:

详见注释(皮一下很开心)

代码:

 

#include<bits/stdc++.h>
using namespace std;
int a[10000005];
int n,mx=-100;
int b[1000005];
int cnt[1000005];
void get_factor(){
    for (int i = 1;  i <= 1000000; i++){//题目中说明数字不超过100w 
        if(!b[i]) continue;//数字必须存在,不存在则跳过 
        for (int j = 1; j <= 1000000 / i; j++){
        	// b[i*j]为真,表示i这个数字的j倍也是存在的,那么显然i*j能整除i,
			//所以我们要统计i*j这个数字能被整除多少次时就要把i这个数字出现的次数累加上去
			//i这个数字出现的次数就是b[i]
			//我们用cnt[i*j] 来累计i*j这个数字被整除的次数,所以有cnt[i*j]+=b[i]; 
             if(b[i*j]) cnt[i*j]+=b[i]; 
             
             //这个地方很关键,因为j是从1开始的,所以要把自己整除自己的那一次减掉
             //但是j初始化是必须从1而不能从2开始,因为i这个数字有可能出现了多次 
             //其实下面这一句可以不要,然后在输出答案的地方减1,及输出cnt[a[i]]-1就可以了。 
             if(i*j==i) cnt[i]--; 
             
        }       
    } 
}
int main(){
    scanf("%d",&n);//输入n个人 
    for(int i=1;i<=n;i++){
        scanf("%d",&a[i]);//输入每个人手中的数字 
        b[a[i]]++;//因为每个人手中的数字可能重复,所以用桶算法 
    }
    get_factor();
    for(int i=1;i<=n;i++)
        cout<<cnt[a[i]]<<endl;
    return 0;
} 

 

 

 

 

posted @ 2019-08-02 22:35  双子最可爱啦  阅读(151)  评论(0编辑  收藏  举报