select count(*)很慢

yangant 2010-10-28 04:15:18
select count(*) from A , B where A.id=b.aid
就这一条简单的count语句,却很慢,得接近2秒!但如果select count(*) from A(或B)时间马上就到0.0几秒,差的太多了吧!

A表和B表各有20多万条记录吧。而且,id列都是主健,B表的aid也已经建了nomal索引了。我看过执行计划,一个表是index fast full scan(cost 250 cardinality 246962 ) ,一个表是index unique scan (cost 1 cardinality 1) ,都不是全表扫描了呀,

请问还有优化的可能没,怎么优化?
...全文
1375 19 打赏 收藏 转发到动态 举报
写回复
用AI写文章
19 条回复
切换为时间正序
请发表友善的回复…
发表回复
fanpeng613 2010-11-01
  • 打赏
  • 举报
回复
转载 Oracle SQL 执行顺序 收藏

--查询的逻辑执行过程,来自技术内幕

(8) SELECT (9) DISTINCT (11) <TOP_specification> <select_list>
(1) FROM <left_table>
(3) <join_type> JOIN <right_table>
(2) ON <join_condition>
(4) WHERE <where_condition>
(5) GROUP BY <group_by_list>
(6) WITH {CUBE | ROLLUP}
(7) HAVING <having_condition>
(10) ORDER BY <order_by_list>

--------------------------------------------------------------------------------------

1.FROM:对FROM子句中前两个表执行笛卡尔积 生成虚拟表VT1
2.ON:对VT1表应用ON筛选器 只有满足 <join_condition>为真的行才被插入VT2
3.OUTER(JOIN):如果指定了OUTER JOIN 保留表(preserved table)中未找到的行将行作为外部行添加到VT2 生成T3
如果FROM包含两个以上表 则对上一个联结生成的结果表和下一个表重复执行步骤1和步骤3 直接结束
4.WHERE:对VT3应用WHERE筛选器 只有使 <where_condition>为TRUE的行才被插入VT4
5.GROUP BY:按GROUP BY子句中的列列表 对VT4中的行分组 生成VT5
6.CUBE|ROLLUP:把超组(Supergroups)插入VT6 生成VT6
7.HAVING:对VT6应用HAVING筛选器 只有使 <having_condition>为TRUE的组才插入VT7
8.SELECT:处理SELECT列表 产生VT8
9.DISTINCT:将重复的行从VT8中去除 产生VT9
10.ORDER BY:将VT9的行按ORDER BY子句中的列列表排序 生成一个游标 VC10
11.TOP:从VC10的开始处选择指定数量或比例的行 生成VT11 并返回调用者


你这是对AB表生成笛卡尔积,20万*20万,在400亿条记录里进行where筛选,2秒已经很快了兄弟。
建议把select count(*) from A , B where A.id=b.aid 修改为
select count(1)
from A
inner join B on A.id=b.aid
count(1)和count(*)的效率区分以及使用上有什么不同,建议自己到网上看看,或者自己试下会有比较深的理解的!
minitoy 2010-10-29
  • 打赏
  • 举报
回复
因为两列都有索引,且数据量差不多.oracle必然先从某表索引取一条数据,然后比较该条数据的id在另一表的索引中是否存在,所以一个表走的索引全扫描,一个表走的索引唯一扫描.
minitoy 2010-10-29
  • 打赏
  • 举报
回复
...
没法提升速度了,这已经是最好的结果了.
zkl516321905 2010-10-29
  • 打赏
  • 举报
回复
将*改成a.id试试啊,如果用*是没有走索引的
YY_MM_DD 2010-10-29
  • 打赏
  • 举报
回复
索引扫描本来就是采用的单块扫描的方式,他是通过索引存储的索引值和rowid进行查询,
一个rowid对应的只有一个数据块,如果是全表扫描的方式他就是根据一次性从磁盘读多块的
方式,具体一次读取多少块取决于db_file_multiblock_read_count
minitoy 2010-10-29
  • 打赏
  • 举报
