May 23, 2006 at 12:10 pm
I need to execute a dynamic SQL string within a function, but I get an error on the EXECUTE statement saying I cannot run the EXECUTE within a function.
Offending code as follows:
SET @strSQL = '
SELECT @intCount = ' + @DBName + '.dbo.DieLifetimeStrokeCount(2)'
EXECUTE sp_executesql @strSQL, N'@intCount int OUTPUT', @intCount OUTPUT
The reason I'm doing this is because the function needs to total the results of the above statement from several different databases, the names of which are stored in a table. So I am cursoring thru the databases and running this statement and gathering the total.
I would like this cursor to be done within the function.
Any ideas would be very appreciated!
May 23, 2006 at 12:28 pm
Functions have a lot of limitations on them that will make this hard to do. Without knowing more details about what you're doing, how about using a Stored Procedure instead. I would think you could total your results in a Temp Table, and might be able to avoid the cursor in the process. This is all just guessing since you only supplied two lines of code though, but you might look into it.
May 23, 2006 at 12:48 pm
I have been hoping to be able to run EXEC in a function. It is so handy if can be done. But I could not. Instead I use stored procedure with OUTPUT parameter.
I feel it has some advantages in using stored proc over function. It allow you to do much more and it is easy to debug.
May 23, 2006 at 1:01 pm
Unfortunately the only use of EXEC allowed in UDF's is to call an extended stored procedure (XP). This limitation is imposed because of determinism. UDF's ideally should be deterministic; SP's are considered non-deterministic because they can potentially cause side effects to the database and return different return values for the same input values.
As an aside, XP's can also cause side effects. BOL states that if you call an XP (that can potentially have side effects on the database) from within a UDF, don't rely on consistent results.
May 23, 2006 at 9:11 pm
How about putting this code in a trigger or sp to run when the db list is updated:
@sql nvarchar(4000)
@dbs table(dbname sysname)
function dbo.fn_tot_alldbs
@sql = N'
@dbs select name from sys.databases --databaselist
@sql = @sql + 'select @intcount = @intcount + ' + d.dbname + '.dbo.DieLifetimeStrokeCount(2)
from @dbs d
@sql = @sql + 'return @intcount
sp_executesql @sql
dbo.fn_tot_alldbs(0)
The output is like:
Tim Wilkinson
"If it doesn't work in practice, you're using the wrong theory"
- Immanuel Kant
May 24, 2006 at 7:08 am
Thanks for everyone's input. It has really helped me confirm and understand the issue better. stax68, yr solution is intriguing! I would have never thought of that but it seems like it would work! Even tho I personally try to stay away from triggers because I find them difficult to work with from a maintenance standpoint (debugging, figuring out what's going on in the system, remember that they're there, etc), it would really simplify my code in general to have this function in place. I think I'll try that. Thanks!!
ps - I love this site!
May 24, 2006 at 7:47 am
Glad to be of help.
Re trigger phobia: If the db list is only updated by stored procs, you could put the code in there. I know what you mean about triggers, but sometimes (however much normalisation you do - 5th normal anyone?) you need them to enforce consistency rules, and I'd much rather have them in my schema than have some novice VBer maintaining them, and the web guys trying to copy what she does, etc.etc...Then there's DDL triggers in v9, which definitely are worth the work - at last, version control of database objects without cumbersome procedures! But that's another story.
A quick caveat: this solution is of course only suitable because the target table is (I assume) both very small and very infrequently updated. You wouldn't want a function to be constantly dropped and recreated (you should also ensure drop and recreate are in a transaction), and you wouldn't want the recreation process to involve any lengthy data access process.
Tim Wilkinson
"If it doesn't work in practice, you're using the wrong theory"
- Immanuel Kant
May 24, 2006 at 8:16 am
Noted - thanks!
May 24, 2006 at 8:45 am
Well ... that almost got me there, but unfortunately the database that the 'databases' table resides on is different from where I want the function to reside. And I guess I cannot specify the database where I want the function even if I hard code the name (which I don't want to do anyways). I could possibly leave the function in the 'databases' database... I'll have to think about that. Thanks.
alter function databaseX.dbo.fn_tot_alldbsLifetimeStrokes(@DieKeyID int)
error from above stmt:
'CREATE FUNCTION' does not allow specifying the database name as a prefix to the object name.
May 24, 2006 at 9:37 am
Hmmm... Can you give more details about exactly what you're trying to do? Like where do you want this function to exist? Are you using SQL 2K or 2K5? What table are the database names stored in, and where is it located? What exactly does the DieLifetimeStrokeCount function do?
I suspect there's a pretty simple and efficient answer to what you want to do, but I'm having trouble determining exactly what you do want to do from your posts...
May 24, 2006 at 7:06 pm
Try adding a USE statement to the dynamic SQL.
May 24, 2006 at 9:01 pm
You could put the code on the remote server (where the db list resides) and cahnge it to create a stored proc which could then be executed from yuor local server by a remote procedure call...
Tim Wilkinson
"If it doesn't work in practice, you're using the wrong theory"
- Immanuel Kant
May 25, 2006 at 6:57 am
You guys know too much. Thx for all the ideas.
End solution for me was: Use the trigger idea and move the db list table (that has the trigger) to the same database as where the function should reside. We decided that table should be in that database anyways.
It works like a charm. Thanks!
July 10, 2014 at 5:40 pm
It appears that if you put the name of the database in front of the call to the function, it will use the tables in that database. For example I created a function to list the columns within a table. I wanted to compare the two tables to see if there had been any changes to the column list between two databases.
select db1.dbo.fn_GetColumnList('Table1'), db2.dbo.fn_GetColumnList('Table1')
Which shows that I did indeed have two different column lists.
July 10, 2014 at 9:31 pm
Robert Davis (5/24/2006)
Try adding a USE statement to the dynamic SQL.<DT><FONT color=#1111ff>set</FONT><FONT size=2><FONT color=#1111ff> @sql</FONT> </FONT><FONT color=#808080 size=2>=</FONT><FONT size=2> N</FONT><FONT color=#ff0000 size=2>' USE [OtherDatabase] <DT>create function dbo.fn_tot_alldbs(@yourparameter int) <DT>returns int <DT>as <DT>--auto generated function code <DT>begin <DT>declare @intCount int <DT>select @intcount = 0 <DT>'</FONT></DT>
I think you'll find that'll return an error about how CREATE must be the first thing in the batch.
--Jeff Moden
Change is inevitable... Change for the better is not.
Viewing 15 posts - 1 through 14 (of 14 total)
You must be logged in to reply to this topic. Login to reply