挑战
给定一个正整数(K)
输出一个均匀随机整数(Y)
之间[0, K)
。
如果Y > 0
假设K = Y
并重复该过程,直到Y = 0
。
规则
- 输入必须首先打印
- 您想要的输出格式
- Your program must finish.
0
must be the final output, Optionally an empty line instead0
挑战
给定一个正整数(K)
输出一个均匀随机整数(Y)
之间[0, K)
。
如果Y > 0
假设K = Y
并重复该过程,直到Y = 0
。
规则
0
must be the final output, Optionally an empty line instead 0
Answers:
.uOW
.uOW Full program. Takes an integer from STDIN and outputs a list to STDOUT. .u Cumulative fixed-point. Apply the given function with a given initial value, which is implicitly assigned to the input, until a result that has occurred before is found. Returns the list of intermediate results. W Conditional application. If the argument (the current value) is truthy, then apply the function below, otherwise leave it unchanged. O Random integer in the range [0, N). IOW: at each iteration of .u, assign a variable N to the current value, starting with the input. If N is not 0, then choose a random integer in [0, N), else return N unchanged. Whenever we encounter a 0, the next iteration must also result in a 0, and therefore the loop stops there.
f(_){printf("%d\n",_);(_=rand()%_)&&f(_);}
Uses short-circuited logical and.
f(_){ // f(int _) {
printf("%d\n",_); // print argument and a newline
(_=rand()%_) // set _ to rand()%_
&&f(_);} // short-circuit AND to recursively call f if _ not zero
f(_){printf("%d\n",_=rand()%_);_&&f(_);}
Uses short-circuited logical and.
f(_){ // f(int _) {
printf("%d\n", // print an integer and a newline
_= // The integer is _ which we set to...
rand()%_); // a random value modulo the input _
_&&f(_);} // short-circuit AND to recursively call f if _ not zero
rand()%_
is not uniform
>0
and cat(n,"")
(empty string) will also work.
k=scan();while(x<-sample(1:k-1,1))k=c(x,k);cat(rev(k),0)
n=scan();while(print(n))n=sample(n,1)-1
`tYrqt
` % Do...while
t % Duplicate. Takes input (implicit) the first time
Yr % Uniform random integer from 1 to n, included
q % Subtract 1
t % Duplicate. This will be used as loop condition
% End (implicit). Proceeds with next iteration if non-zero
% Display stack (implicit)
Pepe is a programming language made by user Soaku.
REeErEErReEEreeEREEeEEree
Explanation:
REeErEErReEEreeEREEeEEree # full program
REeE # input as num, in stack 1
rEE # create loop in stack 2 with name 0
rReEE # - output and preserve the number in stack 1
reeE # - output a newline "\n"
REEeEE # - random number by 0 to input
ree # goto loop with name 0 if stack 1 is not equal
to stack 2
{$_,(^*).pick...0}
Anonymous code block that returns a list of values. If you don't mind the numbers being ranges, you can do:
{^$_,^*.pick...0}
for 17 bytes. Funnily enough, another builtin random function, roll
, has the same behaviour in this instance for the same amount of bytes.
XƬ0
This is a monadic link (function) that prints an array and returns 0.
XƬ0 Monadic link. Argument: n
XƬ Pseudo-randomly pick (X) an integer k in [1, ..., n], set n = k, and repeat.
Do this 'til (Ƭ) the results are no longer unique and return the array of
unique results, including the initial value of n.
This stops once X returns k with argument k. The second k will be omitted
from the return value.
0 Print the resulting array and set the return value to 0.
ẉ?ℕ₁-₁ṙ↰
ẉ Write the input followed by a linebreak
?ℕ₁ The input must be in [1, …, +∞)
-₁ṙ Generate an integer in [0, …, input - 1] uniformly at random
↰ Recursive call with that random integer as the new input
The recursion will stop once ?ℕ₁
fails, that is, when the input is 0
.
Δ=L<Ω0M
Explanation
Δ # loop until value doesn't change
= # print current value
L<Ω # push a random number in ([1 ... X] - 1)
# will return -1 when X=0
0M # push max of that and 0
Δ=ݨΩ0M
is equivalent.
[:}:? ::]^:a:
On the subway, so apologies for lack of TIO (hopefully there isn’t a lack of correctness).
Outputs a list of values.
Presumably the APL approach will be shorter, but this is what I thought of.
^:a:
apply repeatedly until convergence, storing intermediate results in an array.
?
random integer in range [0, K)
for K
greater than 0. For 0, it gives a random integer in range (0,1)
. For a floating point number, it errors.
::]
catch an error for an input to ?
and instead of erroring, output the input that caused the error.
}:
get rid of the last value in the array (this is so that a floating point number isn’t output).
?.
, but I don’t think I’m using that.
-1 byte thanks to @Arnauld
f=n=>[n,...n?f(Math.random()*n|0):[]]
new Date%n
doesn't really work here, since it doesn't change fast enough to be useful for generating multiple random numbers
f(k){printf("%d ",k);k?f(rand()%k):0;}
Ungolfed
void f(int k){
printf("%d ",k);
if(k)
f(rand()%k);
}
&&
; also, you may want to consider seeding the RNG in your main
function: Try it online!
W
~O
This basically implements the algorithm:
To translate the Pyth into the algorithm, we can mostly just examine what each character means. Since Pyth is written in prefix notation (i.e. * + 1 2 3
is (1 + 2) * 3
) we can start from the left and fill in the arguments as we go.
W
begins a traditional while loop. The first statement after it is the loop condition and the second statement after it is the loop body. If the second statement is empty it becomes a no-op. This while works exactly like the Python while, so it will evaluate non-zero integers as True and zero as false.
The first statement after the while begins with the newline character. This corresponds to Pyth's "print and return with a newline" function. This takes one argument, which is then printed and also returned unmodified. This allows us to print the intermediate steps while also performing the needed operations.
The argument passed to this print function begins with ~
which is a bit special. If the character immediately after ~
is a variable it takes two arguments, otherwise it takes one. Since O
is not a variable ~
will consume only one argument. ~
functions a bit like +=
does in many conventional languages, though the closest operator would be the post-increment operator ++
from C
. You may know that x++
will be like using x
as the current value, but thereafter x
will be x+1
. ~
is the same idea, but generalised to whatever the result of the first argument is. How it picks what variable to assign to will be addressed later.
The argument of ~
is O
which is very simple. When its one argument is an integer O
returns a value from 0 to one less than that integer uniformly at random.
Now you may have noticed O
does not have an argument. Here the Pyth interpreter kindly fills in a guess, which here is the variable Q
. Q
has a special meaning in Pyth: whenever it is present in a program the Pyth program begins with assigning Q
to the input of the program. Since this is the first variable occurring in ~
's argument Q
is also now the variable that ~
will assign a value to.
Summed up our "readable" program might look like:
while print_and_return( assign_variable( Q, unif(0, Q-1) ) ):
pass
And one sample "run-through" might look like:
O
returns 3, ~
returns 5, \n
returns and prints 5 which is trueO
returns 0, ~
returns 3, \n
returns and prints 3 which is trueO
returns something irrelevant, ~
returns 0, \n
returns and prints 0 which is falseAnonymous tacit prefix function. Assumes ⎕IO
(Index Origin) to be 0
, which is default on many systems. Returns the final value (0) in addition to printing while run.
{⌊?⎕←⍵}⍣=
{
…}⍣=
apply the following function until stable:
⎕←⍵
output the argument
?
return a uniformly distributed random number in the range 0 through that–1
⌊
round down (because ?0
gives a (0,1) float)
Some idiot™ forgot to print the initial value first.
f(K){while(K)printf("%d\n",K,K=rand()%K);}
Don't panic.
f(K){while(K)printf("%d\n",K),K=rand()%K;}
. Still you got my +1 for an equal solution!
f(K){while(K)printf("%d\n",K,K=rand()%K);}
Straightforward implementation. Takes input K in ecx
and outputs to a buffer in ebx
.
0000000a <start>:
a: 0f c7 f0 rdrand %eax
d: 31 d2 xor %edx,%edx
f: f7 f1 div %ecx
11: 89 13 mov %edx,(%ebx)
13: 83 c3 04 add $0x4,%ebx
16: 89 d1 mov %edx,%ecx
18: 85 c9 test %ecx,%ecx
1a: 75 ee jne a <start>
1c: c3 ret
f=lambda k:print(k)or f(hash('.'*k)%k)
Probably not the most cryptographically secure random number generator but to the human eye it looks random enough...
k=0
is acceptable.
hash()
tries to maintain but doesn't guarantee this property. For that task you should use the random
module.
random.randrange()
: from random import*;f=lambda k:print(k)or f(randrange(k))
While Ans
Disp Ans
int(randAns
End
Ans
-4 bytes from Misha Lavrov
Takes input in Ans
as 50:prgmNAME
.
TI-Basic is a tokenized language. All tokens used here are one byte.
Explanation:
While Ans # 3 bytes, While the number we hold is not zero:
Disp Ans # 3 bytes, Display it on its own line
int(randAns # 4 bytes, and replace it with a number randomly
# chosen from 0 to one less than it (inclusive)
End # 2 bytes, end While loop
Ans # 1 byte, Display (and return) zero
An 11-byte solution suggested by Misha Lavrov that requires pressing enter
after each line following the first.
Ans
While Ans
Pause int(randAns
End
int(Ansrand
is shorter. Also, using Pause
instead of Disp
, you can make the only statement in the loop be Pause int(Ansrand
, which also updates Ans
.
from random import*
k=input()
while k:print k;k=randrange(k)
Saved
while 1:print k;k=randint(0,~-k)
should work (with an error at the end)
while 1:print k;k=randrange(k)
saves two.
#import<cstdio>
#import<cstdlib>
#define d printf("%i ",x
int p(int x){d);while(x>0)d=rand()%x);}
Usage
int main() {
p(100);
}
This is my first code golf attempt. Any feedback or remarks are welcome.
Edit: Removed the main function as suggested to make it valid.
-Dd='printf("%i,",x'
instead of the #define
would save you some bytes (-4), and is allowed as long as you count the bytes towards your result (because it is a non-standard preprocessor directive. You can also leave out the imports (at least with -std=c++98
and -w
, which don't count for bytes), and the variable types. So, you'd have p(x){d);while(x>0)d=rand()%x;}
and -Dd='printf("%i,",x'
.
:nao:0=?;0_1>:{:}(?\\
}(?!\$2*1>x~\$+1$*2/\~00:{{:@}
8+1.\~:{:}\+>$1+f
~~@~~47*0.\(a2*1@@?!.
+2B for -v flag
><>'s only source of randomness comes from the 'x' instruction, which sets the instruction pointer's direction to a random value. As such, generating a random number from 0 to n isn't trivial.
I first calculate how many bits are required to represent a number in the range [0,n), then generate random bits to generate a random number. This leaves the possibility that it'll generate a number slightly larger than n, in which case we just discard it and try again.
:nao Print the current number followed by a newline
:0=?; If the current n is 0, terminate
Calculate how many random bits we need to be generating:
0 1 Initialise the variables for this loop:
numberOfBits = 0, maxValue = 1
:{:}(?\ If maxValue >= n, break out of the loop
*2 maxValue *= 2
$+1$ numberOfBits += 1
Generate the random number:
~ Delete maxValue
00 Initialise randomNumber = 0, i = 0
}(?!\ :{{:@} If i >= numberOfBits, break out of the (inner) loop
$2* randomNumber *= 2
_
1>x~\ The random bit: If the IP goes up or
\+> left, it'll be redirected back onto the 'x',
if it goes down, it adds one to randomNumber
If it goes right, it does nothing to randomNumber
$1+ increment i
8+1. f Jump back to the start of the inner loop
After we've generated our number, check that it's actually below n
~ Delete i
:{:} ( ? Test that the number is less than n
a2*1 . If it's not, jump back to the start
of the number generation section
@~~ Otherwise delete the old value of n, and numberOfBits
47*0. Then jump back to the start of the program
@(k)eval('while k;disp(k);k=randi(k)-1;end;0')
Sample output:
>> @(k)eval('while k;disp(k);k=randi(k)-1;end;0')
ans(5)
ans =
@(k)eval('while k;disp(k);k=randi(k)-1;end;0')
5
3
2
1
ans =
0
k=randi(k)-1
for a few bytes less.
.+
*
L$`.
$.`
+¶<)G?`
Try it online! Explanation:
+
Repeat until the value stops changing (i.e. 0).
¶<)
Print the value before each pass through the loop.
.+
*
Convert to unary.
L$`.
$.`
Create the range and convert to decimal.
G?`
Pick a random element.
QWQ=OQQ
+1 to print the initial input value.
While Q is truthy, set Q to be a random integer between 0 and Q and print Q.
Not the shortest Pyth answer but I'm just learning and only posting because of the recent discussion about no-one using Pyth any more :)
=
and ~
use the first variable of an expression as the variable that will be assigned if one isn't specified. For example ~hT
will set T
to 11 while returning 10. The only other fancy trick is that the newline character prints its input and then returns that value unmodified, so we can have an empty loop body. Let me know if something else is confusing :)
-3 bytes by actually doing what the specs say it should do.
import System.Random
f 0=pure[0]
f x=randomRIO(0::Int,x-1)>>=fmap(x:).f
-4 bytes thanks AdmBorkBork
filter f{$_;if($_){Random $_|f}}
Testscript:
filter f{$_;if($_){Random $_|f}}
100 |f
Output:
100
61
20
8
6
3
0
for($a="$args";$a;$a=Random $a){$a}
Full program. Takes input $args
, stores it into $a
, and enters a for
loop. Each iteration we're checking whether $a
is still positive (as 0
is falsey in PowerShell). Then we leave $a
on the pipeline and move to the next iteration, where we set $a
to be the result of Get-Random $a
, which returns an integer in the range 0..($a-1)
.
(Ab)uses the fact that PowerShell outputs an additional trailing newline in lieu of outputting the final zero (allowed by the rules as currently written).
"$args"
- nice. I was stuck on $args[0]
in this case