SQL Server-NTEXT列と文字列操作


11

というNTEXT列のあるテーブルがありますcomments。私は2番目の文字列を持っている、のは、それを呼びましょう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完全で完全な悪夢です。非推奨のデータ型であることに気づきましたが、問題のアプリケーションを記述していません。

回答:


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];

この声明を手伝ってくれたAndriy Mに感謝しREPLICATEます。


10

コードの観点から見ると、変換しnvarchar(max)て元に戻すのntextは簡単ですが、すべての(おそらく非常に大きな)値を変換して書き換えることを意味し、すべてのCPUとログのオーバーヘッドが意味します。

代わりにを使用することUPDATETEXTです。これはと同様ntextに非推奨ですが、ロギングのオーバーヘッドを大幅に削減できます。欠点は、テキストポインタを使用することを意味し、一度に1行のみを操作します。

次のコード例では、カーソルを使用してその制限を回避しています。前者はで直接機能する数少ない関数の 1つであるため、PATINDEX代わりにを使用しています。CHARINDEXntext

サンプルデータ

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;
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.