Sdílet prostřednictvím


Úprava dat Large-Value (max) v ADO.NET

Datové typy velkého objektu (LOB) jsou ty, které překračují maximální velikost řádku 8 kilobajtů (kB). SQL Server poskytuje max specifikátor pro varchar, nvarchara varbinary datové typy umožňující ukládání hodnot tak velké jako 2^32 bajtů. Sloupce tabulky a proměnné Transact-SQL mohou určovat varchar(max), nvarchar(max)nebo varbinary(max) datové typy. V ADO.NET max lze datové typy načíst pomocí parametru DataReadera lze je také zadat jako hodnoty vstupních i výstupních parametrů bez jakéhokoli speciálního zpracování. U velkých varchar datových typů je možné data načíst a aktualizovat přírůstkově.

Datové max typy lze použít pro porovnání, jako proměnné typu Transact-SQL, a pro zřetězení. Lze je také použít v klauzulích DISTINCT, ORDER BY, GROUP BY příkazu SELECT a také v agregacích, spojeních a poddotazech.

Další informace naleznete v tématu Použití Large-Value datových typů.

omezení typu Large-Value

Následující omezení platí pro max datové typy, které neexistují pro menší datové typy:

  • A sql_variant nemůže obsahovat velký varchar datový typ.

  • Velké varchar sloupce nelze zadat jako klíčový sloupec v indexu. Jsou povoleny v zahrnutém sloupci v neklasifikovaném indexu.

  • Velké varchar sloupce nelze použít jako klíčové sloupce pro particionování.

Práce s typy Large-Value v Transact-SQL

Funkce Transact-SQL OPENROWSET je jednorázová metoda připojení a přístupu ke vzdáleným datům. Obsahuje všechny informace o připojení potřebné pro přístup ke vzdáleným datům ze zdroje dat OLE DB. OPENROWSET lze odkazovat v klauzuli FROM dotazu, jako by se jednalo o název tabulky. Lze na ni také odkazovat jako na cílovou tabulku příkazu INSERT, UPDATE nebo DELETE, která podléhá schopnostem zprostředkovatele OLE DB.

Funkce OPENROWSET obsahuje BULK poskytovatele sady řádků, který umožňuje číst data přímo ze souboru bez načtení dat do cílové tabulky. To umožňuje použití OPENROWSET v jednoduchém příkazu INSERT SELECT.

Argumenty OPENROWSET BULK možností poskytují významnou kontrolu nad tím, kde začít a končit čtení dat, jak řešit chyby a jak se data interpretují. Můžete například určit, že se datový soubor čte jako jeden řádek, sada řádků s jedním sloupcem typu varbinary, varcharnebo nvarchar.

Následující příklad vloží fotku do tabulky ProductPhoto v ukázkové databázi AdventureWorks. Při použití BULK OPENROWSET zprostředkovatele musíte zadat pojmenovaný seznam sloupců, i když do každého sloupce nevkládáte hodnoty. Primární klíč v tomto případě je definován jako sloupec identity a může být vynechán ze seznamu sloupců. Všimněte si, že na konec OPENROWSET příkazu musíte zadat také název korelace, což je v tomto případě ThumbnailPhoto. To koreluje se sloupcem ProductPhoto v tabulce, do které se soubor načítá.

INSERT Production.ProductPhoto (  
    ThumbnailPhoto,
    ThumbnailPhotoFilePath,
    LargePhoto,
    LargePhotoFilePath)  
SELECT ThumbnailPhoto.*, null, null, N'tricycle_pink.gif'  
FROM OPENROWSET
    (BULK 'c:\images\tricycle.jpg', SINGLE_BLOB) ThumbnailPhoto  

Aktualizace dat pomocí UPDATE .WRITE

Příkaz Transact-SQL UPDATE má novou syntaxi zápisu pro úpravu obsahu , varchar(max)nvarchar(max)nebo varbinary(max) sloupců. To vám umožní provádět částečné aktualizace dat. Syntax UPDATE .WRITE je zde zobrazena ve zkrácené podobě:

Aktualizace

{ <object> }

NASTAVIT

