Ask Ben: Getting A Random Date From A Date Range In SQL

Posted October 3, 2006 at 7:44 AM by Ben Nadel

Tags: SQL, Ask Ben

How can I select a random date from a date range in SQL server. I basically want to do RandRange() for dates in sql? Does that make sense?

I have to admit, this one totally stumped me at first. I rarely work with random data in SQL. If I need random data I usually just build a query in ColdFusion. I tried looking up random methods for SQL, but it turns out that things like RAND() are only called once per table, NOT per row. I got some great ideas from Jon Galloway over at http://weblogs.asp.net/jgalloway/archive/2004/03/18/92498.aspx. He supplied the way to come up with a random date:

  • cast(
  • cast( getdate() as int )
  • -5 * rand( cast( cast( newid() as binary(8) ) as int ) )
  • as datetime
  • )

I am not exactly sure what the casting of the UUID to a binary(8) does, but I see that it converts to an integer.

After some experimentation, I see that casting UUIDs to binary can result in either a negative or positive number. For a date range, I needed to work with a positive number. Therefor I use ABS() to force a positive outcome. Once we have that, all we need is the date range part. That's where our old friend date-math comes in. You can think of a random date from a date range as adding a random number of days to the start date (>= 0) such that the resultant date does not exceed the end range date.

Once you think about it that way, we have all the pieces we need:

  • -- First, let's declare the date range. I am declaring this
  • -- here for the demo, but this could be done anyway you like.
  • DECLARE @date_from DATETIME;
  • DECLARE @date_to DATETIME;
  •  
  • -- Set the start and date dates. In this case, we are using
  • -- the month of october, 2006.
  • SET @date_from = '2006-10-01';
  • SET @date_to = '2006-10-30';
  •  
  • -- Select random dates.
  • SELECT
  • (
  • -- Remember, we want to add a random number to the
  • -- start date. In SQL we can add days (as integers)
  • -- to a date to increase the actually date/time
  • -- object value.
  • @date_from +
  • (
  • -- This will force our random number to be GTE 0.
  • ABS(
  •  
  • -- This will give us a HUGE random number that
  • -- might be negative or positive.
  • CAST(
  • CAST( NewID() AS BINARY(8) )
  • AS INT
  • )
  • )
  •  
  • -- Our random number might be HUGE. We can't have
  • -- exceed the date range that we are given.
  • -- Therefore, we have to take the modulus of the
  • -- date range difference. This will give us between
  • -- zero and one less than the date range.
  • %
  •  
  • -- To get the number of days in the date range, we
  • -- can simply substrate the start date from the
  • -- end date. At this point though, we have to cast
  • -- to INT as SQL will not make any automatic
  • -- conversions for us.
  • CAST(
  • (@date_to - @date_from)
  • AS INT
  • )
  • )
  • )

Hope that helps.




Reader Comments

Jan 31, 2007 at 10:59 PM // reply »
1 Comments

Just wanted to say thanks for this post, it was exactly what I was after!

Cheers,

Steve.


Feb 1, 2007 at 7:39 AM // reply »
10,640 Comments

Steve,

Glad to help.


Nov 26, 2008 at 8:55 AM // reply »
3 Comments

Great post. Thanks a lot!


Nov 26, 2008 at 11:39 AM // reply »
3 Comments

Hey how about including a random time in that date time as well?


Nov 27, 2008 at 8:59 AM // reply »
3 Comments

This works as well.

select DATEADD(second, rand()*86400, DATEADD(d, rand()*(CAST((@date_to - @date_from) AS INT )), @date_from))

86400, being the number of seconds in a day.

Cheers.


Feb 10, 2009 at 1:21 PM // reply »
1 Comments

Thanks! This script helped me a lot in trying to generate a buncha test data where we're tracking stats based on unique datetimes.


Mar 29, 2009 at 8:24 PM // reply »
1 Comments

Nice code and great explanation.

Thank you for that.


CRH
Jul 7, 2009 at 1:48 PM // reply »
1 Comments

Here's a slightly modified version... selecting a random time during an 8hr period during a particular day. (28800 is the number of seconds in 8 hours.) I send the NEWID() through RAND() to guarantee a unique timestamp for each row.

DECLARE @date_from DATETIME;
DECLARE @date_to DATETIME;

SET @date_from = '2009-06-29 08:00:00.000';
SET @date_to = '2009-06-29 16:00:00.000';

