みんなのプロコン2018 決勝 A- Uncommon
解法
ある整数 i について配列内の i と互いに素な整数の数を数えるのは大変そうなので、i と互いに素でない数を数えることにする。
i を素因数分解し、例えば素因数が p, q, r だったとき、配列 a に含まれる i と互いに素でないでない整数の数は以下のように求められる。
(配列 a に含まれる i と互いに素でないでない整数の数) = (配列 a に含まれる p の倍数の数) + (配列 a に含まれる q の倍数の数) + (配列 a に含まれる r の倍数の数) - (配列 a に含まれる p*q の倍数の数) - (配列 a に含まれる q*r の倍数の数) - (配列 a に含まれる r*p の倍数の数) + (配列 a に含まれる p*q*r の倍数の数)
配列 a 内の中にある x の倍数の数をカウントしたいが、 は なので、以下のような愚直なループで求まる。
for x in 2..n: cur = x while cur <= max: sum[x] += count[cur] cur += x
コード
/// Thank you tanakh!!! /// https://qiita.com/tanakh/items/0ba42c7ca36cd29d0ac8 macro_rules! input { (source = $s:expr, $($r:tt)*) => { let mut iter = $s.split_whitespace(); input_inner!{iter, $($r)*} }; ($($r:tt)*) => { let mut s = { use std::io::Read; let mut s = String::new(); std::io::stdin().read_to_string(&mut s).unwrap(); s }; let mut iter = s.split_whitespace(); input_inner!{iter, $($r)*} }; } macro_rules! input_inner { ($iter:expr) => {}; ($iter:expr, ) => {}; ($iter:expr, $var:ident : $t:tt $($r:tt)*) => { let $var = read_value!($iter, $t); input_inner!{$iter $($r)*} }; } macro_rules! read_value { ($iter:expr, ( $($t:tt),* )) => { ( $(read_value!($iter, $t)),* ) }; ($iter:expr, [ $t:tt ; $len:expr ]) => { (0..$len).map(|_| read_value!($iter, $t)).collect::<Vec<_>>() }; ($iter:expr, chars) => { read_value!($iter, String).chars().collect::<Vec<char>>() }; ($iter:expr, usize1) => { read_value!($iter, usize) - 1 }; ($iter:expr, $t:ty) => { $iter.next().unwrap().parse::<$t>().expect("Parse error") }; } fn get_primes(n: usize) -> Vec<usize> { let mut is_prime = vec![true; n + 1]; is_prime[0] = false; is_prime[1] = false; let mut primes = vec![]; for i in 2..(n + 1) { if is_prime[i] { primes.push(i); let mut pos = i * 2; while pos < is_prime.len() { is_prime[pos] = false; pos += i; } } } primes } use std::collections::BTreeSet; fn main() { input!(n: usize, m: usize, a: [usize; n]); let primes = get_primes(350); let mut divisors = vec![vec![]; m + 1]; let mut set = BTreeSet::new(); for i in 1..(m + 1) { let mut a = i; for &prime in primes.iter() { if a % prime == 0 { divisors[i].push(prime); set.insert(prime); while a % prime == 0 { a /= prime; } } if prime * prime > a { break; } } if a > 1 { divisors[i].push(a); set.insert(a); } } let max: usize = *a.iter().max().unwrap(); let mut a_count = vec![0; max + 1]; for &a in a.iter() { a_count[a] += 1; } let mut divide_count = vec![0; m + 1]; for i in 2..(m + 1) { let mut cur = i; let mut count = 0; while cur <= max { count += a_count[cur]; cur += i; } divide_count[i] = count; } for i in 1..(m + 1) { let divisors = &divisors[i]; let n = divisors.len(); let mut ans: i64 = 0; for mask in 1..(1 << n) { let mut count_ones = 0; let mut t = 1; for i in 0..n { if mask & (1 << i) != 0 { t *= divisors[i]; count_ones += 1; } } let sum = divide_count[t] as i64; let sum = if count_ones % 2 == 1 { sum } else { -sum }; ans += sum; } println!("{}", a.len() as i64 - ans); } }