Inventory sequence
You are encouraged to solve this task according to the task description, using any language you may know.
To build the inventory sequence, first take inventory of what numbers are already in the sequence, then, starting with 0 add the count of each number in turn to the end of the sequence. When you reach a number that has a count of 0, stop, add the 0 to the end of the sequence, and then restart the inventory from 0.
- E.G.
- Start taking inventory; how many 0s are there? 0. Add a 0 to the end of the sequence and restart the inventory. (0)
- How many 0s are there? 1. Add a 1 to the end of the sequence. How many 1s are there? 1. Add a 1 to the end of the sequence. How many 2s are there? 0. Add a 0 to the end of the sequence and restart the inventory. (0 1 1 0)
- and so on.
- Task
- Find and display the first 100 elements of the sequence.
- Find and display the position and value of the first element greater than or equal to 1000.
- Stretch
- Find and display the position and value of the first element greater than or equal to 2000, 3000 ... 10,000.
- Plot a graph of the first 10,000 elements of the sequence.
- See also
8080 Assembly
PRINT equ 9 ; CP/M call to print a string
org 100h
lxi h,tally ; Zero out the array
lxi d,11000 * 2 ; Two bytes each
mvi b,0
zero: mov m,b
inx h
dcx d
mov a,d
ora e
jnz zero
restrt: lxi b,0 ; BC = current number
takinv: mov d,b ; Look up count of current number
mov e,c
call talidx
mov e,m ; DE = current element
inx h
mov d,m
lxi h,tabct ; Any items left to print for the table?
mov a,m
ora a
jz cktrsh
dcr m ; In any case there's now one less
call atoi
lxi h,numbuf ; Print the current number
call prstr
lxi h,tabcol ; Done with the row yet?
dcr m
jnz cktrsh
mvi m,10 ; Yes, reset counter and print a newline
lxi h,nl
call prstr
cktrsh: lhld trshld ; Has the threshold been reached?
call cdehl
jc increm
xchg ; Yes, fill in the numbers
call atoi ; Threshold (from HL)
xchg
push d ; And add 1000 to it while we've got it here
lxi d,1000
dad d
pop d
shld trshld
lxi h,trsout
call numat
call atoi ; Our current element
lxi h,numout
call numat
lxi h,trsfmt ; Print the element and its index
call prstr
increm: call talidx ; HL = address of count of element
call inxm ; Increment count
call incidx ; Increment index
inx b ; Increment the search number
mov a,d ; Reached zero?
ora e
jnz takinv ; If not, keep going
lhld trshld ; Once we've done enough, stop
lxi d,-10001
dad d
jnc restrt
ret
tabcol: db 10 ; Column counter for table
tabct: db 100 ; First 100 items are to be printed
trshld: dw 1000 ; Initial threshold
; 16-bit compare DE and HL
cdehl: mov a,d
cmp h
rnz
mov a,e
cmp l
ret
; 16-bit increment of [HL]
inxm: inr m
rnz
inx h
inr m
ret
; Get HL = &tally[DE]
talidx: push d
xchg
dad h
lxi d,tally
dad d
pop d
ret
; print string at HL
prstr: push h
push d
push b
xchg
mvi c,PRINT
call 5
pop b
pop d
pop h
ret
; copy number buffer to HL
numat: push d
push b
lxi d,numbuf
mvi b,5
numcpy: ldax d
mov m,a
inx d
inx h
dcr b
jnz numcpy
pop b
pop d
ret
; set numbuf to DE as decimal number
atoi: push h
push d
push b
mvi a,5 ; Space out the buffer
lxi h,nbufe ; End of number buffer
push h ; Keep it on the stack
aspc: dcx h
mvi m,' '
dcr a
jnz aspc
lxi b,-10
adgt: xchg ; Number in HL
lxi d,-1
adgtlp: inx d ; Extract digit
dad b
jc adgtlp
mvi a,'0'+10 ; ASCII digit in A
add l
pop h ; Get buffer pointer back
dcx h
mov m,a
push h
mov a,d ; Any digits left?
ora e
jnz adgt ; If so, next digits
pop b ; Otherwise we're done
pop b
pop d
pop h
ret
numbuf: db '*****'
nbufe: db '$'
; Format to print first number above threshold, ends in index
trsfmt: db 'First > '
trsout: db '*****: '
numout: db '***** at '
; Index counter, kept as ASCII decimal so we don't need 32-bit math
idx: db ' 0'
nl: db 13,10,'$' ; Newline string which also ends index
; Increment index
incidx: lxi h,nl-1
inclp: mov a,m
cpi ' ' ; Space should be set to 1
jz one
inr m
cpi '9'
rnz
mvi m,'0'
dcx h
jmp inclp
one: mvi m,'1'
ret
tally equ $ ; Array stored at the end of the program
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
8086 Assembly
cpu 8086
org 100h
section .text
xor ax,ax ; Zero out the array
mov di,tally
mov cx,11000
rep stosw
mov bp,1000 ; BP = next threshold
mov cx,640Ah ; CH = table counter, CL = column counter
restrt: xor bx,bx ; BX = current number
takinv: shl bx,1
mov si,[tally+bx] ; SI = current element
shr bx,1
test ch,ch ; Any items left to print for the table?
jz cktrsh ; If not check threshold
dec ch
mov di,format.cell ; Print cell
mov dx,di
mov ax,si
call atoi
mov ah,9
int 21h
dec cl ; Row done?
jnz cktrsh
mov dx,format.nl
mov cl,10
int 21h
cktrsh: cmp si,bp ; Has the threshold been reached?
jna increm
mov ax,bp ; Yes, fill in the numbers
mov di,format.trs
call atoi
mov ax,si
mov di,format.num
call atoi
mov dx,format ; Print the string
mov ah,9
int 21h
add bp,1000 ; Next threshold
increm: mov di,si ; Increment count of current element
shl di,1
inc word [tally+di]
call incidx ; Increment current index
inc bx ; Increment search number
test si,si ; Reached zero?
jnz takinv ; If not, keep going
cmp bp,10000 ; Last threshold reached?
jna restrt ; If not, start over
int 20h ; Otherwise we're done
; Convert AX to decimal and store at DI (must be 5-char buffer)
atoi: push ax
push bx
push cx
push dx
mov cx,5
add di,cx
mov bx,10
.digit: xor dx,dx
div bx
add dl,'0'
dec di
dec cx
mov [di],dl
test ax,ax
jnz .digit
jcxz .out
mov al,' '
dec di
std
rep stosb
cld
.out: pop dx
pop cx
pop bx
pop ax
ret
; Increment index as ASCII in place
incidx: mov di,format.nl-1
.loop: cmp byte [di],' ' ; Space -> 1
je .one
inc byte [di]
cmp byte [di],'9'
ja .carry
ret
.carry: mov byte [di],'0'
dec di
jmp .loop
.one: mov byte [di],'1'
ret
; Format to print first number above threshold
format: db "First > "
.trs: db "*****: "
.num: db "***** at "
.idx: db " 0"
.nl: db 13,10,'$'
.cell: db "*****$" ; cell placeholder
section .bss
align 2
tally: resw 11000
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
ALGOL 68
Calculates the sequence elements without storing them.
BEGIN # find elements of the inventory sequence #
INT next to show := 1 000; # next value to show first element > #
INT max to show = 10 000; # last value to show first element > #
INT max number = max to show + 1 000; # max. element value to consider #
[ 0 : max number ]INT occurs; # number of times each number occurs #
FOR i FROM LWB occurs TO UPB occurs DO occurs[ i ] := 0 OD;
INT seq pos := 0; # current end of the sequence #
WHILE next to show <= max to show DO
FOR n FROM 0 WHILE next to show <= max to show
AND BEGIN
INT element := occurs[ n ];
seq pos +:= 1;
IF seq pos <= 100 THEN
print( ( " ", whole( element, -4 ) ) );
IF seq pos MOD 10 = 0 THEN print( ( newline ) ) FI
ELIF element > next to show THEN
print( ( "Element ", whole( seq pos, -8 )
, " (", whole( element, -8 )
, ") is first > ", whole( next to show, -6 )
, newline
)
);
next to show +:= 1 000
FI;
IF element < max number THEN
occurs[ element ] +:= 1
FI;
element /= 0
END
DO SKIP OD
OD
END
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 Element 24256 ( 1001) is first > 1000 Element 43302 ( 2009) is first > 2000 Element 61709 ( 3001) is first > 3000 Element 81457 ( 4003) is first > 4000 Element 98705 ( 5021) is first > 5000 Element 121343 ( 6009) is first > 6000 Element 151757 ( 7035) is first > 7000 Element 168805 ( 8036) is first > 8000 Element 184429 ( 9014) is first > 9000 Element 201789 ( 10007) is first > 10000
APL
invseq←{
t←1000×⍳10
seq←0{
(⍺+1)∇⍣(n>0)⊢⍵,n←+/⍺=⍵
}⍣{
⍵∨.>⊃⌽t
}⊢⍬
⎕←'First 100 elements:'
⎕←10 10⍴seq
loc←{⊃⍸seq>⍵}¨t
⎕←'⊂First > ⊃,I5,⊂: ⊃,I5,⊂ at ⊃,I6'⎕FMT t,seq[loc],[1.5]loc
}
- Output:
Note that APL arrays are 1-indexed by default.
First 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24256 First > 2000: 2009 at 43302 First > 3000: 3001 at 61709 First > 4000: 4003 at 81457 First > 5000: 5021 at 98705 First > 6000: 6009 at 121343 First > 7000: 7035 at 151757 First > 8000: 8036 at 168805 First > 9000: 9014 at 184429 First > 10000: 10007 at 201789
BASIC
10 DEFINT A-Z
20 DIM I(11000)
30 T=1000
40 N=0
50 E=I(N)
60 IF S#<100 THEN PRINT USING "####";E;
70 IF E>T THEN PRINT USING "First > #####: ##### at ######";T;E;S#: T=T+1000
80 S#=S#+1
90 I(E)=I(E)+1
100 N=N+1
110 IF E>0 THEN 50
120 IF T<=10000 THEN 40
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
C++
#include <cstdint>
#include <iomanip>
#include <iostream>
#include <map>
#include <vector>
std::vector<uint32_t> inventory_sequence(uint32_t max_term) {
uint32_t term = 0;
std::vector<uint32_t> result = { term };
std::map<uint32_t, uint32_t> inventory = { { term, 1 } };
while ( result.back() < max_term ) {
inventory.insert({ term, 0 });
const uint32_t count = inventory[term];
term = ( count == 0 ) ? 0 : term + 1;
if ( inventory.find(count) == inventory.end() ) {
inventory.emplace(count, 1);
} else {
inventory[count]++;
}
result.emplace_back(count);
}
return result;
}
int main() {
std::vector<uint32_t> sequence = inventory_sequence(10'000);
uint32_t thousands = 1'000;
std::cout << "The first 100 numbers of the inventory sequence:" << "\n";
for ( uint64_t i = 0; i < sequence.size(); ++i ) {
const uint32_t number = sequence[i];
if ( i < 100 ) {
std::cout << std::setw(2) << number << ( i % 20 == 19 ? "\n" : " " );
} else if ( i == 100 ) {
std::cout << "\n";
} else if ( number >= thousands ) {
std::cout << "The first element ≥ " << std::setw(5) << thousands << " is "
<< std::setw(5) << number << " which occurs at index " << std::setw(6) << i << "\n";
thousands += 1'000;
}
}
}
- Output:
The first 100 numbers of the inventory sequence: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 The first element ≥ 1000 is 1001 which occurs at index 24255 The first element ≥ 2000 is 2009 which occurs at index 43301 The first element ≥ 3000 is 3001 which occurs at index 61708 The first element ≥ 4000 is 4003 which occurs at index 81456 The first element ≥ 5000 is 5021 which occurs at index 98704 The first element ≥ 6000 is 6009 which occurs at index 121342 The first element ≥ 7000 is 7035 which occurs at index 151756 The first element ≥ 8000 is 8036 which occurs at index 168804 The first element ≥ 9000 is 9014 which occurs at index 184428 The first element ≥ 10000 is 10007 which occurs at index 201788
CLU
grow = proc (arr: array[int], n: int)
while array[int]$high(arr) < n do
array[int]$addh(arr, 0)
end
end grow
invseq = iter () yields (int)
tally: array[int] := array[int]$[0:0]
num: int := 0
while true do
grow(tally, num)
el: int := tally[num]
yield(el)
grow(tally, el)
tally[el] := tally[el] + 1
if el=0
then num := 0
else num := num+1
end
end
end invseq
start_up = proc ()
po: stream := stream$primary_output()
index: int := 0
threshold: int := 1000
for n: int in invseq() do
if index<100 then
stream$putright(po, int$unparse(n), 5)
if index // 10 = 9 then stream$putl(po, "") end
end
if n > threshold then
stream$puts(po, "First > ")
stream$putright(po, int$unparse(threshold), 5)
stream$puts(po, ": ")
stream$putright(po, int$unparse(n), 5)
stream$puts(po, " at ")
stream$putright(po, int$unparse(index), 6)
stream$putl(po, "")
threshold := threshold + 1000
end
index := index + 1
if threshold > 10000 then break end
end
end start_up
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
Cowgol
include "cowgol.coh";
sub len(num: uint32): (length: uint8) is
length := 1;
while num>=10 loop
length := length + 1;
num := num / 10;
end loop;
end sub;
sub printtab32(num: uint32, size: uint8) is
var spaces := size - len(num);
while spaces > 0 loop
print_char(' ');
spaces := spaces - 1;
end loop;
print_i32(num);
end sub;
sub printtab16(num: uint16, size: uint8) is
printtab32(num as uint32, size);
end sub;
var tally: uint16[11000];
var threshold: uint16 := 1000;
var index: uint32 := 0;
MemZero(&tally[0] as [uint8], @bytesof tally);
while threshold <= 10000 loop
var number: uint16 := 0;
loop
var element := tally[number];
if index < 100 then
printtab16(element, 5);
if index % 10 == 9 then
print_nl();
end if;
end if;
if element > threshold then
print("First > ");
printtab16(threshold, 5);
print(": ");
printtab16(element, 5);
print(" at ");
printtab32(index, 6);
print_nl();
threshold := threshold + 1000;
end if;
index := index + 1;
number := number + 1;
tally[element] := tally[element] + 1;
if element == 0 then
break;
end if;
end loop;
end loop;
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
Draco
proc main() void:
[11000]word tally;
word threshold, number, element;
ulong index;
for number from 0 upto 10999 do tally[number] := 0 od;
index := 0;
threshold := 1000;
while threshold <= 10000 do
number := 0;
while
element := tally[number];
if index < 100 then
write(element:5);
if index % 10 = 9 then writeln() fi
fi;
if element > threshold then
writeln("First > ",threshold:5,": ",element:5," at ",index:6);
threshold := threshold + 1000
fi;
index := index + 1;
number := number + 1;
tally[element] := tally[element] + 1;
element /= 0
do od
od
corp
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
EasyLang
sysconf zero_based
repeat
cnts[] &= 0
n = cnts[i]
cnts[n] += 1
if len cnts[] <= 100
write n & " "
.
i += 1
if n = 0
i = 0
.
until n > 1000
.
print ""
print len cnts[] & " " & n
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 24256 1001
FreeBASIC
Dim As Integer max = 10000
Dim As Integer inv()
Dim As Integer counts(max + 100)
counts(0) = 1
Dim As Integer lower = 100
Dim As Integer upper = 1000
Dim As Boolean done = False
Dim As Integer ix = 0
While Not done
Dim As Integer i = 0, c = 0
Do
Dim As Integer j = counts(i)
If Ubound(inv) < max Then
Redim Preserve inv(ix+1)
inv(ix+1) = j
End If
counts(j) += 1
ix += 1
If Ubound(inv) >= lower Then
Print "Inventory sequence, first 100 elements:"
For c = 0 To 99
Print Using "###"; inv(c);
If (c+1) Mod 20 = 0 Then Print
Next
lower = max + 1
End If
If j = 0 Then Exit Do
If j >= upper Then
Print Using !"\nFirst element >= ##,### is ##,### at index ###,###"; upper; j; ix;
If j >= max Then done = True: Exit Do
upper += 1000
End If
i += 1
Loop
Wend
Sleep
J
nextinv=: ((*@] * 1+{.@[), }.@[ , ]) +/@({. = }.)
10 10$}.nextinv^:100]0 NB. first 100 elements of inventory sequence
0 1 1 0 2 2 2 0 3 2
4 1 1 0 4 4 4 1 4 0
5 5 4 1 6 2 1 0 6 7
5 1 6 3 3 1 0 7 9 5
3 6 4 4 2 0 8 9 6 4
9 4 5 2 1 3 0 9 10 7
5 10 6 6 3 1 4 2 0 10
11 8 6 11 6 9 3 2 5 3
2 0 11 11 10 8 11 7 9 4
3 6 4 5 0 12 11 10 9 13
({:,_2+#)nextinv^:(1000>{:)^:_]0 NB. first value of at least 1000 and its index
1001 24255
The inventory sequence has a "hidden value" which is the number that we are searching for, and counting. So, here, we include it as the first element of the representation of an inventory subsequence. And nextinv
calculates both the next "hidden value" as well as the corresponding subsequence which incorporates a count of how many times the current "hidden value" appeared.
For the task, we iterate inductively from the initial state (either a 100 times or until we find a value which is greater than 1000).
It would be more efficient to maintain counts of each integer so far encountered, but that efficiency is not necessary for this task (and would require more code).
That said, here's a faster implementation:
invseq=: {{
cnt=. 0, seq=. i. nxt=. 0
while. -. u seq do.
k=. nxt{cnt
nxt=. (*k)*nxt+1
cnt=. (1+k{cnt) k} cnt=. cnt {.~ (2+k)>.#cnt
seq=. seq, k
end.
}}
stretch=: }.nextinv^:(10000>{:)^:_]0 NB. or
stretch=: (1e4<:{:) invseq NB. equivalent, faster approach
(,~ {&stretch) {.I.2000<stretch NB. first value greater than 2000 and its index
2009 43301
(,~ {&stretch) {.I.3000<stretch
3001 61708
(,~ {&stretch) {.I.4000<stretch
4003 81456
(,~ {&stretch) {.I.5000<stretch
5021 98704
(,~ {&stretch) {.I.6000<stretch
6009 121342
(,~ {&stretch) {.I.7000<stretch
7035 151756
(,~ {&stretch) {.I.8000<stretch
8036 168804
(,~ {&stretch) {.I.9000<stretch
9014 184428
(,~ {&stretch) {.I.10000<stretch
10007 201788
require'plot'
plot 1e4{.stretch
Java
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public final class InventorySequence {
public static void main(String[] args) {
List<Integer> inventorySequence = inventorySequence(10_000);
int thousands = 1_000;
System.out.println("The first 100 numbers of the inventory sequence:");
for ( int i = 0; i < inventorySequence.size(); i++ ) {
final int number = inventorySequence.get(i);
if ( i < 100 ) {
System.out.print(String.format("%2d%s", number, ( i % 20 == 19 ? "\n" : " " )));
} else if ( i == 100 ) {
System.out.println();
} else if ( number >= thousands ) {
System.out.println(String.format("%s%5d%s%5d%s%6d",
"The first element ≥ ", thousands, " is ", number, " which occurs at index ", i));
thousands += 1_000;
}
}
}
private static List<Integer> inventorySequence(int maxTerm) {
int term = 0;
List<Integer> result = new ArrayList<Integer>(List.of(term ));
Map<Integer, Integer> inventory = new HashMap<Integer, Integer>(Map.of( 0, 1 ));
while ( result.getLast() < maxTerm ) {
final int count = inventory.computeIfAbsent(term, n -> 0 );
term = ( count == 0 ) ? 0 : term + 1;
inventory.merge(count, 1, Integer::sum);
result.addLast(count);
}
return result;
}
}
- Output:
The first 100 numbers of the inventory sequence: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 The first element ≥ 1000 is 1001 which occurs at index 24255 The first element ≥ 2000 is 2009 which occurs at index 43301 The first element ≥ 3000 is 3001 which occurs at index 61708 The first element ≥ 4000 is 4003 which occurs at index 81456 The first element ≥ 5000 is 5021 which occurs at index 98704 The first element ≥ 6000 is 6009 which occurs at index 121342 The first element ≥ 7000 is 7035 which occurs at index 151756 The first element ≥ 8000 is 8036 which occurs at index 168804 The first element ≥ 9000 is 9014 which occurs at index 184428 The first element ≥ 10000 is 10007 which occurs at index 201788
jq
Works with both jq and gojq, the C and Go implementations of jq
... but note that gojq takes about 5 times longer (and requires much more memory) to complete the task.
With minor modifications, the program below also works quite snappily with jaq, the Rust implementation of jq.
The definition of `_nwise` can be omitted if using the C implementation of jq.
def _nwise($n):
def n: if length <= $n then . else .[0:$n] , (.[$n:] | n) end;
n;
# Emit the inventory sequence ad infinitum
def inventory_sequence:
{num: 0,
emit: 0,
inventory: {} }
| foreach range(0; infinite) as $n (.;
.emit = (.inventory[.num|tostring] // 0)
| if .emit == 0 then .num = 0 else .num += 1 end
| .inventory[.emit|tostring] += 1 )
| .emit ;
# Report on the progress of an arbitrary sequence, indefinitely
# Emit [.next, $x, .n]
def probe(s; $gap):
foreach s as $x ({n: 0, next: $gap};
.n += 1
| if $x >= .next then .emit = {next, $x, n} | .next += $gap
else .emit = null
end)
| select(.emit).emit;
def lpad($len): tostring | ($len - length) as $l | (" " * $l)[:$l] + .;
def task($n):
[limit($n; inventory_sequence)] | _nwise(10) | map(lpad(3)) | join(" ");
task(100),
"",
(limit(10; probe(inventory_sequence; 1000))
| "First element >= \(.next) is \(.x) at index \(.n - 1)")
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1000 is 1001 at index 24255 First element >= 2000 is 2009 at index 43301 First element >= 3000 is 3001 at index 61708 First element >= 4000 is 4003 at index 81456 First element >= 5000 is 5021 at index 98704 First element >= 6000 is 6009 at index 121342 First element >= 7000 is 7035 at index 151756 First element >= 8000 is 8036 at index 168804 First element >= 9000 is 9014 at index 184428 First element >= 10000 is 10007 at index 201788
Julia
""" rosettacode.org/wiki/Inventory_sequence """
using Printf
using Counters
using Plots
struct InventorySequence
inventory::Counter{Int}
InventorySequence() = new(counter([0]))
end
function Base.iterate(i::InventorySequence, num = 0)
nextval = i.inventory[num]
num = nextval == 0 ? 0 : num + 1
i.inventory[nextval] += 1
return nextval, num
end
const thresholds = [1000 * j for j in 1:10]
const toplot = Int[]
for (i, n) in enumerate(InventorySequence())
if i <= 100
print(rpad(n, 4), i % 20 == 0 ? "\n" : "")
elseif n >= thresholds[begin]
@printf("First element >= %d5: %d5 in position %d.\n", popfirst!(thresholds), n, i)
length(thresholds) == 0 && break
end
length(toplot) < 10000 && push!(toplot, n)
end
plot(toplot)
- Output:
Similar to Python output.
MAD
NORMAL MODE IS INTEGER
DIMENSION TALLY(11000)
DIMENSION ROW(10)
VECTOR VALUES ROWF = $10(I5)*$
VECTOR VALUES TRSHF = $8HFIRST > ,I5,2H: ,I5,S1,3HAT ,I6*$
RWIX = 0
INTERNAL FUNCTION(X)
ENTRY TO TBLOUT.
ROW(RWIX) = X
RWIX = RWIX + 1
WHENEVER RWIX.L.10, FUNCTION RETURN 0
PRINT FORMAT ROWF,ROW(0),ROW(1),ROW(2),ROW(3),ROW(4),
0 ROW(5),ROW(6),ROW(7),ROW(8),ROW(9)
RWIX = 0
END OF FUNCTION
TRSHLD = 1000
IDX = 0
RESTRT NUM = 0
TAKINV ELEM = TALLY(NUM)
WHENEVER IDX.L.100, TBLOUT.(ELEM)
WHENEVER ELEM.G.TRSHLD
PRINT FORMAT TRSHF,TRSHLD,ELEM,IDX
TRSHLD = TRSHLD + 1000
END OF CONDITIONAL
IDX = IDX+1
NUM = NUM+1
TALLY(ELEM) = TALLY(ELEM)+1
WHENEVER ELEM.G.0, TRANSFER TO TAKINV
WHENEVER TRSHLD.LE.10000, TRANSFER TO RESTRT
END OF PROGRAM
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 FIRST > 1000: 1001 AT 24255 FIRST > 2000: 2009 AT 43301 FIRST > 3000: 3001 AT 61708 FIRST > 4000: 4003 AT 81456 FIRST > 5000: 5021 AT 98704 FIRST > 6000: 6009 AT 121342 FIRST > 7000: 7035 AT 151756 FIRST > 8000: 8036 AT 168804 FIRST > 9000: 9014 AT 184428 FIRST > 10000: 10007 AT 201788
Mathematica /Wolfram Language
(*Function to generate the inventory sequence*)
InventorySequence[terms_] :=
Module[{num = 0, alst = {0}, inventory, c}, inventory = <|0 -> 1|>;
Table[c = Lookup[inventory, num, 0];
num = If[c == 0, 0, num + 1];
alst = Append[alst, c];
inventory[c] = Lookup[inventory, c, 0] + 1;, {n, 2, terms}];
alst]
(*Generate the inventory sequence*)
biglist = InventorySequence[201790];
(*Print first 100 elements of the sequence*)
partitioned = Partition[Take[biglist, 100], 10];
Do[Print[Row[partitioned[[i]], " "]], {i, Length[partitioned]}]
(*Find and print the first occurrences of elements>=thresholds*)
thresholds = 1000 Range[1, 10];
firstOccurrences =
Reap[Do[If[biglist[[i]] >= thresholds[[1]],
Sow[{thresholds[[1]], biglist[[i]], i}];
thresholds = Rest[thresholds];
If[Length[thresholds] == 0, Break[]];], {i, Length[biglist]}]][[
2, 1]];
(*Print the formatted results*)
Do[Print["First element \[GreaterEqual] ", firstOccurrence[[1]],
" is ", firstOccurrence[[2]], " at index ",
firstOccurrence[[3]]], {firstOccurrence, firstOccurrences}]
(*Plot the first 10,000 elements of the sequence*)
ListPlot[biglist[[1 ;; 10000]], Joined -> True,
PlotStyle -> {Thin, Blue}, PlotRange -> Full]
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1000 is 1001 at index 24256 First element >= 2000 is 2009 at index 43302 First element >= 3000 is 3001 at index 61709 First element >= 4000 is 4003 at index 81457 First element >= 5000 is 5021 at index 98705 First element >= 6000 is 6009 at index 121343 First element >= 7000 is 7035 at index 151757 First element >= 8000 is 8036 at index 168805 First element >= 9000 is 9014 at index 184429 First element >= 10000 is 10007 at index 201789
Miranda
main :: [sys_message]
main = [Stdout "First 100 elements:\n",
Stdout (table 10 5 (take 100 invseq)),
Stdout (lay (map disp (zip2 thresholds firsts)))
]
where thresholds = [1000, 2000..10000]
firsts = first [(>x) | x <- thresholds] invseq
disp (t,(i,x)) = concat ["First > ",
rjustify 5 (shownum t),
": ",
rjustify 5 (shownum x),
" at ",
rjustify 6 (shownum i)]
table :: num->num->[num]->[char]
table w cw nums = lay [ concat (map (rjustify cw . show) row)
| row <- group w nums]
group :: num->[*]->[[*]]
group sz [] = []
group sz ls = take sz ls : group sz (drop sz ls)
invseq :: [num]
invseq = f [] 0
where f acc i = el : f (inc el acc) i'
where el = get i acc
i' = 0, if el=0
i' = i+1
get :: num->[(num,num)]->num
get n [] = 0
get n ((n,x):ns) = x
get n (m:ns) = get n ns
inc :: num->[(num,num)]->[(num,num)]
inc n [] = [(n, 1)]
inc n ((n,x):ns) = (n,x+1):ns
inc n (m:ns) = m:inc n ns
first :: [(*->bool)]->[*]->[(num,*)]
first ps = f ps 0
where f [] n xs = []
f (p:ps) n (x:xs) = (n, x) : f ps (n+1) xs, if p x
f (p:ps) n (x:xs) = f (p:ps) (n+1) xs
- Output:
First 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
Nim
import std/[strformat, tables]
import gnuplot
iterator inventorySequence(): (int, int) =
var counts: CountTable[int]
var idx = -1
while true:
var i = 0
while true:
let n = counts[i]
inc idx
counts.inc(n)
yield (idx, n)
if n == 0: break
inc i
echo "First 100 elements:"
var x, y: seq[int]
var lim = 1000
for idx, n in inventorySequence():
if idx < 10000:
x.add idx
y.add n
if idx <= 100:
stdout.write &"{n:>2}"
stdout.write if idx mod 10 == 0: '\n' else: ' '
if idx == 100: echo()
elif n >= lim:
echo &"First element ⩾ {lim:>5} is {n:>5} at index {idx:>6}"
lim += 1000
if lim > 10000: break
withGnuPlot:
plot(x, y, "Inventory sequence", "with impulses lw 0.5")
png("inventory_sequence.png")
- Output:
First 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 8 First element ⩾ 1000 is 1001 at index 24255 First element ⩾ 2000 is 2009 at index 43301 First element ⩾ 3000 is 3001 at index 61708 First element ⩾ 4000 is 4003 at index 81456 First element ⩾ 5000 is 5021 at index 98704 First element ⩾ 6000 is 6009 at index 121342 First element ⩾ 7000 is 7035 at index 151756 First element ⩾ 8000 is 8036 at index 168804 First element ⩾ 9000 is 9014 at index 184428 First element ⩾ 10000 is 10007 at index 201788
Perl
use strict;
use warnings;
use feature 'say';
use List::AllUtils <max firstidx>;
use GD::Graph::bars;
sub comma { reverse ((reverse shift) =~ s/.{3}\K/,/gr) =~ s/^,//r }
sub table { my $t = 20 * (my $c = 1 + length max @_); ( sprintf( ('%'.$c.'d')x@_, @_) ) =~ s/.{1,$t}\K/\n/gr }
my($i, @inventory, %i) = 0;
do {
my $count = $i{$i} // 0;
$i = $count ? $i+1 : 0;
++$i{$count};
push @inventory, $count
} until $inventory[-1] > 10_000;
say "Inventory sequence, first 100 elements:\n" . table @inventory[0..99]; say '';
for my $n (map { $_ * 1000 } 1..10) {
my $i = firstidx { $_ >= $n } @inventory;
printf "First element >= %6s is %6s in position: %s\n", comma($n), comma($inventory[$i]), comma $i;
}
# graph
my @data = ( [0..5000], [@inventory[0..5000]] );
my $graph = GD::Graph::bars->new(800, 600);
$graph->set(
title => 'Inventory sequence',
y_max_value => 250,
x_tick_number => 5,
r_margin => 10,
dclrs => [ 'blue' ],
) or die $graph->error;
my $gd = $graph->plot(\@data) or die $graph->error;
open my $fh, '>', 'Perl-inventory-sequence.png';
binmode $fh;
print $fh $gd->png();
close $fh;
- Output:
Inventory sequence, first 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1,000 is 1,001 in position: 24,255 First element >= 2,000 is 2,009 in position: 43,301 First element >= 3,000 is 3,001 in position: 61,708 First element >= 4,000 is 4,003 in position: 81,456 First element >= 5,000 is 5,021 in position: 98,704 First element >= 6,000 is 6,009 in position: 121,342 First element >= 7,000 is 7,035 in position: 151,756 First element >= 8,000 is 8,036 in position: 168,804 First element >= 9,000 is 9,014 in position: 184,428 First element >= 10,000 is 10,007 in position: 201,788
Phix
-- demo\rosetta\Inventory_sequence.exw with javascript_semantics function inventory(integer limit) sequence inv = {0}, counts = {1} integer ix = 0, thousands = 1000 while true do integer i = 0 while true do integer j = iff(i>=length(counts)?0:counts[i+1]) inv &= j while j>=length(counts) do counts &= 0 end while counts[j+1] += 1 ix += 1 if length(inv)=100 then printf(1,"Inventory sequence, first 100 elements:\n%s\n", {join_by(inv,1,20,"",fmt:="%3d")}) end if if j=0 then exit end if if j>=thousands then printf(1,"First element >= %,6d is %,6d at index %,7d\n", {thousands, j, ix}) if j>=limit then return inv[1..limit] end if thousands += 1000 end if i += 1 end while end while end function constant lim = 1e4 sequence x = tagset(lim), y = inventory(lim) include pGUI.e include IupGraph.e function get_data(Ihandle graph) integer {w,h} = IupGetIntInt(graph,"SIZE") IupSetInt(graph,"XTICK",iff(w<500?iff(w<350?iff(w<250?5000:2500):2000):1000)) IupSetInt(graph,"YTICK",iff(h<350?iff(h<200?iff(h<150? 200: 100): 80): 40)) return {{x,y,CD_BLUE}} end function IupOpen() Ihandle graph = IupGraph(get_data,"XMIN=0,XMAX=10000,YMIN=0,YMAX=400"), dlg = IupDialog(graph,`TITLE=gGraph,SIZE=320x240,MINSIZE=240x140`) IupShow(dlg) if platform()!=JS then IupMainLoop() end if
- Output:
Inventory sequence, first 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1,000 is 1,001 at index 24,255 First element >= 2,000 is 2,009 at index 43,301 First element >= 3,000 is 3,001 at index 61,708 First element >= 4,000 is 4,003 at index 81,456 First element >= 5,000 is 5,021 at index 98,704 First element >= 6,000 is 6,009 at index 121,342 First element >= 7,000 is 7,035 at index 151,756 First element >= 8,000 is 8,036 at index 168,804 First element >= 9,000 is 9,014 at index 184,428 First element >= 10,000 is 10,007 at index 201,788
PL/M
100H:
BDOS: PROCEDURE (FN, ARG); DECLARE FN BYTE, ARG ADDRESS; GO TO 5; END BDOS;
EXIT: PROCEDURE; GO TO 0; END EXIT;
PRINT: PROCEDURE (STR); DECLARE STR ADDRESS; CALL BDOS(9, STR); END PRINT;
PRINT$NUM: PROCEDURE (N, W);
DECLARE BUF (6) BYTE INITIAL ('.....$');
DECLARE N ADDRESS, (I, W) BYTE;
DO I=0 TO 4; BUF(I) = ' '; END;
I = 5;
DIGIT:
BUF(I := I-1) = '0' + N MOD 10;
IF (N := N/10) > 0 THEN GO TO DIGIT;
CALL PRINT(.BUF(5-W));
END PRINT$NUM;
DECLARE INV$SEQ (24500) ADDRESS;
CALC$SEQ: PROCEDURE;
DECLARE (ITEM, SEARCH$IDX, SEARCH$NUM) ADDRESS;
INV$SEQ(0) = 0;
SEARCH$NUM = 0;
DO ITEM = 1 TO LAST(INV$SEQ);
CALL PRINT$NUM(ITEM, 5);
CALL PRINT(.(13,'$'));
INV$SEQ(ITEM) = 0;
DO SEARCH$IDX = 0 TO ITEM-1;
IF INV$SEQ(SEARCH$IDX) = SEARCH$NUM THEN
INV$SEQ(ITEM) = INV$SEQ(ITEM) + 1;
END;
IF INV$SEQ(ITEM) = 0
THEN SEARCH$NUM = 0;
ELSE SEARCH$NUM = SEARCH$NUM + 1;
END;
END CALC$SEQ;
FIND$FIRST: PROCEDURE (N) ADDRESS;
DECLARE (N, I) ADDRESS;
DO I = 0 TO LAST(INV$SEQ);
IF INV$SEQ(I) >= N THEN RETURN I;
END;
END FIND$FIRST;
CALL CALC$SEQ;
DECLARE I ADDRESS;
CALL PRINT(.('FIRST 100 ITEMS OF INVENTORY SEQUENCE:',13,10,'$'));
DO I=0 TO 99;
CALL PRINT$NUM(INV$SEQ(I), 4);
IF I MOD 10 = 9 THEN CALL PRINT(.(13,10,'$'));
END;
CALL PRINT(.'FIRST ELEMENT >= 1000: $');
CALL PRINT$NUM(INV$SEQ(I := FIND$FIRST(1000)), 5);
CALL PRINT(.' AT $');
CALL PRINT$NUM(I, 5);
CALL EXIT;
EOF
- Output:
FIRST 100 ITEMS OF INVENTORY SEQUENCE: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 FIRST ELEMENT >= 1000: 1001 AT 24255
Python
''' rosettacode.org/wiki/Inventory_sequence '''
from collections import Counter
from matplotlib.pyplot import plot
def inventory_sequence(terms):
''' From the code by Branicky at oeis.org/A342585 '''
num, alst, inventory = 0, [0], Counter([0])
for n in range(2, terms+1):
c = inventory[num]
num = 0 if c == 0 else num + 1
alst.append(c)
inventory.update([c])
return alst
biglist = inventory_sequence(201_790)
thresholds = [1000 * j for j in range(1, 11)]
for i, k in enumerate(biglist):
if i < 100:
print(f'{k:<4}', end='\n' if (i + 1) % 20 == 0 else '')
elif k >= thresholds[0]:
print(f'\nFirst element >= {thresholds.pop(0):5}: {k:5} in position {i:6}')
if len(thresholds) == 0:
break
plot(biglist[:10_000], linewidth=0.3)
plt.show()
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1000: 1001 in position 24255 First element >= 2000: 2009 in position 43301 First element >= 3000: 3001 in position 61708 First element >= 4000: 4003 in position 81456 First element >= 5000: 5021 in position 98704 First element >= 6000: 6009 in position 121342 First element >= 7000: 7035 in position 151756 First element >= 8000: 8036 in position 168804 First element >= 9000: 9014 in position 184428 First element >= 10000: 10007 in position 201788
Quackery
[ 0 unrot witheach
[ over =
rot + swap ]
drop ] is occurs ( n [ --> n )
[ 2dup occurs
tuck join
dip [ 0 != tuck * + ] ] is additem ( n [ --> n [ )
0 [] 100 times additem
witheach
[ dup 10 < if sp
echo sp
i^ 10 mod 9 = if cr ]
drop
cr
0 []
[ additem
dup -1 peek 999 >
until ]
say "Element #"
dup size 1 - echo
say " is "
-1 peek echo
say "." cr
drop
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 Element #24255 is 1001.
Raku
use Lingua::EN::Numbers;
my ($i, %i) = 0;
my @inventory = (^∞).map: {
my $count = %i{$i} // 0;
$i = $count ?? $i+1 !! 0;
++%i{$count};
$count
}
say "Inventory sequence, first 100 elements:\n" ~
@inventory[^100].batch(20)».fmt("%2d").join: "\n";
say '';
for (1..10).map: * × 1000 {
my $k = @inventory.first: * >= $_, :k;
printf "First element >= %6s is %6s in position: %s\n",
.&comma, @inventory[$k].&comma, comma $k;
}
use SVG;
use SVG::Plot;
my @x = ^10000;
'Inventory-raku.svg'.IO.spurt:
SVG.serialize: SVG::Plot.new(
background => 'white',
width => 1000,
height => 600,
plot-width => 950,
plot-height => 550,
x => @x,
values => [@inventory[@x],],
title => "Inventory Sequence - First {+@x} values (zero indexed)",
).plot: :lines;
- Output:
Inventory sequence, first 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1,000 is 1,001 in position: 24,255 First element >= 2,000 is 2,009 in position: 43,301 First element >= 3,000 is 3,001 in position: 61,708 First element >= 4,000 is 4,003 in position: 81,456 First element >= 5,000 is 5,021 in position: 98,704 First element >= 6,000 is 6,009 in position: 121,342 First element >= 7,000 is 7,035 in position: 151,756 First element >= 8,000 is 8,036 in position: 168,804 First element >= 9,000 is 9,014 in position: 184,428 First element >= 10,000 is 10,007 in position: 201,788
converted to a .png to reduce size for display here:
Refal
$ENTRY Go {
, <InvSeqTo 210000>: e.Inv
, <First 100 e.Inv>: (e.100) e.1
, <Each (Mul 1000) <Iota 1 10>>: e.Ts
= <Prout 'First 100 elements:'>
<Tbl 10 5 e.100>
<Each (DispFind (e.Inv)) e.Ts>;
};
Iota {
s.E s.E = s.E;
s.S s.E = s.S <Iota <+ 1 s.S> s.E>;
};
DispFind {
t.L s.T, <Find t.L s.T>: s.X s.N =
<Prout 'First > ' <Fmt 5 s.T> ': '
<Fmt 5 s.X> ' at ' <Fmt 6 s.N>>;
};
Find {
(s.X e.Y) s.T = <Find (s.X e.Y) 0 s.T>;
(s.X e.Y) s.N s.T, <Compare s.X s.T>: {
'+' = s.X s.N;
s.1 = <Find (e.Y) <+ 1 s.N> s.T>;
};
};
Tbl {
s.W s.CW = ;
s.W s.CW e.X, <First s.W e.X>: (e.R) e.Y =
<Prout <Each (Fmt s.CW) e.R>>
<Tbl s.W s.CW e.Y>;
};
Fmt {
s.W s.N,
<Rep s.W ' '> <Symb s.N>: e.F,
<Last s.W e.F>: (e.1) e.2 = e.2;
};
Rep {
0 s.C = ;
s.N s.C = s.C <Rep <- s.N 1> s.C>;
};
Each {
(e.F) = ;
(e.F) t.I e.X = <Mu e.F t.I> <Each (e.F) e.X>;
};
InvSeqTo {
s.Num = <InvSeqTo () () s.Num>;
(e.X) (e.L) s.Num,
<SeqStep (e.X) (e.L) 0>: (e.X2) (e.L2),
<Lenw e.L2>: s.Len e.L2,
<Compare s.Len s.Num>: {
'-' = <InvSeqTo (e.X2) (e.L2) s.Num>;
s.1 = e.L2;
};
};
SeqStep {
(e.X) (e.L) s.N,
<GetN (e.X) s.N>: s.C,
<IncN (e.X) s.C>: e.X2,
e.L s.C: e.L2,
s.C: {
0 = (e.X2) (e.L2);
s.C = <SeqStep (e.X2) (e.L2) <+ 1 s.N>>;
};
};
GetN {
(e.X (s.N s.C) e.Y) s.N = s.C;
(e.X) s.N = 0;
};
IncN {
(e.X (s.N s.C) e.Y) s.N = e.X (s.N <+ 1 s.C>) e.Y;
(e.X) s.N = e.X (s.N 1);
};
- Output:
First 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
RPL
For efficiency reasons, two different programs are needed to generate the sequence or search for the first high value.
« → max « { 0 1 1 0 } @ need to start with a non-null cycle to have ∑LIST work WHILE DUP SIZE max < REPEAT 0 max FOR j DUP 1 « j == » DOLIST ∑LIST @ count occurrences in the list IF DUP NOT THEN max 'j' STO END + NEXT END 1 max SUB » 'INVT' STO @ ( n → { a(1)..a(n)} ) « DUP 1 + { } + 0 CON -1 → max counts j « 2 CF 1 DO 'counts' 'j' INCR 1 + IFERR GET THEN DROP2 0 END IF DUP NOT THEN -1 'j' STO END IF DUP max > THEN "element" →TAG SWAP "position" →TAG 2 SF ELSE 'counts' SWAP 1 + DUP2 GET 1 + PUT 1 + END UNTIL 2 FS? END » » 'INVT1ST' STO @ ( n → 1st_value_>_n pos )
100 INVT 1000 INVT1ST
- Output:
3: { 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 } 2: element: 1001 1: position: 24256
Ruby
Not actually counting but keeping count in a hash:
n = 0
counter = Hash.new(0)
inventory = loop.lazy.map do
c = counter[n]
counter[c] += 1
c == 0 ? n = 0 : n += 1
c
end
inventory.first(100).each_slice(10){|s| puts "%4d"*s.size % s}
puts
(1000..10000).step(1000).each do |t|
counter.clear
puts "First element >= #{t} : %d index %d" % inventory.with_index.detect{|e,i| e > t}
end
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1000 : 1001 index 24255 First element >= 2000 : 2009 index 43301 First element >= 3000 : 3001 index 61708 First element >= 4000 : 4003 index 81456 First element >= 5000 : 5021 index 98704 First element >= 6000 : 6009 index 121342 First element >= 7000 : 7035 index 151756 First element >= 8000 : 8036 index 168804 First element >= 9000 : 9014 index 184428 First element >= 10000 : 10007 index 201788
SETL
program inventory_sequence;
loop init
inv := {};
next := 1000;
while next <= 10**4 do
loop init
i := 0;
until el = 0 do
el := inv(i) ? 0;
if (seq +:= 1) <= 100 then
nprint(lpad(str el, 5));
if seq mod 10 = 0 then print; end if;
elseif el > next then
print("First > " + lpad(str next, 10) + ": "
+ lpad(str el, 10) + " at" + lpad(str(seq-1), 10));
next +:= 1000;
end if;
inv(el) +:= 1;
i +:= 1;
end loop;
end loop;
end program;
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First > 1000: 1001 at 24255 First > 2000: 2009 at 43301 First > 3000: 3001 at 61708 First > 4000: 4003 at 81456 First > 5000: 5021 at 98704 First > 6000: 6009 at 121342 First > 7000: 7035 at 151756 First > 8000: 8036 at 168804 First > 9000: 9014 at 184428 First > 10000: 10007 at 201788
Wren
import "dome" for Window
import "graphics" for Canvas, Color
import "./plot" for Axes
import "./iterate" for Stepped
import "./fmt" for Fmt
var max = 10000
var inv = [0]
var counts = List.filled(max + 100, 0) // say
counts[0] = 1
var lower = 100
var upper = 1000
var done = false
var ix = 0
while (!done) {
var i = 0
while(true) {
var j = counts[i]
if (inv.count < max) inv.add(j)
counts[j] = counts[j] + 1
ix = ix + 1
if (inv.count >= lower) {
System.print("Inventory sequence, first 100 elements:")
Fmt.tprint("$2d", inv[0..99], 20)
System.print()
lower = max + 1
}
if (j == 0) break
if (j >= upper) {
Fmt.print("First element >= $,6d is $,6d at index $,7d", upper, j, ix)
if (j >= max) {
done = true
break
}
upper = upper + 1000
}
i = i + 1
}
}
// generate points for the plot
var Pts = (0...max).map { |i| [i, inv[i]] }.toList
class Main {
construct new() {
Window.title = "Inventory sequence - first 10,000 elements."
Canvas.resize(1000, 600)
Window.resize(1000, 600)
Canvas.cls(Color.white)
var axes = Axes.new(100, 500, 800, 400, 0..10000, 0..450)
axes.draw(Color.black, 2)
var xMarks = Stepped.new(0..10000, 500)
var yMarks = Stepped.new(0..400, 50)
axes.mark(xMarks, yMarks, Color.black, 2)
var xMarks2 = Stepped.new(0..10000, 1000)
var yMarks2 = Stepped.new(0..400, 100)
axes.label(xMarks2, yMarks2, Color.black, 2, Color.black)
axes.lineGraph(Pts, Color.blue, 2)
}
init() {}
update() {}
draw(alpha) {}
}
var Game = Main.new()
- Output:
Terminal output:
Inventory sequence, first 100 elements: 0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1,000 is 1,001 at index 24,255 First element >= 2,000 is 2,009 at index 43,301 First element >= 3,000 is 3,001 at index 61,708 First element >= 4,000 is 4,003 at index 81,456 First element >= 5,000 is 5,021 at index 98,704 First element >= 6,000 is 6,009 at index 121,342 First element >= 7,000 is 7,035 at index 151,756 First element >= 8,000 is 8,036 at index 168,804 First element >= 9,000 is 9,014 at index 184,428 First element >= 10,000 is 10,007 at index 201,788
XPL0
include xpllib; \for Print
def Size = 201_790;
int Seq(Size), SeqEnd, Num, Count, N, Thresh;
[SeqEnd:= 0;
loop [Num:= 0;
repeat Count:= 0;
for N:= 0 to SeqEnd-1 do
if Num = Seq(N) then Count:= Count+1;
Seq(SeqEnd):= Count;
SeqEnd:= SeqEnd+1;
if SeqEnd >= Size then quit;
Num:= Num+1;
until Count = 0;
];
Thresh:= 1000;
for N:= 0 to SeqEnd-1 do
if N < 100 then
[Print("%3.0f", float(Seq(N)));
if rem(N/20) = 19 then CrLf(0);
]
else if Seq(N) >= Thresh then
[Print("First element >= %5.0f: %5.0f in position %6.0f\n",
float(Thresh), float(Seq(N)), float(N));
if Thresh >= 10000 then return;
Thresh:= Thresh + 1000;
];
]
- Output:
0 1 1 0 2 2 2 0 3 2 4 1 1 0 4 4 4 1 4 0 5 5 4 1 6 2 1 0 6 7 5 1 6 3 3 1 0 7 9 5 3 6 4 4 2 0 8 9 6 4 9 4 5 2 1 3 0 9 10 7 5 10 6 6 3 1 4 2 0 10 11 8 6 11 6 9 3 2 5 3 2 0 11 11 10 8 11 7 9 4 3 6 4 5 0 12 11 10 9 13 First element >= 1000: 1001 in position 24255 First element >= 2000: 2009 in position 43301 First element >= 3000: 3001 in position 61708 First element >= 4000: 4003 in position 81456 First element >= 5000: 5021 in position 98704 First element >= 6000: 6009 in position 121342 First element >= 7000: 7035 in position 151756 First element >= 8000: 8036 in position 168804 First element >= 9000: 9014 in position 184428 First element >= 10000: 10007 in position 201788