按:之前看pg的执行计划,多次看到不同的排序方式,但不知何意。偶遇此篇讲解pg执行计划三种排序方式,备忘一下。

Sorting

Sorting is one of the most fundamental operations in database systems and understanding how they work inside is crucial in optimizing them.

This blog post mainly focuses on postgres(a kick ass relational database), but it can be translated to other databases in algorithmic terms.

Setting the stage

Below are the details of my setup.

Database version : 9.5 (Latest version at the time of this writing).

Operating system : Cent OS 7

Table structure is as below

Total Rows : 1 Million Rows.All values are unique via random data population.

Terminologies : If you are not familiar with any of the technical terms mentioned here, refer Postgres Guide.

Disk merge sort - When data does not fit in memory

If we consider a query like below,

Select * from users order by userid;

Running explain analyze, gives us the plan as follows (it actually runs the query).

performance_test=# explain analyze select * from users order by userid;

  Sort  (cost=352825.84..355325.84 rows=1000000 width=219) (actual time=1601.031..2033.621 rows=1000000 loops=1)
Sort Key: userid
Sort Method: external merge Disk: 225872kB
-> Seq Scan on users (cost=0.00..41250.00 rows=1000000 width=219) (actual time=0.477..225.933 rows=1000000 loops=1)
Planning time: 0.298 ms
Execution time: 2170.056 ms
(6 rows)

Of course, in a real life situation nobody would use select * and select without a limit, but for demo purposes this should be fine.

External merge is much like Merge sort. It is used when the data does not fit in memory. This is probably the slowest sort method since there is lot of disk thrashing involved i.e it takes data from disk sorts it and then puts it back, it takes total table data/memory it can fit.

There are two different memory areas we can talk about mainly in postgres. shared_buffers is where all of the table data i.e the cached data from tables is stored. work_mem or worker memory is what is used by postgres for sorting, joins etc.,

So when we say that the data does not fit in memory, it means that work_mem is too low.

Quick sort - Completely in memory

If we bump up worker memory(say 300MB) and change the query slightly as follows.

Select userid from users order by userid;

We have a new query plan as below.

performance_test=# explain analyze select * from users order by userid;

  Sort  (cost=140907.84..143407.84 rows=1000000 width=219) (actual time=719.208..879.849 rows=1000000 loops=1)
Sort Key: userid
Sort Method: quicksort Memory: 290202kB
-> Seq Scan on users (cost=0.00..41250.00 rows=1000000 width=219) (actual time=0.039..266.260 rows=1000000 loops=1)
Planning time: 0.079 ms
Execution time: 966.795 ms
(6 rows)

Postgres uses a well known sorting algorithm called Quick sort to accomplish in memory sorting. There are certain variations from a vanilla quick sort, you can lookup the source code to understand in much deeper detail.

This will definitely be faster than external merge, since all of the data is brought into memory and then the sorting is done in memory itself.

Heapsort - Sorting with a limit

Let’s look at a real life use case query.

Select userid from users order by userid limit 10;

Plan is changed as below.

performance_test=# explain analyze select userid from users order by userid limit 10;

 Limit  (cost=62859.64..62859.67 rows=10 width=8) (actual time=483.900..483.901 rows=10 loops=1)
-> Sort (cost=62859.64..65359.64 rows=1000000 width=8) (actual time=483.898..483.899 rows=10 loops=1)
Sort Key: userid
Sort Method: top-N heapsort Memory: 25kB
-> Seq Scan on users (cost=0.00..41250.00 rows=1000000 width=8) (actual time=0.045..300.833 rows=1000000 loops=1)
Planning time: 0.078 ms
Execution time: 483.933 ms
(7 rows)

Underlying problem is the same, order all the rows by the column userid and select the top 10. The default sort order is ascending order in postgres.

You might wonder how does the plan change, it has to sort all of the data anyway to get the top 10/bottom 10 rows. The answer is yes, but what changes is the memory usage for sorting. Since we need only a limited set of rows, there is no need to sort all of them inside memory.

To achieve this, postgres maintains a heap with a bounded size. It consumes the input values in sequence. After it fills the heap up to the target number of tuples/rows it starts looking each new value to see if it’s larger than all current values/smaller than all current values, or fits within the heap.

If it’s larger than all current values (ascending sort in our case) it gets discarded, since we have enough values already. If it’s smaller than all current values or than some current values, it’s inserted at the appropriate point in the heap, everything gets shifted down by one, and it bumps the last entry off the heap.

