Conteo de semiprime sin cuadrados


8

Definición

Un semiprime sin cuadrados es un número natural que es el producto de dos números primos distintos.

La tarea

Dado un número natural n, cuente todas las semiprimes sin cuadrados menores o iguales que n.

Detalles

Escriba una función o procedimiento que acepte un único parámetro entero y cuente todos los semiprimes sin cuadrados menores o iguales a su parámetro. El recuento debe ser un valor de retorno de una llamada de función o imprimirse en STDOUT.

Puntuación

La respuesta con el menor número de caracteres gana.

En caso de empate, se utilizarán los siguientes criterios en orden:

  1. Persona más alta

  2. Mejor complejidad temporal

  3. La peor complejidad espacial

Ejemplos

f(1)     = 0
f(62)    = 18
f(420)   = 124
f(10000) = 2600


Ups, lo siento, pero no, esa secuencia no es del todo correcta debido a la restricción de congruencia (por ejemplo, 35 = 5 * 7 y 55 = 5 * 11 no están incluidos). Agregaré algunas soluciones de ejemplo a este problema en particular momentáneamente.
ardnew


¿Qué sucede si un idioma no tiene STDOUT (como javascript)? Uso console.log?
Inkbug

@Inkbug ¿javascript no es capaz de devolver un valor de una función?
nuevo el

Respuestas:


7

J, 50 40 38 37 caracteres

f=:3 :'+/y<:}.~.,(~:/**/)~p:i._1&p:y'

Uso:

   f 1
0
   f 62
18
   f 420
124
   f 10000
2600

Gracias a FUZxxl .

