David Steinsland – informatikkstudent og webutvikler

Innlegg merket med «matematikk»

Java: Primtallgenerering og -faktorisering

| Ingen kommentarer »

Hvordan kan man finne alle primtall mindre enn k? Hvordan sjekke om et gitt heltall n er primtall? Hvordan kan vi primtallsfaktorisere ethvert tall?

Først av alt må man etablere et par matematiske sannheter:

  1. Et primtall er et helt tall større enn 1 som bare er delelig med seg selv og med 1
  2. Et heltall k er også primtall dersom det ikke er delelig med andre primtall mindre enn eller lik kvadratrota til k.
  3. Et hvert heltall større enn 1 kan skrives som produktet av ett eller flere primtall.

Så, hvordan kan man finne et primtall?

  1.  Gitt et heltall k, begynner vi å sjekke om tallet er delelig med det minste primtallet, 2.
  2. Om det er delelig, kan vi trekke konklusjonen om at tallet ikke er primtall.
  3. Dersom det ikke er delelig, går vi videre til 3. Dette gjentar vi for alle primtall \leq \sqrt{k}.
  4. Om ingen tall er delelige, er tallet et primtall.

Vi kan nå begynne å skrive Java-koden vår, og en kan for eksempel ende opp med noe slikt:


public boolean isPrime (int n)
{
	int q = (int) Math.sqrt (n);

	boolean isPrime = true;

	for (int i = 2; i < q; i++)
 	{
		if (n % i == 0)
		{
			return false;
		}
	}

	return true;
}

Det første vi gjør er å ta kvadratrota til heltallet vårt. Deretter kjører vi gjennom alle heltall 2 \leq i \leq \sqrt{n}. Dersom tallet skulle være delelig med et annet, da er det ikke et primtall. Når løkken er gjennomført, vet vi at ingen tall er delelige og at heltallet vårt er primtall.

Primtallsfaktorisering

Ved aritmetikkens fundamentalteorem vet vi at et heltall større enn 1 er enten et primtall, eller et produkt av ett eller flere primtall.

For eksempel er: 7007 = 7 \cdot 7 \cdot 11 \cdot 13


public Integer[] primeFactorization (int a)
{
	ArrayList factors = new ArrayList();

	// fetch primes less than or equal to sqrt(a).
	// if no primes are returned, the number itself is a prime.
	Integer[] primes = primes ((int)Math.sqrt(a));

	int i = 0;
	while (i < primes.length)
 	{
 		int p = primes[i];
 		if (Arrays.asList (primes).contains(a) || isPrime (a))
 		{
 			// the remaining number is now a prime.
 			break;
 		}
 		if (a % p == 0)
 		{
 			// add the prime to our list, and execute the division
 			factors.add (p);
 			a = a / p;
 		}
 		else
 		{
 			// Continue to next prime, if the current one does not
 			// divide
 			i++;
 		}
 	}
 	// The number than remains should be a prime itself.
 	if (a > 1)
	{
		factors.add (a);
	}

	return (Integer[]) factors.toArray(new Integer[factors.size()]);
}

Her ble det kanskje litt komplisert med det første, men prosessen er igrunn ganske enkel:

  1. primes() til å gi oss et int-array med primtall (bruker funksjonen isPrime())
  2. Deretter går vi gjennom alle primtallene, og sjekker om tallet a er delelig med p. Dersom det er delelig, så legger vi primtallet p i samlingen vår og utfører divisjonen \frac{a}{p}.
  3. I neste runde tester vi enten det samme primtallet som i forrige runde, eller så tester vi det neste primtallet. Det er først når det gjeldende primtallet ikke er delelig, at vi går til neste.
  4. Til slutt skal det resterende tallet selv være et primtall, men vi ønsker ikke å inkludere tallet 1 som faktor.

Eksempel

Om vi bruker tallet 7007 om igjen, så vil vi i Java-funksjonen få følgende primtall:

{2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83}. Det er alle primtall som er mindre enn kvadratroten til 7007.

Deretter tester vi først om 2 er delelig med 7007. Det er det ikke, så vi går videre til neste primtall. Det er først når vi kommer til 7 at vi finner et primtall som er delelig med 7007.

Vi legger da til tallet 7 i listen vår, og utfører divisjonen \frac{7007}{7} = 1001. I neste runde tester vi tallet 7 en gang til, og igjen så er det delelig. Vi utfører da \frac{1001}{7} = 143.

Når vi i påfølgende runde tester 7 igjen, finner vi ut at det ikke lengre er delelig. Men det er primtallet 11! Det siste primtallet vil da være 13.

Dette gir oss følgende i resultat: {7, 7, 11, 13} som forteller oss hvilke primtallsfaktorer som gir 7007.

Fagvalg 2010: Valget er tatt

| 8 kommentarer »

Som jeg skrev tidligere, uttrykket jeg min fortvilelse da jeg fant ut at det var ingen måte jeg kunne unngå matematikk (R2) i tredjeklasse til høsten.

Nå, derimot, ser jeg løsningen og er villig til å ta den.

Da jeg tilfeldigvis gikk gjennom fagplanen for Informasjonsteknologi 1, som også tilfeldigvis er et realfag,  så jeg at jeg kunne mesteparten av pensumet allerede. Siden reglene sier at jeg er nødt til å fortsette med minst to realfag i tredjeklasse som jeg tar nå i andreklasse, meldte jeg meg opp som privatist i IT-1 med eksamen til våren.

Da løste plutselig alt seg!

Jeg ble litt forbauset da jeg så at timeantallet mitt nå er øket til 37 timer (mot 30 timer/uke som er påkrevd). Dette betyr at jeg har så mange timer til overs, at jeg kun trenger ta to programfag til neste år, og ikke tre som jeg ellers måtte ha gjort (2 realfag + 1 fra et annet programområde).

Videre betyr dette at jeg da kan fortsette med Kjemi 2 og Informasjonsteknologi 2. Jeg er da nødt til å ta sistnevnte som privatist, men det vil gi meg 10 timer mindre pr. uke enn alle andre, da jeg kun har ett programfag i hele tredjeklasse (på skolen da, vel å merke). Sweet!

PS: Jeg har ingenting i mot matematikk; faget er både interessant og utfordrende (kom for øvrig på tredjeplass på Abelkonkurransen, runde 1). Det som foreligger bak, er at jeg ikke trenger R2 til de studiene jeg har tenkt å ta (i skrivende stund). Selvsagt er det nyttig med matematikk, og jeg hadde sikkert ikke hatt vondt av å tatt R2 heller.

Nå har jeg iallfall åpnet dørene, slik jeg har muligheten til å ta R2 eller velge det bort. Istedenfor å bli presset til å ta et fag jeg ikke ønsker, kan jeg nå velge/vrake det i fred og ro.

Om det er flere som skulle havne i samme situasjon som meg: ta kontakt med Eksamenskontoret/privatistkontoret i hjemfylket ditt; de kan svare på det meste!