The Algorithm used is heap sort which makes use of the heap data structure to sort things. This is also a classical well known algorithm.

Normally, unless you have very wide rows and you are selecting more rows that cannot fit it in memory, then heap sort is what will be used.

There won’t be much speed difference between a quicksort and a heap-sort, but what will be significant is the memory usage which can be seen clearly from both the query execution plans. When allocating memory is costly i.e when you have lesser free memory to allocate immediately, often happens when your machine is not scaling well to the memory needs, then there can be huge differences in speed.

Using indexes for sorting

All of the algorithms above involve a great deal of overhead that is common in all databases i.e the disk seek. No matter what algorithm is used, it has to first fetch all of the rows and then sort the data. Disk as we all know is probably the slowest and most time consuming component in a computing system.

We can avoid disk seek by using indexes. A B-Tree are commonly used to speed up sorting, where conditions, group by and a whole lot of other use cases.

Creating an index on the userid column is pretty straightforward as below.

create index on users(userid);

The table definition also lists that there is an index.

There is a great speed up in query execution time.

performance_test=# explain analyze select userid from users order by userid limit 10;

  Limit  (cost=0.42..0.68 rows=10 width=8) (actual time=0.030..0.041 rows=10 loops=1)
-> Index Only Scan using users_userid_idx on users (cost=0.42..25980.42 rows=1000000 width=8) (actual time=0.029..0.034 rows=10 loops=1)
Heap Fetches: 0
Planning time: 0.096 ms
Execution time: 0.070 ms
(5 rows)

Since indexes are already ordered, it has to just lookup the values in the indexes which is much faster. After all a b-tree offers a lookup of O(logN) in asymptotic complexity time. A typical query that uses indexes will be in the order of milliseconds.

An index can be used in descending sort as well.

select userid from users order by userid desc limit 10;
performance_test=# explain analyze select userid from users order by userid desc limit 10;

  Limit  (cost=0.42..0.68 rows=10 width=8) (actual time=0.085..0.092 rows=10 loops=1)
-> Index Only Scan Backward using users_userid_idx on users (cost=0.42..25980.42 rows=1000000 width=8) (actual time=0.084..0.089 rows=10 loops=1)
Heap Fetches: 0
Planning time: 0.142 ms
Execution time: 0.121 ms
(5 rows)

Indexes can also be cached entirely in memory, since they are smaller in size.

If we want to sort by multiple columns then we have to create an index appropriately.

create index on users(name desc,userid asc);

The above index called as a composite index, can be used for a query such as below,

Select userid,name from users order by name desc,userid asc limit 10;
performance_test=# explain analyze Select userid,name from users order by name desc,userid asc limit 10;

  Limit  (cost=0.55..1.12 rows=10 width=59) (actual time=0.025..0.059 rows=10 loops=1)
-> Index Only Scan using users_name_userid_idx on users (cost=0.55..57240.55 rows=1000000 width=59) (actual time=0.024..0.030 rows=10 loops=1)
Heap Fetches: 0
Planning time: 0.114 ms
Execution time: 0.088 ms
(5 rows)

Optimization

I have listed four methods in which postgres does sorting.

  • External merge (Slowest, because of increased disk I/O)
  • Quick sort (Faster than external merge, best suited for sorting large data sets)
  • Top-N heapsort (Faster than quicksort, stops when the limit of data is gathered)
  • Sorting using indexes (Fastest, just lookup, no sorting)

It is important to understand that these algorithms by themselves are not slow. Postgres just uses the best algorithm for the job.

Optimizing Order by

  • First step is to use an index. Almost always this is the rule of thumb.
  • Check if your query has a limit clause. It makes no sense to order and select all of the rows in the table.
  • Bump up worker memory for sorting. This has to be done carefully since each order by, sorting and other operations can use this individually, i.e if it is configured to be 128MB, then if there are 10 queries running, the total memory usage would 10 * 128 MB.
  • In-memory tables. As said earlier, the slowest part of a database operation is disk access. An in-memory table can be used when it is not feasible to create indexes on all columns. This is usually done when a user does sorting on the application UI. Postgres does not have anything natively that supports/keeps the entire table in memory but there is a way to do this via IMCS. But you do not need to do this, since indexes are optimized to fit in memory and there is no need to cache/store the whole table in memory.

注:

  • External merge (Slowest, because of increased disk I/O)
  • Quick sort (Faster than external merge, best suited for sorting large data sets)
  • Top-N heapsort (Faster than quicksort, stops when the limit of data is gathered)
  • Sorting using indexes (Fastest, just lookup, no sorting)

参考:

https://madusudanan.com/blog/all-you-need-to-know-about-sorting-in-postgres/

All you need to know about sorting in Postgres的更多相关文章

  1. HDU Cow Sorting (树状数组)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2838 Cow Sorting Problem Description Sherlock's N (1  ...

  2. 1306. Sorting Algorithm 2016 12 30

    1306. Sorting Algorithm Constraints Time Limit: 1 secs, Memory Limit: 32 MB Description One of the f ...

  3. 算法:POJ1007 DNA sorting

    这题比较简单,重点应该在如何减少循环次数. package practice; import java.io.BufferedInputStream; import java.util.Map; im ...

  4. U3D sorting layer, sort order, order in layer, layer深入辨析

    1,layer是对游戏中所有物体的分类别划分,如UIlayer, waterlayer, 3DModelLayer, smallAssetsLayer, effectLayer等.将不同类的物体划分到 ...

  5. WebGrid with filtering, paging and sorting 【转】

    WebGrid with filtering, paging and sorting by Jose M. Aguilar on April 24, 2012 in Web Development A ...

  6. ASP.NET MVC WebGrid – Performing true AJAX pagination and sorting 【转】

    ASP.NET MVC WebGrid – Performing true AJAX pagination and sorting FEBRUARY 27, 2012 14 COMMENTS WebG ...

  7. poj 1007:DNA Sorting(水题,字符串逆序数排序)

    DNA Sorting Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 80832   Accepted: 32533 Des ...

  8. ural 1252. Sorting the Tombstones

    1252. Sorting the Tombstones Time limit: 1.0 secondMemory limit: 64 MB There is time to throw stones ...

  9. CF#335 Sorting Railway Cars

    Sorting Railway Cars time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

随机推荐

  1. 适配iPhoneX、iPhoneXs、iPhoneXs Max、iPhoneXr 屏幕尺寸及安全区域

    此篇文章是对上一篇文章(http://www.ifiero.com/index.php/archives/611)的进一步补充,主要说明如何适配Apple的最新三款手机iPhoneXs.iPhoneX ...

  2. Centos7下部署activeMQ消息队列服务

    #1.下载activeMQlinux包 http://activemq.apache.org/activemq-5100-release.html   下载linux的activeMQ包 #2.使用X ...

  3. 小米 OJ 编程比赛 02 月常规赛

    Carryon 数数字 描述 Carryon 最近迷上了数数字,然后 Starry 给了他一个区间[l,r] ,然后提了几个要求, 需要将 ll 到 rr 之间的数全部转化成 16 进制,然后连起来. ...

  4. Matlab 图象操作函数讲解

    h = imrect;pos = getPosition(h); 这个函数用来获取图象上特定区域的坐标,其中pos的返回值中有四个参数[xmin,ymin,width,height],特定区域的左上角 ...

  5. 4.安装hive

      下载安装包并解压安装元数据库配置hive添加hvie环境变量修改hive-env.sh修改hive配置文件初始化metastore使用hive cli配置hivemestore配置hiveserv ...

  6. Thunder团队第三周 - Scrum会议6

    Scrum会议6 小组名称:Thunder 项目名称:i阅app Scrum Master:宋雨 工作照片: 代秋彤照相,所以图片中没有该同学. 参会成员: 王航:http://www.cnblogs ...

  7. Acm hust 1.25

    闲着无聊做了点hust上 acm的训练题 A题 http://acm.hust.edu.cn/vjudge/contest/view.action?cid=104738#problem/A 看了20分 ...

  8. k邻近算法理解及代码实现

    github:代码实现 本文算法均使用python3实现 1 KNN   KNN(k-nearest neighbor, k近邻法),故名思议,是根据最近的 $ k $ 个邻居来判断未知点属于哪个类别 ...

  9. 每个zone的low memory是怎么计算出来的

    内核都是试图让活动页和不活动页的数量均衡 在分配内存时每次都会唤醒wakeup_swapd,这个函数会在 现在是不是已经没有全局的LRU表了?已经都变成per cgroup级别的LRU表了吗? ina ...

  10. [剑指Offer] 62.二叉搜索树的第k个结点

    题目描述 给定一颗二叉搜索树,请找出其中的第k大的结点.例如, 5 / \ 3 7 /\ /\ 2 4 6 8 中,按结点数值大小顺序第三个结点的值为4. [思路]遍历二叉搜索树,存入一个vector ...