MySQL中order by中關於NULL值的排序問題


MySQL中order by 排序遇到NULL值的問題 MySQL數據庫,在order by排序的時候,如果存在NULL值,那么NULL是最小的,ASC正序排序的話,NULL值是在最前面的。 如果我們想讓NULL排在后面,讓非NULL的行排在前面該怎么做呢?


MySQL數據庫在設計的時候,如果字段允許NULL值,那么對該字段進行排序的時候需要注意那些值為NULL的行。


我們知道NULL的意思表示什么都不是,或者理解成“未知”也可以,它與任何值比較的結果都是false,


默認情況下,MySQL會認為NULL值比其他類型的數據小, 也就是說,在order by排序的時候,NULL是最小的,ASC正序排序的話,NULL值是在最前面的。 如果我們想讓NULL排在后面,讓非NULL的行排在前面該怎么做呢?


下面我們通過一個小例子,來說明這個情況。


首先,創建一個測試數據表 test_user,


mysql> create table test_user(

id int unsigned not null auto_increment,

username varchar(10) not null,

age int,

primary key (id))

engine=myisam default charset=utf8 ;

Query OK, 0 rows affected (0.14 sec)


填充6條數據,其中3條設置了age值,另外3條age值為NULL


mysql> insert into test_user values(1,'user1',28),(2,'user2',30);

Query OK, 2 rows affected (0.00 sec) Records: 2 Duplicates: 0 Warnings: 0


mysql> insert into test_user(username) values('user3'),('user4'),('user5');

Query OK, 3 rows affected (0.00 sec) Records: 3 Duplicates: 0 Warnings: 0


mysql> insert into test_user values(6,'user6',23);

Query OK, 1 row affected (0.00 sec)


mysql> select * from test_user;

+----+-----------+------+ | id | username | age | +----+-----------+------+

| 1 | user1 | 28 |

| 2 | user2 | 30 |

| 3 | user3 | NULL |

| 4 | user4 | NULL |

| 5 | user5 | NULL |

| 6 | user6 | 23 |

+----+-----------+------+ 6 rows in set (0.00 sec)


我們按照age字段從小到大排序,我們看到NULL值是最小的,排在了最前面


mysql> select * from test_user order by age;

+----+-----------+------+ | id | username | age | +----+-----------+------+

| 3 | user3 | NULL | | 4 | user4 | NULL |

| 5 | user5 | NULL |

| 6 | user6 | 23 |

| 1 | user1 | 28 |

| 2 | user2 | 30 |

+----+-----------+------+ 6 rows in set (0.00 sec)


按照age字段從大到小排序,我們看到NULL值確實是最小的,排在了最后面


mysql> select * from test_user order by age desc;

+----+-----------+------+ | id | username | age | +----+-----------+------+

| 2 | user2 | 30 |

| 1 | user1 | 28 |

| 6 | user6 | 23 |

| 3 | user3 | NULL |

| 4 | user4 | NULL |

| 5 | user5 | NULL |

+----+-----------+------+ 6 rows in set (0.00 sec)


這里有個需要注意的事項,就是NULL值本身是無法排序的,也就是說一個NULL是無法和另外一個NULL比較的。 你可能已經發現了,那三個NULL值的行,不管正序還是倒序,順序都是一致的,當然也可能不一致。


那如果在對age進行正序排序的時候,我們希望NULL值的記錄在最后面顯示,那該如何做呢?


1、重新生成一列,比如agenull,利用is null操作符,把NULL值的行變成1,非NULL值的行變成0,先對該字段排序,再對age排序


mysql> select *,age is null as agenull from test_user order by agenull,age; +----+-----------+------+---------+ | id | username | age | agenull | +----+-----------+------+---------+ | 6 | user6 | 23 | 0 | | 1 | user1 | 28 | 0 | | 2 | user2 | 30 | 0 | | 3 | user3 | NULL | 1 | | 4 | user4 | NULL | 1 | | 5 | user5 | NULL | 1 | +----+-----------+------+---------+ 6 rows in set (0.01 sec)


2、直接利用isnull函數對age列求值,跟第一種方法的道理是一樣的


mysql> select * from test_user order by isnull(age),age; +----+-----------+------+ | id | username | age | +----+-----------+------+ | 6 | user6 | 23 | | 1 | user1 | 28 | | 2 | user2 | 30 | | 3 | user3 | NULL | | 4 | user4 | NULL | | 5 | user5 | NULL | +----+-----------+------+ 6 rows in set (0.00 sec)


3、還可以利用MySQL中的一個小技巧,在字段前面加上一個負號,也就是減號,ASC改成DESC ,DESC改成ASC


mysql> select * from test_user order by -age desc; +----+-----------+------+ | id | username | age | +----+-----------+------+ | 6 | user6 | 23 | | 1 | user1 | 28 | | 2 | user2 | 30 | | 3 | user3 | NULL | | 4 | user4 | NULL | | 5 | user5 | NULL | +----+-----------+------+ 6 rows in set (0.00 sec)


所以,在設計數據庫的時候,如果某個字段要進行排序的話,最好不要為NULL。







免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM