Maison > Java > javaDidacticiel > Comment puis-je déterminer rapidement si un grand entier est un carré parfait à l'aide d'opérations au niveau du bit ?

Comment puis-je déterminer rapidement si un grand entier est un carré parfait à l'aide d'opérations au niveau du bit ?

DDD
Libérer: 2025-01-01 07:23:10
original
908 Les gens l'ont consulté

How Can I Quickly Determine if a Large Integer is a Perfect Square Using Bitwise Operations?

Bien que l'algorithme soit environ 35 % plus rapide que le code que vous avez donné, les résultats réels peuvent varier selon les différents processeurs (x86) et langages de programmation (C/C). La méthode de cet article est divisée en trois parties :

  1. Filtrer les réponses évidentes : inclure les nombres négatifs, vérifier les 4 derniers chiffres (il a été constaté que vérifier les 6 derniers chiffres n'est pas utile), répondez 0. (Lors de la lecture du code suivant, veuillez noter que mon entrée est un int64. Le produit de deux nombres premiers différents, donc le carré modulo 255 n'a qu'un reste d'environ 1/8. Cependant, d'après mon expérience, les coûts d'utilisation de l'opérateur modulo (%) dépassent les avantages, j'ai donc utilisé une petite astuce impliquant 255 pour calculer le reste. (Mieux ou pire, je n'ai pas utilisé l'astuce consistant à lire les octets individuels du mot, juste ET au niveau du bit et en décalant.)

    if( x < 0 || (x&2) || ((x & 7) == 5) || ((x & 11) == 8) )
        return false;
    if( x == 0 )
        return true;
    Copier après la connexion
  2. J'ai utilisé une table précalculée pour vérifier si le reste est un nombre carré. .
  3. int64 y = x;
    y = (y & 4294967295LL) + (y >> 32); 
    y = (y & 65535) + (y >> 16);
    y = (y & 255) + ((y >> 8) & 255) + (y >> 16);
    // At this point, y is between 0 and 511.  More code can reduce it farther.
    Copier après la connexion
    Essayer de calculer la racine carrée en utilisant une méthode similaire au lemme de Hensel

     : Avant cela, j'utilisais deux La recherche divise tous les restes élevés par des puissances de 2 :

    if( bad255[y] )
        return false;
    // However, I just use a table of size 512
    Copier après la connexion
  4. À ce stade, pour que notre nombre soit un nombre carré, son module doit être 1 sur 8.
  5. La structure de base du lemme de Hensel est la suivante. (Remarque : code non testé ; si cela ne fonctionne pas, essayez t=2 ou 8.)

    if((x & 4294967295LL) == 0)
        x >>= 32;
    if((x & 65535) == 0)
        x >>= 16;
    if((x & 255) == 0)
        x >>= 8;
    if((x & 15) == 0)
        x >>= 4;
    if((x & 3) == 0)
        x >>= 2;
    Copier après la connexion
    L'idée est qu'à chaque itération, vous ajoutez un bit à r," La racine carrée de (Notez que si r est la racine carrée de la puissance de .) Puisque notre racine carrée réelle est inférieure à 2 ^ 32, nous pouvons alors vérifier si r ou t/2-r est la racine carrée réelle de x. Dans mon code actuel, j'ai utilisé la boucle modifiée suivante :

    if((x & 7) != 1)
        return false;
    Copier après la connexion
    Le gain de vitesse ici peut être obtenu de trois manières : valeur de départ précalculée (équivalente à environ 10 itérations de boucle), quitter la boucle plus tôt et ignorez certaines valeurs t. Pour la dernière partie, j'observe z=r-x*x et j'utilise des astuces pour régler t à la plus grande puissance de 2 divisée par z. Cela me permet d'ignorer les valeurs t qui n'affectent pas la valeur r de toute façon. Ma valeur de départ précalculée a sélectionné la racine carrée "la moins positive" modulo 8192 dans mon cas.

    int64 t = 4, r = 1;
    t <<= 1; r += ((x - r * r) & t) >> 1;
    t <<= 1; r += ((x - r * r) & t) >> 1;
    t <<= 1; r += ((x - r * r) & t) >> 1;
    // Repeat until t is 2^33 or so.  Use a loop if you want.
    Copier après la connexion

    Même si ce code ne fonctionne pas pour vous plus rapidement, j'espère que vous apprécierez certaines des idées. Le code de test complet est le suivant, y compris les tableaux précalculés.

    int64 r, t, z;
    r = start[(x >> 3) & 1023];
    do {
        z = x - r * r;
        if( z == 0 )
            return true;
        if( z < 0 )
            return false;
        t = z & (-z);
        r += (z & t) >> 1;
        if( r > (t >> 1) )
            r = t - r;
    } while( t <= (1LL << 33) );
    Copier après la connexion

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

source:php.cn
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal