当你想在某些时候只用byref参数调用一个函数时,你做了什么(在最近的PHP中)?
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 ...)
。
答案 0 :(得分:3)
这在用户区代码中是不可能的。对于手上的内部函数,您可以在arginfo结构中指定PREFER_REF
标志。如果可能,这将传递值-ref,否则传递-val。
在我看来,这个功能不会暴露在用户态代码中。在任何情况下,引用都是一项棘手的业务,并且根据您传递内容的确切方式使其变得更糟,可以选择不同的行为。
我的建议:不要完全使用引用。你只会遇到问题。