Usando CTE:
DECLARE @Begin DATETIME
DECLARE @End DATETIME
SELECT @Begin = '20110101', @End = '20120101'
;WITH ranges(DateFrom, DateTo) AS
(
SELECT @Begin, DATEADD(DAY, -1, DATEADD(MONTH, 1, @begin))
UNION ALL
SELECT DATEADD(month, 1, DateFrom), DATEADD(DAY, -1, DATEADD(MONTH, 2, DateFrom))
FROM ranges
WHERE DateFrom < @End
)
SELECT * FROM ranges
OPTION(MAXRECURSION 0)
E não usando CTE:
DECLARE @Begin DATETIME
DECLARE @End DATETIME
SELECT @Begin = '20110101', @End = '20120101'
SELECT DATEADD(MONTH, n.Number, @Begin) DateFrom, DATEADD(day, -1, DATEADD(MONTH, n.Number+1, @Begin)) DateTo
FROM master.dbo.spt_values n
WHERE
n.Number < DATEDIFF(MONTH, @begin, @end)
AND n.Type = 'P'
Se você precisar incluir janeiro de 2012 também, use este
DECLARE @Begin DATETIME
DECLARE @End DATETIME
SELECT @Begin = '20110101', @End = '20120101'
SELECT DATEADD(MONTH, n.Number, @Begin) DateFrom, DATEADD(day, -1, DATEADD(MONTH, n.Number+1, @Begin)) DateTo
FROM master.dbo.spt_values n
WHERE
n.Number <= DATEDIFF(MONTH, @begin, @end)
AND n.Type = 'P'
E CTE:
DECLARE @Begin DATETIME
DECLARE @End DATETIME
SELECT @Begin = '20110101', @End = '20120101'
;WITH ranges(DateFrom, DateTo) AS
(
SELECT @Begin, DATEADD(DAY, -1, DATEADD(MONTH, 1, @begin))
UNION ALL
SELECT DATEADD(month, 1, DateFrom), DATEADD(DAY, -1, DATEADD(MONTH, 2, DateFrom))
FROM ranges
WHERE DATEADD(month, 1, DateFrom) < @End
)
SELECT * FROM ranges
OPTION(MAXRECURSION 0)