SQL Server - столбцы NTEXT и манипуляции со строками

11

У меня есть таблица с NTEXTименем столбца comments. У меня есть вторая строка, давайте назовем ее anothercommentvarchar), которая должна быть помещена в данную commentsстроку после слова UPDATEHERE.

Приведение к nvarchar(max)усечению commentsстроки, поэтому я не могу использовать подобные CHARINDEX()( Msg 8152, Level 16, State 10, Line 2 String or binary data would be truncated.). Я использовал, datalength()чтобы проверить, что есть несколько тысяч столбцов, которые> 8000 символов.

Пример того, чего я хочу достичь (хотя и с гораздо более длинными строками):

Комментарии - This is a test UPDATEHERE This is the end of the test

еще один комментарий - . This is inserted.

Результирующая строка - This is a test UPDATEHERE. This is inserted. This is the end of the test

Я понимаю, что это нормально с varchar()/ nvarchar(), но ntextэто полный и полный кошмар для работы. Я понимаю, что это устаревший тип данных, но я не написал соответствующее приложение.

Philᵀᴹ
источник

Ответы:

8

Преобразование в nvarchar(max)должно работать, если вы не делаете что-то не так с вашимCHARINDEX()

Попробуйте этот фрагмент кода, он должен выводить то, что вы хотите.

-- Create the table
CREATE TABLE [dbo].[PhilsTable](
    [comment] [ntext] NULL,
    [anothercomment] [nvarchar](50) NULL
) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY];

GO

-- insert very long string
INSERT INTO [dbo].[PhilsTable] (comment, anothercomment) VALUES (N'This is a test UPDATEHERE This is the end of the test' + REPLICATE (CAST(N'x' AS nvarchar(max)), 1000000), 'this goes in here');

-- verify data
SELECT DATALENGTH(comment), *  FROM [dbo].[PhilsTable];

-- perform replace
SELECT CAST(REPLACE(CAST(comment AS NVARCHAR(MAX)),'UPDATEHERE','UPDATEHERE' + anothercomment) AS NTEXT) FROM [dbo].[PhilsTable];

DROP TABLE [dbo].[PhilsTable];

Спасибо Андрею М за помощь с REPLICATEзаявлением.

Том V - попробуйте topanswers.xyz
источник
10

Преобразование в nvarchar(max)и обратно в ntextдействительно упрощает жизнь с точки зрения кода, но это означает преобразование и перезапись всего (возможно, очень большого) значения со всеми затратами ресурсов процессора и журналирования, что подразумевает.

Альтернативой является использование UPDATETEXT. Как и раньше, это не рекомендуется, ntextно это может значительно снизить затраты на ведение журнала. С другой стороны, это означает использование текстовых указателей, и он работает только с одной строкой за раз.

В следующем примере кода используется курсор для обхода этого ограничения и используется PATINDEXвместо него, CHARINDEXтак как первый является одной из немногих функций, которые работают непосредственно с ntext:

Пример данных

CREATE TABLE dbo.PhilsTable
(
    comment ntext NULL,
    anothercomment nvarchar(50) NULL
);

INSERT dbo.PhilsTable
    (comment, anothercomment)
VALUES 
(
    CONVERT(ntext, 
        N'This is a test UPDATEHERE This is the end of the test ' + 
            REPLICATE (CONVERT(nvarchar(max), N'x'), 1000000)), 
    CONVERT(nvarchar(50), N'. This is inserted.')
),
(
    CONVERT(ntext, 
        N'This is a test UPDATEHERE This is the end of the test ' + 
            REPLICATE (CONVERT(nvarchar(max), N'x'), 1000000)), 
    CONVERT(nvarchar(50), N'. This is inserted.')
),
(
    CONVERT(ntext, 
        N'This is a test UPDATEHERE This is the end of the test ' + 
            REPLICATE (CONVERT(nvarchar(max), N'x'), 1000000)), 
    CONVERT(nvarchar(50), N'. This is inserted.')
);

Объявление курсора

DECLARE c 
    CURSOR GLOBAL 
    FORWARD_ONLY 
    DYNAMIC 
    SCROLL_LOCKS 
    TYPE_WARNING
FOR
SELECT
    TxtPtr = TEXTPTR(PT.comment),
    Src = PT.anothercomment,
    Offset = PATINDEX(N'%UPDATEHERE%', PT.comment) + LEN(N'UPDATEHERE') - 1
FROM dbo.PhilsTable AS PT
WHERE
    PT.comment LIKE N'%UPDATEHERE%'; -- LIKE works with ntext

OPEN c;

Цикл обработки

DECLARE 
    @Ptr binary(16),
    @Src nvarchar(50),
    @Offset integer;

SET STATISTICS XML OFF; -- No cursor fetch plans

BEGIN TRANSACTION;

    WHILE 1 = 1
    BEGIN
        FETCH c INTO @Ptr, @Src, @Offset;

        IF @@FETCH_STATUS = -2 CONTINUE; -- row missing
        IF @@FETCH_STATUS = -1 BREAK; -- no more rows

        IF 1 = TEXTVALID('dbo.PhilsTable.comment', @Ptr)
        BEGIN
            -- Modify ntext value
            UPDATETEXT dbo.PhilsTable.comment @Ptr @Offset 0 @Src;
        END;
    END;

COMMIT TRANSACTION;

CLOSE c; DEALLOCATE c;
Пол Уайт 9
источник