Rumah > pangkalan data > tutorial mysql > mysql apakah jadual sementara

mysql apakah jadual sementara

青灯夜游
Lepaskan: 2022-02-17 16:01:41
asal
5437 orang telah melayarinya

Dalam mysql, jadual sementara merujuk kepada jadual yang digunakan buat sementara waktu dan digunakan untuk menyimpan beberapa set hasil perantaraan hanya boleh dilihat dalam sambungan semasa dan akan dipadam secara automatik oleh Mysql apabila sambungan ditutup dan mengosongkan semua ruang.

mysql apakah jadual sementara

Persekitaran pengendalian tutorial ini: sistem windows7, versi mysql8, komputer Dell G3.

Jadual sementara ialah jadual yang digunakan buat sementara waktu.

Jadual sementara ialah jadual yang digunakan oleh MySQL untuk menyimpan beberapa set hasil perantaraan hanya kelihatan dalam sambungan semasa Apabila sambungan ditutup, Mysql akan memadamkan jadual secara automatik dan melepaskan semua ruang.

Gunakan program klien MySQL yang lain untuk menyambung ke pelayan pangkalan data MySQL untuk membuat jadual sementara Jadual sementara hanya akan dimusnahkan apabila program klien ditutup, ia juga boleh dipadamkan secara manual.

Nota: Jadual sementara telah ditambahkan dalam MySQL versi 3.23 Jika versi MySQL anda lebih rendah daripada versi 3.23, anda tidak boleh menggunakan jadual sementara MySQL. Walau bagaimanapun, secara amnya jarang sekali untuk menggunakan versi rendah perkhidmatan pangkalan data MySQL seperti itu lagi

Dua jadual sementara dalam MySQL

Jadual sementara luaran

Jadual sementara yang dicipta oleh CREATE TEMPORARY TABLE ini dipanggil jadual sementara luaran. Jadual sementara ini hanya boleh dilihat oleh pengguna semasa dan akan ditutup secara automatik apabila sesi semasa tamat. Jadual sementara ini boleh dinamakan dengan nama yang sama seperti jadual bukan sementara (jadual bukan sementara tidak akan kelihatan kepada sesi semasa sehingga jadual sementara dipadamkan).

Jadual sementara dalaman

Jadual sementara dalaman ialah jadual sementara ringan khas yang digunakan untuk pengoptimuman prestasi. Jadual sementara jenis ini akan dibuat secara automatik oleh MySQL dan digunakan untuk menyimpan hasil perantaraan operasi tertentu. Operasi ini mungkin termasuk dalam fasa pengoptimuman atau fasa pelaksanaan. Jadual dalaman jenis ini tidak kelihatan kepada pengguna, tetapi melalui EXPLAIN atau SHOW STATUS anda boleh menyemak sama ada MYSQL menggunakan jadual sementara dalaman untuk membantu menyelesaikan operasi. Jadual sementara dalaman memainkan peranan yang sangat penting dalam proses pengoptimuman pernyataan SQL Banyak operasi dalam MySQL bergantung pada jadual sementara dalaman untuk pengoptimuman. Walau bagaimanapun, menggunakan jadual sementara dalaman memerlukan kos untuk mencipta jadual dan mengakses data perantaraan, jadi pengguna harus cuba mengelak daripada menggunakan jadual sementara semasa menulis pernyataan SQL.

Terdapat dua jenis jadual sementara dalaman:

  • Satu ialah jadual sementara HEAP Semua data dalam jadual sementara ini akan disimpan dalam ingatan jadual Operasi tidak memerlukan operasi IO.

  • Yang lain ialah jadual sementara OnDisk Seperti namanya, jadual sementara ini menyimpan data pada cakera. Jadual sementara OnDisk digunakan untuk mengendalikan operasi dengan hasil perantaraan yang agak besar.

Jika data yang disimpan dalam jadual sementara HEAP lebih besar daripada MAX_HEAP_TABLE_SIZE, jadual sementara HEAP akan ditukar secara automatik menjadi jadual sementara OnDisk.

Dalam 5.7, jadual sementara OnDisk boleh memilih untuk menggunakan enjin MyISAM atau enjin InnoDB melalui pembolehubah sistem INTERNAL_TMP_DISK_STORAGE_ENGINE.

Penggunaan biasa jadual sementara luaran

