Есть ли способ получить доступ к строке в качестве дескриптора файла в php?
Я нахожусь на сервере, где я ограничен PHP 5.2.6, что означает, что str_getcsv недоступен для меня. Вместо этого я использую fgetcsv, для работы с которым требуется "Действительный указатель на файл, успешно открытый fopen(), popen() или fsockopen()".
Мой вопрос заключается в следующем: есть ли способ получить доступ к строке в качестве дескриптора файла?
Мой другой вариант - записать строку в текстовый файл, а затем получить к ней доступ через fopen()
, а затем использовать fgetcsv
, но я надеюсь, что есть способ сделать это напрямую, как в perl.
5 answers
Если вы посмотрите в примечаниях пользователя на странице руководства для str_getcsv
, вы найдете эту заметку от Даниэля, в которой предлагается эта функция (цитата) :
<?php
if (!function_exists('str_getcsv')) {
function str_getcsv($input, $delimiter = ",", $enclosure = '"', $escape = "\\") {
$fiveMBs = 5 * 1024 * 1024;
$fp = fopen("php://temp/maxmemory:$fiveMBs", 'r+');
fputs($fp, $input);
rewind($fp);
$data = fgetcsv($fp, 1000, $delimiter, $enclosure); // $escape only got added in 5.3.0
fclose($fp);
return $data;
}
}
?>
Похоже, он делает именно то, о чем вы просили: он использует поток, который указывает на временный файловый дескриптор в памяти, для использования fgetcsv
на нем.
Смотрите Потоки ввода/вывода PHP для документации, в частности, о потоке php://temp
обертка.
Конечно, вы должны проверить, хорошо ли это работает для вас, но, по крайней мере, это должно дать вам представление о том, как этого добиться ;-)
Я в ужасе, что никто не ответил на это решение:
<?php
$string = "I tried, honestly!";
$fp = fopen('data://text/plain,' . $string,'r');
echo stream_get_contents($fp);
#fputcsv($fp, .......);
?>
И жаждущий памяти идеальное решение:
<?php
class StringStream
{
private $Variable = NULL;
protected $fp = 0;
final public function __construct(&$String, $Mode = 'r')
{
$this->$Variable = &$String;
switch($Mode)
{
case 'r':
case 'r+':
$this->fp = fopen('php://memory','r+');
fwrite($this->fp, @strval($String));
rewind($this->fp);
break;
case 'a':
case 'a+':
$this->fp = fopen('php://memory','r+');
fwrite($this->fp, @strval($String));
break;
default:
$this->fp = fopen('php://memory',$Mode);
}
}
final public function flush()
{
# Update variable
$this->Variable = stream_get_contents($this->fp);
}
final public function __destruct()
{
# Update variable on destruction;
$this->Variable = stream_get_contents($this->fp);
}
public function __get($name)
{
switch($name)
{
case 'fp': return $fp;
default: trigger error('Undefined property: ('.$name.').');
}
return NULL;
}
}
$string = 'Some bad-ass string';
$stream = new StringStream($string);
echo stream_get_contents($stream->fp);
#fputcsv($stream->fp, .......);
?>
Чтобы ответить на ваш общий вопрос, да, вы можете рассматривать переменную как поток файлов.
Http://www.php.net/manual/en/function.stream-context-create.php
Ниже приводится копия и вставка из нескольких разных комментариев к руководству по PHP (поэтому я не могу ручаться за то, насколько оно готово к производству):
<?php
class VariableStream {
private $position;
private $varname;
public function stream_open($path, $mode, $options, &$opened_path) {
$url = parse_url($path);
$this->varname = $url["host"];
$this->position = 0;
return true;
}
public function stream_read($count) {
$p=&$this->position;
$ret = substr($GLOBALS[$this->varname], $p, $count);
$p += strlen($ret);
return $ret;
}
public function stream_write($data){
$v=&$GLOBALS[$this->varname];
$l=strlen($data);
$p=&$this->position;
$v = substr($v, 0, $p) . $data . substr($v, $p += $l);
return $l;
}
public function stream_tell() {
return $this->position;
}
public function stream_eof() {
return $this->position >= strlen($GLOBALS[$this->varname]);
}
public function stream_seek($offset, $whence) {
$l=strlen(&$GLOBALS[$this->varname]);
$p=&$this->position;
switch ($whence) {
case SEEK_SET: $newPos = $offset; break;
case SEEK_CUR: $newPos = $p + $offset; break;
case SEEK_END: $newPos = $l + $offset; break;
default: return false;
}
$ret = ($newPos >=0 && $newPos <=$l);
if ($ret) $p=$newPos;
return $ret;
}
}
stream_wrapper_register("var", "VariableStream");
$csv = "foo,bar\ntest,1,2,3\n";
$row = 1;
if (($handle = fopen("var://csv", "r")) !== FALSE) {
while (($data = fgetcsv($handle, 1000, ",")) !== FALSE) {
$num = count($data);
echo "<p> $num fields in line $row: <br /></p>\n";
$row++;
for ($c=0; $c < $num; $c++) {
echo $data[$c] . "<br />\n";
}
}
fclose($handle);
}
?>
Конечно, для вашего конкретного примера можно использовать более простые потоковые методы.
Вы можете использовать дескрипторы потоков, такие как php://память, чтобы достичь того, что вам нужно. Просто откройте, перепишите, перемотайте назад, и вы сможете использовать fgetcsv.
К сожалению, это невозможно. Вы не можете обрабатывать строку так, как если бы это был поток из файла. Вам действительно придется сначала записать строку в файл, а затем открыть указанный файл с помощью fopen
.
А теперь, что касается очевидной части, вы рассматривали возможность обновления?