Difference between revisions of "SQL Snippets"
From SpenchWiki
(Slow spatial) |
|||
(5 intermediate revisions by the same user not shown) | |||
Line 1: | Line 1: | ||
− | * | + | == Snippets == |
+ | |||
+ | * Update index statistics: | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | EXEC sp_updatestats | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | * Server version: | ||
+ | <syntaxhighlight lang="tsql"> | ||
+ | SELECT @@VERSION | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | * Case-sensitive string comparison: | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | WHERE (ComponentName.Name = @name COLLATE SQL_Latin1_General_CP1_CS_AS) | ||
+ | </syntaxhighlight> | ||
* Finding a sproc in the meta-database | * Finding a sproc in the meta-database | ||
Line 9: | Line 26: | ||
AND SPECIFIC_SCHEMA = N'<Schema_Name, sysname, Schema_Name>' | AND SPECIFIC_SCHEMA = N'<Schema_Name, sysname, Schema_Name>' | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | |||
− | |||
* Declaring a table variable | * Declaring a table variable | ||
Line 39: | Line 54: | ||
DECLARE @t TABLE (L bigint, ID bigint, V nvarchar(64)) | DECLARE @t TABLE (L bigint, ID bigint, V nvarchar(64)) | ||
INSERT @t | INSERT @t | ||
− | EXEC @i = uspFindLanguageVersion ' | + | EXEC @i = uspFindLanguageVersion 'My Language' |
SELECT V FROM @t | SELECT V FROM @t | ||
SELECT @i AS 'Return' | SELECT @i AS 'Return' | ||
Line 84: | Line 99: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | * MSDN links | + | == Notes == |
+ | |||
+ | * [http://www.mssqlcity.com/Articles/Undoc/UndocExtSP.htm Undocumented procedures] | ||
+ | |||
+ | * Cannot have ORDER BY clause in a view's definition | ||
+ | |||
+ | * Performing replication on sprocs with '''GO''' not commented out using standard SQL '''--''' comment will result in replication failure: | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | -- Drop stored procedure if it already exists | ||
+ | /*IF EXISTS ( | ||
+ | SELECT * | ||
+ | FROM INFORMATION_SCHEMA.ROUTINES | ||
+ | WHERE SPECIFIC_NAME = N'uspFindComponent' | ||
+ | ) | ||
+ | DROP PROCEDURE uspFindComponent | ||
+ | --GO | ||
+ | |||
+ | CREATE PROCEDURE uspFindComponent | ||
+ | */ALTER PROCEDURE [dbo].[uspFindComponent] | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | == MSDN links == | ||
+ | * [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/udb9/html/eb0639a2-f7f3-40ad-8b65-dbefadd4b41e.htm Special data types] | ||
* [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/1054c76e-0fd5-4131-8c07-a6c5d024af50.htm INSERT] | * [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/1054c76e-0fd5-4131-8c07-a6c5d024af50.htm INSERT] | ||
* [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/864b393f-225f-4895-8c8d-4db59ea60032.htm CREATE FUNCTION] | * [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/864b393f-225f-4895-8c8d-4db59ea60032.htm CREATE FUNCTION] | ||
Line 96: | Line 134: | ||
* [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/d410e06e-061b-4c25-9973-b2dc9b60bd85.htm SET @local_variable] | * [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/tsqlref9/html/d410e06e-061b-4c25-9973-b2dc9b60bd85.htm SET @local_variable] | ||
* [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/fulltxt9/html/5686d3ac-b84c-46a6-8b98-34a5bd674ded.htm Performing Prefix Searches] | * [ms-help://MS.SQLCC.v9/MS.SQLSVR.v9.en/fulltxt9/html/5686d3ac-b84c-46a6-8b98-34a5bd674ded.htm Performing Prefix Searches] | ||
+ | |||
+ | == More snippets == | ||
* Derived query | * Derived query | ||
Line 138: | Line 178: | ||
GO | GO | ||
</syntaxhighlight> | </syntaxhighlight> | ||
+ | |||
+ | * Clever EXISTS ( SELECT 1 ) JOIN trick | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | SELECT r2.* | ||
+ | FROM ( | ||
+ | SELECT TOP 50 CustomerID | ||
+ | FROM ( | ||
+ | SELECT MovieID | ||
+ | FROM Ratings | ||
+ | WHERE CustomerID = 915 | ||
+ | ) q | ||
+ | JOIN Ratings r | ||
+ | ON r.MovieID = q.MovieID | ||
+ | AND CustomerID <> 915 | ||
+ | AND EXISTS | ||
+ | ( | ||
+ | SELECT 1 | ||
+ | FROM Ratings re | ||
+ | WHERE re.MovieID = 1 | ||
+ | AND re.CustomerID = r.CustomerID | ||
+ | ) | ||
+ | GROUP BY | ||
+ | CustomerID | ||
+ | ORDER BY | ||
+ | COUNT(*) DESC | ||
+ | ) ro | ||
+ | JOIN Ratings r2 | ||
+ | ON r2.MovieID = 1 | ||
+ | AND r2.CustomerID = ro.CustomerID | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | * Spatial index statistics (useful for debugging slow query or plan warnings) | ||
+ | |||
+ | If query is slow (especially when using a JOIN or limiting row count using TOP or ROW_NUMBER), force plan to use spatial index with (no quotes): | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | WITH (INDEX(<spatial index name>)) | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | This will avoid the case where a plan is generated that suggests it's faster to seek over all rows and filter by a spatial predicate, instead of actually using the spatial index. | ||
+ | |||
+ | Plan utilising spatial index might show a warning on the "Clustered Index Seek (Spatial)" saying "Columns with no statistics" (Cell_Attributes, SRID, pk0). This may have something to do with the sub-optimal plan being generated when the query has certain conditions (e.g. TOP 690 is slow, while TOP 691 and above is fast). | ||
+ | |||
+ | <syntaxhighlight lang="tsql"> | ||
+ | declare c cursor static for | ||
+ | select object_id, name, index_id from sys.indexes where type_desc = 'SPATIAL' | ||
+ | open c | ||
+ | declare @obj int | ||
+ | declare @statsname varchar(max) | ||
+ | declare @indid int | ||
+ | declare @tname varchar(max) | ||
+ | fetch next from c into @obj, @statsname, @indid | ||
+ | while (@@fetch_status = 0) | ||
+ | begin | ||
+ | declare @internaltablename varchar(max) | ||
+ | declare @stmt varchar(max) | ||
+ | fetch next from c into @obj, @statsname, @indid | ||
+ | select top 1 @tname = name from sys.objects where object_id = @obj | ||
+ | set @internaltablename = 'sys.extended_index_' + cast(@obj as varchar(max)) + '_' + cast(@indid as varchar(max)) | ||
+ | select 'Showing statistics for ', @tname | ||
+ | select @stmt = 'dbcc show_statistics (''' + @internaltablename + ''', ''' + @statsname + ''')' | ||
+ | exec (@stmt) | ||
+ | end | ||
+ | close c | ||
+ | deallocate c | ||
+ | </syntaxhighlight> | ||
+ | |||
+ | [http://social.msdn.microsoft.com/Forums/en/sqlspatial/thread/746e8c67-1e89-43cf-87e7-fec7620ff7e2 Spatial index statistics missing] |
Latest revision as of 12:18, 13 April 2011
Contents
[hide]Snippets
- Update index statistics:
EXEC sp_updatestats
- Server version:
SELECT @@VERSION
- Case-sensitive string comparison:
WHERE (ComponentName.Name = @name COLLATE SQL_Latin1_General_CP1_CS_AS)
- Finding a sproc in the meta-database
SELECT *
FROM INFORMATION_SCHEMA.ROUTINES
WHERE SPECIFIC_NAME = N'uspFindComponent'
AND SPECIFIC_SCHEMA = N'<Schema_Name, sysname, Schema_Name>'
- Declaring a table variable
DECLARE @TableVar TABLE (Cola int PRIMARY KEY, Colb char(3))
INSERT INTO @TableVar VALUES (1, 'abc')
INSERT INTO @TableVar VALUES (2, 'def')
SELECT * FROM @TableVar
GO
- WHERE EXISTS condition
SELECT Version FROM LanguageVersion WHERE EXISTS
(
SELECT * FROM LanguageVersion
)
- Selecting results into a table variable & CTE using WITH
DECLARE @i int
DECLARE @t TABLE (L bigint, ID bigint, V nvarchar(64))
INSERT @t
EXEC @i = uspFindLanguageVersion 'My Language'
SELECT V FROM @t
SELECT @i AS 'Return'
GO
WITH DirReps(ManagerID, DirectReports) AS
(
SELECT Version, Language
FROM LanguageVersion AS e
)
SELECT ManagerID, DirectReports
FROM DirReps
ORDER BY ManagerID;
GO
- TRANSACTION
BEGIN TRANSACTION;
BEGIN TRY
-- Statements
END TRY
BEGIN CATCH
SELECT
ERROR_NUMBER() AS ErrorNumber,
ERROR_SEVERITY() AS ErrorSeverity,
ERROR_STATE() as ErrorState,
ERROR_PROCEDURE() as ErrorProcedure,
ERROR_LINE() as ErrorLine,
ERROR_MESSAGE() as ErrorMessage;
IF @@TRANCOUNT > 0
ROLLBACK TRANSACTION;
RETURN(1);
END CATCH;
IF @@TRANCOUNT > 0
COMMIT TRANSACTION;
RETURN(0);
Notes
- Cannot have ORDER BY clause in a view's definition
- Performing replication on sprocs with GO not commented out using standard SQL -- comment will result in replication failure:
-- Drop stored procedure if it already exists
/*IF EXISTS (
SELECT *
FROM INFORMATION_SCHEMA.ROUTINES
WHERE SPECIFIC_NAME = N'uspFindComponent'
)
DROP PROCEDURE uspFindComponent
--GO
CREATE PROCEDURE uspFindComponent
*/ALTER PROCEDURE [dbo].[uspFindComponent]
MSDN links
- Special data types
- INSERT
- CREATE FUNCTION
- CREATE PROCEDURE
- WITH
- Using Special Data
- EXECUTE
- LIKE
- OBJECT_ID
- SET @local_variable
- Performing Prefix Searches
More snippets
- Derived query
SELECT ComponentName.Name, derivedtbl_1.ID
FROM ComponentName CROSS JOIN
(SELECT ID
FROM Component) AS derivedtbl_1
WHERE derivedtbl_1.ID = ComponentName.Component
- INNER JOIN
SELECT ComponentReference.Reference, ComponentReference.Component, Reference.Location, ComponentReference.ID
FROM ComponentReference INNER JOIN
Reference ON ComponentReference.Reference = Reference.ID
WHERE (ComponentReference.Component = 983)
- Generating a GUID
DECLARE @id uniqueidentifier
SET @id = NEWID()
DECLARE @txt nvarchar(36)
SET @txt = CONVERT(nvarchar(36), @id)
SELECT @id, @txt, LEN(@txt)
- Procedure for converting a hexadecimal string into the original binary data
CREATE PROCEDURE dbo.uspHexToBinary
@s nvarchar(MAX),
@b varbinary(MAX) OUT
AS
DECLARE @sql nvarchar(MAX)
SET @sql = 'SELECT @b = ' + @s
EXEC sp_executesql @sql, N'@b VARBINARY(MAX) OUTPUT', @b OUTPUT
GO
- Clever EXISTS ( SELECT 1 ) JOIN trick
SELECT r2.*
FROM (
SELECT TOP 50 CustomerID
FROM (
SELECT MovieID
FROM Ratings
WHERE CustomerID = 915
) q
JOIN Ratings r
ON r.MovieID = q.MovieID
AND CustomerID <> 915
AND EXISTS
(
SELECT 1
FROM Ratings re
WHERE re.MovieID = 1
AND re.CustomerID = r.CustomerID
)
GROUP BY
CustomerID
ORDER BY
COUNT(*) DESC
) ro
JOIN Ratings r2
ON r2.MovieID = 1
AND r2.CustomerID = ro.CustomerID
- Spatial index statistics (useful for debugging slow query or plan warnings)
If query is slow (especially when using a JOIN or limiting row count using TOP or ROW_NUMBER), force plan to use spatial index with (no quotes):
WITH (INDEX(<spatial index name>))
This will avoid the case where a plan is generated that suggests it's faster to seek over all rows and filter by a spatial predicate, instead of actually using the spatial index.
Plan utilising spatial index might show a warning on the "Clustered Index Seek (Spatial)" saying "Columns with no statistics" (Cell_Attributes, SRID, pk0). This may have something to do with the sub-optimal plan being generated when the query has certain conditions (e.g. TOP 690 is slow, while TOP 691 and above is fast).
declare c cursor static for
select object_id, name, index_id from sys.indexes where type_desc = 'SPATIAL'
open c
declare @obj int
declare @statsname varchar(max)
declare @indid int
declare @tname varchar(max)
fetch next from c into @obj, @statsname, @indid
while (@@fetch_status = 0)
begin
declare @internaltablename varchar(max)
declare @stmt varchar(max)
fetch next from c into @obj, @statsname, @indid
select top 1 @tname = name from sys.objects where object_id = @obj
set @internaltablename = 'sys.extended_index_' + cast(@obj as varchar(max)) + '_' + cast(@indid as varchar(max))
select 'Showing statistics for ', @tname
select @stmt = 'dbcc show_statistics (''' + @internaltablename + ''', ''' + @statsname + ''')'
exec (@stmt)
end
close c
deallocate c