对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引。
如果没有查询条件,则每次查询所有的行。实际应用中,一般要指定查询的条件。对记录进行过滤。
查询的语法:
select 字段列表 from 表名列表 where 条件列表 group by 分组字段 having 分组之后的条件 order by 排序 limit 分页限定
1、基础查询
SELECT * FROM 表名;
Select的优化:
任何地方都不要使用 select * from t ,用具体的字段列表代替“*”,不要返回用不到的任何字段。学习阶段偷懒才写*号。
2、where条件查询
SELECT 字段名 FROM 表名 WHERE 条件;
注意:
*尽量将表字段定义为not null约束,这样不需要判断是否为null,如:int not null default -1,这是由于在mysql中含有空值的列很难进行查询优化,null值会使索引以及索引的统计信息变得很复杂。
*通常使用特殊的数据进行占位,如:int not null default 0,String not null default ‘’,但是会产生歧义,如果数据恰好是一个空字符串,就会和默认值空字符串一样,如果数据恰好是0的话也是一样的,尽量使用不会出现在常规业务逻辑行中的数据作为默认,表示null数据,如用:int not null default -1,如年龄的话,可能有0,所以不用0来表示默认,而用-1,因为年龄不会出现-1。
*应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索引而进行全表扫描,如:select id from t where num is null, 可以在num上设置默认值-1,确保表中num列没有null值,然后这样查询:select id from t where num=-1
*尽量避免在 where 子句中使用 or 来连接条件,否则将导致引擎放弃使用索引而进行全表扫描,可以使用union all.如果使用Or,则要保证两边的条件都有索引可以用。如:select id from t where num=10 or num=20,可以这样查询:
select id from t where num=10 union all select id from t where num=20
*in 和 not in 也要慎用,否则会导致全表扫描,如:select id from t where num in(1,2,3);
状态值即使在字段上增加了索引,往往索引不能起作用,如:gender 0,1,2表示男、女、保密,select * from student where gender in (1,2)。故状态值一般不使用索引,因为使用状态值时会导致一个状态值同时能够匹配到大量的记录;对于大量的记录,mysql有时会认为使用索引的开销比全表扫描还要大,从而主动放弃索引。即当你的条件容易匹配大量数据时,索引不一定能用得上。 对于连续的数值,能用 between 就不要用 in 了
*Like查询不能以通配符开头,如:user like ‘马%’,否则索引不会被使用。字符串比较时,不能使用包含逻辑,如:subject like ‘%PHP%’,因为这将导致全表扫描,效率太低了,若要提高效率,可以考虑全文检索。Mysql中全文索引不支持中文,用第三方提供的全文索引。
3、分组查询
语法:group by 分组字段
注意:分组之后查询的字段,要么是分组字段,要么是聚合函数,不能写其他字段,因为写其他字段没有任何意义,我们现在把男生或女生当成一个整体看,故不应该出现个人的信息,而应该是整体的共性内容。
分组之前的条件用where,分组之后的条件用having。
Where和having的区别:
1、 where在分组之前进行限定,如果不满足条件,则不参与分组。Having是在分组之后进行限定,如果不满足结果就不会被查询出来。
2、 where后不可以进行聚合函数的判断,而having可以进行聚合函数的判断。我们可以给聚合函数取别名,having后进行判断时使用别名进行判断。
4、排序查询
语法:order by 排序字段1 排序方式1,排序字段2 排序方式2…
经常作为查询条件在where或order by 语句中出现的列要建立索引。
当使用两个排序条件时,只要当第一排序条件一样时,才会使用第二排序条件。每一个排序条件都可以指定不同的排序方式。
5、分页查询
语法:limit 开始的索引 每页查询的条数
分页查询:limit 起始值从 0 开始, 长度
第一个参数注意是开始的索引,不是当前页。
开始的索引 = (当前页码-1)*每页显示的条数
limit是一个“方言”,limit关键字只能在mysql中使用。
LIMIT 子句可以被用于强制 SELECT 语句返回指定的记录数。LIMIT 接受一个或两个数字参数。
SELECT * FROM table LIMIT 5,10; // 检索记录行 6-15
如果只给定一个参数,它表示返回最大的记录行数目,换句话说,LIMIT n 等价于 LIMIT 0,n。
mysql> SELECT * FROM table LIMIT 5; //检索前 5 个记录行