เป็นวิธีที่เร็วที่สุดในการแปลง ints 1: 1 เป็นเลขฐานสองแบบ null ได้คืออะไร?


14

ส่วนหนึ่งของภาระงานของฉันใช้ฟังก์ชั่น CLR ที่ใช้อัลกอริทึมแฮชที่น่ากลัวเพื่อเปรียบเทียบแถวเพื่อดูว่ามีการเปลี่ยนแปลงค่าคอลัมน์ใด ๆ หรือไม่ ฟังก์ชั่น CLR รับสตริงไบนารีเป็นอินพุตดังนั้นฉันต้องการวิธีที่รวดเร็วในการแปลงแถวเป็นสตริงไบนารี ฉันคาดว่าจะแฮชประมาณ 10 พันล้านแถวในระหว่างการทำงานเต็มรูปแบบดังนั้นฉันต้องการรหัสนี้ให้เร็วที่สุด

ฉันมีตารางประมาณ 300 ตารางที่มีสกีมาแตกต่างกัน สำหรับจุดประสงค์ของคำถามนี้โปรดสมมติโครงสร้างตารางอย่างง่ายที่มีค่าเป็น 32 INTคอลัมน์ ฉันได้ให้ข้อมูลตัวอย่างรวมถึงวิธีการเปรียบเทียบผลลัพธ์ที่ด้านล่างของคำถามนี้

แถวต้องถูกแปลงเป็นสตริงไบนารีเดียวกันหากค่าคอลัมน์ทั้งหมดเหมือนกัน แถวต้องถูกแปลงเป็นสตริงไบนารีที่แตกต่างกันหากค่าคอลัมน์ใด ๆ แตกต่างกัน ตัวอย่างเช่นรหัสง่าย ๆ ดังต่อไปนี้จะไม่ทำงาน:

CAST(COL1 AS BINARY(4)) + CAST(COL2 AS BINARY(4)) + ..

มันไม่จัดการ NULL อย่างถูกต้อง ถ้าCOL1เป็น NULL สำหรับแถวที่ 1 และCOL2เป็น NULL สำหรับแถวที่ 2 ดังนั้นทั้งสองแถวจะถูกแปลงเป็นสตริง NULL ฉันเชื่อว่าการจัดการ NULL ที่ถูกต้องเป็นส่วนที่ยากที่สุดในการแปลงทั้งแถวอย่างถูกต้อง ค่าที่อนุญาตทั้งหมดสำหรับคอลัมน์ INT นั้นเป็นไปได้

หากต้องการคำถามล่วงหน้า:

  • หากเป็นเรื่องสำคัญเวลาส่วนใหญ่ (90% +) คอลัมน์จะไม่เป็น NULL
  • ฉันต้องใช้ CLR
  • ฉันต้องแฮชหลายแถวนี้ ฉันไม่สามารถคงแฮชได้
  • ฉันเชื่อว่าฉันไม่สามารถใช้โหมดแบทช์สำหรับการแปลงเนื่องจากการมีฟังก์ชัน CLR

เป็นวิธีที่เร็วที่สุดในการแปลง 32 INTคอลัมน์nullable เป็นBINARY(X)หรือVARBINARY(X)สตริงอะไร

ข้อมูลตัวอย่างและรหัสตามสัญญา:

-- create sample data
DROP TABLE IF EXISTS dbo.TABLE_OF_32_INTS;

CREATE TABLE dbo.TABLE_OF_32_INTS (
    COL1 INT NULL,
    COL2 INT NULL,
    COL3 INT NULL,
    COL4 INT NULL,
    COL5 INT NULL,
    COL6 INT NULL,
    COL7 INT NULL,
    COL8 INT NULL,
    COL9 INT NULL,
    COL10 INT NULL,
    COL11 INT NULL,
    COL12 INT NULL,
    COL13 INT NULL,
    COL14 INT NULL,
    COL15 INT NULL,
    COL16 INT NULL,
    COL17 INT NULL,
    COL18 INT NULL,
    COL19 INT NULL,
    COL20 INT NULL,
    COL21 INT NULL,
    COL22 INT NULL,
    COL23 INT NULL,
    COL24 INT NULL,
    COL25 INT NULL,
    COL26 INT NULL,
    COL27 INT NULL,
    COL28 INT NULL,
    COL29 INT NULL,
    COL30 INT NULL,
    COL31 INT NULL,
    COL32 INT NULL
);

INSERT INTO dbo.TABLE_OF_32_INTS WITH (TABLOCK)
SELECT 0, 123, 12345, 1234567, 123456789
, 0, 123, 12345, 1234567, 123456789
, 0, 123, 12345, 1234567, 123456789
, 0, 123, 12345, 1234567, 123456789
, 0, 123, 12345, 1234567, 123456789
, 0, 123, 12345, 1234567, 123456789
, NULL, -876545321
FROM
(
    SELECT TOP (1000000) ROW_NUMBER() OVER (ORDER BY (SELECT NULL)) RN
    FROM master..spt_values t1
    CROSS JOIN master..spt_values t2
) q
OPTION (MAXDOP 1);


GO


-- procedure to test performance
CREATE OR ALTER PROCEDURE #p AS 
BEGIN

SET NOCOUNT ON;

DECLARE
@counter INT = 0,
@dummy VARBINARY(8000);

WHILE @counter < 10
BEGIN
    SELECT @dummy = -- this code is clearly incomplete as it does not handle NULLs
        CAST(COL1 AS BINARY(4)) + 
        CAST(COL2 AS BINARY(4)) + 
        CAST(COL3 AS BINARY(4)) + 
        CAST(COL4 AS BINARY(4)) + 
        CAST(COL5 AS BINARY(4)) + 
        CAST(COL6 AS BINARY(4)) + 
        CAST(COL7 AS BINARY(4)) + 
        CAST(COL8 AS BINARY(4)) + 
        CAST(COL9 AS BINARY(4)) + 
        CAST(COL10 AS BINARY(4)) + 
        CAST(COL11 AS BINARY(4)) + 
        CAST(COL12 AS BINARY(4)) + 
        CAST(COL13 AS BINARY(4)) + 
        CAST(COL14 AS BINARY(4)) + 
        CAST(COL15 AS BINARY(4)) + 
        CAST(COL16 AS BINARY(4)) + 
        CAST(COL17 AS BINARY(4)) + 
        CAST(COL18 AS BINARY(4)) + 
        CAST(COL19 AS BINARY(4)) + 
        CAST(COL20 AS BINARY(4)) + 
        CAST(COL21 AS BINARY(4)) + 
        CAST(COL22 AS BINARY(4)) + 
        CAST(COL23 AS BINARY(4)) + 
        CAST(COL24 AS BINARY(4)) + 
        CAST(COL25 AS BINARY(4)) + 
        CAST(COL26 AS BINARY(4)) + 
        CAST(COL27 AS BINARY(4)) + 
        CAST(COL28 AS BINARY(4)) + 
        CAST(COL29 AS BINARY(4)) + 
        CAST(COL30 AS BINARY(4)) + 
        CAST(COL31 AS BINARY(4)) + 
        CAST(COL32 AS BINARY(4))
    FROM dbo.TABLE_OF_32_INTS
    OPTION (MAXDOP 1);

    SET @counter = @counter + 1;
END;

SELECT cpu_time
FROM sys.dm_exec_requests
WHERE session_id = @@SPID;

END;

GO

-- run procedure
EXEC #p;

(ฉันจะยังคงใช้แฮชเหมือนผีในผลลัพธ์ไบนารีนี้เวิร์กโหลดใช้แฮชรวมและใช้ค่าแฮชสำหรับแฮชบิวด์ตัวใดตัวหนึ่งฉันไม่ต้องการค่าไบนารียาวในบิลด์แฮชเนื่องจากต้องใช้มากเกินไป หน่วยความจำ.)

คำตอบ:


6

สิ่งที่เกี่ยวกับการใช้BINARY(5)และการแปลงค่า NULL ให้อยู่นอกขอบเขตสำหรับ INT:

SELECT @dummy =
    ISNULL(CAST(COL1  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL2  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL3  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL4  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL5  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL6  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL7  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL8  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL9  AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL10 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL11 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL12 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL13 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL14 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL15 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL16 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL17 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL18 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL19 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL20 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL21 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL22 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL23 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL24 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL25 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL26 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL27 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL28 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL29 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL30 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL31 AS BINARY(5)), 0x0100000000) + 
    ISNULL(CAST(COL32 AS BINARY(5)), 0x0100000000)
FROM dbo.TABLE_OF_32_INTS
OPTION (MAXDOP 1);

11

บนเครื่องของฉัน (SQL Server 2017) ฟังก์ชั่น C # SQLCLR ต่อไปนี้ทำงานได้เร็วกว่าbinary(5)ความคิดประมาณ 30% เร็วกว่าความคิด 35% CONCAT_WSและใช้เวลาตอบรับครึ่งชั่วโมง

มันต้องUNSAFEได้รับอนุญาตและใช้พอยน์เตอร์ การใช้งานนั้นเชื่อมโยงกับข้อมูลการทดสอบ

สำหรับวัตถุประสงค์ในการทดสอบวิธีที่ง่ายที่สุดในการทำให้แอสเซมบลีที่ไม่ปลอดภัยนี้ทำงานคือการตั้งค่าฐานข้อมูลTRUSTWORTHYและปิดใช้งานตัวเลือกการกำหนดค่าความปลอดภัยที่เข้มงวด clrหากจำเป็น

รหัสรวบรวม

เพื่อความสะดวกCREATE ASSEMBLYบิตที่คอมไพล์จะอยู่ที่https://gist.github.com/SQLKiwi/72d01b661c74485900e7ebcfdc63ab8e

Stub ฟังก์ชัน T-SQL

CREATE FUNCTION dbo.NullableIntsToBinary
(
    @Col01 int, @Col02 int, @Col03 int, @Col04 int, @Col05 int, @Col06 int, @Col07 int, @Col08 int, 
    @Col09 int, @Col10 int, @Col11 int, @Col12 int, @Col13 int, @Col14 int, @Col15 int, @Col16 int, 
    @Col17 int, @Col18 int, @Col19 int, @Col20 int, @Col21 int, @Col22 int, @Col23 int, @Col24 int, 
    @Col25 int, @Col26 int, @Col27 int, @Col28 int, @Col29 int, @Col30 int, @Col31 int, @Col32 int
)
RETURNS binary(132) 
WITH EXECUTE AS CALLER
AS EXTERNAL NAME Obbish.UserDefinedFunctions.NullableIntsToBinary;

รหัสแหล่งที่มา

แหล่งข้อมูล C # อยู่ที่https://gist.github.com/SQLKiwi/64f320fe7fd802a68a3a644aa8b8af9f

หากคุณรวบรวมสิ่งนี้ด้วยตนเองคุณต้องใช้ Class Library (.dll) เป็นประเภทโครงการเป้าหมายและตรวจสอบตัวเลือกอนุญาตการสร้างรหัสที่ไม่ปลอดภัย

วิธีการแก้ปัญหารวม

เนื่องจากในที่สุดคุณต้องการคำนวณ SpookyHash ของข้อมูลไบนารีที่ส่งคืนข้างต้นคุณสามารถเรียก SpookyHash ภายในฟังก์ชัน CLR และส่งคืนแฮชขนาด 16 ไบต์

