Example 8.35: Grab true (not pseudo) random numbers; passing API URLs to functions or macros
[This article was first published on SAS and R, and kindly contributed to R-bloggers]. (You can report issue about the content on this page here)
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.
Usually, we’re content to use a pseudo-random number generator. But sometimes we may want numbers that are actually random– an example might be for randomizing treatment status in a randomized controlled trial.Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.
The site Random.org provides truly random numbers based on radio static. For long simulations, its quota system may prevent its use. But for small to moderate needs, it can be used to provide truly random numbers. In addition, you can purchase larger quotas if need be.
The site provides APIs for several types of information. We’ll write functions to use these to pull vectors of uniform (0,1) random numbers (of 10^(-9) precision) and to check the quota. To generate random variates from other distributions, you can use the inverse probability integral transform (section 1.10.8).
The coding challenge here comes in integrating quotation marks and special characters with function and macro calls.
SAS
In SAS, the challenging bit is to pass the desired number of random numbers off to the API, though the macro system. This is hard because the API includes the special characters ?, ", and especially &. The ampersand is used by the macro system to denote the start of a macro variable, and is used in APIs to indicate that an additional parameter follows.
To avoid processing these characters as part of the macro syntax, we have to enclose them within the macro quoting function %nrstr. We use this approach twice, for the fixed pieces of the API, and between them insert the macro variable that contains the number of random numbers desired. Also note that the sequence %" is used to produce the quotation mark. Then, to unmask the resulting character string and use it as intended, we %unquote it. Note that the line break shown in the filename statement must be removed for the code to work.
Finally, we read data from the URL (section 1.1.6) and transform the data to take values between 0 and 1.
%macro rands (outds=ds, nrands=); filename randsite url %unquote(%nrstr(%"http://www.random.org/integers/?num=) &nrands%nrstr(&min=0&max=1000000000&col=1&base=10&format=plain&rnd=new%")); proc import datafile=randsite out = &outds dbms = dlm replace; getnames = no; run; data &outds; set &outds; var1 = var1 / 1000000000; run; %mend rands; /* an example macro call */ %rands(nrands=25, outds=myrs);
The companion macro to find the quota is slightly simpler, since we don’t need to insert the number of random numbers in the middle of the URL. Here, we show the quota in the SAS log; the file print syntax, shown in Example 8.34, can be used to send it to the output instead.
%macro quotacheck; filename randsite url %unquote(%nrstr(%"http://www.random.org/quota/?format=plain%")); proc import datafile=randsite out = __qc dbms = dlm replace; getnames = no; run; data _null_; set __qc; put "Remaining quota is " var1 "bytes"; run; %mend quotacheck; /* an example macro call */ %quotacheck;
R
Two R functions are shown below. While the problem isn’t as difficult as in SAS, it is necessary to enclose the character string for the URL in the as.character() function (section 1.4.1).
truerand = function(numrand) { read.table(as.character(paste("http://www.random.org/integers/?num=", numrand, "&min=0&max=1000000000&col=1&base=10&format=plain&rnd=new", sep="")))/1000000000 } quotacheck = function() { line = as.numeric(readLines("http://www.random.org/quota/?format=plain")) return(line) }
To leave a comment for the author, please follow the link and comment on their blog: SAS and R.
R-bloggers.com offers daily e-mail updates about R news and tutorials about learning R and many other topics. Click here if you're looking to post or find an R/data-science job.
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.