5

コードのこの部分があり、タイトルにエラーが表示されます。

$k各テーブルに 1 から 5 までのカウントがあります。

$myarray少なくとも 3 つから 4 つの名前で構成されます。

エラーは次の行で発生し$qu .=ます...

これまでに試したこと:変数名$iを行$v{$i}に変更します$qu .=

それで、クエリを繰り返す可能性はありますか? 配列の数と同じ数の s が句に含まれるようにするにはANDWHERE

while ($k<=5) {
  $queryname = "SELECT Name FROM description";
  $qname = mysql_query($queryname,$link);

  while ($reihe = mysql_fetch_object($qname)) {
    {
      $unse = unserialize($reihe->Name);
      {
        foreach ($unse as $j=>$h)
          foreach ($h as $m) {
            $myarray = preg_split('/ |\s| /',$m);
            {
              echo "<br>";
              $array_empty = $myarray;
              $empty_elements = array("");
              $myarray = array_diff($array_empty,$empty_elements);
              var_dump($myarray);
              for ($i=1; $i<=count($myarray); $i++) {
                $v{$i} = $myarray[$i];
                echo $v{$i};
                $esc{$i} = strtolower(mysql_escape_string($v{$i}));
                echo "<br>" . $esc{$i} . "<br>";
                $qu = "SELECT * FROM `table ID=$k` WHERE";
                $qu{$i} .= "AND `table ID=$k`.`name` LIKE '%$esc{$i}%'";
              }
            }
          }
        {
          $test_a = mysql_query($qu,$link) or die (mysql_error());
          echo "<br>";
          var_dump($test_a);
          for ($x=0; $x<mysql_num_rows($test_a); $x++) {
            $row = mysql_result($test_a,$x,'data1');
            $namee = mysql_result($test_a,$x,'data2');
            echo 'data1' . $row . '<br>';
            echo 'data2' . $namee . '<br>';
          }
        }
      }
    }
  }
  $k++;
}
4

1 に答える 1

6

あなたはいくつかの基本的なPHP構文を誤解しているようです。

マニュアルに記載されているように:

文字列s内の文字は、のように角かっこ使用して、文字列の後に目的の文字のゼロベースのオフセットを指定することでアクセスおよび変更できます。この目的のために、文字列を文字の配列と考えてください。関数とは、複数の文字を抽出または置換する場合に使用できます。$str[42]substr()substr_replace()

注: 同じ目的で、のように中括弧を使用して文字列にアクセスすることもできます。$str{42}

したがって、コード全体で中括弧を使用すると(for制御構造のスコープを定義するものを除く)、完全に誤りであり、意図したとおりに実行されません。

次に、コードから、シリアル化されたPHPオブジェクトにリレーショナルデータを格納しているように見えます。これは、MySQLのようなRDBMSを使用することの多くの利点を打ち負かします。やむを得ない理由がない限り、PHPオブジェクトは正規化された形式で保存する必要があります。たとえば、プロパティが名前の暗黙の文字列を保持する配列であるシリアル化されたPHPオブジェクトを含むフィールドをdescription持つ各レコードではなく、テーブル内の関連レコードを参照する新しいテーブルにそのような名前を格納するだけです。NamedescriptionNamesdescription

CREATE TABLE descriptionNames (
  descriptionID INT NOT NULL,
  name VARCHAR(50),
  PRIMARY KEY (descriptionId, name),
  FOREIGN KEY (descriptionId) REFERENCES description (descriptionId)
);

また、、などの名前の5つの(概略的に)同一のテーブルがあるようTable ID=1ですTable ID=2。その場合は、おそらく5つのテーブルを1つに結合し、必要に応じて、レコードがどのテーブルから発生したかを示す列を作成する必要があります。また、テーブル名を変更して、空白などの特殊文字を使用しないようにすることをお勧めします。テーブル名=を適切に引用し忘れた場合にのみ、問題や混乱が生じる可能性があります。例えば:

ALTER TABLE `Table ID=1`
  RENAME TO CombiTable,
  ADD COLUMN FromTableID TINYINT NOT NULL;

UPDATE CombiTable SET FromTableID = 1;

INSERT INTO CombiTable
  SELECT *, 2 FROM `Table ID=2` UNION ALL
  SELECT *, 3 FROM `Table ID=3` UNION ALL
  SELECT *, 4 FROM `Table ID=4` UNION ALL
  SELECT *, 5 FROM `Table ID=5`;

SELECT * FROM CombiTable;  -- check everything is okay

DROP TABLE `Table ID=2`, `Table ID=3`, `Table ID=4`, `Table ID=5`;

いずれにせよ、あなたは古代のmysql_*機能を使うべきではありません。それらはもはや維持されておらず、コミュニティは非推奨プロセスを開始しています。代わりに、プリペアドステートメントについて学び、 PDO抽象化レイヤーまたは改良されたMySQLi拡張機能のいずれかを使用する必要があります。

既存のデータ構造を維持する必要がある場合は、PDOを使用して次のようにすることができます。

$dbh = new PDO("mysql:dbname=$dbname;charset=utf8", $user, $password);

$qry = $dbh->query("SELECT Name FROM description");
$myarray = array();
while ($reihe = $dbh->fetchColumn())
  foreach (unserialize($reihe) as $h)
    foreach ($h as $m)
      array_merge($myarray, preg_split("/\s+/", $m, -1, PREG_SPLIT_NO_EMPTY));

for ($k = 1; $k <= 5; $k++) {
  $qry = $dbh->prepare("SELECT * FROM `table ID=$k` WHERE " . implode(" OR ",
    array_pad(array(), count($myarray), "name LIKE CONCAT('%', ?, '%')")
  ));

  $qry->execute($myarray);
  while($row = $qry->fetch()) {
    echo "data1:$row[data1]<br/>";
    echo "data2:$row[data2]<br/>";
  }
}

ただし、提案した新しいデータ構造を使用すると、次のことを行うだけで済みます。

$dbh = new PDO("mysql:dbname=$dbname;charset=utf8", $user, $password);

$qry = $dbh->query("
  SELECT   *
  FROM     CombiTable JOIN descriptionNames USING (name)
  WHERE    FromTableID BETWEEN 1 AND 5   -- in case you have others?
  ORDER BY FromTableID
");

while ($row = $qry->fetch()) {
  echo "data1:$row[data1]<br/>";
  echo "data2:$row[data2]<br/>";
}
于 2012-05-24T18:11:35.230 に答える