0

(最近のPHPでは)byref引数を使用して関数を呼び出したい場合はどうしますか?

function blah_byval ($arg) { /* some code */ }
function blah_byref (&$arg) { /* same code */ }

私には遅れているようです。私は何かが欠けているに違いありません。

質問は基本的に架空のものですが、私にはユースケースがあります。進行中のキャプチャの実装は次のとおりです。

function captcha_image($string = "", $create = true, $session = "") {

  # outputs a transparent gif image 16 pixels high by 60 pixels wide containing
  # a string of six lowercase letters, either specified by $string or generated
  # randomly. returns this string, or false in the event of an error.

  # once a captcha image is created, the string it contains should be saved so
  # that it can be compared later with user input. to skip saving, set $create
  # to false.

  # if called from a web server, content-type is set.

  if (!preg_match("/^[a-z]{6}$/", $string)) $string = string_random(6);
  if (php_sapi_name() != "cli") header("Content-Type: image/gif");
  if ($ihandle = @imagecreatetruecolor(60, 16)) {
    $background = imagecolorallocate($ihandle, 255, 255, 255);
    $foreground = imagecolorallocate($ihandle, 128, 128, 128);
    imagecolortransparent($ihandle, $background);
    imagefilledrectangle($ihandle, 0, 0, 60, 16, $background);
    imagestring($ihandle, 5, 0, 0, $string, $foreground);
    imagegif($ihandle, NULL);
    imagedestroy($ihandle);
    if ($create) return captcha_create($string, $session);
    return $string;
  }
  return false;
}

function captcha_compare($string, $session = "", $destroy = true) {

  # returns true if $string is associated with a certain session, which is
  # specified by $session or, if no such session exists, by session_identity().
  # if $string is not associated with the session, returns false.

  # once a string is matched to a session, it should be destroyed to prevent
  # re-use. to allow re-use, $destroy may be set to false.

  global $cfgimage;
  if (!session_exists($session)) $session = session_identity();
  $get_return = false;
  if ($chandle = @fopen($cfgimage['captcha']['file'], "rb")) {
    flock($chandle, LOCK_SH);
    while (!feof($chandle)) {
      $line = explode(" ", trim(fgets($chandle)), 3);
      if (($line[2] === $string) && ($line[1] == $session) && ($line[0] > (time() - $cfgimage['captcha']['keepalive']))) {
        $get_return = true;
        break;
      }
    }
    fclose($chandle);
  }
  if (($destroy) && ($get_return)) captcha_destroy($session);
  return $get_return;
}

function captcha_create($string, $session = "") {

  # associates a string of six lowercase letters with a session, either
  # specified by $session or by session_identity(). fails if $string is not
  # well-formed. returns true if successful, or false.

  global $cfgimage;
  if (!preg_match("/^[a-z]{6}$/", $string)) return false;
  if (!session_exists($session)) $session = session_identity();
  if ($thandle = @tmpfile()) {
    if ($chandle = @fopen($cfgimage['captcha']['file'], "rb")) {
      flock($chandle, LOCK_SH);
      while (!feof($chandle)) {
        $line = explode(" ", trim(fgets($chandle)), 3);
        if (($line[1] != $session) && ($line[0] > (time() - $cfgimage['captcha']['keepalive']))) fputs($thandle, implode(" ", $line) . "\n");
      }
      fclose($chandle);
      fseek($thandle, 0);
      if ($chandle = @fopen($cfgimage['captcha']['file'], "cb")) {
        if (flock($chandle, LOCK_EX)) {
          fputs($chandle, time() . " " . $session . " " . $string . "\n");
          while (!feof($thandle)) fputs($chandle, fgets($thandle));
        }
        fclose($chandle);
      }
    }
    flock($thandle, LOCK_UN);
    fclose($thandle);
    return true;
  }
  return false;
}

function captcha_destroy($session = "") {

  # dis-associates any existing captcha string with a session, either specified
  # by $session or by session_identity(). does not return any value.

  global $cfgimage;
  if (!session_exists($session)) $session = session_identity();
  if ($thandle = @tmpfile()) {
    if ($chandle = @fopen($cfgimage['captcha']['file'], "rb")) {
      flock($chandle, LOCK_SH);
      while (!feof($chandle)) {
        $line = explode(" ", trim(fgets($chandle)), 3);
        if (($line[1] != $session) && ($line[0] > (time() - $cfgimage['captcha']['keepalive']))) fputs($thandle, implode(" ", $line) . "\n");
      }
      fclose($chandle);
      fseek($thandle, 0);
      if ($chandle = @fopen($cfgimage['captcha']['file'], "cb")) {
        if (flock($chandle, LOCK_EX)) {
          while (!feof($thandle)) fputs($chandle, fgets($thandle));
        }
        fclose($chandle);
      }
    }
    flock($thandle, LOCK_UN);
    fclose($thandle);
  }
}

captcha_image()では、return captcha_create($string, $session)以前はcaptcha_create(&$string ...)。でした。

4

1 に答える 1

3

これは、ユーザーランド コードでは不可能です。一方の内部関数についてはPREFER_REF、arginfo 構造体でフラグを指定できます。これは、可能な場合は値を ref で渡し、そうでない場合は by-val で渡します。

私の目には、この機能がユーザーランド コードに公開されていないのは良いことです。いずれにせよ、参照はトリッキーなビジネスであり、それらをオプションにすると、正確に物を渡す方法に応じて異なる動作が発生するだけで、さらに悪化します。

私の提案: 参照を完全に使用しないでください。あなたはそれらに問題があるだけです。

于 2012-07-11T21:46:50.073 に答える