MySQL对具有字母和数字字符的产品代码进行排序

时间:2013-01-05 01:31:02

标签: mysql sql sorting

我必须对产品名称的查询输出进行排序,其中名称包含字母和数字字符。

我已经找到了将变量转换为数值(+0等)的各种解决方案,并且它们对产品名称的数字部分进行排序。但是产品名称字符串的前一部分具有不同的长度,因此名称不按字母顺序排序:

Post Lantern PL1
Post Lantern PL2
Post Lantern PL10
Post Lantern PL22

Landscape Light LV1
Landscape Light LV2
Landscape Light LV10
Landscape Light LV11

我猜这些较短的名字是先排序的吗?

我希望结果自然排序:按字母顺序排列,数字也按照自然顺序排列。我试过了:

ORDER by CAST(`product_name` AS DECIMAL), product_name
...
ORDER by product_name+0

较短的名称首先排序,即使它们在字母表后面。最后一部分中的数字需要按数字顺序排列。

1 个答案:

答案 0 :(得分:0)

这是一个很长的查询,正在寻找你正在寻找的东西。不过,我不确定性能。您可能还想对几个记录进行一些测试以确保。

SELECT
    *
    ,SUBSTRING(
        REVERSE(CAST(REVERSE(CONCAT(`product_name`,'8')) AS UNSIGNED)),1,
        CHARACTER_LENGTH(
            REVERSE(CAST(REVERSE(CONCAT(`product_name`,'8')) AS UNSIGNED))
        )-1
    ) AS 'numericVal'
FROM `some_table`
ORDER BY
    SUBSTRING(`product_name`,1,CHAR_LENGTH(`product_name`)-CHAR_LENGTH(`numericVal`)),
    CAST(`numericVal` AS UNSIGNED INTEGER)

CONCAT()函数中的 8 用于以零结尾的数字。否则,当你逆转,例如字符串“etc30”并解析数字 3 ,而不是 03 。因此,将其反转将再次产生 3 而不是 30

您可以使用您喜欢的任何单个数字(零除外)更改CONCAT()函数中的两个 8

[编辑2]

这是故障。

# Example record "Post Lantern PL10"...
SELECT
    *
    ,SUBSTRING(                                         # 5a) substring of this is calculated
        REVERSE(                                        # 4) gets re-reversed into "108"
            CAST(                                       # 3) gets casted into an integer so "801" part is parsed
                REVERSE(                                # 2) gets reversed: "801LP nretnaL tsoP"
                    CONCAT(`product_name`,'8')          # 1) is concatenated with an 8: "Post Lantern PL108"
                )
            AS UNSIGNED)
        ),
        1,                                              # 5b) from the first character (index is 1 for this in SQL)
        CHARACTER_LENGTH(                               # 5c) and the length is recalculated (steps 1-4 repeated)
            REVERSE(
                CAST(
                    REVERSE(
                        CONCAT(`product_name`,'8')
                    )
                AS UNSIGNED)
            )
        )-1                                             # 5d1) minus 1 because at the beginning we appended an 8 and we
                                                        # 5d2) want to get rid of it now, so we're dropping the last digit
    ) AS 'numericVal'
FROM `some_table`
ORDER BY                                                        # 6) order by
    SUBSTRING(`product_name`,                                   # 7a) first, substring `product_name`
        1,                                                      # 7b) from the first character
        CHAR_LENGTH(`product_name`)-CHAR_LENGTH(`numericVal`)   # 7c) with the total length - length of numeric part
    ),
    CAST(`numericVal` AS UNSIGNED INTEGER)                      # 8a) then, by the numeric part, which gets casted into
                                                                # 8b) an integer for accurate numeric ordering

[编辑1]

我认为你拥有的最佳镜头(完全控制不同的数据)是将product_name分成3列 - product_name(如“景观灯” ),product_class(或其他,如“LV”)和product_version(数字部分)。