การใช้ตัวอย่างขึ้นอยู่กับตารางที่มีส่วนผสมของชนิดข้อมูลคอลัมน์ที่https://gist.github.com/SQLKiwi/6f82582a4ad1920c372fac118ec82460 ซึ่งรวมถึงอัลกอริทึมแบบ Spooky Hash ที่ไม่ปลอดภัยซึ่งมาจากSpookilySharpของ Jon Hanna และรหัสต้นฉบับโดเมนสาธารณะCดั้งเดิมโดย Bob Jenkins


7

คอลัมน์มีสี่ไบต์ของค่าที่ได้รับอนุญาตซึ่งตรงกับขนาดของที่INT BINARY(4)กล่าวอีกนัยหนึ่งค่าที่เป็นไปได้ทั้งหมดของ BINARY (4) จะถูกจับคู่กับค่าที่เป็นไปได้ของINTคอลัมน์ ดังนั้นถ้าไม่มีค่าที่ไม่ได้รับอนุญาตในINTคอลัมน์ไม่มีการแทนที่ NULL อย่างปลอดภัย ไม่ว่าคอลัมน์นั้นจะเป็น NULL หรือไม่จะต้องถูกเข้ารหัสแยกต่างหาก BINARY(4)มันก็ไม่สามารถใส่ภายใน

วิธีหนึ่งในการทำเช่นนี้คือด้วยบิตแมป NULL พิจารณารหัสต่อไปนี้:

CAST(       
    CASE WHEN COL1 IS NOT NULL THEN 0 ELSE 1 END | 
    CASE WHEN COL2 IS NOT NULL THEN 0 ELSE 2 END | 
    CASE WHEN COL3 IS NOT NULL THEN 0 ELSE 4 END | 
    CASE WHEN COL4 IS NOT NULL THEN 0 ELSE 8 END | 
    CASE WHEN COL5 IS NOT NULL THEN 0 ELSE 16 END | 
    CASE WHEN COL6 IS NOT NULL THEN 0 ELSE 32 END | 
    CASE WHEN COL7 IS NOT NULL THEN 0 ELSE 64 END | 
    CASE WHEN COL8 IS NOT NULL THEN 0 ELSE 128 END
AS BINARY(1))

ไม่ว่าคอลัมน์แปดคอลัมน์นั้นเป็น NULL พอดีกับไบต์เดียวหรือไม่ นิพจน์เหล่านี้สามารถเปรียบเทียบระหว่างแถวเพื่อตรวจสอบว่าคอลัมน์เดียวกันทั้งหมดเป็น NULL หรือไม่เป็น NULL ด้วยข้อมูลเพิ่มเติมนั้นจะปลอดภัยเมื่อต้องการแทนที่ค่าคอลัมน์ NULL ด้วยสิ่งใดก็ตามที่ไม่ใช่ NULL ฉันพบว่าCAST(ISNULL(COL1, 0) AS BINARY(4))เป็นวิธีที่เร็วที่สุดแม้ว่ารูปแบบอื่น ๆ เช่นเป็นISNULL(CAST(COL1 AS VARBINARY(4)), 0x)ไปได้

เป็นการยากที่จะพิสูจน์อะไรเลย แต่ฉันพบว่ารายละเอียดต่อไปนี้เป็นวิธีที่เร็วที่สุด:

  • การใช้ 0 สำหรับ NOT NULL ในบิตแมปเนื่องจากฉันรู้ว่าค่าในคอลัมน์ส่วนใหญ่จะไม่เป็น NULL
  • ใช้ bitwise หรือแทนการเพิ่มสำหรับบิตแมป
  • ตรวจสอบว่าค่าคอลัมน์เป็น NULL ซึ่งตรงข้ามกับค่าไบนารีที่แปลงแล้ว

เกณฑ์มาตรฐานใช้เวลาประมาณ 27.5 CPU วินาทีในเครื่องของฉัน ขออภัยขั้นตอนบิตแมป NULL ใช้เวลาประมาณหนึ่งในสามของเวลานั้น มันจะดีถ้ามีวิธีที่เร็วกว่านี้

นี่คือโซลูชันที่สมบูรณ์:

