我在PHP中使用fputcsv输出数据库查询的逗号分隔文件。在Ubuntu中打开gedit中的文件时,它看起来是正确的 - 每个记录都有一个换行符(没有可见的换行符,但是你可以告诉每个记录是分开的,在OpenOffice电子表格中打开它可以让我正确地查看文件。)
但是,我们将这些文件发送到Windows上的客户端,并且在他们的系统上,该文件作为一个很长的行。在Excel中打开它,它根本无法识别多行。
我在这里阅读了几个非常相似的问题,包括this one,其中包含指向真实信息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文本编辑器中打开文件时,它仍会显示为一条长行。
答案 0 :(得分:34)
// 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);
}
}
答案 1 :(得分:21)
这是@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;
}
答案 2 :(得分:6)
使用php函数fputcsv只写\ n并且无法自定义。这使得该函数对于microsoft环境毫无价值,尽管有些软件包也会检测linux换行符。
fputcsv的好处仍然让我在发送到文件之前深入研究替换换行符的解决方案。这可以通过首先将fputcsv流式传输到php temp流中的构建来完成。然后将换行符调整为您想要的任何值,然后保存到文件中。像这样:
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);
答案 3 :(得分:2)
或者,您可以以本机unix格式输出(仅限\ n),然后在生成的文件上运行unix2dos,以在适当的位置转换为\ r \ n。请注意,您的数据不包含\ n。另外,我看到你正在使用〜的默认分隔符。尝试\ t的默认分隔符。
答案 4 :(得分:2)
正如webbiedave指出的那样(thx!)可能最干净的方法是使用流过滤器。
它比其他解决方案稍微复杂一点,但是甚至适用于写入后不可编辑的流(如使用$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);
...
答案 5 :(得分:1)
我一直在处理类似情况。这是我发现的一个解决方案,它输出带有Windows友好行结尾的CSV文件。
http://www.php.net/manual/en/function.fputcsv.php#90883
我无法使用,因为我正在尝试将文件流式传输到客户端而无法使用fseek。
答案 6 :(得分:-1)
windows需要\r\n
作为换行符/回车返回组合以显示单独的行。
答案 7 :(得分:-3)
我最终在专家交流中得到了答案;这是有效的:
function my_fputcsv($handle, $fieldsarray, $delimiter = "~", $enclosure ='"'){
$glue = $enclosure . $delimiter . $enclosure;
return fwrite($handle, $enclosure . implode($glue,$fieldsarray) . $enclosure.PHP_EOL);
}
代替标准fputcsv。