Sqlserver
 sql >> Datenbank >  >> RDS >> Sqlserver

Assoziationstabelle auf Spalte mit mehreren Werten reduzieren?

Ich habe eine CLR-Aggregatfunktion erstellt das nimmt ein varchar -Spalte und gibt alle ihre Werte getrennt durch Kommas zurück. Mit anderen Worten, es fügt mehrere Zeichenfolgen zu einer durch Kommas getrennten Liste zusammen. Ich bin sicher, dass seine Leistung viel besser ist als jeder T-Sql-Trick .

Wie jede Aggregatfunktion kann sie in Kombination mit group by verwendet werden . Zum Beispiel:

SELECT id, name, desc, JoinStrings(CONVERT(VARCHAR(20), category_id))
FROM product p
INNER JOIN category_products c ON p.category_id = c.category_id
GROUP BY id, name, desc

Hier ist der C#-Code zum Erstellen der CLR-Assembly in Sql Server 2008:

using System;
using System.Data;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using Microsoft.SqlServer.Server;


[Serializable]
[Microsoft.SqlServer.Server.SqlUserDefinedAggregate(Format.UserDefined, IsInvariantToDuplicates=false, IsInvariantToOrder=false, IsInvariantToNulls=true, MaxByteSize=-1)]
public struct JoinStrings : IBinarySerialize
{
    private char[] sb;
    private int pos;
    public void Init()
    {
        sb = new char[512000];
        pos = 0;
    }

    public void Accumulate(SqlString Value)
    {
        if (Value.IsNull) return;
        char[] src = Value.ToString().ToCharArray();
        Array.Copy(src, 0, sb, pos, src.Length);
        pos += src.Length;
        sb[pos] = ',';
        pos++;
    }

    public void Merge(JoinStrings Group)
    {
        Accumulate(Group.Terminate());
    }

    public SqlString Terminate()
    {
        if (pos <= 0) 
            return new SqlString();
        else
            return new SqlString(new String(sb, 0, pos-1));
    }

    public void Read(System.IO.BinaryReader r)
    {
        this.Init();
        pos = r.ReadInt32();
        r.Read(sb, 0, pos);
    }

    public void Write(System.IO.BinaryWriter w)
    {
        w.Write(pos);
        w.Write(sb, 0, pos);
    }
}

Hier ist der Code zum Erstellen der Funktion (obwohl die Bereitstellung aus Visual Studio automatisch erfolgen sollte):

CREATE AGGREGATE [dbo].[JoinStrings]
(@s [nvarchar](4000))
RETURNS[nvarchar](max)
EXTERNAL NAME [YouAssemblyName].[JoinStrings]