Eban numbers
{draft task}
- Definition
An eban number is a number that has no letter e in it when the number is spelled in ENglish.
Or more literally, spelled numbers that contain the letter e are banned.
The American version of spelling numbers will be used here (as opposed to the British).
2,000,000,000 is two billion, not two milliard.
Only numbers less than one sextillion (1021) will be considered in/for this task.
This will allow optimizations to be used.
- Task
-
- show all eban numbers ≤ 1,000 (in a horizontal format), and a count
- show all eban numbers between 1,000 and 4,000 (inclusive), and a count
- show a count of all eban numbers up and including 10,000
- show a count of all eban numbers up and including 100,000
- show a count of all eban numbers up and including 1,000,000
- show a count of all eban numbers up and including 10,000,000
- show all output here.
- See also
-
- The MathWorld entry: [eban numbersl.
- The OEIS entry: [A6933, eban numbers].
Perl 6
Modular approach, very little is hard coded. Change the order of magnitude limits to adjust the search/display ranges. Change the letter given to the nban sub to change which letter to disallow.
Demonstrate for e-ban and t-ban.
<lang perl6>use Lingua::EN::Numbers::Cardinal;
sub nban (@seq, $n = 'e') { (@seq).map: { next if .&cardinal.contains($n); $_ } }
sub comma { $^i.flip.comb(3).join(',').flip }
for 'e', 't' -> $n {
my @ban = flat ((1 .. 99),).map: *.&nban($n);
my @orders = ( (2 .. 9).map({ 10**$_*1, 10**$_*2 ... 10**$_*9 }).map: *.&nban($n) );
for @orders -> @order { next unless +@order; my @these; for @order { my @extend = flat $_, flat @ban X+ $_; @these.append: @extend; } @ban.append: @these } say "\nDisplaying output for {$n}-ban:"; put my @k = @ban.grep: * <= 100; say "Up to 100: ", +@k; say '=' x 40;
put my @j = @ban.grep: 1000 <= * <= 4000; say "From 1,000 to 4,000: ", +@j; say '=' x 40;
for (4 .. 9).map: { 10**$_ } -> $e { my $f = @ban.first( * >= $e, :k ); say "Up to {comma $e}: ", +@ban[^$f]; say '=' x 40; }
}</lang>
- Output:
Displaying output for e-ban: 2 4 6 30 32 34 36 40 42 44 46 50 52 54 56 60 62 64 66 Up to 100: 19 ======================================== 2000 2002 2004 2006 2030 2032 2034 2036 2040 2042 2044 2046 2050 2052 2054 2056 2060 2062 2064 2066 4000 From 1,000 to 4,000: 21 ======================================== Up to 10,000: 79 ======================================== Up to 100,000: 399 ======================================== Up to 1,000,000: 399 ======================================== Up to 10,000,000: 1599 ======================================== Up to 100,000,000: 7999 ======================================== Up to 1,000,000,000: 7999 ======================================== Displaying output for t-ban: 1 4 5 6 7 9 11 100 Up to 100: 8 ======================================== From 1,000 to 4,000: 0 ======================================== Up to 10,000: 55 ======================================== Up to 100,000: 55 ======================================== Up to 1,000,000: 55 ======================================== Up to 10,000,000: 391 ======================================== Up to 100,000,000: 391 ======================================== Up to 1,000,000,000: 2743 ========================================
REXX
Programming note: REXX has no shortcuts for if statements, so the multiple if statements weren't combined into one. <lang rexx>/*REXX program to display eban numbers (those that don't have an "e" their English name)*/ numeric digits 20 /*support some gihugic numbers for pgm.*/ parse arg $ /*obtain optional arguments from the cL*/ if $= then $= '1 1000 1000 4000 1 -10000 1 -100000 1 -1000000 1 -10000000'
do k=1 by 2 to words($) /*step through the list of numbers. */ call banE word($, k), word($, k+1) /*process the numbers, from low──►high.*/ end /*k*/
exit /*stick a fork in it, we're all done. */ /*──────────────────────────────────────────────────────────────────────────────────────*/ banE: procedure; parse arg x,y,_; z= reverse(x) /*obtain the number to be examined. */
tell= y>=0 /*Is HI non-negative? Display eban #s.*/ #= 0 /*the count of eban numbers (so far).*/ do j=x to abs(y) /*probably process a range of numbers. */ if hasE(j) then iterate /*determine if the number has an "e". */ #= # + 1 /*bump the counter of eban numbers. */ if tell then _= _ j /*maybe add to a list of eban numbers. */ end /*j*/ if _\== then say strip(_) /*display the list (if there is one). */ say; say # ' eban numbers found for: ' x " " y; say copies('═', 105) return
/*──────────────────────────────────────────────────────────────────────────────────────*/ hasE: procedure; parse arg x; z= reverse(x) /*obtain the number to be examined. */
do k=1 by 3 /*while there're dec. digit to examine.*/ @= reverse( substr(z, k, 3) ) /*obtain 3 dec. digs (a period) from Z.*/ if @==' ' then return 0 /*we have reached the "end" of the num.*/ uni= right(@, 1) /*get units dec. digit of this period. */ if uni//2==1 then return 1 /*if an odd digit, then not an eban #. */ if uni==8 then return 1 /*if an eight, " " " " " */ tens=substr(@, 2, 1) /*get tens dec. digit of this period. */ if tens==1 then return 1 /*if teens, then not an eban #. */ if tens==2 then return 1 /*if twenties, " " " " " */ if tens>6 then return 1 /*if 70s, 80s, 90s, " " " " " */ hun= left(@, 1) /*get hundreds dec. dig of this period.*/ if hun==0 then iterate /*if zero, then there is more of number*/ if hun\==' ' then return 1 /*any hundrEd (not zero) has an "e". */ end /*k*/ /*A "period" is a group of 3 dec. digs */ return 0 /*in the number, grouped from the right*/</lang>
- output when using the default inputs:
2 4 6 30 32 34 36 40 42 44 46 50 52 54 56 60 62 64 66 19 eban numbers found for: 1 1000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════ 2000 2002 2004 2006 2030 2032 2034 2036 2040 2042 2044 2046 2050 2052 2054 2056 2060 2062 2064 2066 4000 21 eban numbers found for: 1000 4000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════ 79 eban numbers found for: 1 -10000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════ 399 eban numbers found for: 1 -100000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════ 399 eban numbers found for: 1 -1000000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════ 1599 eban numbers found for: 1 -10000000 ═════════════════════════════════════════════════════════════════════════════════════════════════════════