Randomize points on a disc

14

3

I read about circles somewhere, and just now learned about discs (it's actually a pretty common concept) and thought about codegolf.

Your task is to randomize a point/several points on a disc with the radius 1.

Rules:

  • All points must have an equal probability to be generated
  • Floating point coordinates must be used; the minimum requirement is two decimals (e.g. the points (0.12, -0.45) or (0.00, -1.00) are valid)
  • You get -20 bytes if your program actually displays the bounding circle and the point(s) generated in it. Coordinates still has to be valid but not displayed, and the image generated has to be at least 201 by 201 pixels in size
  • You get -5 bytes if your program takes the number of points to be generated as input on stdin
  • If you decide not to plot the bounding circle and the point(s), your program has to output the point(s) generated on the format (x, y) or (x,y) on stdout
  • If you decide to take the number of generated points as input, but not to plot it - your program has to output all the randomized points on the format stated above with or without one space in between

Shortest submission in bytes wins!

sweerpotato

Posted 2015-09-06T16:41:59.597

Reputation: 2 457

1@sweerpotato Yes, please specify that all the points in and on the circle are valid. I did not realize you meant both. Also, this question seems like it would fit a code-golf challenge better than a popularity-contest, but that's just my opinion. – cole – 2015-09-06T16:59:57.457

5"Do XYZ in a creative way" is the classic Bad Popcon Question™. What one person considers creative is what another person considers the obvious way. – Peter Taylor – 2015-09-06T17:01:54.360

Out of curiosity, why a 201x201 pixel output requirement for plots? – JohnE – 2015-09-06T17:39:19.903

@JohnE I suggested 201x201 pixels as it matches the required 2 decimal place accuracy – trichoplax – 2015-09-06T17:39:49.840

May we output the coordinates as complex numbers? For example: 0.3503082505747327+0.13499221288682994j. – orlp – 2015-09-06T18:00:15.533

If you decide to output the points graphically you may calculate the points as complex numbers as long as the coordinates are valid. If you decide to use stdout you have to output the coordinates on the format stated – sweerpotato – 2015-09-06T18:13:34.320

In what format(s) may we output the points when we need to give back a list of points (to qualify for the -5)? – orlp – 2015-09-06T18:29:34.657

On the format stated with or without one space in between the generated points (last bullet) – sweerpotato – 2015-09-06T18:30:45.370

Is newlines also ok (so one coordinate per line)? Or must it always be spaces? – orlp – 2015-09-06T18:31:12.320

No newlines, one or zero spaces in between the generated points – sweerpotato – 2015-09-06T18:31:47.767

extra points: demonstrate that uniform density in both polar coordinates and carthesian system both give valid output (unless you consider that the limit of precision defines a point, which therefore is not a point, but a square) – njzk2 – 2015-09-08T02:29:29.777

Can we remove the space after the comma in the output? – J F – 2015-09-08T10:31:10.613

Answers

5

Pyth, 26 - 5 = 21 bytes

VQp(sJ*@OZ2^.n1*yOZ.l_1)eJ

Takes the number of coordinates to generate on stdin, and outputs them on stdout like such:

(-0.5260190768964058, -0.43631187015380823)(-0.12127959509302746, -0.08556306418467638)(-0.26813756369750996, -0.4564539715526493)

Uses a strategy similar to @MartinBüttner, generating polar coordinates and radii, except it does it using complex exponentiation.

orlp

Posted 2015-09-06T16:41:59.597

Reputation: 37 067

You can remove the p, can't you? It just changes the output to separate lines. – PurkkaKoodari – 2015-09-07T09:00:50.980

@Pietu1998 That's not allowed, see the comments on the main question. – orlp – 2015-09-07T09:06:05.507

Oh, a‍ll right. – PurkkaKoodari – 2015-09-07T09:07:58.143

16

CJam, 28 27 bytes

PP+mr_mc\ms]1.mrmqf*"(,)".\

