32

我在 PHP 中使用 fputcsv 来输出数据库查询的逗号分隔文件。在 Ubuntu 中在 gedit 中打开文件时,它看起来是正确的 - 每条记录都有一个换行符(没有可见的换行符,但您可以知道每条记录是分开的,在 OpenOffice 电子表格中打开它可以让我正确查看文件。)

但是,我们将这些文件发送到 Windows 上的客户端,在他们的系统上,文件以一大而长的行出现。在 Excel 中打开它,它根本不识别多行。

我在这里阅读了几个非常相似的问题,包括这个,其中包括一个链接到真正信息丰富的Great Newline Schism解释。

不幸的是,我们不能只告诉我们的客户在“更智能”的编辑器中打开文件。他们需要能够在 Excel 中打开它们。是否有任何编程方式来确保添加正确的换行符,以便可以在任何操作系统上的电子表格程序中打开文件?

我已经在使用自定义函数来强制在所有值周围加上引号,因为 fputcsv 对此是有选择性的。我试过做这样的事情:

function my_fputcsv($handle, $fieldsarray, $delimiter = "~", $enclosure ='"'){

        $glue = $enclosure . $delimiter . $enclosure;

    return fwrite($handle, $enclosure . implode($glue,$fieldsarray) . $enclosure."\r\n");

}

但是当文件在 Windows 文本编辑器中打开时,它仍然显示为一个长行。

4

8 回答 8

42
// Writes an array to an open CSV file with a custom end of line.
//
// $fp: a seekable file pointer. Most file pointers are seekable, 
//   but some are not. example: fopen('php://output', 'w') is not seekable.
// $eol: probably one of "\r\n", "\n", or for super old macs: "\r"
function fputcsv_eol($fp, $array, $eol) {
  fputcsv($fp, $array);
  if("\n" != $eol && 0 === fseek($fp, -1, SEEK_CUR)) {
    fwrite($fp, $eol);
  }
}
于 2011-11-01T00:10:16.193 回答
26

这是@John Douthat 的好答案的改进版本,保留了使用自定义分隔符和附件并返回 fputcsv 原始输出的可能性:

function fputcsv_eol($handle, $array, $delimiter = ',', $enclosure = '"', $eol = "\n") {
    $return = fputcsv($handle, $array, $delimiter, $enclosure);
    if($return !== FALSE && "\n" != $eol && 0 === fseek($handle, -1, SEEK_CUR)) {
        fwrite($handle, $eol);
    }
    return $return;
}
于 2014-01-23T01:03:25.450 回答
6

使用 php 函数 fputcsv 只写入 \n 并且不能自定义。这使得该功能对于 microsoft 环境毫无价值,尽管一些软件包也会检测到 linux 换行符。

fputcsv 的好处仍然让我深入研究一种解决方案,以便在发送到文件之前替换换行符。这可以通过首先将 fputcsv 流式传输到 php 临时流中的构建来完成。然后将换行符调整为您想要的任何内容,然后保存到文件中。像这样:

function getcsvline($list,  $seperator, $enclosure, $newline = "" ){
    $fp = fopen('php://temp', 'r+'); 

    fputcsv($fp, $list, $seperator, $enclosure );
    rewind($fp);

    $line = fgets($fp);
    if( $newline and $newline != "\n" ) {
      if( $line[strlen($line)-2] != "\r" and $line[strlen($line)-1] == "\n") {
        $line = substr_replace($line,"",-1) . $newline;
      } else {
        // return the line as is (literal string)
        //die( 'original csv line is already \r\n style' );
      }
    }

        return $line;
}

/* to call the function with the array $row and save to file with filehandle $fp */
$line = getcsvline( $row, ",", "\"", "\r\n" );
fwrite( $fp, $line);
于 2011-03-11T09:19:05.940 回答
4

正如webbiedave指出的(谢谢!)可能最干净的方法是使用流过滤器。

它比其他解决方案复杂一点,但甚至适用于写入后不可编辑的流(例如使用 下载$handle = fopen('php://output', 'w');

这是我的方法:

class StreamFilterNewlines extends php_user_filter {
    function filter($in, $out, &$consumed, $closing) {

        while ( $bucket = stream_bucket_make_writeable($in) ) {
            $bucket->data = preg_replace('/([^\r])\n/', "$1\r\n", $bucket->data);
            $consumed += $bucket->datalen;
            stream_bucket_append($out, $bucket);
        }
        return PSFS_PASS_ON;
    }
}

stream_filter_register("newlines", "StreamFilterNewlines");
stream_filter_append($handle, "newlines");

fputcsv($handle, $list, $seperator, $enclosure);
...
于 2015-01-16T13:57:25.070 回答
2

或者,您可以以原生 unix 格式输出(仅限 \n),然后在生成的文件上运行 unix2dos 以在适当的位置转换为 \r\n。请注意您的数据不包含 \n's 。另外,我看到您使用的默认分隔符为 ~ 。尝试使用默认分隔符 \t 。

于 2010-11-02T17:41:05.297 回答
1

我一直在处理类似的情况。这是我发现的一个解决方案,它输出带有 Windows 友好行尾的 CSV 文件。

http://www.php.net/manual/en/function.fputcsv.php#90883

我无法使用,因为我正在尝试将文件流式传输到客户端并且无法使用 fseeks。

于 2010-11-10T00:43:57.403 回答
-1

windows 需要\r\n作为换行符/回车符组合以显示单独的行。

于 2010-11-02T17:37:26.703 回答
-3

我最终确实在专家交流会上得到了答案;这是有效的:

function my_fputcsv($handle, $fieldsarray, $delimiter = "~", $enclosure ='"'){
   $glue = $enclosure . $delimiter . $enclosure;
   return fwrite($handle, $enclosure . implode($glue,$fieldsarray) . $enclosure.PHP_EOL);
}

用于代替标准 fputcsv。

于 2010-11-16T18:27:52.003 回答