将PHP PDO迁移到Laravel PDO会导致参数号无效

问题描述 投票:1回答:1

我正在将PHP PDO实例转换为Laravel 5.5。我改变了/到的数据库连接:

//$dbc = new PDO ('mysql:host='. $DB_HOST .';dbname='. $DB_NAME, $DB_USER, $DB_PASS);
$dbc = \Illuminate\Support\Facades\DB::connection('foobardb')->getPdo();

出于某种原因,这导致我的SELECT查询失败。

$query = "SELECT *
          FROM foobartable
          WHERE date_created > :sunday
          AND date_created <= :monday
          AND date_updated > :sunday
          AND date_updated <= :monday";

$stmt = $dbc->prepare($query);
$stmt->bindValue(':monday', $monday, PDO::PARAM_STR);
$stmt->bindValue(':sunday', $sunday, PDO::PARAM_STR);
$stmt->execute();

$results = $stmt->fetchAll(PDO::FETCH_ASSOC);

根据堆栈跟踪,错误是在$stmt->execute();上专门触发的

SQLSTATE [HY093]:参数号无效

我显然有correct parameters,这可以很好地使用PHP的PDO实例,所以为什么它不适用于Laravel?

php mysql laravel pdo laravel-5.5
1个回答
2
投票

由于某种原因,Laravel连接不会自动使用PDO::ATTR_EMULATE_PREPARES。即使是mysql连接。只需在连接数组中添加选项即可解决此问题。

'foobardb' => [
    'driver'    => 'mysql',
    'host'      => env('DB_HOST', 'localhost'),
    'port'      => env('DB_PORT', 3306),
    'database'  => 'foobardb',
    'username'  => env('DB_USERNAME', 'forge'),
    'password'  => env('DB_PASSWORD', ''),
    'charset'   => 'utf8mb4',
    'collation' => 'utf8mb4_unicode_ci',
    'prefix'    => '',
    'strict'    => false,
    'engine'    => null,
    //add addition options here
    'options'   => [PDO::ATTR_EMULATE_PREPARES => true]
],

我不知道这是否仅限于Laravel 5.5,或者是否需要在早期版本中添加该选项。

解决此问题的另一种方法是执行以下操作:

$query = "SELECT *
          FROM foobartable
          WHERE date_created > :sunday
          AND date_created <= :monday
          AND date_updated > :sunday2
          AND date_updated <= :monday2";

然后:

$stmt->bindValue(':monday', $monday, PDO::PARAM_STR);
$stmt->bindValue(':sunday', $sunday, PDO::PARAM_STR);
$stmt->bindValue(':monday2', $monday, PDO::PARAM_STR);
$stmt->bindValue(':sunday2', $sunday, PDO::PARAM_STR);

因此它认为你拥有与拥有bindValues相同数量的PDO令牌。

© www.soinside.com 2019 - 2024. All rights reserved.