Mysql_函数调用优化

MySQL函数优化:理解不确定性和性能影响
函数调用优化

MySQL函数在内部被标记为确定性或不确定性。如果给定参数固定值的函数可以为不同的调用返回不同的结果,则它是不确定的。不确定函数的示例: RAND()UUID()

如果某个函数被标记为不确定的,则将WHERE针对每一行(从一个表中选择时)或行的组合(从多表联接中选择时)评估子句中对该函数的引用。

MySQL还根据参数的类型(参数是表列还是常量值)确定何时评估函数。每当表列更改值时,都必须评估将表列作为参数的确定性函数。

非确定性函数可能会影响查询性能。例如,某些优化可能不可用,或者可能需要更多锁定。以下讨论使用 RAND()但也适用于其他不确定性函数。

假设一个表t具有以下定义:

CREATE TABLE t (id INT NOT NULL PRIMARY KEY, col_a VARCHAR(100));

考虑以下两个查询:

SELECT * FROM t WHERE id = POW(1,2);
SELECT * FROM t WHERE id = FLOOR(1 + RAND() * 49);

由于与主键的相等性比较,两个查询似乎都使用了主键查找,但这仅适用于第一个查询:

  • 第一个查询始终最多产生一行,因为POW()带有常量参数的常量是一个常量值,并用于索引查找。
  • 第二个查询包含一个使用非确定性函数的表达式,该表达式 RAND()在查询中不是常量,但实际上对表的每一行都有一个新值t。因此,查询读取表的每一行,评估每一行的谓词,并输出主键与随机值匹配的所有行。根据id列值和RAND()序列中的值, 它可以是零行,一行或多行 。

非确定性的影响不仅限于 SELECT陈述。该 UPDATE语句使用非确定性函数来选择要修改的行:

UPDATE t SET col_a = some_expr WHERE id = FLOOR(1 + RAND() * 49);

大概目的是最多更新主键与表达式匹配的一行。但是,它可能会更新零,一或多个行,具体取决于 id列值和RAND()序列中的值 。

刚刚描述的行为对性能和复制有影响:

  • 由于不确定函数不会产生恒定值,因此优化器无法使用其他可能适用的策略,例如索引查找。结果可能是表扫描。
  • InnoDB 可能升级为范围键锁,而不是为一个匹配的行获取单行锁。
  • 无法确定执行的更新对于复制是不安全的。

困难源于RAND()对表的每一行都对函数进行一次评估的事实 。为了避免进行多功能评估,请使用以下技术之一:

  • 将包含不确定性函数的表达式移到单独的语句,将值保存在变量中。在原始语句中,将表达式替换为对变量的引用,优化器可以将该变量视为常量值:

    SET @keyval = FLOOR(1 + RAND() * 49);
    UPDATE t SET col_a = some_expr WHERE id = @keyval;
    
  • 将随机值分配给派生表中的变量。此技术使变量在WHERE子句中的比较中使用之前被分配一个值 :

    SET optimizer_switch = 'derived_merge=off';
    UPDATE t, (SELECT @keyval := FLOOR(1 + RAND() * 49)) AS dt
    SET col_a = some_expr WHERE id = @keyval;
    

如前所述,该WHERE子句中的不确定性表达式 可能会阻止优化并导致表扫描。但是,WHERE如果其他表达式是确定性的,则可以部分优化该子句。例如:

SELECT * FROM t WHERE partial_key=5 AND some_column=RAND();

如果优化器可以partial_key用来减少所选行的集合, RAND()则执行的次数更少,这可以减少不确定性对优化的影响。

更多内容欢迎关注我的个人公众号“韩哥有话说”,100G人工智能学习资料,大量后端学习资料等你来拿。

韩哥有话说

### MYSQL_RESULT 的使用说明 `MYSQL_RESULT` 是 MySQL C API 中的一个结构体,用于存储通过 `mysql_store_result()` 或 `mysql_use_result()` 获取的查询结果集。它包含了查询返回的数据表的相关信息。 #### 1. 结构定义 `MYSQL_RESULT` 并不是一个可以直接使用的函数,而是一个由 MySQL 客户端库管理的结果集对象。通常情况下,开发者不会直接操作这个结构体,而是通过一系列接口函数来访问其内容[^3]。 #### 2. 常见关联函数 以下是与 `MYSQL_RESULT` 密切相关的几个重要函数: - **`mysql_store_result()`**: 将整个结果集加载到客户端内存中,并返回一个指向 `MYSQL_RES` 对象的指针。如果查询不返回任何数据,则可能返回 NULL。 - **`mysql_use_result()`**: 不会将结果集一次性加载到内存中,而是允许逐行获取数据。这种方式适合处理非常大的结果集,但需要注意资源释放问题。 - **`mysql_fetch_row(MYSQL_RES *result)`**: 返回当前结果集中的一行记录作为一个数组,每列对应数组中的一个元素。当没有更多行时返回 NULL[^2]。 - **`mysql_num_rows(MYSQL_RES *result)`**: 计算指定结果集中包含多少条记录(仅适用于 `mysql_store_result`) [^3]。 - **`mysql_free_result(MYSQL_RES *result)`**: 当完成对某个结果的操作之后应该调用此方法释放分配给它的内存空间。 #### 3. 示例代码 下面展示如何利用上述提到的功能实现基本的数据检索过程: ```c #include <mysql.h> #include <stdio.h> int main(){ MYSQL *conn; MYSQL_RES *res; MYSQL_ROW row; conn = mysql_init(NULL); /* Connect to database */ if (!mysql_real_connect(conn, "localhost", "root", "", "testdb", 0, NULL, 0)) { fprintf(stderr, "%s\n", mysql_error(conn)); exit(1); } /* Execute SQL query */ if (mysql_query(conn, "SELECT id,name FROM users")) { finish_with_error(conn); } res = mysql_store_result(conn); // Store result set if(res){ printf("Result:\n"); while ((row = mysql_fetch_row(res))) { // Fetch each row one by one. printf("%s %s\n", row[0], row[1]); } mysql_free_result(res); // Free memory after use. }else{ fprintf(stderr,"%d\n",mysql_errno(conn)); } mysql_close(conn); return EXIT_SUCCESS; } ``` 以上程序展示了连接至数据库服务器、执行 SELECT 查询以及遍历输出每一项的过程[^4]。 #### 4. 注意事项 - 如果尝试从非 SELECT 类型语句(如 INSERT/UPDATE)上调用 `mysql_store_result()` ,则总是得到 NULL 指针作为回应。 - 调用了 `mysql_store_result()` 后记得及时清理不再需要的结果集合以免造成泄漏;同样地,在结束应用程序之前关闭所有打开过的链接也是必要的良好习惯。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值