|
database
newsgroups
|
|||||||||||||||||||||||
|
|||||||||||||||||||||||
Is tables created in UDF thread safe?the same time, do they each get their own copy? 2) Is there a reason to use 2 tables - One temp and one to return? Thanks, Dave Example from "CREATE FUNCTION" in Sql Server 2000 Help ========================================================== CREATE FUNCTION fn_FindReports (@InEmpId nchar(5)) RETURNS @retFindReports TABLE (empid nchar(5) primary key, empname nvarchar(50) NOT NULL, mgrid nchar(5), title nvarchar(30)) /*Returns a result set that lists all the employees who report to given employee directly or indirectly.*/ AS BEGIN DECLARE @RowsAdded int -- table variable to hold accumulated results DECLARE @reports TABLE (empid nchar(5) primary key, empname nvarchar(50) NOT NULL, mgrid nchar(5), title nvarchar(30), processed tinyint default 0) -- initialize @Reports with direct reports of the given employee INSERT @reports SELECT empid, empname, mgrid, title, 0 FROM employees WHERE empid = @InEmpId SET @RowsAdded = @@rowcount -- While new employees were added in the previous iteration WHILE @RowsAdded > 0 BEGIN /*Mark all employee records whose direct reports are going to be found in this iteration with processed=1.*/ UPDATE @reports SET processed = 1 WHERE processed = 0 -- Insert employees who report to employees marked 1. INSERT @reports SELECT e.empid, e.empname, e.mgrid, e.title, 0 FROM employees e, @reports r WHERE e.mgrid=r.empid and e.mgrid <> e.empid and r.processed = 1 SET @RowsAdded = @@rowcount /*Mark all employee records whose direct reports have been found in this iteration.*/ UPDATE @reports SET processed = 2 WHERE processed = 1 END -- copy to the result of the function the required columns INSERT @retFindReports SELECT empid, empname, mgrid, title FROM @reports RETURN END GO Hi
Yes, each Temporarty Table Variable as you defined it is only in the SPIDS scope, other SPIDS will not see it. Your UDF will not perform very well as UDF's are not designed for this. I hope you are not using it on a table that is larger than a few rows. This should be a SP, not a UDF. Regards -------------------------------- Mike Epprecht, Microsoft SQL Server MVP Zurich, Switzerland MVP Program: http://www.microsoft.com/mvp Blog: http://www.msmvps.com/epprecht/ Show quote "Dave" wrote: > 1) Is the tables being created thread safe. If 2 users hit this function at > the same time, do they each get their own copy? > 2) Is there a reason to use 2 tables - One temp and one to return? > > Thanks, > Dave > > > Example from "CREATE FUNCTION" in Sql Server 2000 Help > ========================================================== > > CREATE FUNCTION fn_FindReports (@InEmpId nchar(5)) > RETURNS @retFindReports TABLE (empid nchar(5) primary key, > empname nvarchar(50) NOT NULL, > mgrid nchar(5), > title nvarchar(30)) > /*Returns a result set that lists all the employees who report to given > employee directly or indirectly.*/ > AS > BEGIN > DECLARE @RowsAdded int > -- table variable to hold accumulated results > DECLARE @reports TABLE (empid nchar(5) primary key, > empname nvarchar(50) NOT NULL, > mgrid nchar(5), > title nvarchar(30), > processed tinyint default 0) > -- initialize @Reports with direct reports of the given employee > INSERT @reports > SELECT empid, empname, mgrid, title, 0 > FROM employees > WHERE empid = @InEmpId > SET @RowsAdded = @@rowcount > -- While new employees were added in the previous iteration > WHILE @RowsAdded > 0 > BEGIN > /*Mark all employee records whose direct reports are going to be > found in this iteration with processed=1.*/ > UPDATE @reports > SET processed = 1 > WHERE processed = 0 > -- Insert employees who report to employees marked 1. > INSERT @reports > SELECT e.empid, e.empname, e.mgrid, e.title, 0 > FROM employees e, @reports r > WHERE e.mgrid=r.empid and e.mgrid <> e.empid and r.processed = 1 > SET @RowsAdded = @@rowcount > /*Mark all employee records whose direct reports have been found > in this iteration.*/ > UPDATE @reports > SET processed = 2 > WHERE processed = 1 > END > > -- copy to the result of the function the required columns > INSERT @retFindReports > SELECT empid, empname, mgrid, title > FROM @reports > RETURN > END > GO > Mike Epprecht (SQL MVP) wrote:
[snip] > It's not really his fault though, since as he said, it's straight out> Your UDF will not perform very well as UDF's are not designed for this. I > hope you are not using it on a table that is larger than a few rows. This > should be a SP, not a UDF. > of BOL - I think he's just using it for the purposes of illustration. Damien Hi
Here is the answer: 1: as you have created table as a variable, the table will be thread safe. Table will be safe and there is no collision when u create a table as a variable or Temp Table 2: U need not declare a return table, just specify RETURNS TABLE and say RETURN SELECT empid, empname, mgrid, title FROM @reports please let me know if you have any questions -- Show quotebest Regards, Chandra http://chanduas.blogspot.com/ http://groups.msn.com/SQLResource/ --------------------------------------- "Dave" wrote: > 1) Is the tables being created thread safe. If 2 users hit this function at > the same time, do they each get their own copy? > 2) Is there a reason to use 2 tables - One temp and one to return? > > Thanks, > Dave > > > Example from "CREATE FUNCTION" in Sql Server 2000 Help > ========================================================== > > CREATE FUNCTION fn_FindReports (@InEmpId nchar(5)) > RETURNS @retFindReports TABLE (empid nchar(5) primary key, > empname nvarchar(50) NOT NULL, > mgrid nchar(5), > title nvarchar(30)) > /*Returns a result set that lists all the employees who report to given > employee directly or indirectly.*/ > AS > BEGIN > DECLARE @RowsAdded int > -- table variable to hold accumulated results > DECLARE @reports TABLE (empid nchar(5) primary key, > empname nvarchar(50) NOT NULL, > mgrid nchar(5), > title nvarchar(30), > processed tinyint default 0) > -- initialize @Reports with direct reports of the given employee > INSERT @reports > SELECT empid, empname, mgrid, title, 0 > FROM employees > WHERE empid = @InEmpId > SET @RowsAdded = @@rowcount > -- While new employees were added in the previous iteration > WHILE @RowsAdded > 0 > BEGIN > /*Mark all employee records whose direct reports are going to be > found in this iteration with processed=1.*/ > UPDATE @reports > SET processed = 1 > WHERE processed = 0 > -- Insert employees who report to employees marked 1. > INSERT @reports > SELECT e.empid, e.empname, e.mgrid, e.title, 0 > FROM employees e, @reports r > WHERE e.mgrid=r.empid and e.mgrid <> e.empid and r.processed = 1 > SET @RowsAdded = @@rowcount > /*Mark all employee records whose direct reports have been found > in this iteration.*/ > UPDATE @reports > SET processed = 2 > WHERE processed = 1 > END > > -- copy to the result of the function the required columns > INSERT @retFindReports > SELECT empid, empname, mgrid, title > FROM @reports > RETURN > END > GO > You have mimiced 1950's style "scratch tapes" in temp tables, used
assembly language style bit flags and have a loop. That is about as non-relational as you can get. If I undrerstand this, you can replace the adjacency list model of the organizational chart with a nested set model and avoid this kind of horrible coding. Get a copy of TREES & HIERSARCHIES IN SQL if you need details. |
|||||||||||||||||||||||