Are there any benefits to using sql_variant over varchar in SQL Server?

前端 未结 4 1791
日久生厌
日久生厌 2021-02-19 03:29

I currently have a database table setup as follows (EAV - business reasons are valid):

  • Id - int (PK)
  • Key - unique, varchar(15)
  • Value - varchar(10
相关标签:
4条回答
  • 2021-02-19 03:47

    The good thing about sql variant is that you can store several types in a column and you keep the type information.

    Insert into MySettings values ('Name','MyName'); Insert into MySettings values ('ShouesNumber',45); Insert into MySettings values ('MyDouble',31.32);

    If you want to retrieve the type:

    select SQL_VARIANT_PROPERTY ( value , 'BaseType' ) as DataType,* from mysettings
    

    and you have:

    Datatype Name          Value
    -----------------------------
    varchar  Name          MyName
    int      ShoesNumber   45
    numeric  MyDouble      31.32
    

    Unfortunately this has several drawbacks:

    1. not very fast
    2. not well supported by ORM frameworks
    0 讨论(0)
  • 2021-02-19 03:55

    The sql_variant type has its limitations as described well by Zarathos.

    What I find confusing is that you mention varchar(1000) and then 'ouch' about returning a converted nvarchar(4000).

    I would start by saying that it is a good thing that the entire world have finally stopped using local and limited charsets and decided to go all in on Unicode and UTF-8, so you should prefer nvarchar over varchar and ntext over text.

    And the return is nvarchar(4000) and not nchar(4000). Difference is that any varchar is - variable in size, while the plain type char is fixed in size. Returning tuples of char(4000) would be sending a lot of empty waste, but with varchar this is not an issue.

    Okay. But what would be a proper datatype be for you? I would recommend ntext. What was 1000 today could be 10'000 tomorrow. If you have 'a lot of text' that you are not indexing, then perhaps your database should not decide what the limit may be. It's just text.

    ntext also fits well with .NET, as its strings are always in Unicode. Conversion from string to int is also faster in .NET than done by sql server.

    Hope this helps

    0 讨论(0)
  • 2021-02-19 03:59

    If you change the type to sql_variant, you will have to use the IDataRecord.GetValue method. It will preserve the type all the way.

    So in .NET it will allow you to have this kind of code:

    // read an object of SQL underlying data type 'int' stored in an sql_variant column
    object o = myReader.GetValue(); // o.GetType() will be System.Int32
    
    // read an object of SQL underlying data type '(n)varchar' or '(n)char' stored in an sql_variant column
    object o = myReader.GetValue(); // o.GetType() will be System.String
    
    // read an object of SQL underlying data type 'datetime' stored in an sql_variant column
    object o = myReader.GetValue(); // o.GetType() will be System.DateTime
    
    etc...
    

    Of course, it supposes you do the same when saving. Just set SqlParameter.Value to the opaque value, don't use the DbType.

    EAV with various (standard) types as value is the one case where I personally think sql_variant is interesting.

    Of course "SQLServer-focused guys" (read: DBAs) don't like it at all :-) On the SQL Server side, sql_variant is not very practical to use (as noted in the comments), but if you keep it as an opaque "thing" and don't have to use it in SQL procedure code, I think it's ok. So, it's more an advantage on the .NET/OO programming side.

    0 讨论(0)
  • 2021-02-19 04:11

    I don't see that it has been mentioned yet, so I'll mention that a fairly common approach to this problem is a table like this:

    • Id - int (PK)
    • Key - unique, varchar(15)
    • ValueType - Integer (0 - String, 1 - Integer, 3 - Float) (optional)
    • StringValue - varchar(1000)
    • IntValue - Integer
    • FloatValue - Double

    Advantages:

    • Data is saved in its appropriate form (storing ints as strings wastes a lot of bits)
    • You can do fancy queries like WHERE left(key,5) = 'SHOES', and IntValue>5
    • The ValueType column is only useful you are using prefix/suffixes on your keys (to retrieve sets of keys) and the set could be of mixed type. i.e. WHERE left(key,4) = 'Size' and ValueType = 1

    Disadvantages:

    • Everywhere where you use this table you have to ensure that you/get/set the correct Value column
    • If you decide you need/want the ValueType column, you have to ensure that you get/set it correctly.
    0 讨论(0)
提交回复
热议问题