Jadual sementara luaran dikendalikan melalui CREATE TEMPORARY TABLE dan DROP TABLE, tetapi arahan SHOW TABLES memaparkan Apabila memasuki senarai jadual data, anda tidak akan dapat melihat jadual sementara yang anda buat. Dan selepas keluar dari sesi semasa, jadual sementara akan dimusnahkan secara automatik. Sudah tentu, ia juga boleh dimusnahkan secara manual (DROP TABLE).

1. Jenis enjin: hanya boleh: memori (timbunan), myisam, merge, innodb, mysql cluster (cluster) tidak disokong.

2. Beri perhatian kepada beberapa perkara apabila menggunakan jadual sementara luaran:

1) Akaun pangkalan data yang anda gunakan mesti mempunyai kebenaran untuk mencipta jadual sementara;

2) Dalam SQL yang sama, jadual sementara yang sama tidak boleh dikaitkan dua kali, jika tidak, ralat berikut akan dilaporkan;

mysql> select * from temp_table, temp_table as t2;
  error 1137: can't reopen table: 'temp_table'
Salin selepas log masuk

3) Jadual sementara boleh dilihat apabila menubuhkan a sambungan. Apabila menutup, ruang akan dikosongkan dan jadual sementara akan dipadamkan; menamakan semula jadual sementara. Walau bagaimanapun, anda boleh mengubah jadual sebaliknya: anda hanya boleh menggunakan alter table old_tp_table_name rename new_tp_table_name;

6), menjejaskan penggunaan fungsi replikasi;

7), jika anda mengisytiharkan alias untuk jadual, apabila Anda mesti menggunakan alias ini apabila menunjuk ke jadual ini. Lihat "Kemas Kini dan Padam Persatuan Multi-jadual MySQL"

Contoh:

3 Operasi jadual sementara dalam mybatis

mysql> CREATE TEMPORARY TABLE SalesSummary (
    -> product_name VARCHAR(50) NOT NULL
    -> , total_sales DECIMAL(12,2) NOT NULL DEFAULT 0.00
    -> , avg_unit_price DECIMAL(7,2) NOT NULL DEFAULT 0.00
    -> , total_units_sold INT UNSIGNED NOT NULL DEFAULT 0
);
Query OK, 0 rows affected (0.00 sec)

mysql> INSERT INTO SalesSummary
    -> (product_name, total_sales, avg_unit_price, total_units_sold)
    -> VALUES
    -> ('cucumber', 100.25, 90, 2);

mysql> SELECT * FROM SalesSummary;
+--------------+-------------+----------------+------------------+
| product_name | total_sales | avg_unit_price | total_units_sold |
+--------------+-------------+----------------+------------------+
| cucumber     |      100.25 |          90.00 |                2 |
+--------------+-------------+----------------+------------------+
1 row in set (0.00 sec)

mysql> DROP TABLE SalesSummary;  
mysql>  SELECT * FROM SalesSummary;  
ERROR 1146: Table 'RUNOOB.SalesSummary' doesn't exist
Salin selepas log masuk

   <update id="createTempTable">
        CREATE TEMPORARY TABLE IF NOT EXISTS temp 
        SELECT * FROM settlement_temp
         WHERE settle_date=#{settleDate} AND LENGTH(operator) IN(16,32) AND  pay_status IN (&#39;01&#39;,&#39;06&#39;) 
         ORDER BY settle_date,merchant_no
    </update>
Salin selepas log masuk
Penggunaan biasa jadual sementara dalaman
  <!-- 4、删除临时表 -->
    <update id="dropTempTable">  
        DROP TEMPORARY TABLE IF EXISTS settlement_temp;
    </update>
Salin selepas log masuk

Jika pengguna boleh menggunakan jadual sementara dalaman sesedikit mungkin semasa menulis pernyataan SQL untuk pengoptimuman pertanyaan, ia akan bertambah baik dengan berkesan Kecekapan pelaksanaan pertanyaan. Mula-mula kita mentakrifkan jadual t1

Semua operasi berikut adalah berdasarkan jadual t1 sebagai contoh.

CREATE TABLE t1( a int, b int); 
INSERT INTO t1 VALUES(1,2),(3,4);
Salin selepas log masuk

Gunakan petunjuk SQL_BUFFER_RESULT

dalam pernyataan SQL

SQL_BUFFER_RESULT主要用来让MySQL尽早的释放表上的锁。因为如果数据量很大的话,需要较长时间将数据发送到客户端,通过将数据缓冲到临时表中可以有效的减少读锁对表的占用时间。SQL_BUFFER_RESULT见《mysql查询优化之三:查询优化器提示(hint)》

例如:

mysql> explain format=json select SQL_BUFFER_RESULT * from t1;
    EXPLAIN
    {
      "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "2.00"
        },
        "buffer_result": {
          "using_temporary_table": true,
          "table": {
            "table_name": "t1",
            "access_type": "ALL",
        ...
Salin selepas log masuk
  • 如果SQL语句中包含了DERIVED_TABLE。

在5.7中,由于采用了新的优化方式,我们需要使用 set optimizer_switch=’derived_merge=off’来禁止derived table合并到外层的Query中。

例如:

mysql> explain format=json select * from (select * from t1) as tt;
    EXPLAIN
    {
      "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "2.40"
        },
        "table": {
          "table_name": "tt",
          "access_type": "ALL",
          ...
          "materialized_from_subquery": {
            "using_temporary_table": true,
        ...
Salin selepas log masuk
  • 如果我们查询系统表的话,系统表的数据将被存储到内部临时表中。

我们当前不能使用EXPLAIN来查看是否读取系统表数据需要利用到内部临时表,但是可以通过SHOW STATUS来查看是否利用到了内部临时表。

例如:

mysql> select * from information_schema.character_sets;
mysql> show status like &#39;CREATE%&#39;;
Salin selepas log masuk
  • 如果DISTINCT语句没有被优化掉,即DISTINCT语句被优化转换为GROUP BY操作或者利用UNIQUE INDEX消除DISTINCT, 内部临时表将会被使用。

mysql> explain format=json select distinct a from t1;
    EXPLAIN
    {
    {
      "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "1.60"
        },
        "duplicates_removal": {
          "using_temporary_table": true,
        ...
Salin selepas log masuk
  • 如果查询带有ORDER BY语句,并且不能被优化掉。下面几种情况会利用到内部临时表缓存中间数据,然后对中间数据进行排序。

1)如果连接表使用BNL(Batched Nestloop)/BKA(Batched Key Access)
例如:

1))BNL默认是打开的

mysql> explain format=json select * from t1, t1 as t2 order by t1.a;
EXPLAIN
{
  "query_block": {
  "select_id": 1,
  "cost_info": {
    "query_cost": "22.00"
  },
  "ordering_operation": {
    "using_temporary_table": true,
  ...
Salin selepas log masuk

2))关掉BNL后,ORDER BY将直接使用filesort。

mysql> set optimizer_switch=&#39;block_nested_loop=off&#39;;
Query OK, 0 rows affected (0.00 sec)
mysql> explain format=json select * from t1, t1 as t2 order by t1.a;
EXPLAIN
{
   "query_block": {
    "select_id": 1,
    "cost_info": {
      "query_cost": "25.00"
    },
    "ordering_operation": {
      "using_filesort": true,
    ...
Salin selepas log masuk

2)ORDER BY的列不属于执行计划中第一个连接表的列。

例如:

mysql> explain format=json select * from t as t1, t as t2 order by t2.a;
EXPLAIN
{
   "query_block": {
    "select_id": 1,
    "cost_info": {
      "query_cost": "25.00"
    },
    "ordering_operation": {
      "using_temporary_table": true,
    ...
Salin selepas log masuk

3)如果ORDER BY的表达式是个复杂表达式。

那么什么样的ORDER BY表达式,MySQL认为是复杂表达式呢?

1))如果排序表达式是SP或者UDF。

例如:

drop function if exists func1;
delimiter |
create function func1(x int)
returns int deterministic
begin
declare z1, z2 int;
set z1 = x;
set z2 = z1+2;
return z2;
end|
delimiter ;
explain format=json select * from t1 order by func1(a);
{
    "query_block": {
    "select_id": 1,
    "cost_info": {
      "query_cost": "2.20"
    },
    "ordering_operation": {
      "using_temporary_table": true,
    ...
Salin selepas log masuk

2))ORDER BY的列包含聚集函数

为了简化执行计划,我们利用INDEX来优化GROUP BY语句。

例如:

create index idx1 on t1(a);
  explain format=json SELECt a FROM t1 group by a order by sum(a);
  | {
       "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "1.20"
        },
        "ordering_operation": {
          "using_temporary_table": true,
          "using_filesort": true,
          "grouping_operation": {
            "using_filesort": false,
        ...
  drop index idx1 on t1;
Salin selepas log masuk

3))ORDER BY的列中包含有SCALAR SUBQUERY,当然该SCALAR SUBQUERY没有被优化掉。

例如:

explain format=json select (select rand() from t1 limit 1) as a from t1 order by a;        
| {
      "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "1.20"
        },
        "ordering_operation": {
          "using_temporary_table": true,
          "using_filesort": true,
            ...
Salin selepas log masuk

4) 如果查询既带有ORDER BY同时也有GROUP BY语句,但是两个语句使用的列不相同。

注意: 如果是5.7,我们需要将sql_mode设置为非only_full_group_by模式,否则会报错。

同样为了简化执行计划,我们利用INDEX来优化GROUP BY语句。

例如:

set sql_mode=&#39;&#39;;
create index idx1 on t1(b);
explain format=json select t1.a from t1 group by t1.b order by 1;
| {
     "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "1.40"
        },
    "ordering_operation": {
          "using_temporary_table": true,
          "using_filesort": true,
          "grouping_operation": {
            "using_filesort": false,
    ...
drop index idx1 on t1;
Salin selepas log masuk
  • 如果查询带有GROUP BY语句,并且不能被优化掉。下面几种情况会利用到内部临时表缓存中间数据,然后对中间数据进行GROUP BY。

1)如果连接表使用BNL(Batched Nestloop)/BKA(Batched Key Access)。

例如:

explain format=json select t2.a from t1, t1 as t2 group by t1.a;
    | {
        "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "8.20"
        },
        "grouping_operation": {
          "using_temporary_table": true,
          "using_filesort": true,
          "cost_info": {
            "sort_cost": "4.00"
        ...
Salin selepas log masuk

2) 如果GROUP BY的列不属于执行计划中的第一个连接表。

例如:

explain format=json select t2.a from t1, t1 as t2 group by t2.a;
    | {
        "query_block": {
        "select_id": 1,
        "cost_info": {
          "query_cost": "8.20"
        },
        "grouping_operation": {
          "using_temporary_table": true,
          "using_filesort": true,
          "nested_loop": [
        ...
Salin selepas log masuk

3) 如果GROUP BY语句使用的列与ORDER BY语句使用的列不同。

例如:

	set sql_mode=&#39;&#39;;
	explain format=json select t1.a from t1 group by t1.b order by t1.a;
	| {
	   "query_block": {
		"select_id": 1,
		"cost_info": {
		  "query_cost": "1.40"
		},
		"ordering_operation": {
		  "using_filesort": true,
		  "grouping_operation": {
			"using_temporary_table": true,
			"using_filesort": false,
		...
Salin selepas log masuk

4) 如果GROUP BY带有ROLLUP并且是基于多表外连接。

例如:

	explain format=json select sum(t1.a) from t1 left join t1 as t2 on true group by t1.a with rollup;
	| {
		"query_block": {
		"select_id": 1,
		"cost_info": {
		  "query_cost": "7.20"
		},
		"grouping_operation": {
		  "using_temporary_table": true,
		  "using_filesort": true,
		  "cost_info": {
			"sort_cost": "4.00"
		  },
		...
Salin selepas log masuk

5) 如果GROUP BY语句使用的列来自于SCALAR SUBQUERY,并且没有被优化掉。

例如:

explain format=json select (select avg(a) from t1) as a from t1 group by a;
| {
"query_block": {
"select_id": 1,
"cost_info": {
  "query_cost": "3.40"
},
"grouping_operation": {
  "using_temporary_table": true,
  "using_filesort": true,
  "cost_info": {
"sort_cost": "2.00"
  },
...
Salin selepas log masuk

IN表达式转换为semi-join进行优化

1) 如果semi-join执行方式为Materialization

例如:

set optimizer_switch=&#39;firstmatch=off,duplicateweedout=off&#39;;
explain format=json select * from t1 where a in (select b from t1);
| {
"query_block": {
"select_id": 1,
"cost_info": {
  "query_cost": "5.60"
},
"nested_loop": [
  {
"rows_examined_per_scan": 1,
  "materialized_from_subquery": {
"using_temporary_table": true,
"query_block": {
  "table": {
"table_name": "t1",
"access_type": "ALL",
...
Salin selepas log masuk

2) 如果semi-join执行方式为Duplicate Weedout

例如:

set optimizer_switch=&#39;firstmatch=off&#39;;
explain format=json select * from t1 where a in (select b from t1);
| {
"query_block": {
"select_id": 1,
"cost_info": {
  "query_cost": "4.80"
},
"duplicates_removal": {
  "using_temporary_table": true,
  "nested_loop": [
{
...
Salin selepas log masuk
  • 如果查询语句带有UNION,MySQL将利用内部临时表帮助UNION操作消除重复。

例如:

explain format=json select * from t1 union select * from t1;
| {
"query_block": {
"union_result": {
  "using_temporary_table": true,
  "table_name": "<union1,2>",
...
Salin selepas log masuk
  • 如果查询语句使用多表更新。

这里Explain不能看到内部临时表被利用,所以需要查看status。

例如:

update t1, t1 as t2 set t1.a=3;
show status like &#39;CREATE%&#39;;
Salin selepas log masuk
  • 如果聚集函数中包含如下函数,内部临时表也会被利用。

1) count(distinct *)
例如:
explain format=json select count(distinct a) from t1;
2) group_concat
例如:
explain format=json select group_concat(b) from t1;
Salin selepas log masuk

总之,上面列出了10种情况,MySQL将利用内部临时表进行中间结果缓存,如果数据量比较大的话,内部临时表将会把数据存储在磁盘上,这样显然会对性能有所影响。为了尽可能的减少性能损失,我们需要尽量避免上述情况的出现。

MySQL在以下几种情况会创建临时表:

1、UNION查询;
2、用到TEMPTABLE算法或者是UNION查询中的视图;
3、ORDER BY和GROUP BY的子句不一样时;
4、表连接中,ORDER BY的列不是驱动表中的;
5、DISTINCT查询并且加上ORDER BY时;
6、SQL中用到SQL_SMALL_RESULT选项时;
7、FROM中的子查询;
8、子查询或者semi-join时创建的表;
Salin selepas log masuk

EXPLAIN 查看执行计划结果的 Extra 列中,如果包含 Using Temporary 就表示会用到临时表。

当然了,如果临时表中需要存储的数据量超过了上限( tmp-table-size 或 max-heap-table-size 中取其大者),这时候就需要生成基于磁盘的临时表了。

在以下几种情况下,会创建磁盘临时表:

1、数据表中包含BLOB/TEXT列;
2、在 GROUP BY 或者 DSTINCT 的列中有超过 512字符 的字符类型列(或者超过 512字节的 二进制类型列,在5.6.15之前只管是否超过512字节);
3、在SELECT、UNION、UNION ALL查询中,存在最大长度超过512的列(对于字符串类型是512个字符,对于二进制类型则是512字节);
4、执行SHOW COLUMNS/FIELDS、DESCRIBE等SQL命令,因为它们的执行结果用到了BLOB列类型。
Salin selepas log masuk

从5.7.5开始,新增一个系统选项 internal_tmp_disk_storage_engine 可定义磁盘临时表的引擎类型为 InnoDB,而在这以前,只能使用 MyISAM。而在5.6.3以后新增的系统选项 default_tmp_storage_engine 是控制 CREATE TEMPORARY TABLE 创建的临时表的引擎类型,在以前默认是MEMORY,不要把这二者混淆了。

见下例:

mysql> set default_tmp_storage_engine = "InnoDB";
-rw-rw----   1 mysql mysql  8558 Jul  7 15:22 #sql4b0e_10_0.frm -- InnoDB引擎的临时表
-rw-rw----   1 mysql mysql 98304 Jul  7 15:22 #sql4b0e_10_0.ibd
-rw-rw----   1 mysql mysql  8558 Jul  7 15:25 #sql4b0e_10_2.frm
mysql> set default_tmp_storage_engine = "MyISAM";
-rw-rw----   1 mysql mysql     0 Jul  7 15:25 #sql4b0e_10_2.MYD -- MyISAM引擎的临时表
-rw-rw----   1 mysql mysql  1024 Jul  7 15:25 #sql4b0e_10_2.MYI
mysql> set default_tmp_storage_engine = "MEMORY";
-rw-rw----   1 mysql mysql  8558 Jul  7 15:26 #sql4b0e_10_3.frm -- MEMORY引擎的临时表
Salin selepas log masuk

【相关推荐:mysql视频教程

Atas ialah kandungan terperinci mysql apakah jadual sementara. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan