Konwersja liczb całkowitych na angielskie słowa

21

Celem tego kodu golfa jest konwersja liczb całkowitych na angielskie słowa.

Program monituje o wprowadzenie danych. Jeśli to wejście nie jest liczbą całkowitą, wydrukuj NaN. Jeśli jest to liczba całkowita, przekonwertuj ją na angielskie słowa i wydrukuj te słowa. Minimalne wejście: 0 (zero). Maksymalny wkład: 9000 (dziewięć tysięcy).
Tak więc 5zwraca five( wielkość liter nie ma znaczenia) i 500zwraca five hundredlub five-hundred(myślniki nie mają znaczenia).

Kilka innych zasad:

A oneprzed hundredlub thousandjest opcjonalne: one hundredjest poprawne, ale hundredteż (jeśli dane wejściowe są 100oczywiście).

Słowo andna przykład one hundred and forty fivejest również opcjonalne.

Białe znaki mają znaczenie. Tak więc, na 500, five-hundredczy five hundredjest poprawny, ale fivehundrednie jest.

Powodzenia!

ProgramFOX
źródło
Tutaj jest odpowiedź bez golfa rgagnon.com/javadetails/java-0426.html .
Ta odpowiedź w SO robi podobne rzeczy, ale nie jest golfem.
ST3

Odpowiedzi:

7

Perl 281 bajtów

print+0eq($_=<>)?Zero:"@{[((@0=($z,One,Two,Three,Four,Five,@2=(Six,Seven),
Eight,Nine,Ten,Eleven,Twelve,map$_.teen,Thir,Four,@1=(Fif,@2,Eigh,Nine)))
[$_/1e3],Thousand)x($_>999),($0[($_%=1e3)/100],Hundred)x($_>99),
($_%=100)>19?((Twen,Thir,For,@1)[$_/10-2].ty,$0[$_%10]):$0[$_]]}"||NaN

Dodano nowe linie dla poziomego zdrowia psychicznego. Powyższe można wykorzystać interaktywnie lub poprzez przesłanie wartości przez stdin.

Działa poprawnie dla wszystkich wartości całkowitych w zakresie [0, 19999] , wartości poza tym zakresem wykazują niezdefiniowane zachowanie. Wartości niecałkowite zostaną obcięte w kierunku zera i jako takie będą zgłaszane tylko wartości, które są naprawdę nienumeryczne NaN.

Przykładowe użycie:

for $n (14, 42, 762, 2000, 6012, 19791, 1e9, foobar, 17.2, -3) {
  print "$n: ", `echo $n | perl spoken-numbers.pl`, $/;
}

Przykładowe dane wyjściowe:

14: Fourteen
42: Forty Two
762: Seven Hundred Sixty Two
2000: Two Thousand 
6012: Six Thousand Twelve
19791: Nineteen Thousand Seven Hundred Ninety One
1000000000: Thousand 
foobar: NaN
17.2: Seventeen
-3: Nine Hundred Ninety Seven
primo
źródło
„1000000000: tysiąc”? I czy 17.2 nie powinno wypisywać „NaN”?
DavidC,
5
@DavidCarraher „... wartości poza tym zakresem wykazują niezdefiniowane zachowanie . Wartości niecałkowite zostaną obcięte w kierunku zera i jako takie zgłaszane będą tylko wartości, które są naprawdę nienumeryczne NaN.”
primo,
Nie jestem ekspertem od Perla, więc zadaję to pytanie: czy ten program prosi o podanie danych?
ProgramFOX,
@ProgramFOX Zaktualizowałem go, aby odczytał wartość ze standardowego wejścia (jeśli zostanie uruchomiony interaktywnie, wyświetli monit o podanie wartości), a nie jako funkcję.
primo
13

JavaScript (375)

Prawdopodobnie okropna próba, ale tak czy inaczej ...

alert(function N(s,z){return O="zero,one,two,three,four,five,six,seven,eight,nine,ten,eleven,twelve,thir,,fif,,,eigh,,,,twen,,for".split(","),(z?O[s]||O[s-10]||O[s-20]:s<13?N(s,1):s<20?N(s,1)+"teen":s<100?N(a=20+(s/10|0),1)+"ty"+(s%10?" "+N(s%10):""):s<1e3?N(s/100|0)+" hundred"+(s%100?" "+N(s%100):""):s<1e5?N(s/1e3|0)+" thousand"+(s%1e3?" "+N(s%1e3):""):0)||NaN}(prompt()))

Całkiem drukowane (jako funkcja):

function N(s,z) {
  return O = "zero,one,two,three,four,five,six,seven,eight,nine,ten,eleven,twelve,thir,,fif,,,eigh,,,,twen,,for".split(","),
      (z? O[s] || O[s-10] || O[s-20]
       : s < 13?  N(s,1)
       : s < 20?  N(s,1) + "teen"
       : s < 100? N(a=20+(s/10|0),1) + "ty" + (s%10?" "+N(s%10):"")
       : s < 1e3?  N(s/100|0) +  " hundred" + (s%100?" "+N(s%100):"")
       : s < 1e5?  N(s/1e3|0) + " thousand" + (s%1e3?" "+N(s%1e3):"") : 0) || NaN
}

Przykładowa konwersja (pamiętaj, że wyprowadza nawet, NaNgdy jest poza zakresem, tj. Nieprawidłowe dane wejściowe):

540: five hundred forty
4711: four thousand seven hundred eleven
7382: seven thousand three hundred eighty two
1992: one thousand nine hundred ninety two
hutenosa: NaN
1000000000: NaN
-3: NaN
Robaczek świętojański
źródło
+1dość trudno jest zrobić to lepiej w języku takim jak javascript. (możesz usunąć miejsce w N(s,z) {returncelu zaoszczędzenia 1 znaku)
Chłodnica do matematyki
Och, haha, musiałem tęsknić. Wydaje mi się również, że przegapiłem kilka znaków w Ołańcuchu. Naprawię to ...
FireFly,
11

Mathematica 60 57

f = ToString@#~WolframAlpha~{{"NumberName", 1}, "Plaintext"} &

Stosowanie:

f[500]

pięćset

Edytować:

InputString[]~WolframAlpha~{{"NumberName", 1}, "Plaintext"}
alephalpha
źródło
3
To tak naprawdę nie odpowiada na pytanie. Powiedziałem, że użytkownik musi wprowadzić liczbę (na przykład za pomocą wiersza polecenia lub wiersza polecenia), a następnie program powinien wypisać słowa (na przykład w wierszu polecenia lub w oknie komunikatu). Twój kod jest tylko funkcją do jego konwersji, a twój program nie prosi o dane wejściowe.
ProgramFOX,
@ProgramFOX mówi „Użytkownik coś wprowadza”. To nie znaczy „Program monituje o wprowadzenie danych”.
MrZander
@MrZander: Cóż, „Program pyta o dane wejściowe”, tak naprawdę miałem na myśli. Aktualizuję moje pytanie, ale oczywiście byłoby niesprawiedliwe, gdybym nie głosował za odpowiedzią Alephalpha, więc dostał moją +1
ProgramFOX
8

Lisp, 72 56 znaków

Zdaję sobie sprawę 1), że jest stary, i 2) że działa całkowicie w oparciu o standardową bibliotekę, ale fakt, że możesz uzyskać system drukowania c-lisp do tego rodzaju rzeczy, zawsze zachwycał mnie. Ponadto w rzeczywistości pobiera dane wejściowe od użytkownika, konwertuje je i drukuje.

(format t "~:[NaN~;~:*~r~]" (parse-integer (read-line) :junk-allowed t))

Zawiera 72 znaki.

  • :junk-allowed powoduje, że parsowanie liczb całkowitych zwraca zero w przypadku niepowodzenia zamiast zgłaszania błędu.
  • ~:[if-nil~;if-non-nill] warunkowy przewidywany na zero, w razie potrzeby obsługuje NaN
  • ~:* tworzy kopię zapasową interpretacji argumentów, aby ponownie wykorzystać dane wejściowe
  • ~r wypisuje liczbę jako ciąg słów w języku angielskim, zgodnie z żądaniem, z wyjątkiem pełnej poprawionej interpunkcji

Próba:

17823658
seventeen million, eight hundred and twenty-three thousand, six hundred and fifty-eight

192hqfwoelkqhwef9812ho1289hg18hoif3h1o98g3hgq
NaN

Informacje Lisp głównie z Practical Common Lisp .

Edytuj, poprawnie zagrał w golfa do 56 znaków

(format t "~:[NaN~;~:*~r~]"(ignore-errors(floor(read))))

Ta wersja działa raczej inaczej. Zamiast czytać wiersz i konwertować go, wywołuje on czytnik lisp do interpretacji danych wejściowych jako wyrażenia lisp s, próbuje użyć go jako liczby, a jeśli wystąpią jakiekolwiek błędy, ignoruje je, generując zero do warunkowego podania łańcucha formatu. To może być pierwszy przypadek, gdy widziałem seplenienie naprawdę zwięzłego programu ... Zabawa!

  • (read) Wywołuje czytnik / analizator składni w celu odczytania jednego wyrażenia ze standardowego wejścia i przekształcenia go w odpowiedni obiekt
  • (floor) próby konwersji dowolnego typu liczbowego na najbliższą niższą liczbę całkowitą, typy nienumeryczne powodują zgłoszenie błędu
  • (ignore-errors ...) robi to, co jest napisane na puszce, wyłapuje i ignoruje wszelkie błędy w dołączonym wyrażeniu, zwracając zero, aby podać gałąź NaN ciągu formatu
Tom Scogland
źródło
Z pewnością nie ma problemu, że pytanie stare :) Zredagowałem twoją odpowiedź, aby dołączyć nazwę języka i liczbę znaków w nagłówku.
ProgramFOX
Dziękuję za zmiany, jeszcze nie otrzymałem składni Stack * dla tych rzeczy. Wróciłem i naprawiłem błąd, który popełniłem w opisie warunku w ciągu formatu.
Tom Scogland,
3

PHP, 327 310 308 bajtów

<?$a=['',one,two,three,four,five,six,seven,eight,nine,ten,eleven,twelve,thir,0,fif,0,0,eigh];echo($n=$argv[1])>999?$a[$n/1000].' thousand ':'',$n%1000>99?$a[$n/100%10].' hundred ':'',$n?($k=$n%100)<20?($a[$k]?:$a[$k%10]).[teen][$k<13]:[2=>twen,thir,'for',fif,six,seven,eigh,nine][$k/10].'ty '.$a[$k%10]:zero;

przyjmuje liczbę jako parametr, działa dla 0 <= n <= 12999

awaria

// define names
$a=['',one,two,three,four,five,six,seven,eight,nine,
    ten,eleven,twelve,thir,0,fif,0,0,eigh];
// print ...
echo
    ($n=$argv[1])>999?$a[$n/1000].' thousand ':'',                  // thousands
    $n%1000>99?$a[$n/100%10].' hundred ':'',                        // hundreds
    $n?
        // if remains <20:
        ($k=$n%100)<20?
            ($a[$k]?:$a[$k%10]) // no value at index (0,14,16,17,19)? value from index%10
            .[teen][$k<13]      // append "teen" for $k>12
        // else:
        :[2=>twen,thir,'for',fif,six,seven,eigh,nine][$k/10].'ty '  // tens
        .$a[$k%10]                                                  // ones
    // "zero" for $n==0
    :zero
;
Tytus
źródło
2

SAS, 70 znaków

data;window w n;display w;if n=. then put 'NaN';else put n words.;run;

Instrukcje windowi displayotwierają wiersz polecenia SAS. Dane wejściowe nidą do wiersza 1. Wykorzystuje format SAS, w words.którym liczba jest wypisywana jako słowo lub seria słów z odpowiednio „i”, „” i „-”.

Alex A.
źródło
2

PHP

777 znaków

To zdecydowanie okropna próba, ale nie możesz mnie oskarżyć o wykorzystywanie jakichkolwiek luk, a do tego to bardzo szczęśliwy wynik. Dzięki ProgramFOX za wskazówkę.

<?php $i=9212;$b = array('zero','one','two','three','four','five','six','seven','eight','nine');$t='teen';$c = array('ten','eleven','tweleve','thir'.$t,$b[4].$t,'fif'.$t,$b[6].$t,$b[7].$t,$b[8].$t,$b[9].$t);$d = array('','','twenty','thirty','fourty','fifty','sixty','seventy','eighty','ninety');$e='hundred';$f='thousand';$j=str_split($i);if (strlen($i)===1){$a=$b[$i];}elseif (strlen($i)===3){$k=1;$a=$b[$j[0]].' '.$e.' '.x($j,$k);}elseif (strlen($i)===4){$k=2;$a=$b[$j[0]].' '.$f.' '.$b[$j[1]].' '.$e.' '.x($j,$k);}elseif (substr($i, -2, 1)==='1'){$a=$c[$j[1]];}else{$a=$d[$j[0]].' '.$b[$j[1]];}$a = str_replace('zero hundred','',$a);echo $a;function x($j,$k){global $i, $b, $c, $d;if (substr($i, -2, 1)==='1'){return $c[$j[$k+1]];}else{return $d[$j[$k]].' '.$b[$j[$k+1]];}}

Długa ręka

<?php
// Input
$i=9212;
// 0-9
$b = array('zero','one','two','three','four','five','six','seven','eight','nine');
// 10-19 (Very tricky)
$t='teen';
$c = array('ten','eleven','tweleve','thir'.$t,$b[4].$t,'fif'.$t,$b[6].$t,$b[7].$t,$b[8].$t,$b[9].$t); 
// Left digit of 20-99
$d = array('','','twenty','thirty','fourty','fifty','sixty','seventy','eighty','ninety');
// Hundreds
$e='hundred';
// Thousands
$f='thousand';
// Split input
$j=str_split($i);
// 1 digit inputs
if (strlen($i)===1){$a=$b[$i];}
// 3 digit input
elseif (strlen($i)===3){$k=1;$a=$b[$j[0]].' '.$e.' '.x($j,$k);}
// 4 digit input
elseif (strlen($i)===4){$k=2;$a=$b[$j[0]].' '.$f.' '.$b[$j[1]].' '.$e.' '.x($j,$k);}
// 10-19
elseif (substr($i, -2, 1)==='1'){$a=$c[$j[1]];}
// 20-99
else{$a=$d[$j[0]].' '.$b[$j[1]];}
// Fix for thousand numbers
$a = str_replace('zero hundred','',$a);
// Result
echo $a;
// Abstracted function last 2 digits for 3 and 4 digit numbers
function x($j,$k){
    global $i, $b, $c, $d;
    // 10-19
    if (substr($i, -2, 1)==='1'){return $c[$j[$k+1]];}
    // 20-99
    else{return $d[$j[$k]].' '.$b[$j[$k+1]];}
}
Gęś
źródło
1
Myślę, że można skrócić swój kod tworząc tablice jak ten: array('zero','one','two').
ProgramFOX
@ProgramFOX lub nawet ['zero','one','two'](php 5.4+). A jeśli nie masz nic przeciwko E_NOTICE, [zero,one,two]to też zadziała.
primo
Powinienem go zaktualizować, ale 777 to taka szczęśliwa liczba.
Goose
+1 za twoje wysiłki. PHP jest tragicznie niedostatecznie reprezentowane w kodzie golfowym.
primo
1

Python 2.x - 378

Pochodna odpowiedzi Fireflys, chociaż zmieniając Pna miliony lub tryliony itd., Można ją rekurencyjnie wykorzystać do dowolnego zakresu liczb dodatnich. Obsługuje to również wartości do 999,999

O=",one,two,three,four,five,six,seven,eight,nine,ten,eleven,twelve,thir,,fif,,,eigh,,,,twen,thir,for,fif,,,eigh,".split(",")
P=",thousand".split(',')
def N(s,p=0):
 h,s=divmod(s,1000);x=N(h,p+1)if h>0 else" "
 if s<20:x+=O[s]or O[s-10]+["","teen"][s>12]
 elif s<100:x+=(O[s/10+20]or O[s/10])+"ty"+N(s%10)
 else:x+=N(s/100)+"hundred"+N(s%100)
 return x+" "+P[p]
print N(input())

Przykładowy test (wejście jest <<<, wyjście jest >>>):

<<< 1234
>>> one thousand two hundred thirty four

<<< 999999
>>>  nine hundred ninety nine   thousand nine hundred ninety nine

Chociaż, jeśli ktoś może wyjaśnić ten dziwny problem z „przepełnieniem bufora”, mam go, to byłoby bardzo ...

<<< -1
>>>  nine hundred ninety nine

<<< -2
>>>  nine hundred ninety eight

źródło
print divmod(-2,1000) #-> (-1, 998)
primo,
Och, oczywiście. Myślałem, że może to mieć absolutną wartość lub coś w tym rodzaju. Ale jest -1*1000i „reszta” 998.
1

SmileBASIC, 365 trzysta czterdzieści siedem bajtów

DIM N$[22]D$="OneTwoThreeFourFiveSixSevenEightNineTenElevenTwelveThirFourFifSixSevenEighNineTwenFor
WHILE LEN(D$)INC I,D$[0]<"_
INC N$[I],SHIFT(D$)WEND
INPUT N
W=N MOD 100C%=N/100MOD 10M%=N/1E3T=W<20X=W/10>>0?(N$[M%]+" Thousand ")*!!M%+(N$[C%]+" Hundred ")*!!C%+(N$[X+10+(X==2)*8+(X==4)*7]+"ty "+N$[N MOD 10])*!T+N$[W*T]+"teen"*(T&&W>12)+"Zero"*!N

Jeśli ostatnia jedna lub dwie cyfry to 0, występuje spacja końcowa.

12Me21
źródło
0

MOO - 55 znaków

player:tell($string_utils:english_number(read(player)))

Lub, jeśli nie muszę drukować do „standardowego” - 42 znaki: $string_utils:english_number(read(player))

Uwaga: ten kod nie wyświetla żadnego monitu na standardowym wyjściu i drukuje zerozamiast, NaNgdy dane wejściowe nie są liczbą.

Jako bonus, ten kod może obsługiwać dowolną liczbę w granicach języka moo ( 2147483647- -2147483648).

pppery
źródło
0

Wolfram Language 27 40 bajtów

Wykorzystując natywną funkcję IntegerName,

 Check[Input[]~IntegerName~"Words","NaN"]

Powyższe monituje o wprowadzenie danych przez użytkownika. Obecna implementacja zwraca „NaN”, jeśli użytkownik wpisze coś innego niż liczba całkowita.


Niektóre przykłady (ze wstępnie ustawionymi wejściami) :

 Check[243~IntegerName~"Words","NaN"]

dwieście czterdzieści trzy


 Check[1234567890~IntegerName~"Words","NaN"]   

jeden miliard dwieście trzydzieści cztery miliony pięćset sześćdziesiąt siedem tysięcy osiemset dziewięćdziesiąt


 Check["abc"~IntegerName~"Words","NaN"]  

NaN

DavidC
źródło
0

Python 2 , 333 bajty

def f(n):S=str.split;D=S('z one two three four five six seven eight nine');K=' fif six seven eigh nine';k=n/1000;n,m=n/100%10,n%100;e,d=m/10,m%10;return' '.join([k and f(k),'thousand']*(k>0)+[D[n],'hundred']*(n>0)+([S('ten eleven twelve thir four'+K)[d]+'teen'*(d>2)]if 9<m<20else[S('twen thir for'+K)[e-2]+'ty']*(e>0)+[D[d]]*(d>0)))

Wypróbuj online!

To jest dobre dla 1 do 999,999 włącznie.

Chas Brown
źródło
0

Pyth, 239 242 bajtów

L:rjdb6"  +"dAm+cd;"nine"," one two three four five six seven eight""  twen thir for fif six seven eigh"|y_.ey+Wk.e?Y?thZjd,?hZ+@HhZ"ty"""@GeZ@+c"ten eleven twelve"d+R"teen"+c"thir four"d>H5eZ?hZ+@GhZ" hundred"""c.[03_b]1"thousand"c_jQT3"zero

Dane wejściowe są liczbami całkowitymi z zakresu [0–999,999]. Wypróbuj online tutaj . Wyjaśnienie w toku.

Poprzednia wersja, bardzo podobna operacja, ale nie obsługuje 0:

L:rjdb6"  +"dJc" one two three four five six seven eight nine"dKc"  twen thir for fif six seven eigh nine"dy_.ey+Wk.e?Y?thZjd,?hZ+@KhZ"ty"""@JeZ@+c"ten eleven twelve"d+R"teen"+c"thir four"d>K5eZ?hZ+@JhZ" hundred"""c.[03_b]1"thousand"c_jQT3

Objaśnienie poprzedniej wersji:

Implicit: Q=eval(input()), d=" "

Step 1: output formatting helper function
L:rjdb6"  +"d   
L               Define a function, y(b):
   jdb          Join b on spaces
  r   6         Strip whitespace from beginning and end
 :              In the above, replace...
       "  +"    ... strings of more than one space...
            d   ... with a single space

Step 2: Define number lookup lists
Jc"..."dKc"..."d   
  "..."            Lookup string
 c     d           Split the above on spaces
J                  Store in J - this is list of unit names
        Kc"..."d   As above, but storing in K - this is list of tens names, without "ty"

Step 3: Bringing it all together
y_.ey+Wk.e?Y?thZjd,?hZ+@KhZ"ty"""@JeZ@+c"ten eleven twelve"d+R"teen"+c"thir four"d>K5eZ?hZ+@JhZ" hundred"""c.[03_b]1"thousand"c_jQT3   
                                                                                                                                jQT    Get digits of Q
                                                                                                                               _       Reverse
                                                                                                                              c    3   Split into groups of 3
  .e                                                                                                                                   Map the above, element as b, index as k, using:
                                                                                                                _b                       Reverse the digits in the group
                                                                                                            .[03                         Pad the above on the left with 0 to length 3
                                                                                                           c      ]1                     Chop at index 1 - [1,2,3] => [[1],[2,3]]
        .e                                                                                                                               Map the above, element as Z, index as Y, using:
          ?Y                                                                                                                               If second element in the group (i.e. tens and units):
            ?thZ                                                                                                                             If (tens - 1) is non-zero (i.e. 0 or >=2):
                   ?hZ                                                                                                                         If tens is non-zero:
                       @KhZ                                                                                                                      Lookup in tens names
                      +    "ty"                                                                                                                  Append "ty"
                                                                                                                                               Else:
                               ""                                                                                                                Empty string
                  ,                                                                                                                            Create two-element list of the above with...
                                 @JeZ                                                                                                          ... lookup units name
                jd                                                                                                                             Join the above on a space - this covers [0-9] and [20-99]
                                                                                                                                             Else:
                                                                     c"thir four"d                                                             ["thir", "four"]
                                                                    +             >K5                                                          Append last 5 element of tens names ("fif" onwards)
                                                            +R"teen"                                                                           Append "teen" to each string in the above
                                      +c"ten eleven twelve"d                                                                                   Prepend ["ten", "eleven", "twelve"]
                                     @                                               eZ                                                        Take string at index of units column - this covers [10-19]
                                                                                                                                           Else: (i.e. hundreds column)
                                                                                       ?hZ                                                   If hundreds column is non-zero:
                                                                                           @JhZ                                                Lookup units name
                                                                                          +    " hundred"                                      Append " hundred"
                                                                                                         ""                                  Else: empty string
                                                                                                                                         Result of map is two element list of [hundreds name, tens and units name]
      Wk                                                                                                                                 If k is nonzero (i.e. dealing with thousands group)...
     +                                                                                                              "thousand"           ... Append "thousand"
    y                                                                                                                                    Apply output formatting (join on spaces, strip, deduplicate spaces)
                                                                                                                                       Result of map is [units group string, thousands group string]
 _                                                                                                                                     Reverse group ordering to put thousands back in front
y                                                                                                                                      Apply output formatting again, implicit print
Sok
źródło