SELECT
    CAST(ISNULL(COL1, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL2, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL3, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL4, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL5, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL6, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL7, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL8, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL9, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL10, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL11, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL12, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL13, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL14, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL15, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL16, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL17, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL18, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL19, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL20, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL21, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL22, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL23, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL24, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL25, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL26, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL27, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL28, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL29, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL30, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL31, 0) AS BINARY(4)) + 
    CAST(ISNULL(COL32, 0) AS BINARY(4)) + 
    CAST(       
        CASE WHEN COL1 IS NOT NULL THEN 0 ELSE 1 END | 
        CASE WHEN COL2 IS NOT NULL THEN 0 ELSE 2 END | 
        CASE WHEN COL3 IS NOT NULL THEN 0 ELSE 4 END | 
        CASE WHEN COL4 IS NOT NULL THEN 0 ELSE 8 END | 
        CASE WHEN COL5 IS NOT NULL THEN 0 ELSE 16 END | 
        CASE WHEN COL6 IS NOT NULL THEN 0 ELSE 32 END | 
        CASE WHEN COL7 IS NOT NULL THEN 0 ELSE 64 END | 
        CASE WHEN COL8 IS NOT NULL THEN 0 ELSE 128 END
    AS BINARY(1)) + 
    CAST(   
        CASE WHEN COL9  IS NOT NULL THEN 0 ELSE 1 END | 
        CASE WHEN COL10 IS NOT NULL THEN 0 ELSE 2 END | 
        CASE WHEN COL11 IS NOT NULL THEN 0 ELSE 4 END | 
        CASE WHEN COL12 IS NOT NULL THEN 0 ELSE 8 END | 
        CASE WHEN COL13 IS NOT NULL THEN 0 ELSE 16 END | 
        CASE WHEN COL14 IS NOT NULL THEN 0 ELSE 32 END | 
        CASE WHEN COL15 IS NOT NULL THEN 0 ELSE 64 END | 
        CASE WHEN COL16 IS NOT NULL THEN 0 ELSE 128 END
    AS BINARY(1)) + 
    CAST(   
        CASE WHEN COL17 IS NOT NULL THEN 0 ELSE 1 END | 
        CASE WHEN COL18 IS NOT NULL THEN 0 ELSE 2 END | 
        CASE WHEN COL19 IS NOT NULL THEN 0 ELSE 4 END | 
        CASE WHEN COL20 IS NOT NULL THEN 0 ELSE 8 END | 
        CASE WHEN COL21 IS NOT NULL THEN 0 ELSE 16 END | 
        CASE WHEN COL22 IS NOT NULL THEN 0 ELSE 32 END | 
        CASE WHEN COL23 IS NOT NULL THEN 0 ELSE 64 END | 
        CASE WHEN COL24 IS NOT NULL THEN 0 ELSE 128 END
    AS BINARY(1)) + 
    CAST(   
        CASE WHEN COL25 IS NOT NULL THEN 0 ELSE 1 END | 
        CASE WHEN COL26 IS NOT NULL THEN 0 ELSE 2 END | 
        CASE WHEN COL27 IS NOT NULL THEN 0 ELSE 4 END | 
        CASE WHEN COL28 IS NOT NULL THEN 0 ELSE 8 END | 
        CASE WHEN COL29 IS NOT NULL THEN 0 ELSE 16 END | 
        CASE WHEN COL30 IS NOT NULL THEN 0 ELSE 32 END | 
        CASE WHEN COL31 IS NOT NULL THEN 0 ELSE 64 END | 
        CASE WHEN COL32 IS NOT NULL THEN 0 ELSE 128 END
    AS BINARY(1))
FROM dbo.TABLE_OF_32_INTS
OPTION (MAXDOP 1);

5