Prueba de rendimiento

   showtotal_jpm_ ''[f 1[start_jpm_ ''
 Time (seconds)
┌───────┬──────┬────────┬────────┬─────┬────┬───┐
│name   │locale│all     │here    │here%│cum%│rep│
├───────┼──────┼────────┼────────┼─────┼────┼───┤
│f      │base  │0.000046│0.000046│100.0│100 │1  │
│[total]│      │        │0.000046│100.0│100 │   │
└───────┴──────┴────────┴────────┴─────┴────┴───┘
   showtotal_jpm_ ''[f 1[f 62[start_jpm_ ''
 Time (seconds)
┌───────┬──────┬────────┬────────┬─────┬────┬───┐
│name   │locale│all     │here    │here%│cum%│rep│
├───────┼──────┼────────┼────────┼─────┼────┼───┤
│f      │base  │0.000095│0.000095│100.0│100 │2  │
│[total]│      │        │0.000095│100.0│100 │   │
└───────┴──────┴────────┴────────┴─────┴────┴───┘
   showtotal_jpm_ ''[f 1[f 62[f 420[start_jpm_ ''
 Time (seconds)
┌───────┬──────┬────────┬────────┬─────┬────┬───┐
│name   │locale│all     │here    │here%│cum%│rep│
├───────┼──────┼────────┼────────┼─────┼────┼───┤
│f      │base  │0.000383│0.000383│100.0│100 │3  │
│[total]│      │        │0.000383│100.0│100 │   │
└───────┴──────┴────────┴────────┴─────┴────┴───┘
   showtotal_jpm_ ''[f 1[f 62[f 420[f 10000[start_jpm_ ''
 Time (seconds)
┌───────┬──────┬────────┬────────┬─────┬────┬───┐
│name   │locale│all     │here    │here%│cum%│rep│
├───────┼──────┼────────┼────────┼─────┼────┼───┤
│f      │base  │0.084847│0.084847│100.0│100 │4  │
│[total]│      │        │0.084847│100.0│100 │   │
└───────┴──────┴────────┴────────┴─────┴────┴───┘
   showtotal_jpm_ ''[f 1[f 62[f 420[f 10000[f 50000[start_jpm_ ''
 Time (seconds)
┌───────┬──────┬────────┬────────┬─────┬────┬───┐
│name   │locale│all     │here    │here%│cum%│rep│
├───────┼──────┼────────┼────────┼─────┼────┼───┤
│f      │base  │5.014691│5.014691│100.0│100 │5  │
│[total]│      │        │5.014691│100.0│100 │   │
└───────┴──────┴────────┴────────┴─────┴────┴───┘

No soy un teórico como se ha visto aquí en el pasado, pero creo que la complejidad del tiempo es algo así como O (n p 2 ) donde n p es el número de primos hasta e incluyendo el número de entrada n. Esto se basa en la suposición de que la complejidad de mi método (generar una tabla de multiplicación muy grande) supera con creces la complejidad de la función generadora principal incorporada en J.

Explicación

f=:3 :'...'declara un verbo (función) monádico. La entrada al verbo está representada por ydentro de la definición del verbo.

p:i._1&p:yEl p:verbo es los números primos de usos múltiples verbo, y se utiliza de dos maneras diferentes aquí: _1&p:ydevuelve el número de primos menos de yentonces p:i.genera cada uno de ellos. Usando 10 como entrada:

   p:i._1&p:10
2 3 5 7

(~:/**/)~genera la tabla de la que hablé anteriormente. */genera una tabla de multiplicación, ~:/genera una tabla no igual (para eliminar los cuadrados) y ambos se multiplican juntos. Usando nuestra salida anterior como entrada:

   */~2 3 5 7
 4  6 10 14
 6  9 15 21
10 15 25 35
14 21 35 49

   ~:/~2 3 5 7
0 1 1 1
1 0 1 1
1 1 0 1
1 1 1 0

   (~:/**/)~2 3 5 7
 0  6 10 14
 6  0 15 21
10 15  0 35
14 21 35  0

}.~.,ahora convertimos los números en una lista, ,obtenemos los valores únicos ~.y eliminamos el 0 al comienzo}.

   }.~.,(~:/**/)~2 3 5 7
6 10 14 15 21 35

y<: Una comparación con la entrada original para verificar qué valores son válidos:

   10<:6 10 14 15 21 35
1 1 0 0 0 0

+/ y luego sume eso para obtener la respuesta.

   +/1 1 0 0 0 0
2

¿Tiene una versión falsa de este programa (falso como lo opuesto a tácito)? 13 no siempre da el código tácito más eficiente.
FUZxxl

No, no usé 13 en este caso, aunque creo que probablemente hice lo que hubiera hecho si lo hubiera intentado. El código es básicamente: +/-.x<}.~.,(~:/~*[*/])p:i._1&p:[x=.ndonde n es la entrada.
Gareth el

1
¿Por qué no solo f=:3 :'+/-.y<}.~.,(~:/~*[*/])p:i._1&p:y'por 40 caracteres?
FUZxxl

Gracias, nunca consideré usar3 :'...'
Gareth el

¿Publicaría algunos resultados de tiempo para que podamos juzgar la eficiencia del programa?
DavidC

5

Mathematica 65 64 55 51 47 39

Código

Lo siguiente cuenta el número de semiprimes sin cuadrados menores o iguales a n:

FactorInteger@Range@n~Count~{a={_,1},a}

Cualquier factor semiprime sin cuadrados en una estructura de la forma: {{p,1}{q,1}} por ejemplo,

FactorInteger@221
(* out *)
{{13, 1},{17, 1}}

La rutina simplemente cuenta los números en el rango deseado que tienen esta estructura de factores.


Uso

n=62;
FactorInteger@Range@n~Count~{a={_,1},a}

(* out *)
18

Tiempo: todos los ejemplos dados

FactorInteger@Range@#~Count~{a = {_, 1}, a} & /@ {1, 62, 420, 10^4} // Timing

(* out *)
{0.038278, {0, 18, 124, 2600}}

Tiempo: n = 10 ^ 6

Se necesitan menos de cuatro segundos para contar el número de semi-primos sin cuadrados menores o iguales a un millón.

n=10^6;
FactorInteger@Range@n~Count~{a = {_, 1}, a}//Timing
(* out *)
{3.65167, 209867}

Fantástica solución concisa
nuevo

@ardnew Gracias. Disfruté el desafío.
DavidC

¡Agradable! Pregunta: ¿esos caracteres espaciales alrededor del =y después del ,realmente son sintácticamente necesarios?
Todd Lehman

@ToddLehman, tienes razón. Me los quité. (No se habían contado, por lo que el recuento de bytes permanece igual.)
DavidC

4

Python, 115

r=range
p=lambda x:all(x%i for i in r(2,x))
f=lambda x:sum([i*j<=x and p(j)and p(i)for i in r(2,x)for j in r(2,i)])

f=lambda x:sum([(i*j<=x)&p(j)&p(i)for i in r(2,x)for j in r(2,i)])Guarda 5 caracteres.
beary605

@ beary605: Gracias, pero creo que llevará demasiado tiempo sin cortocircuitos.
grc

votandote. demasiados pensamientos sobre itertoolsen mi cabeza.
Ev_genus

4

Jalea , 7 bytes

ŒcfÆf€L

Pruébalo en línea!

Cómo funciona

ŒcfÆf€L  Main link. Argument: n

Œc       Generate all 2-combinations of [1, ..., n], i.e., all pairs [a, b] such
         that 1 ≤ a < b ≤ n.
   Æf€   Compute the prime factorization of each k in [1, ..., n].
  f      Filter; keep only results that appear to the left and to the right.
      L  Take the length.

Wow, hiciste que mi intento pareciera vergonzoso. Gracias por las ideas!
Harry

3

Pitón (139)

from itertools import*;s=lambda n:sum(x*y<=n and x<y for x,y in product(filter(lambda x:all(x%i for i in range(2,x)),range(2,n)),repeat=2))

Proporcione algunos resultados de muestra para que los competidores puedan probar sus programas.


mira, ¡ni siquiera necesitabas los ejemplos! : ^)
ardnew


2

Python 139

def f(n):
 p=[];c=0
 for i in range(2,n+1):
    if all(i%x for x in p):p+=[i]
    c+=any((0,j)[i/j<j]for j in p if i%j==0 and i/j in p)
 return c

2

Golfscript 64

~:ß,{:§,{)§\%!},,2=},0+:©{©{1$}%\;2/}%{+}*{..~=\~*ß>+\0?)+!},,2/

Demostración en línea aquí

Nota: En la demostración anterior, excluí los casos de prueba 420y 10000. Debido a la prueba de primalidad extremadamente ineficiente, no es posible hacer que el programa se ejecute para estas entradas en menos de 5 segundos.


2

Shell, 40

#! / bin / sh

seq $ 1 | factor | awk 'NF == 3 && $ 2! = $ 3' | wc -l

# viejo, 61
#seq $ 1 | factor | awk 'BEGIN {a = 0} NF == 3 && $ 2! = $ 3 {a ++} END {print a}'

Uso:

$ ./count 1
0 0
$ ./count 420
124
$ ./count 10000
2600
$ time ./cnt.sh 1000000
209867

0m23.956s reales
usuario 0m23.601s
sys 0m0.404s

2

Jalea , 14 13 bytes

RÆEḟ0⁼1,1$Ɗ€S

Pruébalo en línea!

RÆEḟ0⁼1,1$Ɗ€S    main function:
RÆE             get the prime factorization Exponents on each of the Range from 1 to N,
          Ɗ€    apply the preceding 1-arg function composition (3 funcs in a row) to each of the prime factorizations:
                (the function is a monadic semiprime checker, as per DavidC's algorithm)
    ḟ0          check if the factors minus the zero exponents...
      ⁼1,1$      ...are equal to the list [1,1]
             S   take the Sum of those results, or number of successes!

La crítica constructiva bienvenida!


2
Esta combinación de y Sse puede convertir en un uso de ċ(Count). Puede obtener hasta 10 bytes usándolo. ¡Te dejaré resolverlo!
Lynn

2

Python 2/3 , 95 94 bytes

lambda n:sum(map(F,range(n+1)))
F=lambda x,v=2:sum(x%i<1and(F(i,0)or 3)for i in range(2,x))==v

Pruébalo en línea!

Publicado en un desafío de 6 años porque establece un nuevo récord de Python, un IMO es un enfoque bastante interesante.

Explicación

lambda n:sum(map(F,range(n+1)))           # Main function, maps `F` ("is it a semiprime?")
                                          #  over the range [0, n]
F=lambda x,v=2:                           # Helper function; "Does `x` factor into `v`
                                          #  distinct prime numbers smaller than itself?"
  sum(                                    # Sum over all numbers `i` smaller than `x`
    x%i<1                                 # If `i` divides `x`,
    and                                   #  then
    (F(i,0)                               #  add 1 if `i` is prime (note that `F(i,0)`
                                          #  is just a primality test for `i`!)
    or 3)                                 #  or `3` if `i` is not prime (to make `F`
                                          #  return `False`)
  for i in range(2,x))
  ==v                                     # Check if there were exactly `v` distinct prime
                                          #  factors smaller than `x`, each with
                                          #  multiplicity 1

Python 2/3 (PyPy) , 88 82 81 bytes

lambda n:sum(sum(x%i<1and(x/i%i>0or 9)for i in range(2,x))==2for x in range(n+1))

Pruébalo en línea!

Basado en un golf de 92 bytes de Value Ink. Se necesita PyPy para interpretar correctamente 0or, porque Python estándar ve esto como un intento de un número octal.



1

Stax , 8 bytes

ߺ@N¬Që↔

Ejecutar y depurarlo

Desempaquetado, sin golf y comentado, se ve así.

F       for 1..n run the rest of the program
  :F    get distinct prime factors
  2(    trim/pad factor list to length 2
  :*    compute product of list
  _/    integer divide by initial number (yielding 0 or 1)
  +     add to running total

Ejecute este




0

Retina , 58 bytes

_
¶_$`
%(`$
$"
,,`_(?=(__+)¶\1+$)
¶1$'
)C`1(?!(__+)\1+¶)
2

Pruébalo en línea!

Toma como entrada unaria con _marca de conteo

Explicación

Un número es un semi-primo sin cuadrado si su factor más grande y más pequeño, excluyéndose a sí mismo y 1, son ambos primos.

_
¶_$`

Toma la entrada y genera cada número unario menor o igual a él, cada uno en su propia línea

%(`

Entonces, para cada número ...

$
$"
,,`_(?=(__+)¶\1+$)
¶1$'

Encuentra su factor más pequeño y más grande, excluyéndose un 1 ...

)C`1(?!(__+)\1+¶)

y cuenta el número de ellos que es primo. Como el factor más pequeño debe ser primo, esto devuelve 1 o 2

2

Cuenta el número total de 2



0

Ruby -rprime , 64 bytes

Sé que hay otra solución de Ruby aquí, pero no quería incluirla en los comentarios ya que fue respondida en 2012 ... y resulta que usar un indicador de programa lo cuenta como un idioma diferente , así que supongo que esto técnicamente no es "Ruby" de todos modos.

Pruébalo en línea!

Explicación

->n{(1..n).count{|i|m=i.prime_division;m.size|m.sum(&:last)==2}}
->n{                                    # Anonymous lambda
    (1..n).count{|i|                    # Count all from 1 to `n` that match
                                        # the following condition
                m=i.prime_division;     # Get the prime factors of `i` as
                                        #  base-exponent pairs (e.g. [2,3])
                m.size                  # Size of factors (# of distinct primes)
                      |                 # bit-or with...
                       m.sum(&:last)    # Sum of the last elements in the pairs
                                        #  (the sum of the exponents)
                                    ==2 # Check if that equals 2 and return.
                                        # Because 2 is 0b10, the bit-or means
                                        #  that the condition is true iff both
                                        #  are either 2 or 0, but because this
                                        #  is a prime factorization, it is
                                        #  impossible to have the number of
                                        #  distinct primes or the sum of the
                                        #  exponents to equal 0 for any number
                                        #  > 1. (And for 1, size|sum == 0.)
    }                                   # End count block
}                                       # End lambda

0

APL (NARS), caracteres 26, bytes 52

{≢b/⍨{(⍵≡∪⍵)∧2=≢⍵}¨b←π¨⍳⍵}

prueba:

  f←{≢b/⍨{(⍵≡∪⍵)∧2=≢⍵}¨b←π¨⍳⍵}
  f 1
0
  f 9
1
  f 62
18
  f 420
124
  f 1000
288
  f 10000
2600
  f 100000
23313

esta es una alternativa más larga (59 caracteres que preferiría)

r←h w;n;t
n←4⋄r←0
n+←1⋄→0×⍳w<n⋄→2×⍳(2≠≢t)∨2≠≢∪t←πn⋄r+←1⋄→2

prueba:

  h 1000000
209867
Al usar nuestro sitio, usted reconoce que ha leído y comprende nuestra Política de Cookies y Política de Privacidad.
Licensed under cc by-sa 3.0 with attribution required.