回复
学习下.
请教,为什么索引会引起单块读呢[Quote=引用 13 楼 47522341 的回复:]
这个查询不应该走索引的。
本来就需要全表扫描,走索引反而只能使用db file sequential read也就是单块读,并且多了对索引的检索步骤。

你尝试使用hash_join,全表扫描来解决。
select /*+ use_hash(A,B) full(A) full(B) */count(*) from A , B where A.id=b.aid
[/Quote]
47522341 2010-10-29
  • 打赏
  • 举报
回复
正确的执行计划应该是
hash join
table full scan A
table full scan B
47522341 2010-10-29
  • 打赏
  • 举报
回复
这个查询不应该走索引的。
本来就需要全表扫描,走索引反而只能使用db file sequential read也就是单块读,并且多了对索引的检索步骤。

你尝试使用hash_join,全表扫描来解决。
select /*+ use_hash(A,B) full(A) full(B) */count(*) from A , B where A.id=b.aid
水中影子 2010-10-29
  • 打赏
  • 举报
回复
使用select count(a.id) ...试试
心中的彩虹 2010-10-28
  • 打赏
  • 举报
回复
[Quote=引用 6 楼 wkc168 的回复:]
引用楼主 yangant 的回复:
select count(*) from A , B where A.id=b.aid
就这一条简单的count语句,却很慢,得接近2秒!但如果select count(*) from A(或B)时间马上就到0.0几秒,差的太多了吧!

A表和B表各有20多万条记录吧。而且,id列都是主健,B表的aid也已经建了nomal索引了。我看过执行计划,一个表是……
[/Quote]
把*改成a.id减少时间
select count(a.id) from A , B where A.id=b.aid
心中的彩虹 2010-10-28
  • 打赏
  • 举报
回复
[Quote=引用楼主 yangant 的回复:]
select count(*) from A , B where A.id=b.aid
就这一条简单的count语句,却很慢,得接近2秒!但如果select count(*) from A(或B)时间马上就到0.0几秒,差的太多了吧!

A表和B表各有20多万条记录吧。而且,id列都是主健,B表的aid也已经建了nomal索引了。我看过执行计划,一个表是index fast full sc……
[/Quote]

select count(*) from A , B where A.id=b.aid --这是两个表连接肯定要花时间的
就这一条简单的count语句,却很慢,得接近2秒!但如果select count(*) from A(或B)时间马上就到0.0几秒,差的太多了吧!
select count(*) from A , B where A.id=b.aid --这是两个表连接肯定要花时间的
单独的话肯定是快些
如果你这两个表经常dml语句 建议试试分析表
analyze table A compute statistics
analyze table B compute statistics

majy 2010-10-28
  • 打赏
  • 举报
回复
如果是9i及以前的版本,不会自动分析,10G开始,默认会自动分析。

不明白分析,那一定要去学一下。 dbms_stats.gather_table_stats(....)等函数的使用
yangant 2010-10-28
  • 打赏
  • 举报
回复
[Quote=引用 3 楼 majy 的回复:]
第二个表inde unique scan不对,也应该刘 index fast full scan就对了,你是不是在B表了建了normal索引后,没有做过表分析或者索引分析(gather_table_stats or gather_index_stats)
[/Quote]
确实没有分析过,不知道怎么分析,分析之后能做什么呀!但B表也是经常查询的一个表,是否oracle会自动自己分析了
majy 2010-10-28
  • 打赏
  • 举报
回复
第二个表inde unique scan不对,也应该刘 index fast full scan就对了,你是不是在B表了建了normal索引后,没有做过表分析或者索引分析(gather_table_stats or gather_index_stats)
  • 打赏
  • 举报
回复
select count(a.id) 看看
YY_MM_DD 2010-10-28
  • 打赏
  • 举报
回复
用存在量词试下看exists
​ 博主介绍:✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌项目名称基于Web的酒店客房管理系统的设计与实现系统说明3.2.1  用户模块功能分析前台模块应主要包括用户登录模块、用户注册模块、查看客房信息模块、客房预定 模块、用户留言模块、充值模块和个人信息维护模块。用户登录模块:用户通过输入注册的的账号和密码,然后进行身份验证,匹配成功 后实现登录功能。用户注册模块:用户输入合法的账号和密码可以实现注册功能。 查看客房信息模块:用户可以通过首页查看客房的价格,图片,详情等信息,从而   可以选择想要预定的房间。 客房预定模块:用户选择自己想要预定的客房后,可以通过输入相关信息进行订房操作。用户留言模块:用户可以向管理员发送留言。 充值模块:用户可以通过添加银行卡再为自己进行充值操作。 个人信息维护模块:用户可以修改自己的姓名、密码、身份证号等信息,还可以查看自己的订单。3.2.2  管理员模块功能分析对于管理员而言,一个好的管理系统总是能让酒店的管理工作事半功倍[7]。管理员 能够通过这个系统对自己的酒店情况一目了然,应该包括客户留言模块、客房管理模 块、订房信息管理模块、入住信息管理模块、统计分析模块、酒店新闻管理模块、会 员信息管理模块、员工信息管理模块、系统用户管理模块、个人信息维护模块。客户留言模块:管理员可以查看并回复用户的留言。客房管理模块:管理员可以管理客房信息,可以添加新的客房,删除已经停用的客房信息,还可以修改现有的客房信息。订房信息管理模块:管理员可以处理用户的订房请求,为用户办理入住手续。入住信息管理模块:对于没有注册的线下客户,也可以办理入住手续。统计分析模块:可以对酒店所有的入住信息进行记录和总结分析。酒店新闻管理模块:管理员可以更新网站上的新闻公告,展示图片等信息。会员信息管理模块:管理员可以查看到所有的注册会员信息,可以对会员信息进行删除,修改,添加操作。员工信息管理模块:管理员可以查看到自己公司所有的员工信息,而且还可以对员工的相关信息进行管理。系统用户管理模块:管理员可以查看到所有的系统管理员信息并对管理员信息进行管理。个人信息维护模块:管理员可以更改自己的登录密码或者是姓名、性别、手机号等 相关个人信息。​编辑 环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。2.IDE环境:IDEA,Eclipse,Myeclipse都可以。推荐IDEA;3.tomcat环境:Tomcat 7.x,8.x,9.x版本均可4.硬件环境:windows 7/8/10 1G内存以上;或者 Mac OS; 5.数据库:MySql 5.7版本;6.是否Maven项目:否;技术栈1. 后端:Spring+SpringMVC+Mybatis2. 前端:JSP+CSS+JavaScript+jQuery使用说明1. 使用Navicat或者其它工具,在mysql中创建对应名称的数据库,并导入项目的sql文件;2. 使用IDEA/Eclipse/MyEclipse导入项目,Eclipse/MyEclipse导入时,若为maven项目请选择maven;若为maven项目,导入成功后请执行maven clean;maven install命令,然后运行;3. 将项目中springmvc-servlet.xml配置文件中的数据库配置改为自己的配置;4. 运行项目,在浏览器中输入http://localhost:8080/ 登录运行截图​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑​编辑 用户管理控制层:package com.houserss.controller;import javax.servlet.http.HttpSession;import org.apache.commons.lang3.StringUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.RequestParam;import org.springframework.web.bind.annotation.ResponseBody;import com.houserss.common.Const;import com.houserss.common.Const.Role;import com.houserss.common.ServerResponse;import com.houserss.pojo.User;import com.houserss.service.IUserService;import com.houserss.service.impl.UserServiceImpl;import com.houserss.util.MD5Util;import com.houserss.util.TimeUtils;import com.houserss.vo.DeleteHouseVo;import com.houserss.vo.PageInfoVo;/** * Created by admin */@Controller@RequestMapping(/user/)public class UserController if (ip != null && ip.length() > 0) String[] ips = ip.split(,);for (int i = 0; i  

17,377

社区成员

发帖
与我相关
我的任务
社区描述
Oracle 基础和管理
社区管理员
  • 基础和管理社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