SELECT DATEADD(second, RAND(ABS(CAST(CAST(NewID() AS BINARY(8)) AS INT)))*28800, DATEADD(d, RAND(ABS(CAST(CAST(NewID() AS BINARY(8)) AS INT)))*(CAST((@date_to - @date_from) AS INT )), @date_from))


Jul 7, 2009 at 6:15 PM // reply »
10,640 Comments

@CRH,

That's a cool idea. A bit hard to read here, but I think I get the point.


RXA
Sep 3, 2009 at 2:01 PM // reply »
1 Comments

How would I add random seconds to each row in MS Access


Sep 25, 2009 at 10:50 AM // reply »
1 Comments

Very good code. It worked fine, the only thing that I have to check is the Date Format, as my SQL Server 2005 is in Portuguese I had to change the format to YYYY-DD-MM

Thank you


Aug 22, 2010 at 5:59 AM // reply »
1 Comments

I did it like that

random date betweem 1-9-2009 to 29-1-2010
(between 0 to 150 days)

DECLARE @RandomNumber float
DECLARE @RandomDayF int
DECLARE @MaxDay int
DECLARE @MinDay int

SET @MaxDay = 150
SET @MinDay = 0

-- random the number of days to add
SELECT @RandomNumber = RAND()

set @randomDayF = ((@MaxDay + 1) - @MinDay) * @RandomNumber + @MinDay

-- add the days
Set @FromDate = (select DATEADD(d, @randomDayF,(SELECT convert(datetime, '1-9-2009',105)))) -- dd-mm-yyyy

--to reduce days
-- @randomDayF = @randomDayF * -1


Dec 17, 2010 at 6:01 AM // reply »
1 Comments

Wow- awesome work guys , it has helped me alot..
Abbi



Post A Comment

Comment Etiquette: Please do not post spam. Please keep the comments on-topic. Please do not post unrelated questions or large chunks of code. And, above all, please be nice to each other - we're trying to have a good conversation here.

Please review the following issues:

Author Name:


Author Email:

Author Website:

Comment:

Supported HTML tags for formatting: <strong>bold</strong>   <em>italic</em>   <code>code</code>







  • Help Wanted - Find Your Next ColdFusion Job
InVision App - Prototyping Made Beautiful With Prototyping Tools Ben Nadel's Company - Epicenter Consulting Recent Blog Comments
Feb 10, 2012 at 7:21 PM
jQuery AJAX Strips Script Tags And Inserts Them After Parent-Most Elements
Update! Instead of $(eval(options.insertAfter)).after(data['insertData']); I now use: var ajaxNode = document.createElement('span'); var parent = $(eval(options.insertAfter))[0].parentNode; ... read »
Feb 10, 2012 at 6:18 PM
jQuery AJAX Strips Script Tags And Inserts Them After Parent-Most Elements
encountered this same, what I consider, jQuery bug last week. I'm building a site in which I load some content via AJAX. This content contains Linkedin share button placeholders which Linkedin API ne ... read »
Feb 10, 2012 at 11:30 AM
Cross-Origin Resource Sharing (CORS) AJAX Requests Between jQuery And Node.js
After you understand the concepts here, this is an awesome cheatsheet for enabling CORS in just about anything http://enable-cors.org/ ... read »
JM
Feb 10, 2012 at 9:10 AM
My Safari Browser SQLite Database Hello World Example
@Amy, Here is a very good tutorial on how to use JOIN: http://www.sqltutorial.org/sqljoin-innerjoin.aspx ... read »
Feb 10, 2012 at 4:42 AM
Building A Twitter-Inspired RESTful API Architecture In ColdFusion
This is great, very useful Ben. I spotted a small typo in the api.cgm listing: <cfthrow type="Unauthroized" /> Cheers Stefan ... read »
Feb 9, 2012 at 10:35 PM
CFDirectory Filtering Uses Pipe Character For Multiple Filters (Thanks Steve Withington)
I was wondering if there would be a filter you could apply so that you got everything but what you included in the filter. As in show me all docs that are not a .pdf. ... read »
Feb 9, 2012 at 10:29 PM
Learning ColdFusion 9: Application-Specific Data Sources
@Ben, No offence, but if people were really wanting advanced features they would be using a platform like ASP.NET MVC. CFML is so structurally compromised as a tag-based scripting language that ... read »
Feb 9, 2012 at 10:03 PM
Subversion - Cleanup Failed To Process The Following Paths
@Leviaguirre, do you still have problems with this? ... read »