Torna indietro   Hardware Upgrade Forum > Software > Programmazione

 Hisense 55U7SE: tuttofare e accessibile, il MiniLED per film, sport e gioco
Hisense 55U7SE: tuttofare e accessibile, il MiniLED per film, sport e gioco
MiniLED di fascia media con local dimming a 192 zone, 144 Hz nativi e audio firmato Devialet. La prova strumentale riscontra colori affidabili e gaming reattivo, per un prodotto molto accessibile e convincente. Ma la soundbar aggiuntiva è quasi d'obbligo
Kindle Scribe Colorsoft: riduce le cornici e diventa a colori, ma il prezzo è alto
Kindle Scribe Colorsoft: riduce le cornici e diventa a colori, ma il prezzo è alto
Amazon porta i colori sul suo Kindle da scrittura più grande: schermo Colorsoft a 11 pollici, processore quad-core, penna premium più reattiva e strumenti IA per le note, sono le note salienti. Il salto di prezzo rispetto al modello in bianco e nero si fa sentire, anche se la percezione è quella di trovarsi di fronte a un prodotto di fascia altissima, per veri appassionati
L'IA cambia tutte le regole della sicurezza tra vulnerabilità e sorveglianza. Intervista al CEO di Proofpoint
L'IA cambia tutte le regole della sicurezza tra vulnerabilità e sorveglianza. Intervista al CEO di Proofpoint
Abbiamo intervistato Sumit Dhawan, CEO di Proofpoint, per capire come stia cambiando il mondo della sicurezza con l'avvento dell'intelligenza artificiale e con il ritmo sempre più serrato a cui vengono trovate vulnerabilità nel software. Un problema significativo, che richiederà del tempo per essere risolto (o quantomeno arginato)
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 07-12-2005, 11:10   #1
shinya
Senior Member
 
L'Avatar di shinya
 
Iscritto dal: Jul 2005
Città: Bologna
Messaggi: 1130
[c++]numeri primi in maniera __efficente__[lungo]

Ciao a tutti! E' un sacco di tempo che sto dietro a questo problema, senza trovare una soluzione funzionale. C'è questo sito, del quale ho postato tempo fa, che raccoglie una serie di problemi in cui ci si può cimentare, mandando poi il programma scritto ad un "tester" che verifica che la soluzione sia corretta e stia nei limiti di tempo.
Questo è un semplice problema di ricerca di numeri primi. Il testo del problema lo potete trovare qui: http://spoj.sphere.pl/problems/PRIME1/

In pratica il programma prende una prima riga in input con il numero n dei casi da valutare e poi n righe con il range in cui calcolare i numeri primi, ad esempio:
Input:
2
1 10
50 100

Il programma deve stampare qualcosa tipo:
2
3
5
...e via di seguito.

Il range è da 1 a 1000000000, e la differenza tra max e min non può essere superiore a 100000. Ma al link di prima trovate tutto il testo del problema descritto in maniera più accurata.

Ok, veniamo alla ciccia. Questo è il miglior tentativo che mi è saltato fuori, ma che ovviamente esce dal tempo massimo:
Codice:
#include <iostream>
#include <vector>

using namespace std;

int main(int argc, char **argv)
{
        unsigned long long min = 0, max = 0, n = 0;

        cin >> n;
        while (n--) {
                cin >> min;
                cin >> max;
                vector<bool> p(max+1);

                for (vector<bool>::size_type i = 0; i < p.size(); i++)
                        p[i] = true;
                p[0] = p[1] = false;

                for (vector<bool>::size_type i = 0; i*i < p.size(); i++)
                        if (true == p[i])
                                for (vector<bool>::size_type j = i+i; j < p.size(); j+=i) 
                                       p[j] = false;

                for (vector<bool>::size_type i = 0; i < p.size(); i++)
                        if (true == p[i] && i > min)
                                cout << i << endl;
        }

        return 0;
}
E' un classico crivello di Eratostene. Spero sia comprensibile. Nel caso non lo fosse ditemelo così aggiungo qualche commento.
Il collo di bottiglia qui è evidente: calcolo sempre tutti i numeri primi da 2 a max. Allora ho pensato:"bene, siccome devo calcolare al massimo i numeri primi fino a 1 miliardo, la cui radice quadrata è < 32000, è sufficiente calcolare i numeri primi fino a 32000 e usare quelli per eliminare tutti i loro multipli nel range richiesto".

Sbaglio?? Voi avreste un metodo migliore?
Il problema è che non sono riuscito ancora a trasformare in codice quest'idea...questo è uno schizzo, nel quale manca la parte in cui devo eliminare i multipli dei numeri primi trovati in precedenza. Non riesco a trovare un modo efficente di farlo

Codice:
#include <iostream>
#include <vector>

using namespace std;

int main(int argc, char **argv)
{
	//trova i numeri primi < 32000
	vector<bool> p(32000);

	for (vector<bool>::size_type i = 0; i < p.size(); i++)
		p[i] = true;
	p[0] = p[1] = false;

	for (vector<bool>::size_type i = 0; i*i < p.size(); i++)
		if (p[i])
			for (vector<bool>::size_type j = i+i; j < p.size(); j+=i)
				p[j] = false;

	for (vector<bool>::size_type i = 0; i < p.size(); i++)
		if (p[i])
			cout << i << endl;

	unsigned long long min = 0, max = 0, n = 0;
	cin >> n;

	while (n--) {
		cin >> min;
		cin >> max;

		//manca codice qui!!!
	}

	return 0;
}
Qualcuno ha qualche suggerimento? Altri metodi??
Grazie e ciao!
shinya è offline   Rispondi citando il messaggio o parte di esso
Old 09-12-2005, 09:17   #2
shinya
Senior Member
 
L'Avatar di shinya
 
Iscritto dal: Jul 2005
Città: Bologna
Messaggi: 1130
up!
shinya è offline   Rispondi citando il messaggio o parte di esso
Old 09-12-2005, 12:39   #3
NA01
Senior Member
 
L'Avatar di NA01
 
Iscritto dal: Jun 2003
Città: Genova
Messaggi: 5676
presupponendo che non ho guardato l'algortimo ( ) ti vorrei dare un paio di dritte.
la prima cosa che ho fatto è stata prendere il tuo codice e togliere tutte le chiamate a p.size() sostituendolo con una variabile che alloca all'inizo.
il tempo di esecuzione senza la stampa è passato da
Quote:
real 0m6.867s
user 0m6.455s
sys 0m0.015s
a

Quote:
real 0m3.819s
user 0m3.595s
sys 0m0.006s
poi ho tolto l'allocazione delle variabili all'interno del for. le ho spostate fuori e ora non dealloca/rialloca le variabili all'interno dei cicli.
mi aspettavo di guadagnare tanto invece siamo quasi a un guadagno nullo.

Quote:
real 0m3.784s
user 0m3.568s
sys 0m0.005s
con le ottimizzazione attive il primo rimane un guadagno sensibile, il secondo che già era nullo scompare quasi del tutto.

a parte questi giochini sugli algoritmi di ricerca dei numeri primi c'era un bel 3d, magari qualcuno si ricorda qualcosa...


ciao!
NA01 è offline   Rispondi citando il messaggio o parte di esso
Old 13-12-2005, 11:24   #4
shinya
Senior Member
 
L'Avatar di shinya
 
Iscritto dal: Jul 2005
Città: Bologna
Messaggi: 1130
Ce l'ho fatta finalmente! Il programma è passato al test.
Se a qualcuno interessa un programmino per calcolare i numeri primi fino a 1 miliardo (e non oltre) in maniera piuttosto efficente, qui c'è il sorgente in C++.

Codice:
#include <iostream>
#include <vector>
#include <cassert>

using namespace std;
typedef unsigned long long bignum;

int main(int argc, char **argv)
{
  // find the first 31626 prime numbers
  // this is because sqrt(1_000_000_000) < 31626
  bignum upto = 31626;
  vector<bool> p(upto);

  for (vector<bool>::size_type i = 0; i < upto; i++)
    p[i] = true;
  p[0] = p[1] = false;

  for (vector<bool>::size_type i = 0; i*i < upto; i++)
    if (p[i])
      for (vector<bool>::size_type j = i+i; j < upto; j+=i)
        p[j] = false;
  // end of prime calculation

  // take n test case in input and compute
  // the prime numbers in the range between
  // min and max
  bignum min = 0, max = 0, n = 0;
  cin >> n;

  while (n--) {
    cin >> min; cin >> max;
    assert(max >= min);

    bignum tmp_size = max - min + 1;
    vector<bignum> tmp(tmp_size);

    // load the vector with the range
    for (vector<bignum>::size_type i = 0; i < tmp_size; i++)
      tmp[i] = min + i;

    // another sieve
    for (vector<bool>::size_type i = 0; i*i < max; i++)
      if (p[i]) {

        // find the first multiple
        vector<bignum>::size_type j = 0;

        // go to the first non-1
        while (tmp[j] == 1)
          j++;

        bignum div = tmp[j] % i;
        if (div != 0)
          j += (i - (div));

        // delete all the other multiple of i in the range
        if (tmp[j] == i) // if tmp[j] is a prime, jump to the next one
          for (vector<bignum>::size_type k = j+i; k < tmp_size; k += i)
            tmp[k] = 1;
        else
          for (vector<bignum>::size_type k = j; k < tmp_size; k += i)
            tmp[k] = 1;
      }

    for (vector<bignum>::size_type i = 0; i < tmp_size; i++)
      if (tmp[i] != 1)
        cout << tmp[i] << "\n";
  }

  return 0;
}
Scusate i commenti in inglese, è un'abitudine. Spero possa servire a qualcuno.
Esempio di utilizzo:
Codice:
shell# ./prime1
1      -> casi di test
1 10  -> range di numeri

2
3
5
7
shinya è offline   Rispondi citando il messaggio o parte di esso
Old 13-12-2005, 12:26   #5
marco.r
Senior Member
 
Iscritto dal: Dec 2005
Città: Istanbul
Messaggi: 1817
del crivello se n'era parlato ampiamente qui:
http://www.hwupgrade.it/forum/showthread.php?t=824152
marco.r è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


 Hisense 55U7SE: tuttofare e accessibile, il MiniLED per film, sport e gioco Hisense 55U7SE: tuttofare e accessibile, il Min...
Kindle Scribe Colorsoft: riduce le cornici e diventa a colori, ma il prezzo è alto Kindle Scribe Colorsoft: riduce le cornici e div...
L'IA cambia tutte le regole della sicurezza tra vulnerabilità e sorveglianza. Intervista al CEO di Proofpoint L'IA cambia tutte le regole della sicurezza tra ...
L'Europa conta nella tecnologia e può essere autonoma. Cosa si è detto al Nextcloud Summit 2026 L'Europa conta nella tecnologia e può ess...
Dreame X60 Pro Ultra Complete: i bracci si estendono sempre di più Dreame X60 Pro Ultra Complete: i bracci si esten...
USA e Cina si combattono su tutto, trann...
UE, entra in vigore il nuovo dazio sui p...
Piccolo fuori, enorme dentro: il nuovo C...
Nothing Phone (4b) si mostra online a un...
TOP 12 offerte Amazon, aggiornata ora: 2...
Nano Banana 2 Lite: immagini in 4 second...
L'AI fa paura anche ad Apple: cambia com...
Switch 2, aggiornamento hardware in arri...
Roborock F25 e F25 ALT in offerta su Ama...
La fabbrica lituana che vuole salvare l'...
Google cambia i backup di Android: ora p...
Alla scoperta di Aiper, robot pulisci pi...
OPPO Reno16, Pro e FS ufficiali: super b...
Saltato l'accordo con BOE: i top di gamm...
Insta360 X4 Air a 299€ invece di 399€: &...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 10:17.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v