ในการทดสอบconcat_wsของฉันเร็วขึ้นเล็กน้อย (18 วินาที) กว่าโซลูชันบิตแมป null ของคุณ (26 วินาที) จะมีข้อมูลมากขึ้นที่จะสลับไปมาดังนั้นคุณอาจเห็นการลดประสิทธิภาพลงที่อื่นและหากคุณต้องการผสมสิ่งนี้กับคอลัมน์อักขระคุณต้องเลือกตัวคั่นอย่างชาญฉลาด

select @dummy = cast(concat_ws('|',
         isnull(cast(T.COL1  as varchar(11)), ''),
         isnull(cast(T.COL2  as varchar(11)), ''),
         isnull(cast(T.COL3  as varchar(11)), ''),
         isnull(cast(T.COL4  as varchar(11)), ''),
         isnull(cast(T.COL5  as varchar(11)), ''),
         isnull(cast(T.COL6  as varchar(11)), ''),
         isnull(cast(T.COL7  as varchar(11)), ''),
         isnull(cast(T.COL8  as varchar(11)), ''),
         isnull(cast(T.COL9  as varchar(11)), ''),
         isnull(cast(T.COL10 as varchar(11)), ''),
         isnull(cast(T.COL11 as varchar(11)), ''),
         isnull(cast(T.COL12 as varchar(11)), ''),
         isnull(cast(T.COL13 as varchar(11)), ''),
         isnull(cast(T.COL14 as varchar(11)), ''),
         isnull(cast(T.COL15 as varchar(11)), ''),
         isnull(cast(T.COL16 as varchar(11)), ''),
         isnull(cast(T.COL17 as varchar(11)), ''),
         isnull(cast(T.COL18 as varchar(11)), ''),
         isnull(cast(T.COL19 as varchar(11)), ''),
         isnull(cast(T.COL20 as varchar(11)), ''),
         isnull(cast(T.COL21 as varchar(11)), ''),
         isnull(cast(T.COL22 as varchar(11)), ''),
         isnull(cast(T.COL23 as varchar(11)), ''),
         isnull(cast(T.COL24 as varchar(11)), ''),
         isnull(cast(T.COL25 as varchar(11)), ''),
         isnull(cast(T.COL26 as varchar(11)), ''),
         isnull(cast(T.COL27 as varchar(11)), ''),
         isnull(cast(T.COL28 as varchar(11)), ''),
         isnull(cast(T.COL29 as varchar(11)), ''),
         isnull(cast(T.COL30 as varchar(11)), ''),
         isnull(cast(T.COL31 as varchar(11)), ''),
         isnull(cast(T.COL32 as varchar(11)), ''))
       as varbinary(8000))
from dbo.TABLE_OF_32_INTS as T
option (maxdop 1)

2
ขอบคุณสำหรับสิ่งนี้. มันเร็วกว่าสำหรับฉันเช่นกัน แต่มันมีข้อเสียในการสร้างสตริงไบนารีที่ยาวขึ้น (191 ตัวอักษร vs 132 สำหรับข้อมูลตัวอย่าง) เราจะพิจารณามันอย่างแน่นอน
โจ Obbish

-1

หากคุณสามารถมั่นใจได้ล่วงหน้าว่าคุณไม่ได้เก็บ int บางอย่างไว้เช่น-2,147,483,648นั้นคุณสามารถทำสิ่งต่อไปนี้:

 SELECT @dummy = 
        coalesce(CAST(COL1 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL2 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL3 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL4 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL5 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL6 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL7 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL8 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL9 AS BINARY(4)),  0x80000000) + 
        coalesce(CAST(COL10 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL11 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL12 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL13 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL14 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL15 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL16 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL17 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL18 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL19 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL20 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL21 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL22 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL23 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL24 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL25 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL26 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL27 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL28 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL29 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL30 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL31 AS BINARY(4)), 0x80000000) + 
        coalesce(CAST(COL32 AS BINARY(4)), 0x80000000) 
    FROM dbo.TABLE_OF_32_INTS

ขอบคุณสำหรับคำตอบ ค่าที่อนุญาตทั้งหมดสำหรับคอลัมน์ INT เป็นไปได้
Joe Obbish
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.