{ column_name = { . WRITE ( výraz , @Offset , @Length ) }

Metoda WRITE určuje, že se změní část hodnoty column_name . Výraz je hodnota, která se zkopíruje do column_name. @Offset je počáteční bod, na kterém se začne výraz psát, a argument @Length určuje délku úseku ve sloupci.

Pokud Pak
Výraz je nastaven na hodnotu NULL. @Length je ignorována a hodnota v column_name je zkrácena v zadaném @Offset.
@Offset je NULL Operace aktualizace připojí výraz na konec existující hodnoty column_name a @Length je ignorováno.
@Offset je větší než délka hodnoty column_name. SQL Server vrátí chybu.
@Length je NULL Operace aktualizace odebere všechna data od @Offset konce column_name hodnoty.

Poznámka:

@Offset Ani @Length nemůže být záporné číslo.

Příklad

Tento Transact-SQL příklad aktualizuje částečnou hodnotu v DocumentSummary, což je nvarchar(max) sloupec v tabulce Document v databázi AdventureWorks. Slovo "součásti" se nahrazuje slovem "funkce" zadáním náhradního slova, počátečním umístěním (posunem) slova, které má být nahrazeno v existujících datech, a počtem znaků, které se mají nahradit (délka). Příklad obsahuje příkazy SELECT před a za příkazEM UPDATE pro porovnání výsledků.

USE AdventureWorks;  
GO  
--View the existing value.  
SELECT DocumentSummary  
FROM Production.Document  
WHERE DocumentID = 3;  
GO  
-- The first sentence of the results will be:  
-- Reflectors are vital safety components of your bicycle.  
  
--Modify a single word in the DocumentSummary column  
UPDATE Production.Document  
SET DocumentSummary .WRITE (N'features',28,10)  
WHERE DocumentID = 3 ;  
GO
--View the modified value.  
SELECT DocumentSummary  
FROM Production.Document  
WHERE DocumentID = 3;  
GO  
-- The first sentence of the results will be:  
-- Reflectors are vital safety features of your bicycle.  

Práce s typy Large-Value v ADO.NET

V ADO.NET můžete pracovat s velkými typy hodnot tak, že nastavíte velké typy hodnot jako objekty v SqlParameter k vrácení sady výsledků, nebo pomocí SqlDataReader k vyplnění SqlDataAdapterDataSet/DataTable. Mezi tím, jak pracujete s velkým typem hodnoty a jeho souvisejícím, menším datovým typem, neexistuje žádný rozdíl.

Načtení dat pomocí getSqlBytes

GetSqlBytes Metodu SqlDataReader lze použít k načtení obsahu varbinary(max) sloupce. Následující fragment kódu předpokládá SqlCommand objekt s názvem cmd , který vybere varbinary(max) data z tabulky a SqlDataReader objekt s názvem reader , který načte data jako SqlBytes.

reader = cmd.ExecuteReader(CommandBehavior.CloseConnection)  
While reader.Read()  
    Dim bytes As SqlBytes = reader.GetSqlBytes(0)  
End While  
reader = cmd.ExecuteReader(CommandBehavior.CloseConnection);  
while (reader.Read())  
    {  
        SqlBytes bytes = reader.GetSqlBytes(0);  
    }  

Načtení dat pomocí GetSqlChars

Metodu GetSqlChars lze použít k načtení obsahu sloupce SqlDataReader nebo varchar(max). Následující fragment kódu předpokládá SqlCommand objekt s názvem cmd , který vybere nvarchar(max) data z tabulky a SqlDataReader objekt s názvem reader , který načte data.

reader = cmd.ExecuteReader(CommandBehavior.CloseConnection)  
While reader.Read()  
    Dim buffer As SqlChars = reader.GetSqlChars(0)  
End While  
reader = cmd.ExecuteReader(CommandBehavior.CloseConnection);  
while (reader.Read())  
{  
    SqlChars buffer = reader.GetSqlChars(0);  
}  

Načtení dat pomocí GetSqlBinary

Metodu GetSqlBinary lze použít k načtení obsahu SqlDataReader sloupce varbinary(max). Následující fragment kódu předpokládá SqlCommand objekt s názvem cmd , který vybere varbinary(max) data z tabulky a SqlDataReader objekt s názvem reader , který načte data jako SqlBinary datový proud.

reader = cmd.ExecuteReader(CommandBehavior.CloseConnection)  
While reader.Read()  
    Dim binaryStream As SqlBinary = reader.GetSqlBinary(0)  
End While  
reader = cmd.ExecuteReader(CommandBehavior.CloseConnection);  
while (reader.Read())  
    {  
        SqlBinary binaryStream = reader.GetSqlBinary(0);  
    }  

Načtení dat pomocí GetBytes

Metoda GetBytesSqlDataReader čte datový proud bajtů ze zadaného posunu sloupce do pole bajtů počínaje zadaným posunem pole. Následující fragment kódu předpokládá SqlDataReader objekt s názvem reader , který načte bajty do bajtového pole. Všimněte si, že na rozdíl od GetSqlBytes, GetBytes vyžaduje, aby byla u vyrovnávací paměti pole stanovena velikost.

While reader.Read()  
    Dim buffer(4000) As Byte  
    Dim byteCount As Integer = _  
    CInt(reader.GetBytes(1, 0, buffer, 0, 4000))  
End While  
while (reader.Read())  
{  
    byte[] buffer = new byte[4000];  
    long byteCount = reader.GetBytes(1, 0, buffer, 0, 4000);  
}  

Načtení dat pomocí metody GetValue

Metoda GetValueSqlDataReader přečte hodnotu ze zadaného posunu sloupce do pole. Následující fragment kódu předpokládá objekt pojmenovaný SqlDataReaderreader, který načte binární data od posunu prvního sloupce a potom řetězcová data od posunu druhého sloupce.

While reader.Read()  
    ' Read the data from varbinary(max) column  
    Dim binaryData() As Byte = CByte(reader.GetValue(0))  
  
    ' Read the data from varchar(max) or nvarchar(max) column  
    Dim stringData() As String = Cstr((reader.GetValue(1))  
End While  
while (reader.Read())  
{  
    // Read the data from varbinary(max) column  
    byte[] binaryData = (byte[])reader.GetValue(0);  
  
    // Read the data from varchar(max) or nvarchar(max) column  
    String stringData = (String)reader.GetValue(1);  
}  

Převod z velkých typů hodnot na typy CLR

Obsah varchar(max) sloupce nebo sloupce nvarchar(max) můžete převést pomocí některé z metod převodu řetězců, například ToString. Následující fragment kódu předpokládá SqlDataReader objekt s názvem reader , který načte data.

While reader.Read()  
    Dim str as String = reader(0).ToString()  
    Console.WriteLine(str)  
End While  
while (reader.Read())  
{  
     string str = reader[0].ToString();  
     Console.WriteLine(str);  
}  

Příklad

Následující kód načte název a LargePhoto objekt z ProductPhoto tabulky v AdventureWorks databázi a uloží ho do souboru. Sestavení je potřeba zkompilovat s odkazem na System.Drawing obor názvů. Metoda GetSqlBytesSqlDataReader vrátí SqlBytes objekt, který zveřejňuje Stream vlastnost. Tento kód použije k vytvoření nového Bitmap objektu a pak ho uloží do gifu ImageFormat.

static void TestGetSqlBytes(int documentID, string filePath)
{
    // Assumes GetConnectionString returns a valid connection string.
    using (SqlConnection connection =
               new(GetConnectionString()))
    {
        SqlCommand command = connection.CreateCommand();
        SqlDataReader reader = default!;
        try
        {
            // Setup the command
            command.CommandText =
                "SELECT LargePhotoFileName, LargePhoto "
                + "FROM Production.ProductPhoto "
                + "WHERE ProductPhotoID=@ProductPhotoID";
            command.CommandType = CommandType.Text;

            // Declare the parameter
            SqlParameter paramID =
                new("@ProductPhotoID", SqlDbType.Int)
                {
                    Value = documentID
                };
            command.Parameters.Add(paramID);
            connection.Open();

            string photoName = default!;

            reader = command.ExecuteReader(CommandBehavior.CloseConnection);

            if (reader.HasRows)
            {
                while (reader.Read())
                {
                    // Get the name of the file.
                    photoName = reader.GetString(0);

                    // Ensure that the column isn't null
                    if (reader.IsDBNull(1))
                    {
                        Console.WriteLine($"{photoName} is unavailable.");
                    }
                    else
                    {
                        SqlBytes bytes = reader.GetSqlBytes(1);
                        using (Bitmap productImage = new(bytes.Stream))
                        {
                            var fileName = filePath + photoName;

                            // Save in gif format.
                            productImage.Save(fileName, ImageFormat.Gif);
                            Console.WriteLine($"Successfully created {fileName}.");
                        }
                    }
                }
            }
            else
            {
                Console.WriteLine("No records returned.");
            }
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.Message);
        }
        finally
        {
            reader?.Dispose();
        }
    }
}
Private Sub GetPhoto(
  ByVal documentID As Integer, ByVal filePath As String)
    ' Assumes GetConnectionString returns a valid connection string.
    Using connection As New SqlConnection(GetConnectionString())
        Dim command As SqlCommand = connection.CreateCommand()
        Dim reader As SqlDataReader
        Try
            ' Setup the command
            command.CommandText =
              "SELECT LargePhotoFileName, LargePhoto FROM" _
                & " Production.ProductPhoto" _
                & " WHERE ProductPhotoID=@ProductPhotoID"
            command.CommandType = CommandType.Text

            ' Declare the parameter
            Dim paramID As SqlParameter =
                New SqlParameter("@ProductPhotoID", SqlDbType.Int)
            paramID.Value = documentID
            command.Parameters.Add(paramID)
            connection.Open()

            Dim photoName As String

            reader =
             command.ExecuteReader(CommandBehavior.CloseConnection)

            If reader.HasRows Then
                While reader.Read()
                    ' Get the name of the file
                    photoName = reader.GetString(0)

                    ' Ensure that the column isn't null
                    If (reader.IsDBNull(1)) Then
                        Console.WriteLine("{0} is unavailable.", photoName)
                    Else
                        Dim bytes As SqlBytes = reader.GetSqlBytes(1)
                        Using productImage As New Bitmap(bytes.Stream)
                            Dim fileName As String = filePath & photoName

                            ' Save in gif format.
                            productImage.Save(
                              fileName, ImageFormat.Gif)
                            Console.WriteLine("Successfully created {0}.", fileName)
                        End Using
                    End If
                End While
            Else
                Console.WriteLine("No records returned.")
            End If
        Catch ex As Exception
            Console.WriteLine("Exception: {0}", ex.Message)
        End Try
    End Using
End Sub

Použití parametrů velkého hodnotového typu

Velké typy hodnot lze použít v SqlParameter objektech stejným způsobem jako menší typy hodnot v SqlParameter objektech. Velké typy hodnot můžete načíst jako SqlParameter hodnoty, jak je znázorněno v následujícím příkladu. Kód předpokládá, že v ukázkové databázi AdventureWorks existuje následující uložená procedura GetDocumentSummary. Uložená procedura přebírá vstupní parametr s názvem @DocumentID a vrací obsah sloupce DocumentSummary ve výstupním parametru @DocumentSummary .

CREATE PROCEDURE GetDocumentSummary
(  
    @DocumentID int,  
    @DocumentSummary nvarchar(MAX) OUTPUT  
)  
AS  
SET NOCOUNT ON  
SELECT  @DocumentSummary=Convert(nvarchar(MAX), DocumentSummary)  
FROM    Production.Document  
WHERE   DocumentID=@DocumentID  

Příklad

Kód ADO.NET vytvoří SqlConnection a SqlCommand objekty pro spuštění uložené procedury GetDocumentSummary a načte souhrn dokumentu, který je uložen jako velký typ hodnoty. Kód předá hodnotu vstupního parametru @DocumentID a zobrazí výsledky předané zpět ve výstupním @DocumentSummary parametru v okně Konzola.

static string? GetDocumentSummary(int documentID)
{
    // Assumes GetConnectionString returns a valid connection string.
    using (SqlConnection connection =
               new(GetConnectionString()))
    {
        connection.Open();
        SqlCommand command = connection.CreateCommand();
        try
        {
            // Set up the command to execute the stored procedure.
            command.CommandText = "GetDocumentSummary";
            command.CommandType = CommandType.StoredProcedure;

            // Set up the input parameter for the DocumentID.
            SqlParameter paramID =
                new("@DocumentID", SqlDbType.Int)
                {
                    Value = documentID
                };
            command.Parameters.Add(paramID);

            // Set up the output parameter to retrieve the summary.
            SqlParameter paramSummary =
                new("@DocumentSummary",
                SqlDbType.NVarChar, -1)
                {
                    Direction = ParameterDirection.Output
                };
            command.Parameters.Add(paramSummary);

            // Execute the stored procedure.
            command.ExecuteNonQuery();
            Console.WriteLine((string)paramSummary.Value);
            return (string)paramSummary.Value;
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.Message);
            return null;
        }
    }
}
Private Function GetDocumentSummary( _
  ByVal documentID As Integer) As String

    ' Assumes GetConnectionString returns a valid connection string.
    Using connection As New SqlConnection(GetConnectionString())
        connection.Open()
        Dim command As SqlCommand = connection.CreateCommand()

        ' Setup the command to execute the stored procedure.
        command.CommandText = "GetDocumentSummary"
        command.CommandType = CommandType.StoredProcedure

        ' Set up the input parameter for the DocumentID.
        Dim paramID As SqlParameter = _
            New SqlParameter("@DocumentID", SqlDbType.Int)
        paramID.Value = documentID
        command.Parameters.Add(paramID)

        ' Set up the output parameter to retrieve the summary.
        Dim paramSummary As SqlParameter = _
            New SqlParameter("@DocumentSummary", _
               SqlDbType.NVarChar, -1)
        paramSummary.Direction = ParameterDirection.Output
        command.Parameters.Add(paramSummary)

        ' Execute the stored procedure.
        command.ExecuteNonQuery()
        Console.WriteLine(paramSummary.Value)
        Return paramSummary.Value.ToString
    End Using
End Function

Viz také