This solution is not rejection-based. I'm generating the points in polar coordinates, but with a non-uniform distribution of the radii to achieve a uniform density of the points.

Test it here.

Explanation

PP+     e# Push 2π.
mr_     e# Get a random float between 0 and 2π, make a copy.
mc\     e# Take the cosine of one copy and swap with the other.
ms]     e# Take the sine of the other copy and wrap them in an array.
        e# This gives us a uniform point on the unit circle.
1.mr    e# Get a random float between 0 and 1.
mq      e# Take the square root. This is the random radius.
f*      e# Multiply x and y by this radius.
"(,)".\ e# Put the resulting numbers in the required format.

Why does it work? Consider a narrow annulus of radius r and (small) width dr. The area is approximately 2π*r*dr (if the annulus is narrow, the inner and outer circumference are almost identical, and the curvature can be ignored, such that the area can be treated as that of a rectangle with side lengths of the circumference and the width of the annulus). So the area increases linearly with the radius. This means that we also want a linear distribution of the random radii, in order to achieve a constant density (at twice the radius, there is twice as much area to fill, so we want twice as many points there).

How do we generate a linear random distribution from 0 to 1? Let's look at the discrete case first. Say, we have a desired distribution of 4 values, like {0.1, 0.4, 0.2, 0.3} (i.e. we want 1 to be 4 times as common as 0, and twice as common as 2; we want 3 three times as common as 0):

enter image description here

How can pick one of the four values with the desired distribution? We can stack them up, pick a uniformly random value between 0 and 1 on the y-axis and pick the segment at that point:

enter image description here

There's a different way to visualise this picking though. We could instead replace each value of the distribution with the accumulation of the values up to that point:

enter image description here

And now we treat the top line of this chart as a function f(x) = y and invert it to get a function g(y) = f-1(y) = x, which we can apply to a uniformly random value in y ∈ [0,1]:

enter image description here

Cool, so how can make use of this to generate a linear distribution of radii? This is the distribution we want:

enter image description here

The first step is to accumulate the values of the distribution. But the distribution is continuous, so instead of summing over all previous values, we take an integral from 0 to r. We can easily solve that analytically: 0r r dr = 1/2 r2. However, we want this to be normalised, i.e. to multiply it by a constant such that this gives 1 for the maximum value of r, so what we really want is r2:

enter image description here

And finally, we invert this to get a function we can apply to a uniform value in [0,1], which we can again do analytically: it's just r = √y, where y is the random value:

enter image description here

This is a fairly useful technique that can often be used to generate simple distributions exactly (it works for any distribution, but for complicated ones the last two steps may have to be solved numerically). However, I wouldn't use it in this particular case in production code, because the square root, sine and cosine are prohibitively expensive: using a rejection-based algorithm is on average much faster, because it only needs addition and multiplication.

Martin Ender

Posted 2015-09-06T16:41:59.597

Reputation: 184 808

1Very nice explanation! – sweerpotato – 2015-09-06T19:43:05.663

2Mmm pictures :D – Beta Decay – 2015-09-06T21:06:32.930

12

Mathematica, 68 44 - 20 = 24 bytes

Many thanks for David Carraher for letting me know about RandomPoint, which saved 24 (!) bytes. Mathematica does have a built-in for everything.

Graphics@{Circle[],Point@RandomPoint@Disk[]}

This plots the point and the bounding circle to qualify for the bonus:

enter image description here

The result is a vector image, so the size specification of 201x201 pixels doesn't really make sense, but by default it does render larger than that.

Martin Ender

Posted 2015-09-06T16:41:59.597

Reputation: 184 808

How about Graphics[{Circle[], Point@RandomPoint@Disk[]}] ? – DavidC – 2015-09-06T19:18:57.023

Be my guest. Also, to save 1 byte...Graphics@{Circle[], Point@RandomPoint@Disk[]} – DavidC – 2015-09-06T19:21:36.477

@DavidCarraher Thanks a lot! :) – Martin Ender – 2015-09-06T19:31:41.133

I don't know Mathematica syntax but surely you can save another byte by removing the space after the ,? – fluffy – 2015-09-07T05:39:10.333

@fluffy I already did in the posted version – Martin Ender – 2015-09-07T06:11:28.807

Ah, so you did. – fluffy – 2015-09-07T20:29:16.993

9

CJam, 31 26 bytes

{];'({2dmr(_}2*@mhi}g',\')

This works by repeatedly generating random points in a square of side length 2 and keeping the first that falls inside the unit disc.

Thanks to @MartinBüttner for golfing off 3 bytes!

Try it online in the CJam interpreter.

How it works

{                  }g       Do:
 ];'(                         Clear the stack and push a left parenthesis.
     {      }2*               Do twice:
      2dmr                      Randomly select a Double between 0 and 2.
          (_                    Subtract 1 and push a copy.
               @              Rotate the copy of the first on top of the stack.
                mh            Compute the Euclidean norm of the vector consisting
                              of the two topmost Doubles on the stack.
                  i           Cast to integer.
                            If the result is non-zero, repeat the loop.
                     ',\    Insert a comma between the Doubles.
                        ')  Push a right parenthesis.

Dennis

Posted 2015-09-06T16:41:59.597

Reputation: 196 637

8

iKe, 53 51 bytes

Nothing particularly special, but I suppose we ought to have at least one graphical solution:

,(80+160*t@&{.5>%(x*x)+y*y}.+t:0N 2#-.5+?9999;cga;3)

plot

Try it in your browser.

Edit: I can shave off two bytes by applying @MartinBüttner's approach for modifying the distribution of polar coordinates. I think it's also quite a bit more direct:

,(80*1+(%?c){x*(cos y;sin y)}'6.282*?c:9999;cga;3)

JohnE

Posted 2015-09-06T16:41:59.597

Reputation: 4 632

3If you'd also draw the bounding circle you'd qualify for -20. – orlp – 2015-09-06T18:01:43.883

1iKe has a raster-based drawing model, making that requirement rather unfair. I think it would cost quite a bit more than 20 characters to render an approximation of a bounding circle as well. – JohnE – 2015-09-06T18:32:16.683

7

Perl, 59 Bytes

while(($x=1-rand 2)**2+($y=1-rand 2)**2>1){};print"($x,$y)"

This is just a simple solution, generating points in a square and rejecting those too far away. My singular golfing trick is to include the assignments inside of the condition.

Edit: In the process of golfing, I found an interesting way to print random points on a circle.

use Math::Trig;$_=rand 2*pi;print"(",sin,",",cos,")"

PhiNotPi

Posted 2015-09-06T16:41:59.597

Reputation: 26 739

7

Octave, 24 53 - 20 = 33 bytes

polar([0:2e-3:1,rand]*2*pi,[ones(1,501),rand^.5],'.')

Generates 501 equally-spaced theta values plus one random number and scales them all to [0..2π]. Then generates 501 1's for the radius of the circle, plus a random radius for the point and takes the square root to ensure uniform distribution over the disc. Then plots all the points as polar coordinates.

enter image description here


Here is a quick demonstration of the distribution (without the unit circle):

polar(2*pi*rand(99),rand(99).^.5,'.')

9801 Points

beaker

Posted 2015-09-06T16:41:59.597

Reputation: 2 349

5

Octave / Matlab, 74 64 bytes

Rejection method, 64 bytes:

u=1;v=1;while u^2+v^2>1
u=rand;v=rand;end
sprintf('(%f,%f)',u,v)

Direct method, 74 bytes (thanks to Martin Büttner for helping me correct two errors):

t=rand*2*pi;r=1-abs(1-sum(rand(2,1)));sprintf('(%f,%f)',r*cos(t),r*sin(t))

Luis Mendo

Posted 2015-09-06T16:41:59.597

Reputation: 87 464

5

R, 99 95 81-20 = 79 75 61 Bytes

symbols(0,0,1,i=F,asp=1,ylim=c(-1,1));points(complex(,,,runif(9),runif(9,-1)*pi))

Use the complex number construction to build the x/y's from polar coordinates. Taking the input was a bit expensive and there is probably a better way to do this. The ylim and xlim is to ensure the entire circle is plotted and the asp ensures the points are shown under the circle symbol.

Thanks to @jbaums and @flodel for the savings

Try it here

MickyT

Posted 2015-09-06T16:41:59.597

Reputation: 11 735

runif(9,0,1) can be simplified to runif(9) – jbaums – 2015-09-06T20:58:07.827

@jbaums, thanks ... one of things that I always seem to forget:) – MickyT – 2015-09-06T21:37:58.520

Can shave 14: symbols(0,0,1,i=F,asp=1,ylim=c(-1,1));points(complex(,,,runif(9),runif(9,-1)*pi)) – flodel – 2015-09-06T22:07:13.473

@flodel very nice thank you. – MickyT – 2015-09-06T22:51:02.417

Another teeny saving: yli works in place of ylim. – jbaums – 2015-09-08T07:24:57.687

4

Processing /Java 141 bytes-20=121

the requirement for 201*201 being the minimum size requires me to put in the setup method since Processing.org defaults to 200x200 :(

void setup(){noFill();size(201,201);}void draw(){float f=10,a=PI*2*random(),r=random();point(f+f*sin(a)*r,f+f*cos(a)*r);ellipse(f,f,f*2,f*2)}

Timothy Groote

Posted 2015-09-06T16:41:59.597

Reputation: 245

I didn't know processing/java was allowed, neat! – J Atkin – 2015-09-09T12:42:50.710

4

QBasic, 138 bytes - 20 - 5 = 113

INPUT n
r=200
SCREEN 12
RANDOMIZE TIMER
CIRCLE(r,r),r
PAINT(r,r)
FOR i=1TO n
DO
x=RND*r*2
y=RND*r*2
LOOP UNTIL POINT(x,y)
PSET(x,y),1
NEXT

Takes user input and draws the disc and points. Tested on QB64.

This is a pretty basic "throw at the dartboard and keep what sticks" strategy. The catch is that "what sticks" is not determined mathematically but graphically: a white disc is plotted on a black background, and then randomly generated points are rejected until they are not black. The points themselves are drawn in blue (though it's hard to tell when they're single pixels--click on the image to enlarge).

DLosc

Posted 2015-09-06T16:41:59.597

Reputation: 21 213

3

awk - 95 - 5 = 90

{
    for(;$1--;printf"("(rand()<.5?x:-x)","(rand()<.5?y:-y)")")
        while(1<(x=rand())^2+(y=rand())^2);
}

Since i wasn't quite sure about the rand()<.5 part, I did some distribution testing with this, using this script:

BEGIN{ srand() }
{ 
    split("0 0 0 0", s)
    split("0 0 0 0", a)

    for(i=$1; i--; )
    {
        while( 1 < r2 = ( x=rand() )^2 + ( y=rand() )^2 );

        x = rand()<.5 ? x : -x
        y = rand()<.5 ? y : -y

        ++s[ x>=0 ? y>=0 ? 1 : 4 : y>=0 ? 2 : 3 ]

        ++a[ r2>.75 ? 1 : r2>.5 ? 2 : r2>.25 ? 3 : 4]
    }

    print "sector distribution:"
        for(i in s) print "sector " i ": " s[i]/$1

    print "quarter area distribution:"
        for(i in a) print "ring " i ":   " a[i]/$1
}

which for an input of 1e7 gives me this result, after I sipped once or twice at my coffee:

1e7
sector distribution:
sector 1: 0.250167
sector 2: 0.249921
sector 3: 0.249964
sector 4: 0.249948
quarter area distribution:
ring 1:   0.24996
ring 2:   0.25002
ring 3:   0.250071
ring 4:   0.249949

which I think is quite alright.

A little explanation:
After scribbling about for a while it turned out, that if you want to divide the disc into four rings with equal area, the radii where you would have to cut are sqrt(1/4), sqrt(1/2) and sqrt(3/4). Since the actual radius of the point I test would be sqrt( x^2 + y^2), I can skip the square rooting all together. The 1/4, 2/4, 3/4 "coincidence" might be related to what M. Buettner pointed out earlier.

Cabbie407

Posted 2015-09-06T16:41:59.597

Reputation: 1 158

3

HPPPL, 146 (171-20-5) bytes

EXPORT r(n)BEGIN LOCAL R,A,i,Q;RECT();Q:=118.;ARC_P(Q,Q,Q);FOR i FROM 1 TO n DO R:=√RANDOM(1.);A:=RANDOM(2*π);PIXON_P(G0,IP(Q+Q*R*COS(A)),IP(Q+Q*R*SIN(A)));END;FREEZE;END;

Example for 10000 points (including timing in seconds for the real device):

Randomize points on a disc, timing

The function itself is called by r(n). The rest in the image above is only for timing purposes.

Result (disc diameter is 236 pixels):

enter image description here

The version above does not store the point coordinates, so I wrote a version that takes two parameters r(n,p). n is the number of points and p=0 returns the points to the terminal, p=1 plots the points and the disc), in case storing coordinates is mandatory. This version is 283 (308-20-5) bytes long:

EXPORT r(n,p)BEGIN LOCAL R,A,j,Q,x,y;Q:=118.0;CASE IF p==0 THEN print() END IF p==1 THEN RECT();ARC_P(Q,Q,Q) END END;FOR j FROM 1 TO n DO R:=√RANDOM(1.0);A:=RANDOM(2*π);x:=R*COS(A);y:=R*SIN(A);CASE IF p==0 THEN print("("+x+", "+y+")") END IF p==1 THEN PIXON_P(G0,IP(Q+Q*x),IP(Q+Q*y)) END END;END;FREEZE;END;

The ungolfed version:

EXPORT r(n,p)
BEGIN
LOCAL R,A,j,Q,x,y;
  Q:=118.0;
  CASE
    IF p==0 THEN print() END
    IF p==1 THEN RECT();ARC_P(Q,Q,Q) END
  END;
  FOR j FROM 1 TO n DO
    R:=√RANDOM(1.0);
    A:=RANDOM(2*π);
    x:=R*COS(A);
    y:=R*SIN(A);
    CASE
      IF p==0 THEN print("("+x+", "+y+")") END
      IF p==1 THEN PIXON_P(G0,IP(Q+Q*x),IP(Q+Q*y)) END
    END;
  END;
  FREEZE;
END;

Terminal output for r(10,0):

Randomize points on a disc terminal output

r(10,1) shows the disc with the points, like shown above.

M L

Posted 2015-09-06T16:41:59.597

Reputation: 2 865

2

JavaScript, 75 bytes

Rejection-based:

do x=(r=()=>4*Math.random()-2)(),y=r()
while(x*x+y*y>1)
alert(`(${[x,y]})`)

Direct method (80 bytes):

alert(`(${[(z=(m=Math).sqrt((r=m.random)()))*m.sin(p=m.PI*2*r()),z*m.cos(p)]})`)

Ypnypn

Posted 2015-09-06T16:41:59.597

Reputation: 10 485

2

Python, 135 130 bytes

from random import*
def r():return uniform(-1,1)
p=[]
while not p:
    x,y=r(),r()
    if x**2+y**2<=1:p=x,y
print'(%.2f, %2f)'%p

Removed the **0.5 thanks to @jimmy23013’s suggestion (because it is a unit circle, I am now checking whether the distance squared between (x, y) and (0, 0) is equal to 12. This is the same thing).

This also freed me to remove the parentheses.

J F

Posted 2015-09-06T16:41:59.597

Reputation: 281

I think you don't need the **0.5. – jimmy23013 – 2015-09-08T13:19:18.147

@jimmy23013 Thank you! removed. – J F – 2015-09-08T14:25:45.457