Mysql
 sql >> Baza danych >  >> RDS >> Mysql

Błąd wstawiania PHP/SQL podczas używania nazwanych symboli zastępczych

Twoje $userData musi mieć dokładnie te same symbole zastępcze związane z Twoim oświadczeniem, nie więcej i nie mniej. Zobacz PDOStatement::execute dokumentacja , część, która mówi „Nie możesz powiązać więcej wartości niż określono”.

Musisz przygotować swój argument do execute() dokładnie pasować do twoich wiązań. Jest to łatwe dzięki array_intersect_key() jeśli poprawnie ułożysz swoje tablice. Zwykle umieszczam to w funkcji, która również zajmie się prefiksami, jak poniżej:

// Adds a prefix to a name for a named bind placeholder
function prefix($name) {
    return ':'.$name;
}

// like 'prefix()', but for array keys
function prefix_keys($assoc) {
    // prefix STRING keys
    // Numeric keys not included
    $newassoc = array();
    foreach ($assoc as $k=>$v) {
        if (is_string($k)) {
            $newassoc[prefix($k)] = $v;
        }
    }
    return $newassoc;
}

// given a map of datakeyname=>columnname, and a table name, returns an
// sql insert string with named bind placeholder parameters.
function makeInsertStmt($tablename, $namemap) {
    $binds = array_map('prefix', array_keys($namemap));
    return 'INSERT INTO '.$tablename.' ('.implode(',',$namemap).') VALUES ('
    .implode(',',$binds).')';
}

// returns an array formatted for an `execute()`
function makeBindData($data, $namemap) {
    // $data assoc array, $namemap name->column mapping
    return prefix_keys(array_intersect_key($data, $namemap));
}

// example to demonstrate how these pieces fit together
function RunTestInsert(PDO $pdo, $userData) {
    $tablename = 'UserDetails';
    // map "key in $userData" => "column name"
    // do not include ':' prefix in $userData
    $namemap = array(
      'firstName'       => "FirstName",
      'lastName'        => "LastName",
      'address'         => "Address",
      'city'            => "City",
      'county'          => "County",
      'postCode'        => "PostCode",
      'phone'           => "Phone",
      'mobile'          => "Mobile",
      'sex'             => "Sex",
      'DOB'             => "DOB",
      'fundraisingAim'  => "FundraisingAim",
      'weeksAim'        => "WeeksAim",
      'lengthsAim'      => "LengthsAim",
      'hearAbout'       => "HearAboutID",
      'motivation'      => "MotivationID",
      'welcomePackPref' => "WelcomePackID",
      'contactPref'     => "ContactPrefID",
      'title'           => "TitleID",
    );
    $sql = makeInsertStmt($tablename, $namemap);
    $binddata = makeBindData($userData, $namemap);

    $pstmt = $pdo->prepare($sql);
    $pstmt->execute($binddata);
}

Zaletą takiej abstrakcji jest to, że nie musisz się martwić o same parametry wiązania.



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Synchronizacja baz danych online/offline — MySQL/PHP

  2. Projektowanie bazy danych:do EAV czy nie do EAV?

  3. Wydajny zewnętrzny harmonogram dzięki MySQL i ejabberd

  4. Wstaw zawartość pliku do kolumny tabeli MySQL

  5. Ponowna próba zakleszczenia ActiveRecord3