深入理解 mysqli 风格转换与现代数据库操作实践

深入理解 mysqli 风格转换与现代数据库操作实践

本文深入探讨了 `mysqli` 数据库扩展中对象式与过程式两种编程风格的转换与应用,并指出对象式 `mysqli` 并非真正的面向对象编程,但仍是推荐的现代实践。文章通过具体代码示例演示了如何将对象式 `mysqli` 转换为过程式,并剖析了常见错误。同时,强烈建议采用 pdo 或简化版对象式 `mysqli`,以提升代码的健壮性和可维护性,强调了错误报告的重要性,并推荐使用 `fetch_all` 简化结果集处理。

mysqli 数据库扩展的两种编程风格

mysqli 扩展提供了两种与 MySQL 数据库交互的编程风格:对象式(Object-Oriented Style)和过程式(Procedural Style)。这两种风格在功能上是等价的,主要区别在于函数的调用方式和参数传递。

对象式风格:通过创建 mysqli 类的实例,然后调用该实例的方法来执行数据库操作。例如:$mysqli->prepare()。过程式风格:直接调用以 mysqli_ 或 mysqli_stmt_ 开头的函数,通常需要将 mysqli 连接对象或 mysqli_stmt 语句对象作为第一个参数传入。例如:mysqli_prepare($mysqli, …)。

值得注意的是,使用对象式 mysqli 并不意味着您的代码就是完全面向对象的。它仅仅是提供了一种更现代、更结构化的 API 调用方式,而无需深入理解完整的 OOP 范式。

对象式 mysqli 到过程式的转换

将对象式 mysqli 代码转换为过程式风格,核心在于将对象方法调用替换为对应的全局函数,并确保将 mysqli 连接对象或语句对象作为第一个参数传递。

以下是常见的对象式 mysqli 方法及其对应的过程式函数:

对象式方法 过程式函数 描述

$mysqli = new mysqli(…)mysqli_connect(…)建立数据库连接$mysqli->prepare(…)mysqli_prepare($mysqli, …)准备 SQL 语句$stmt->bind_param(…)mysqli_stmt_bind_param($stmt, …)绑定参数到预处理语句$stmt->execute()mysqli_stmt_execute($stmt)执行预处理语句$stmt->get_result()mysqli_stmt_get_result($stmt)获取结果集对象$result->num_rowsmysqli_num_rows($result)获取结果集行数$result->fetch_assoc()mysqli_fetch_assoc($result)从结果集中获取关联数组行$stmt->close()mysqli_stmt_close($stmt)关闭预处理语句$mysqli->close()mysqli_close($mysqli)关闭数据库连接

根据上述映射规则,将以下对象式 mysqli 代码:

