Skip to content

Instantly share code, notes, and snippets.

@gushwell
Last active July 4, 2018 12:03
Show Gist options
  • Save gushwell/541e5e28ffdd9667611bef0a9e0e637b to your computer and use it in GitHub Desktop.
Save gushwell/541e5e28ffdd9667611bef0a9e0e637b to your computer and use it in GitHub Desktop.
var mebius = new Mebius(100);
int n = mebius[10];
var mebius = new Mebius(100);
int n = mebius[10];
μ(1) = 1
μ(2) = -1
μ(3) = -1
μ(4) = 0
μ(5) = -1
μ(6) = 1
μ(7) = -1
μ(8) = 0
μ(9) = 0
μ(10) = 1
μ(11) = -1
μ(12) = 0
μ(13) = -1
μ(14) = 1
μ(15) = 1
μ(16) = 0
μ(17) = -1
μ(18) = 0
μ(19) = -1
μ(20) = 0
μ(21) = 1
μ(1) = 1
μ(2) = -1
μ(3) = -1
μ(4) = 0
μ(5) = -1
μ(6) = 1
μ(7) = -1
μ(8) = 0
μ(9) = 0
μ(10) = 1
μ(11) = -1
μ(12) = 0
μ(13) = -1
μ(14) = 1
μ(15) = 1
μ(16) = 0
μ(17) = -1
μ(18) = 0
μ(19) = -1
μ(20) = 0
μ(21) = 1
using System;
using System.Linq;
namespace Gushwell.Etude {
class Program {
static void Main(string[] args) {
// 1 - 100 までのメビウス関数を求める
int upper = 100;
Mebius mebius = new Mebius(upper);
for (int i = 1; i <= upper; i++) {
Console.WriteLine("μ({0}) = {1}", i, mebius[i]);
}
Console.ReadLine();
}
}
class Mebius {
private int[] mebius;
public Mebius(int maxnum) {
mebius = Enumerable.Repeat(1, maxnum + 1).ToArray();
foreach (int p in PrimeNumber.Enumerate().TakeWhile(n => n <= maxnum)) {
for (int i = p; i <= maxnum; i += p)
mebius[i] *= -1;
int p2 = p * p;
for (int pp = p2; pp <= maxnum; pp += p2)
mebius[pp] = 0;
}
}
public int this[int n] {
get {
return mebius[n];
}
}
}
}
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
namespace Gushwell.Etude {
public static class PrimeNumber {
public static IEnumerable<int> Enumerate() {
// 2,3は既知の素数とする
var primes = new List<int>() { 2, 3 };
foreach (var p in primes)
yield return p;
// 4以上の整数から素数を列挙する。int.MaxValueを超えたときには対処していない
int ix = 0;
while (true) {
int prime1st = primes[ix];
int prime2nd = primes[++ix];
// ふるい用の配列の下限、上限を求め、配列を確保する。
var lower = prime1st * prime1st;
var upper = prime2nd * prime2nd - 1;
// ふるいは、[4:8], [9:24], [25:48], [49:120]... と変化する。
// []内の数値は、配列の下限と上限
var sieve = new BoundedBoolArray(lower, upper);
// 求まっている素数を使い、ふるいに掛ける
foreach (var prime in primes.Take(ix)) {
var start = (int)Math.Ceiling((double)lower / prime) * prime;
for (int index = start; index <= upper; index += prime)
sieve[index] = true;
}
// ふるいに掛けられて残った値が素数。これを列挙する。
// 併せて、求まった素数は、primesリストに記憶していく。
// この素数が次にふるいに掛ける際に利用される。
for (int i = lower; i <= upper; i++) {
if (sieve[i] == false) {
primes.Add(i);
yield return i;
}
}
}
}
}
// 下限、上限が指定できるbool型配列
class BoundedBoolArray {
private BitArray _array;
private int _lower;
public BoundedBoolArray(int lower, int upper) {
_array = new BitArray(upper - lower + 1);
_lower = lower;
}
public bool this[int index] {
get {
return _array[index - _lower];
}
set {
_array[index - _lower] = value;
}
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment