Ordinare i divisori di un numero in base alla scomposizione in fattori primi


23

Dato un input di un numero intero ≥ 2, emette un elenco dei suoi divisori ordinati per esponente nelle sue prime fattorizzazioni, in ordine crescente, ordinando prima per primo più grande, poi per secondo più grande, e così via.

Ad esempio, prendi l'intero 72, che è 2 3 3 2 . Ha i divisori

1     3^0 · 2^0
2     3^0 · 2^1
3     3^1 · 2^0
4     3^0 · 2^2
6     3^1 · 2^1
8     3^0 · 2^3
9     3^2 · 2^0
12    3^1 · 2^2
18    3^2 · 2^1
24    3^1 · 2^3
36    3^2 · 2^2
72    3^2 · 2^3

Se ordinato in ordine crescente dagli esponenti in base ai fattori primi, con i numeri primi più grandi che hanno la priorità, questo diventa

1     3^0 · 2^0
2     3^0 · 2^1
4     3^0 · 2^2
8     3^0 · 2^3
3     3^1 · 2^0
6     3^1 · 2^1
12    3^1 · 2^2
24    3^1 · 2^3
9     3^2 · 2^0
18    3^2 · 2^1
36    3^2 · 2^2
72    3^2 · 2^3

Si noti che l'elenco viene ordinato prima in base all'ordine dell'esponente di 3, quindi in base all'esponente di 2. Puoi anche pensare a questo come leggere da sinistra a destra e dall'alto verso il basso attraverso la seguente griglia:

        2^0  2^1  2^2  2^3

3^0     1    2    4    8
3^1     3    6    12   24
3^2     9    18   36   72

Casi test:

2 => 1 2
72 => 1 2 4 8 3 6 12 24 9 18 36 72
101 => 1 101
360 => 1 2 4 8 3 6 12 24 9 18 36 72 5 10 20 40 15 30 60 120 45 90 180 360
3780 => 1 2 4 3 6 12 9 18 36 27 54 108 5 10 20 15 30 60 45 90 180 135 270 540 7 14 28 21 42 84 63 126 252 189 378 756 35 70 140 105 210 420 315 630 1260 945 1890 3780
30030 => 1 2 3 6 5 10 15 30 7 14 21 42 35 70 105 210 11 22 33 66 55 110 165 330 77 154 231 462 385 770 1155 2310 13 26 39 78 65 130 195 390 91 182 273 546 455 910 1365 2730 143 286 429 858 715 1430 2145 4290 1001 2002 3003 6006 5005 10010 15015 30030
65536 => 1 2 4 8 16 32 64 128 256 512 1024 2048 4096 8192 16384 32768 65536
74088 => 1 2 4 8 3 6 12 24 9 18 36 72 27 54 108 216 7 14 28 56 21 42 84 168 63 126 252 504 189 378 756 1512 49 98 196 392 147 294 588 1176 441 882 1764 3528 1323 2646 5292 10584 343 686 1372 2744 1029 2058 4116 8232 3087 6174 12348 24696 9261 18522 37044 74088

Dato che si tratta di , vince il codice più breve in byte.

Risposte:


8

05AB1E , 6 byte

Codice:

ÑÒí{€P

Spiegazione:

Ñ       # Get the divisors of input.
 Ò      # Factorize each.
  í     # Reverse each.
   {    # Sort the array.
    €P  # Product each.

Utilizza la codifica CP-1252 . Provalo online! .


1
Noice: p (ben fatto)
spasso

8

Gelatina , 8 7 byte

ÆDÆfU$Þ

Provalo online! Grazie a @Dennis per -1 byte.

ÆD         Array of divisors, e.g. 24 -> [1, 2, 4, 8, 3, 6, 12, 24]
      Þ    Sort by...
     $       Combine previous two links...
  Æf           Factorise each, e.g. ['', [2], [3], [2, 2], [2, 3], [2, 2, 2],
                   [2, 2, 3], [2, 2, 2, 3]]
    U          Upend/reverse each sublist

2
ÆDÆfU$Þ(usando il nuovo ordinamento di Jelly), salva un byte.
Dennis,

7

Pyth, 10 byte

+1{*Mt_DyP

Provalo online: dimostrazione

Purtroppo il prodotto su un elenco vuoto non è definito come 1 in Pyth. Questo costa tre byte extra.

Spiegazione:

+1{*Mt_DyPQ   implicit Q (=input number) at the end
         PQ   prime factorization of input
        y     powerset
      _D      order by reversed subsets
     t        remove the empy subset
   *M         compute the product of each subsets
  {           remove duplicates
+1            prepend 1

7

Gelatina , 12 10 byte

2 byte grazie a @ Sp3000.

ÆE'ḶUṚŒpUṚÆẸ
ÆEU'ḶŒpUÆẸ

Provalo online!

Suite di test.

ÆE            Array of exponents, e.g. 24 -> [3, 1] since 24 = 2^3*3^1
  U           Upend/reverse, e.g. [1, 3]
   ‘Ḷ         Range of each, from 0, e.g. [[0, 1], [0, 1, 2, 3]]
     Œp       Cartesian product, e.g. [[0, 0], [0, 1], ..., [1, 3]]
       U      Upend, reversing the innermost lists
        ÆẸ    Inverse of ÆE, converting exponents back into a number

Ringraziamenti a @ Sp3000 per aver fornito il formato della spiegazione.


7

Python 2, 85 byte

n=input()
p,=L=[1]
while~-n:
 l=L;p+=1
 while n%p<1:L=l+[x*p for x in L];n/=p
print L

Nessuna fattorizzazione, nessun ordinamento. Implementazione ricorsiva della stessa lunghezza:

f=lambda n,p=2:1/n*[1]or n%p and f(n,p+1)or[x*c for x in f(n/p)for c in[1,p][x%p<1:]]

5

In realtà, 19 byte

;÷#o♂w♂RS`"iⁿ"£Mπ`M

Provalo online!

Spiegazione:

;÷#o♂w♂RS`"iⁿ"£Mπ`M
;                    duplicate input
 ÷                   divisors
  #o                 include input in divisors list (note to self: fix this bug)
    ♂w               factor each integer into a list of [prime, exponent] pairs
      ♂R             reverse each list, so that the largest prime comes first
        S            sort the list
         `"iⁿ"£Mπ`M  for each factorization:
          "iⁿ"£M       for each [prime, exponent] pair:
           iⁿ            push prime**exponent
                π      product

5

JavaScript, 78 byte

f=(n,p=2,a=[1],b=a)=>n<2?a:n%p?f(n,p+1,a):f(n/p,p,a.concat(b=b.map(m=>m*p)),b)

Basato sull'idea di @ xnor, anche se non ho capito il suo codice, quindi ho dovuto reimplementarlo da zero. L'algoritmo di base è che si inizia con [1] e si moltiplica per [1, ..., pᵏ] per ogni pᵏ nella fattorizzazione primaria di n, anche se poiché non ho una fattorizzazione primaria o un prodotto cartesiano, devo farlo tutto ricorsivamente. Esempio:

n=72 p=2 a=[1] b=[1]
n=36 p=2 a=[1,2] b=[2]
n=18 p=2 a=[1,2,4] b=[4]
 n=9 p=2 a=[1,2,4,8] b=[8]
 n=9 p=3 a=[1,2,4,8] b=[1,2,4,8]
 n=3 p=3 a=[1,2,4,8,3,6,12,24] b=[3,6,12,24]
 n=1 p=3 a=[1,2,4,8,3,6,12,24,9,18,36,72] b=[9,18,36,72]

Ho appena ricordato quando eri a 10k .. ora quasi a 14k. Continuate così!!
NiCk Newman,

2

R, 196 byte

n=scan()
if(n<4)c(1,n)else{
r=2:n
d=NULL
while(n>1){i=r[min(which(n%%r==0))];d=c(d,i);n=n/i}
m=unique(d)
b=table(d)
l=list()
for(i in 1:length(m))l[[i]]=m[i]^(0:b[i])
apply(expand.grid(l),1,prod)}

Questo sarà inefficiente come diamine perché difficilmente ho resistito alla tentazione di usare library(primes). Crea un vettore ddi tutti i fattori primi dell'input, calcola la loro frequenza (numero di occorrenze), quindi calcola il prodotto cartesiano di tutti i possibili poteri (da 0 alla rispettiva frequenza b[i]), a cui prodviene applicata la funzione. Dang it, casi speciali di 2 e 3! Altrimenti, questa è una bella vetrina della gestione dei dataframe R e delle funzioni vettoriali / operazioni per riga (e persino la tablefunzione puramente statistica !).

Naturalmente, la sua efficienza può essere migliorata al costo di 15 byte usando r=2:ceiling(sqrt(n)), se qualcuno se ne frega. Ecco una versione ungolfed più bella:

factorise <- function(n){
  if (n<4) c(1,n) else { # Now that all special cases have been handled
    r=2:ceiling(sqrt(n)) # We check all divisors smaller than the square root
    d=NULL # Initiate the variable for divisors
    while (n>1) {
      i=r[min(which(n%%r==0))] # Check the first divisor with a zero remainder
      d=c(d,i) # Append it to the list of divisors
      n=n/i   # Divide by it and check again
    }
    m=unique(d) # Get unique divisors, and they are already sorted
    b=table(d) # Count their frequencies
    l=list() # Initiate a list of all possible powers of unique factors
    for(i in 1:length(m)) l[[i]]=m[i]^(0:b[i]) # Calculate powers
    apply(expand.grid(l),1,prod) # Make a cartesian dataframe and row-multiply
  }
}

2

Mathematica 150 byte

f[t_]:=Thread@{#,IntegerExponent[t,#]&/@#}&@Prime@Range@PrimePi@Max@FactorInteger[t][[All,1]];Times@@@(#^#2&@@@#&/@Sort[Reverse/@(f@#&/@Divisors@#)])&

2

Brachylog , 3 byte

fḋᵒ

Provalo online!

Il codice legge più o meno come il titolo della sfida: "i fattori dell'input, ordinati in base alle loro prime scomposizioni". Accertarsi che questa bellezza a 3 byte abbia effettivamente superato i casi di test usando solo il senso integrato di Brachylog su come ordinare gli elenchi ha finito per richiedermi di copiare e incollare tutti quei tanti numeri nel REPL Clojure, dove gli elementi dell'elenco sono separati da spazi bianchi e le virgole sono spazi bianchi, ma si è scoperto che funziona davvero.


2

APL (Dyalog Extended) , 17 byte

Mille grazie a ngn e Adám per il loro aiuto nel giocare a golf con entrambi questi programmi APL in The APL Orchard , un ottimo posto per imparare APL e ottenere aiuto APL.

∊×⍀/⌽{⊂×\1,⍵}⌸⍨⍭⎕

Provalo online!

Ungolfing

∊×⍀/⌽{⊂×\1,⍵}⌸⍨⍭⎕

                  Gets evaluated input from stdin.
                  Gives us a list of the prime factors of our input.
                   Example for 720: 2 2 2 2 3 3 5
     {      }⌸⍨     groups our prime factors by the keys in the left argument,
                   and  passes the prime factors as both arguments,
                   grouping all the identical primes together
                   before running a {} dfn on them
      ⊂×\1,⍵       We append 1 to each group, get a list of powers of each prime,
                   and enclose the groups to remove 0s from uneven rows.
                 This reverses the prime power groups.
 ×⍀/              This multiplies all the powers together into
                   a matrix of the divisors of our input.
                   (Same as ∘.×/ in Dyalog Unicode)
                  And this turns the matrix into 
                   a list of divisors sorted by prime factorization.
                   We print implicitly, and we're done.

APL (Dyalog Unicode) , SBCS da 29 byte

{∊∘.×/⌽{⊂×\1,⍵}⌸⍨¯2÷/∪∧\⍵∨⍳⍵}

Provalo online!

Ungolfing

{∊∘.×/⌽{⊂×\1,⍵}⌸⍨¯2÷/∪∧\⍵∨⍳⍵}

{                           }  A dfn, a function in brackets.
                        ⍵∨⍳⍵   We take the GCD of our input with 
                               all the numbers in range(1, input).
                     ∪∧\       This returns all the unique LCMs of
                               every prefix of our list of GCDs.
                               Example for 72: 1 2 6 12 24 72.
                 ¯2÷/          We divide pairwise (and in reverse)
                               by using a filter window of negative two 2).
                               Example for 72: 2 3 2 2 3, our prime factors.
       {      }⌸⍨               groups our prime factors by the keys in the left argument,
                               and  passes the prime factors as both arguments,
                               grouping all the identical primes together
                               before running a {} dfn on them
           1,⍵                 We append 1 to each group.
        ⊂×\                    Then we get a list of powers of each prime,
                               and enclose the groups to remove 0s from uneven rows.
                              This reverses the prime power groups.
  ∘.×/                         This multiplies all the powers together into 
                               a matrix of the divisors of our input.
                              And this turns the matrix into a list of divisors
                               sorted by prime factorization.
                               We return implicitly, and we're done.

1

J, 32 31 byte

[:(*/@#~>:#:[:i.[:*/>:)&|./2&p:

Afferra gli elenchi di numeri primi ed esponenti dell'intero di input, inverti ciascuno e costruisci i divisori da quello.

uso

   f =: [:(*/@#~>:#:[:i.[:*/>:)&|./2&p:
   f 2
1 2
   f 72
1 2 4 8 3 6 12 24 9 18 36 72
   f 101
1 101

Spiegazione

[:(*/@#~>:#:[:i.[:*/>:)&|./2&p:  Input: n
                           2&p:  Factor n as a list where the first row are the primes
                                 and the second are their exponents
[:                     &|./      Reverse each list
                    >:           Increment each exponent by 1
                [:*/             Reduce it using multiplication
            [:i.                 Construct a range from 0 to that product exclusive
        >:                       The list of each exponent incremented
          #:                     Reduce each number in the previous range as a mixed base
                                 using the incremented exponents
      #~                         For each mixed base value in that range, copy from
                                 the list of primes that many times
   */@                           Reduce the copied primes using multiplication
                                 Return this list of products as the result

1

Rubino, 71 byte

Questa risposta si basa sulla risposta di Python 2 di xnor.

->n{a,=t=[1];(s=t;a+=1;(t=s+t.map{|z|z*a};n/=a)while n%a<1)while n>1;t}

Un'alternativa della stessa lunghezza è:

->n{a,=t=[1];(a+=1;(t+=t.map{|z|z*a};n/=a)while n%a<1)while n>1;t.uniq}

Ungolfing:

def f(num)
  factor = 1
  list = [1]
  while num != 1
    s = list
    factor += 1
    while num % factor == 0
      list = s + list.map{|z| z*factor}
      num /= factor
    end
  end
  return list
end

def g(num)
  factor = 1
  list = [1]
  while num != 1
    factor += 1
    while num % factor == 0
      list += list.map{|z| z*factor}
      num /= factor
    end
  end
  return list.uniq
end

1

Japt , 12 9 byte

â mk ñÔ®×

-3 byte grazie a @Shaggy

Provalo online!


Questo dovrebbe funzionare per 9 byte.
Shaggy,

@Shaggy Oh sì, ho dimenticato che semplici funzioni dovrebbero essere definite in questo modo, anche se l'ho appena suggerito per lol solo ASCII
Quintec,


0

Mathematica, 56 byte

1##&@@@Tuples@Reverse[#^Range[0,#2]&@@@FactorInteger@#]&
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.