function build_calendar($month, $year){    $mysqli = new mysqli('localhost', 'root', '', 'ems');    $stmt = $mysqli->prepare("SELECT * FROM bookings WHERE MONTH(event_date) = ? AND YEAR(event_date)=?");    $stmt->bind_param('ss', $month, $year);    $bookings = array();    if($stmt->execute()){        $result = $stmt->get_result();        if($result->num_rows > 0){            while($row = $result->fetch_assoc()){                $bookings[] = $row['event_date'];            }            $stmt->close();        }    }    // ... 其他逻辑}

转换为过程式风格,其代码如下:

function build_calendar_procedural($month, $year){    $mysqli = mysqli_connect('localhost', 'root', '', 'ems');    // 检查连接是否成功    if (mysqli_connect_errno()) {        die("连接失败: " . mysqli_connect_error());    }    $stmt = mysqli_prepare($mysqli, "SELECT * FROM bookings WHERE MONTH(event_date) = ? AND YEAR(event_date)=?");    // 检查预处理是否成功    if (!$stmt) {        die("预处理失败: " . mysqli_error($mysqli));    }    mysqli_stmt_bind_param($stmt, 'ss', $month, $year);    $bookings = array();    if (mysqli_stmt_execute($stmt)) {        $result = mysqli_stmt_get_result($stmt);        if ($result && mysqli_num_rows($result) > 0) {            while ($row = mysqli_fetch_assoc($result)) {                $bookings[] = $row['event_date'];            }            mysqli_free_result($result); // 释放结果集        }        mysqli_stmt_close($stmt);    } else {        die("执行失败: " . mysqli_stmt_error($stmt));    }    mysqli_close($mysqli); // 关闭连接    return $bookings;}

注意事项:

在转换过程中,初学者常犯的错误是尝试将参数直接传递给 mysqli_query() 函数,例如:

$connection = mysqli_connect('localhost', 'root', '', 'ems');$query = "SELECT * FROM bookings WHERE MONTH(event_date) = ? AND YEAR(event_date)=?";$get_dates = mysqli_query($connection, $query); // 错误用法!

这会导致 mysqli_fetch_assoc() 接收到的 $result 参数类型不正确,从而抛出 TypeError。mysqli_query() 函数适用于执行简单的、非预处理的 SQL 语句,或者已经将参数直接嵌入到 SQL 字符串中的语句(但这容易导致 SQL 注入)。对于带有占位符(?)的参数化查询,必须使用预处理语句(mysqli_prepare、mysqli_stmt_bind_param、mysqli_stmt_execute 等)来安全地执行。

推荐的现代数据库操作实践

尽管 mysqli 提供了两种风格,但为了代码的健壮性、可维护性和安全性,我们强烈推荐以下实践:

1. 优先使用 PDO

PHP Data Objects (PDO) 是一个更通用、更灵活的数据库抽象层,它支持多种数据库系统(如 MySQL, PostgreSQL, SQLite 等)。PDO 提供了统一的 API,使得在不同数据库之间切换更加容易,并且在预处理语句方面提供了更简洁、更强大的功能。

以下是使用 PDO 实现相同功能的示例:

function build_calendar_pdo(PDO $pdo, $month, $year): array{    $stmt = $pdo->prepare("SELECT event_date FROM bookings WHERE MONTH(event_date) = :month AND YEAR(event_date) = :year");    $stmt->bindParam(':month', $month, PDO::PARAM_INT);    $stmt->bindParam(':year', $year, PDO::PARAM_INT);    $stmt->execute();    return $stmt->fetchAll(PDO::FETCH_COLUMN); // 直接获取所有event_date列的值}// 示例连接和调用try {    $pdo = new PDO('mysql:host=localhost;dbname=ems;charset=utf8mb4', 'root', '');    $pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); // 设置错误模式为抛出异常    $bookings = build_calendar_pdo($pdo, 1, 2020);    // var_dump($bookings);} catch (PDOException $e) {    die("数据库操作失败: " . $e->getMessage());}

PDO 的优势在于其统一的接口、更强大的错误处理机制和更好的灵活性。

2. 简化和优化对象式 mysqli

如果您确实需要使用 mysqli 且不打算转向 PDO,可以通过以下方式简化和优化对象式 mysqli 代码:

a. 启用错误报告

默认情况下,mysqli 的错误处理可能不够直观。通过设置 mysqli_report,可以让 mysqli 在发生错误时抛出异常,从而更好地捕获和处理错误。

mysqli_report(MYSQLI_REPORT_ERROR | MYSQLI_REPORT_STRICT);$mysqli = new mysqli('localhost', 'root', '', 'ems');// 此时如果连接失败,将直接抛出异常,无需手动检查 mysqli_connect_errno()

MYSQLI_REPORT_ERROR 会报告操作错误,而 MYSQLI_REPORT_STRICT 会将警告转换为异常,这使得错误处理更加一致和可靠。

b. 使用 fetch_all 简化结果集处理

对于需要获取所有结果集的场景,mysqli_result::fetch_all() 方法可以大大简化代码,避免手动循环 fetch_assoc()。

mysqli_report(MYSQLI_REPORT_ERROR | MYSQLI_REPORT_STRICT);$mysqli = new mysqli('localhost', 'root', '', 'ems');function build_calendar_simplified(mysqli $mysqli, $month, $year): array{    $stmt = $mysqli->prepare("SELECT event_date FROM bookings WHERE MONTH(event_date) = ? AND YEAR(event_date) = ?");    $stmt->bind_param('ss', $month, $year);    $stmt->execute();    $result = $stmt->get_result();    return $result->fetch_all(MYSQLI_ASSOC); // 获取所有行作为关联数组}// 示例调用$bookings = build_calendar_simplified($mysqli, 1, 2020);// var_dump($bookings);

在这个简化版本中,函数签名中明确了 $mysqli 参数的类型为 mysqli 对象,这增加了代码的可读性和健壮性。fetch_all(MYSQLI_ASSOC) 会直接返回一个包含所有结果行的二维关联数组,省去了手动循环和构建数组的步骤。

总结

在 mysqli 的对象式和过程式风格之间,虽然过程式风格可以实现相同的功能,但对象式风格通常被认为是更现代、更易于维护的选择。然而,无论选择哪种风格,都应遵循预处理语句的最佳实践来防止 SQL 注入。更进一步,我们强烈推荐使用 PDO 进行数据库操作,因为它提供了更高级的抽象、更统一的接口和更强大的错误处理机制。如果坚持使用 mysqli,请务必启用错误报告并利用 fetch_all 等功能来简化代码,提升开发效率和代码质量。

以上就是深入理解 mysqli 风格转换与现代数据库操作实践的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1331296.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月12日 17:22:46
下一篇 2025年12月12日 17:22:52

相关推荐

  • 深入理解 PHP 配置:php.ini 与 .user.ini 的异同与应用

    本文旨在详细解析 php 配置中 `php.ini` 和 `.user.ini` 文件的关键区别、作用范围及其适用场景。我们将探讨 `php.ini` 作为全局配置的特性,以及 `.user.ini` 如何在特定目录下实现配置覆盖,并强调其在 php-fpm/fastcgi 环境下的依赖性,同时提供…

    2025年12月12日
    000
  • 简化PHP条件判断:优化复杂If语句的实践

    本文旨在探讨PHP中复杂条件逻辑的简化策略,通过分析一个具体的if-elseif结构案例,展示如何将其重构为更简洁、易读且易于维护的形式。我们将深入理解原始代码的意图,对比简化后的逻辑,并强调在重构过程中验证业务需求的重要性,以确保代码优化在提升可读性的同时,不改变原有功能。 在软件开发中,条件判断…

    2025年12月12日
    000
  • 将HTML内容作为纯文本代码展示的PHP教程

    本教程详细介绍了如何使用php将html文件内容读取出来,并将其作为纯文本(即代码形式)在网页或邮件中展示。核心方法是利用`htmlspecialchars()`函数对html实体进行转义,并结合`preg_replace()`将换行符转换为html的“标签,从而确保原始代码结构和格式的正确呈现。…

    2025年12月12日
    000
  • 在Laravel中利用MySQL通配符实现含连字符/空格的灵活搜索

    本文介绍如何在laravel应用中,利用mysql的`_`通配符,实现对包含连字符或空格的商品名称进行灵活且不区分大小写的搜索。通过将用户输入的搜索词中的连字符和空格替换为`_`通配符,可以有效地匹配数据库中格式多样的记录,避免常见的语法错误,提升搜索的健壮性。 在开发Web应用时,我们经常需要实现…

    2025年12月12日
    000
  • 使用FPDI在PHP中合并PDF并智能适配页面尺寸与方向

    本教程详细阐述了如何在php中利用fpdi库合并多个pdf文件,并解决因源文件页面尺寸或方向不一致导致的裁剪问题。通过动态获取每个导入页面的尺寸和方向信息,fpdi能够智能地为新页面设置正确的布局,确保所有内容完整无损地呈现,从而实现高效且兼容性强的pdf合并操作。 引言 在Web应用开发中,经常会…

    2025年12月12日
    000
  • MySQL中特殊字符编码的最佳实践:为什么选择utf8mb4

    在MySQL数据库处理包含特殊字符的数据时,选择正确的字符编码至关重要,否则可能导致数据乱码或查询失败。本文将深入探讨处理 `éšš+á` 等特殊字符的挑战,比较不同字符集的兼容性,并强烈推荐使用 `utf8mb4` 作为全面支持多语言和特殊符号的最佳解决方案,同时提供配置指南。 理解MySQL字符…

    2025年12月12日
    000
  • 解决Laravel项目在GitHub上仅显示README文件的指南

    本教程旨在解决Laravel项目推送到GitHub后,仅显示`README.md`文件而项目目录缺失的问题。文章将详细解释导致此问题的常见原因,并提供一套标准的Git命令流程,包括`git add .`、`git commit`和`git push -f`,以确保所有必要的项目文件都能正确上传并显示…

    2025年12月12日
    000
  • 使用PHP自动化SFTP文件下载:基于SSH密钥认证的实践指南

    本文详细介绍了如何在php环境中,利用ssh密钥认证自动化sftp文件下载。针对传统`ssh2`扩展连接失败和`passthru`多命令执行的挑战,文章提供了一个简洁高效的单行sftp命令解决方案,通过直接指定源文件路径实现文件传输,并探讨了其工作原理、注意事项及适用场景,帮助开发者快速实现sftp…

    2025年12月12日
    000
  • PHP中精确查找逗号分隔字符串中的数字:避免子字符串匹配陷阱

    本文详细阐述了在php中如何准确判断逗号分隔字符串中是否包含特定数字,而非其子串。针对`strpos`可能导致的误判,教程介绍了使用`explode`函数将字符串拆分为数组,再结合`in_array`函数进行精确匹配的解决方案,确保在处理此类数据时获得准确可靠的查找结果。 在PHP开发中,我们经常需…

    2025年12月12日
    000
  • Laravel调度器:实现季度任务的提前执行策略

    本文深入探讨了如何在Laravel命令调度器中实现季度任务的提前执行。虽然Laravel的`quarterly()`方法默认在季度首日运行,但通过灵活运用`cron()`方法,可以精确或近似地将任务调度到季度开始前的一周,以满足特定业务需求,并提供了应对月份天数差异的策略。 在Laravel应用开发…

    2025年12月12日
    000
  • 使用PHP和MySQL高效查询最频繁数据项的教程

    本教程详细介绍了如何利用php和mysql高效地查询并展示数据库中某一列出现频率最高的数据项。文章重点讲解了sql的`count()`和`group by`聚合函数,以及在php中执行查询、处理结果和进行关键错误处理的最佳实践,确保数据检索的准确性和代码的健壮性。 在数据驱动的应用程序中,经常需要识…

    2025年12月12日
    000
  • 在 cPanel 应用管理器中通过 PHP 访问环境变量的实用指南

    本文详细介绍了如何在 cpanel 应用管理器中设置的环境变量通过 php 程序进行访问。教程涵盖了 `$_env`、`getenv()` 和 `$_server` 三种主要的 php 方法,并提供了相应的代码示例和使用注意事项,旨在帮助开发者在共享主机环境中安全、高效地管理应用配置,确保变量的正确…

    2025年12月12日
    000
  • PHP模板怎么组件复用_PHP模板组件复用方法及模块化开发。

    通过组件复用可提升PHP模板开发效率与可维护性,常用方式包括:一、使用include/require引入公共文件实现结构分离;二、封装函数动态渲染组件并集中管理;三、采用Twig等模板引擎利用继承与宏定义高级特性;四、以类和对象组织模块,实现高内聚低耦合设计。 在进行PHP模板开发时,若需要在多个页…

    2025年12月12日
    000
  • PHP怎么处理POST表单提交_PHP使用$_POST接收表单数据指南

    首先确认表单method=”post”且action指向正确PHP文件,输入字段需有name属性;在PHP中用$_POST[‘字段名’]获取数据;通过isset()和empty()检查数据是否存在并过滤;使用htmlspecialchars()防XSS…

    2025年12月12日
    000
  • Laravel Blade 教程:在 href 属性中正确传递动态数据

    本教程将详细介绍如何在 laravel blade 模板中,利用 blade 语法和命名路由,安全且优雅地将数据库中的动态数据(如记录编号)传递到 html “ 标签的 `href` 属性中,从而生成动态链接。我们将探讨常见的错误用法、正确的插值方法,以及推荐的命名路由实践,以提高代码的…

    2025年12月12日
    000
  • WordPress短代码内容CSS样式化指南:以当前用户显示名称为例

    本教程详细介绍了如何在wordpress中为自定义短代码(例如显示当前用户名称的短代码)添加css样式。通过修改短代码使其输出带有特定css类的html结构,并结合css规则,实现对短代码内容的精确视觉控制,同时确保仅在用户登录时显示内容,提升用户体验和网站美观度。 在WordPress开发中,短代…

    2025年12月12日
    000
  • PHP图片怎么缩放_PHP图片缩放方法及响应式图片处理。

    首先使用GD库或ImageMagick在PHP中对图片进行等比缩放,再通过HTML的srcset、picture标签及CSS的max-width、object-fit等属性实现响应式显示。 如果您在处理网页中的图片时发现图片尺寸过大或显示不适应屏幕,可能需要对PHP生成或调用的图片进行缩放,并确保前…

    2025年12月12日
    000
  • php怎么用if_PHP条件语句if/else使用与逻辑判断方法

    使用if、else和elseif语句可实现PHP中的条件分支控制。首先通过if判断条件是否成立,成立则执行对应代码块;若不成立且存在else,则执行else中代码。当需处理多个互斥条件时,可使用多个elseif进行依次判断,最后用else处理默认情况。此外,可在if语句内部嵌套新的if结构以应对复合…

    2025年12月12日
    000
  • 解决Apple M1 Mac上Xdebug 3无法在phpinfo中显示的问题

    本教程详细阐述了在Apple M1芯片Mac上安装Xdebug 3后,其无法在`phpinfo()`中显示的问题。核心在于M1架构与默认PECL安装命令的兼容性。文章提供了针对ARM64架构的正确安装方法,以及必要的PHP配置和验证步骤,确保Xdebug能在Web环境中正常工作,从而实现PHP代码的…

    2025年12月12日
    000
  • WordPress短代码与CSS实现当前用户显示名称样式化

    本教程详细介绍了如何在wordpress中创建并优化一个短代码,用于显示当前登录用户的名称。通过将用户名称封装在带有特定css类的html “标签中,并结合`is_user_logged_in()`函数确保仅对登录用户显示,从而实现对用户名称文本的灵活css样式控制。 在WordPres…

    2025年12月12日
    000

发表回复

登录后才能评论
关注微信