Funções com valor de tabela CLR
Aplica-se a:SQL Server
Uma função com valor de tabela é uma função definida pelo usuário que retorna uma tabela.
O SQL Server estende a funcionalidade de funções com valor de tabela, permitindo que você defina uma função com valor de tabela em qualquer linguagem gerenciada. Os dados são retornados de uma função com valor de tabela por meio de um objeto IEnumerable
ou IEnumerator
.
Para funções com valor de tabela, as colunas do tipo de tabela de retorno não podem incluir colunas de carimbo de data/hora ou colunas de tipo de dados de cadeia de caracteres não Unicode (como char, varchare texto). A restrição de NOT NULL
não é suportada.
Diferenças entre funções com valor de tabela Transact-SQL e CLR
Transact-SQL funções com valor de tabela materializam os resultados de chamar a função em uma tabela intermediária. Como eles usam uma tabela intermediária, eles podem oferecer suporte a restrições e índices exclusivos sobre os resultados. Esses recursos podem ser úteis quando grandes resultados são retornados.
Em contraste, as funções com valor de tabela CLR (Common Language Runtime) representam uma alternativa de streaming. Não há exigência de que todo o conjunto de resultados seja materializado em uma única tabela. O objeto IEnumerable
retornado pela função gerenciada é chamado diretamente pelo plano de execução da consulta que chama a função com valor de tabela e os resultados são consumidos de maneira incremental. Esse modelo de streaming garante que os resultados possam ser consumidos imediatamente após a primeira linha estar disponível, em vez de esperar que toda a tabela seja preenchida. Também é uma alternativa melhor se você tiver um grande número de linhas retornadas, porque elas não precisam ser materializadas na memória como um todo. Por exemplo, uma função gerenciada com valor de tabela pode ser usada para analisar um arquivo de texto e retornar cada linha como uma linha.
Implementar funções com valor de tabela
Implemente funções com valor de tabela como métodos em uma classe em um assembly do .NET Framework. Seu código de função com valor de tabela deve implementar a interface IEnumerable
. A interface IEnumerable
é definida no .NET Framework. Os tipos que representam matrizes e coleções no .NET Framework já implementam a interface IEnumerable
. Isso facilita a escrita de funções com valor de tabela que convertem uma coleção ou uma matriz em um conjunto de resultados.
Parâmetros com valor de tabela
Os parâmetros com valor de tabela são tipos de tabela definidos pelo usuário que são passados para um procedimento ou função e fornecem uma maneira eficiente de passar várias linhas de dados para o servidor. Os parâmetros com valor de tabela fornecem funcionalidade semelhante às matrizes de parâmetros, mas oferecem maior flexibilidade e integração mais estreita com o Transact-SQL. Eles também fornecem o potencial para um melhor desempenho.
Os parâmetros com valor de tabela também ajudam a reduzir o número de viagens de ida e volta ao servidor. Em vez de enviar várias solicitações para o servidor, como com uma lista de parâmetros escalares, os dados podem ser enviados para o servidor como um parâmetro com valor de tabela. Um tipo de tabela definido pelo usuário não pode ser passado como um parâmetro com valor de tabela para, ou ser retornado de, um procedimento armazenado gerenciado ou função em execução no processo do SQL Server. Para obter mais informações sobre parâmetros com valor de tabela, consulte Usar parâmetros com valor de tabela (Mecanismo de Banco de Dados).
Parâmetros de saída e funções com valor de tabela
As informações podem ser retornadas de funções com valor de tabela usando parâmetros de saída. O parâmetro correspondente na função de valor de tabela de código de implementação deve usar um parâmetro pass-by-reference como argumento. Visual Basic .NET não oferece suporte a parâmetros de saída da mesma maneira que o C#. Você deve especificar o parâmetro por referência e aplicar o atributo <Out()>
para representar um parâmetro de saída, como no exemplo a seguir:
Imports System.Runtime.InteropServices
...
Public Shared Sub FillRow ( <Out()> ByRef value As SqlInt32)
Definir uma função com valor de tabela no Transact-SQL
A sintaxe para definir uma função com valor de tabela CLR é semelhante à de uma função com valor de tabela Transact-SQL, com a adição da cláusula EXTERNAL NAME
. Por exemplo:
CREATE FUNCTION GetEmpFirstLastNames()
RETURNS TABLE (
FirstName NVARCHAR (4000),
LastName NVARCHAR (4000)
)
AS EXTERNAL NAME MyDotNETAssembly.[MyNamespace.MyClassname].GetEmpFirstLastNames;
As funções com valor de tabela são usadas para representar dados em forma relacional para processamento posterior em consultas como:
SELECT *
FROM func();
SELECT *
FROM tbl
INNER JOIN func() AS f
ON tbl.col = f.col;
SELECT *
FROM tbl AS t
CROSS APPLY func(t.col);
As funções com valor de tabela podem retornar uma tabela quando:
Criado a partir de argumentos de entrada escalares. Por exemplo, uma função com valor de tabela que usa uma cadeia de números delimitada por vírgulas e os transforma em uma tabela.
Gerado a partir de dados externos. Por exemplo, uma função com valor de tabela que lê o log de eventos e o expõe como uma tabela.
Observação
Uma função com valor de tabela só pode executar o acesso a dados por meio de uma consulta Transact-SQL no método InitMethod
e não no método FillRow
. O InitMethod
deve ser marcado com a propriedade atributo SqlFunction.DataAccess.Read
se uma consulta Transact-SQL for executada.
Uma função com valor de tabela de exemplo
A função com valor de tabela a seguir retorna informações do log de eventos do sistema. A função usa um único argumento de cadeia de caracteres contendo o nome do log de eventos para ler.
Código de exemplo
using System;
using System.Data.Sql;
using Microsoft.SqlServer.Server;
using System.Collections;
using System.Data.SqlTypes;
using System.Diagnostics;
public class TabularEventLog
{
[SqlFunction(FillRowMethodName = "FillRow")]
public static IEnumerable InitMethod(String logname)
{
return new EventLog(logname).Entries;
}
public static void FillRow(Object obj, out SqlDateTime timeWritten, out SqlChars message, out SqlChars category, out long instanceId)
{
EventLogEntry eventLogEntry = (EventLogEntry)obj;
timeWritten = new SqlDateTime(eventLogEntry.TimeWritten);
message = new SqlChars(eventLogEntry.Message);
category = new SqlChars(eventLogEntry.Category);
instanceId = eventLogEntry.InstanceId;
}
}
Declarar e usar a função de exemplo com valor de tabela
Depois que a função com valor de tabela de exemplo é compilada, ela pode ser declarada em Transact-SQL como esta:
USE master;
-- Replace SQL_Server_logon with your SQL Server user credentials.
GRANT EXTERNAL ACCESS ASSEMBLY TO [SQL_Server_logon];
-- Modify the following line to specify a different database.
ALTER DATABASE master
SET TRUSTWORTHY ON;
-- Modify the next line to use the appropriate database.
CREATE ASSEMBLY tvfEventLog
FROM 'D:\assemblies\tvfEventLog\tvfeventlog.dll'
WITH PERMISSION_SET = EXTERNAL_ACCESS;
GO
CREATE FUNCTION ReadEventLog
(@logname NVARCHAR (100))
RETURNS TABLE (
logTime DATETIME,
Message NVARCHAR (4000),
Category NVARCHAR (4000),
InstanceId BIGINT)
AS EXTERNAL NAME tvfEventLog.TabularEventLog.InitMethod;
GO
Objetos de banco de dados Visual C++ compilados com /clr:pure
não têm suporte para execução no SQL Server. Por exemplo, esses objetos de banco de dados incluem funções com valor de tabela.
Para testar o exemplo, tente o seguinte código Transact-SQL:
-- Select the top 100 events,
SELECT TOP 100 *
FROM dbo.ReadEventLog(N'Security') AS T;
GO
-- Select the last 10 login events.
SELECT TOP 10 T.logTime,
T.Message,
T.InstanceId
FROM dbo.ReadEventLog(N'Security') AS T
WHERE T.Category = N'Logon/Logoff';
GO
Exemplos
Os exemplos de código Transact-SQL neste artigo usam o banco de dados de exemplo
Um. Retornar os resultados de uma consulta do SQL Server
O exemplo a seguir mostra uma função com valor de tabela que consulta um banco de dados do SQL Server.
Nomeie seu arquivo de código-fonte FindInvalidEmails.cs ou FindInvalidEmails.vb.
using System;
using System.Collections;
using System.Data;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using Microsoft.SqlServer.Server;
public partial class UserDefinedFunctions {
private class EmailResult {
public SqlInt32 CustomerId;
public SqlString EmailAdress;
public EmailResult(SqlInt32 customerId, SqlString emailAdress) {
CustomerId = customerId;
EmailAdress = emailAdress;
}
}
public static bool ValidateEmail(SqlString emailAddress) {
if (emailAddress.IsNull)
return false;
if (!emailAddress.Value.EndsWith("@adventure-works.com"))
return false;
// Validate the address. Put any more rules here.
return true;
}
[SqlFunction(
DataAccess = DataAccessKind.Read,
FillRowMethodName = "FindInvalidEmails_FillRow",
TableDefinition="CustomerId int, EmailAddress nvarchar(4000)")]
public static IEnumerable FindInvalidEmails(SqlDateTime modifiedSince) {
ArrayList resultCollection = new ArrayList();
using (SqlConnection connection = new SqlConnection("context connection=true")) {
connection.Open();
using (SqlCommand selectEmails = new SqlCommand(
"SELECT " +
"[CustomerID], [EmailAddress] " +
"FROM [AdventureWorksLT2022].[SalesLT].[Customer] " +
"WHERE [ModifiedDate] >= @modifiedSince",
connection)) {
SqlParameter modifiedSinceParam = selectEmails.Parameters.Add(
"@modifiedSince",
SqlDbType.DateTime);
modifiedSinceParam.Value = modifiedSince;
using (SqlDataReader emailsReader = selectEmails.ExecuteReader()) {
while (emailsReader.Read()) {
SqlString emailAddress = emailsReader.GetSqlString(1);
if (ValidateEmail(emailAddress)) {
resultCollection.Add(new EmailResult(
emailsReader.GetSqlInt32(0),
emailAddress));
}
}
}
}
}
return resultCollection;
}
public static void FindInvalidEmails_FillRow(
object emailResultObj,
out SqlInt32 customerId,
out SqlString emailAdress) {
EmailResult emailResult = (EmailResult)emailResultObj;
customerId = emailResult.CustomerId;
emailAdress = emailResult.EmailAdress;
}
};
Compile o código-fonte para uma DLL e copie a DLL para o diretório raiz da unidade C. Em seguida, execute a seguinte consulta Transact-SQL.
USE AdventureWorksLT2022;
GO
IF EXISTS (SELECT name
FROM sysobjects
WHERE name = 'FindInvalidEmails')
DROP FUNCTION FindInvalidEmails;
GO
IF EXISTS (SELECT name
FROM sys.assemblies
WHERE name = 'MyClrCode')
DROP ASSEMBLY MyClrCode;
GO
CREATE ASSEMBLY MyClrCode
FROM 'C:\FindInvalidEmails.dll'
WITH PERMISSION_SET = SAFE;
GO
CREATE FUNCTION FindInvalidEmails
(@ModifiedSince DATETIME)
RETURNS TABLE (
CustomerId INT,
EmailAddress NVARCHAR (4000))
AS EXTERNAL NAME MyClrCode.UserDefinedFunctions.[FindInvalidEmails];
GO
SELECT *
FROM FindInvalidEmails('2000-01-01');
GO