fputcsv和换行符代码

Emm*_*myS 28 php newline fputcsv

我在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");

}
Run Code Online (Sandbox Code Playgroud)

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

Joh*_*hat 38

// 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);
  }
}
Run Code Online (Sandbox Code Playgroud)


luc*_*rio 24

这是@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;
}
Run Code Online (Sandbox Code Playgroud)


Bob*_*kes 6

使用php函数fputcsv只写\n而无法自定义.这使得该功能对于微软环境毫无价值,尽管有些软件包也会检测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);
Run Code Online (Sandbox Code Playgroud)


Emm*_*myS -3

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

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

用于代替标准 fputcsv。

  • 我知道这是一个旧答案,但如果我没有提到这是专家交流中的一个糟糕答案,那我就是失职了。它甚至不处理双引号转义。请改用流过滤器。这就是他们在那里的原因。http://www.php.net/manual/en/function.stream-filter-register.php (3认同)