Pähkinäkö?

Allerginen ko asialle.

Eli kui voisin toteuttaa int to string(dec) härvelin C;llä jos mulla ei ole käytössä kertoalaskua tai jakolaskua?

Eli siis sellaisesta suolatusta vehkeestä olisi kyse jossa ei C-kirjastossa jako tai kertolaskua ole.

Itse saan toimimaan int to srting(hex) jolloin int- tyyppinen muttuja muutetaan kannasta suoraan 16-kantaiseksi merkkijonoksi.

Eli esim 78 on "4E".

Mutta tarve olisi 10-kantaisille.

Kiitos.

3

353

    Vastaukset

    Anonyymi (Kirjaudu / Rekisteröidy)
    5000
    • ...

      Kerto- ja jakolasku on melko helppoa toteuttaa itse, ei siinä ole muusta kuin lisäyksestä, vertailusta ja bittien oikealle/vasemmalle pyörityksestä.

      Mutta tämän voit tehdä vaikka näin. Tuossa on otettu myös inhottavat rajatapaukset huomioon.

      #include

      unsigned int lukutaulu[] = {
      1000000000, /* miljardit */
      100000000,
      10000000,
      1000000,
      100000,
      10000,
      1000,
      100,
      10,
      1 };

      void tulostaLuku(int syoteluku, char * buf)
      {
      int kohta = 0;
      int i;
      int chr;
      int poistaEtuNollat = 1;
      int maxKohta = sizeof(lukutaulu)/sizeof(int);
      /*
      Rajatapaus: Luku on pienin mahdollinen negatiivinen luku
      Tata ei voi esittaa positiivisena lukuna!
      Siksi luku pitaa muuntaa unsigned muotoon ensiksi.
      */
      unsigned int luku;

      if(syoteluku=lukutaulu[kohta];luku -= lukutaulu[kohta])
      chr ;
      if(!(poistaEtuNollat && chr=='0'))
      {
      poistaEtuNollat = 0;
      *buf = chr;
      }

      kohta ;
      }
      /*
      Jos luku oli 0, tulostetaan edes jotain.
      */
      if(poistaEtuNollat)
      *buf = '0';
      *buf = 0;
      }

      void testaa(int luku)
      {
      char buffer[11];
      tulostaLuku(luku, buffer);
      puts(buffer);
      }

      int main()
      {
      testaa(54);
      testaa(-54);
      testaa(-10000);
      testaa(0);
      testaa(-1);
      testaa(0x7fffffff); /* suurin 32-bittinen luku */
      testaa(0x80000000); /* pienin */
      }

      • tosin aika "kallis"

        Tuolla kalliilla meinaan tilannetta jossa: M on muunnettava luku, L muunnokseen kuluvat loopit.
        M L
        ----------------
        1 1
        9 9
        10 1
        100 1
        200 2
        900 9
        990 18
        ...
        999 27
        ...
        999999999 81

        Muuten hyvä.


      • ...
        tosin aika "kallis" kirjoitti:

        Tuolla kalliilla meinaan tilannetta jossa: M on muunnettava luku, L muunnokseen kuluvat loopit.
        M L
        ----------------
        1 1
        9 9
        10 1
        100 1
        200 2
        900 9
        990 18
        ...
        999 27
        ...
        999999999 81

        Muuten hyvä.

        Se sisin looppi on ihan vähennyslaskua rekistereissä (toivottavasti). Jakolasku, toinen vaihtoehto, ei paljoa auta, koska looppikierrosten määrä moninkertaistuisi, vaikka jaetaan vain 10:llä.

        Binaarihakukin (ensiksi vähennettäisiin esim. 80000, sitten 40000, 20000, 10000, jne. veisi vähintään saman ajan ylimääräisen logiikan vuoksi.

        Pelkkään 32-bittiseen kertolaskuun menee pahimmillaan 32 loopin kierrosta. Jakolaskussa menee jaettavan ja jakajan ekan ykkösbitin hakuun jo kohtuullinen määrä looppeja. Naivisti jakajaa vasemmalle shiftailussakin kestää. Binaarihakujakolasku taas vaatisi 48-bittisen rekisterin toteuttamisen. Eli ≤≤16 -≥ ≤≤ 24/ ≤≤ 8, jne. Tai sitten pitäisi rajoittaa jakaja 16 bittiin. 14 bittiset luvut voi tietysti jakaa 10:llä nopsasti vaikka "tmp = (x≤≤3) (x≤≤2); tmp = (tmp≤≤8) (tmp≤≤4) x; x = tmp≥≥15;", mutta mitäpä tuokaan auttaa ilman nopeaa tapaa laskea 10 jakojäännös... Tietysti voisi kertoa 10 (x≤≤3) x x ja vähentää sen jaettavasta... ja mikrokontrollerin aikaa palaa.

        Onko siinä prossussa muuten 32 shift left/right immediate argumentilla? Eli x≤≤7, vai tekeekö se 7 kertaa x≤≤1. En tietysti tarkoita x≤≤y.

        Jos nopeus on tärkeää, cacheta vaikka 10 viimeisintä lukua. Koodaa konekielellä. Huijaa. Esim. laskurin saa nopeiten manipuloimalla itse numeron ASCII-merkkijonoa! Käsittele numeroita konekielellä *vain* BCD-aritmetiikalla (melkein kaikki taskulaskimet tekevät näin).

        #include ≤stdio.h≥
        #include ≤stdlib.h≥
        #include ≤stdint.h≥

        char * tulostaLuku(uint32_t x, char * buf)
        {
        …uint32_t jaettu10;
        …uint32_t jakojaannos;
        …*buf--=0;
        …if(x)
        …while(x≥0)
        …{
        ……/* 32-bit: toimii alle 1301720 */
        ……/* 64-bit: toimii alle 4194309 */
        ……jaettu10 = (x≤≤2) (x≤≤3);
        ……jaettu10 = (jaettu10≤≤8) (jaettu10≤≤4) jaettu10 (((jaettu10≤≤4) x jaettu10)≥≥8);
        ……jaettu10 ≥≥= 15;

        ……/* 32-bit: alle 81920
        ……… 64-bit: alle 262149
        ……*/
        /*
        ……jaettu10 = (x≤≤2) (x≤≤3);
        ……jaettu10 = (jaettu10≤≤4) jaettu10 (((jaettu10≤≤4) jaettu10 x)≥≥8);
        ……jaettu10 ≥≥= 11;
        */
        ……/* 32/64-bit: 16389 */
        /* jaettu10 = (x≤≤2) (x≤≤3);
        ……jaettu10 = (jaettu10≤≤8) (jaettu10≤≤4) x;
        ……jaettu10 ≥≥= 15;
        */
        ……jakojaannos = x - jaettu10 -jaettu10 - (jaettu10≤≤3);
        ……*buf-- = '0' jakojaannos;
        ……x=jaettu10;
        …}
        …else
        …{
        ……*buf='0';
        ……return buf;
        …}
        …return buf;
        }

        int main()
        {
        …char buf[32];
        …char checkbuf[32];
        …uint64_t i;
        …char *buffer = buf 16;
        …char *retbuf;
        …for(i=0; i≤300000000; i )
        …{
        ……sprintf(checkbuf, "%lu", i); //itoa(i, checkbuf);
        ……retbuf = tulostaLuku(i, buf);
        ……if(strcmp(checkbuf, retbuf))
        ……{
        ………printf("virhe %lu (%s)\n", i, retbuf);
        ………break;
        ……}
        …}
        }


    Ketjusta on poistettu 0 sääntöjenvastaista viestiä.

    Luetuimmat keskustelut

    1. SDP jo 100 % suositumpi kuin persut

      Kertoo Hesarin uusin kannatuskysely. Demareiden kannatus on miltei tuplat verrattuna persuihinl. Suomen kansa ei selväst
      Maailman menoa
      23
      4882
    2. Voiko normaali ihminen ryhtyä vasemmistolaiseksi?

      Tätä jäin pohdiskelemaan.
      Maailman menoa
      214
      4368
    3. SDP haluaa 40 000 nettomaahanmuuttajaa

      SDP:n Suunnanmuutos-vaihtoehtobudjetissa, käy ilmi, että demarit itse asiassa vaativat räjähdysmäistä ”työperäisen” maah
      Maailman menoa
      147
      3837
    4. Orpo: Velkajarrua vastustavaa puoluetta vaikea ajatella hallitukseen

      No Minja Koskelan kommunistipuolue jäi ulos tuosta. Kaikki eduskuntapuolueet vasemmistoliittoa lukuun ottamatta sopivat
      Maailman menoa
      141
      3308
    5. Mikä tämä henkilö mahtaa touhuta Parkanossa

      Kamalaa https://www.ylasatakunta.fi/teksti/pirkanmaan-karajaoikeus-vangitsi-koiran-tappamisesta-epaillyn-6.68.127794.b58
      Parkano
      36
      1828
    6. Hienoa! Eduskunta luopui käteisen käytöstä

      Nyt tuo sama muutos pitää saada myös muuhun yhteiskuntaan. Käteistähän ei tarvitse tänä päivänä enää kuin rikolliset.
      Maailman menoa
      57
      1668
    7. Ikävä sinua mies

      Vuosia kuluu, mutta tunteet ei ole hävinnyt. Tasoittuneet toki, kun ei olla nähty. Järki palannut päähän kuitenkin. Se i
      Ikävä
      19
      1668
    8. Sulla on avaimet ja keinot

      Jos haluat jatkaa tutustumista. Itse olen niin jäässä etten pysty tekemään enää mitään. Pidempi keppi johon on helpompi
      Ikävä
      25
      1415
    9. Orpo loukkaantui fasismiin viittaavasta sanavalinnasta

      Mutta miksi loukkaantui? Orpohan on tehnyt yhteistyötä fasistien kanssa jo vuonna 2019, siis jo neljä vuotta ennen loukk
      Maailman menoa
      27
      1368
    10. Kiinnostaa - ei kiinnosta - kiinnostaapas

      Selittäkää hämmentyneelle miksi miehiä ei ikinä kiinnosta silloin, kun sitä olisi itsekin kiinnostunut? Sitten kun siirt
      Sinkut
      127
      1238
    Aihe