下一步就是读取另一张较大的表并对联结键应用散列函数。然后利用得到的散列值对已经在内存中的散列表进行探测以寻找匹配的行数据所在的散列桶。每个散列桶都有一个放在其中的数据行列表(通过一个位图来表示)。这个列表用来与探测行进行匹配。如果匹配成功,则返回这一行数据,否则丢弃。较大的表只读取一次,并检查其中的每一行来寻找匹配。hash联结的内层表被多次读取,较大的表是驱动表,因此仅读取一次,而较小的散列表则被探测很多次。
1. 建表和插入测试数据参考:http://www.jb51.cc/article/p-chkofhzn-zc.html
2. select s.*,c.class_name from student s,class c where s.class_id=c.id;
这个查询将会向下面的伪代码一样来处理结果:
deltermine the smaller row set,or in the case of an outer join,use the outer joined table(根据返回的行数选择内表)
select s.id,s.name,s.class_id from student shash the class_id column and build a hash table
select c.class_name,c.id from class c
hash the id column and probe the hash table
if match made,check bitmap to confirm row match
if not match made,discard the row.
首先内(右)表扫描加载到内存HASH表,hash key为JOIN列.
然后外(左)表扫描,并与内存中的HASH表进行关联,输出最终结果
3. 执行计划
postgres=# explain(analyze,verbose,buffers) select s.*,class c where s.class_id=c.id; QUERY PLAN ----------------------------------------------------------------------------------------------------------------------- Hash Join (cost=1.23..13.34 rows=384 width=44) (actual time=0.035..0.289 rows=384 loops=1) Output: s.id,s.class_id,c.class_name Hash Cond: (s.class_id = c.id) -- HASH join key,c.id Buffers: shared hit=4 -> Seq Scan on public.student s (cost=0.00..6.84 rows=384 width=12) (actual time=0.012..0.098 rows=384 loops=1) Output: s.id,s.class_id Buffers: shared hit=3 -> Hash (cost=1.10..1.10 rows=10 width=36) (actual time=0.012..0.012 rows=10 loops=1) Output: c.class_name,c.id Buckets: 1024 Batches: 1 Memory Usage: 1kB -- 内(右)表加载到内存,hash key是join key c.id,如果Batches大于1则会用到临时文件 Buffers: shared hit=1 -> Seq Scan on public.class c (cost=0.00..1.10 rows=10 width=36) (actual time=0.003..0.006 rows=10 loops=1) Output: c.class_name,c.id Buffers: shared hit=1 Total runtime: 0.359 ms (15 rows)