IFS separate a string like “Hello”,“World”,“this”,“is, a boring”, “line”

前端 未结 3 1569
夕颜
夕颜 2021-01-29 07:10

I\'m trying to parse a .csv file and I have some problems with IFS. The file contains lines like this:

\"Hello\",\"World\",\"this\",\"is, a boring\",\"line\"
         


        
3条回答
  •  慢半拍i
    慢半拍i (楼主)
    2021-01-29 07:20

    bashlib provides a csvline function. Assuming you've installed it somewhere in your PATH:

    line='"Hello","World","this","is, a boring","line"'
    
    source bashlib
    csvline <<<"$line"
    printf '%s\n' "${CSVLINE[@]}"
    

    ...output from the above being:

    Hello
    World
    this
    is, a boring
    line
    

    To quote the implementation (which is copyright lhunath, the below text being taken from this specific revision of the relevant git repo):

    #  _______________________________________________________________________
    # |__ csvline ____________________________________________________________|
    #
    #       csvline [-d delimiter] [-D line-delimiter]
    #
    # Parse a CSV record from standard input, storing the fields in the CSVLINE array.
    #
    # By default, a single line of input is read and parsed into comma-delimited fields.
    # Fields can optionally contain double-quoted data, including field delimiters.
    #
    # A different field delimiter can be specified using -d.  You can use -D
    # to change the definition of a "record" (eg. to support NULL-delimited records).
    #
    csvline() {
        CSVLINE=()
        local line field quoted=0 delimiter=, lineDelimiter=$'\n' c
        local OPTIND=1 arg
        while getopts :d: arg; do
            case $arg in
                d) delimiter=$OPTARG ;;
            esac
        done
    
        IFS= read -d "$lineDelimiter" -r line || return
        while IFS= read -rn1 c; do
            case $c in
                \")
                    (( quoted = !quoted ))
                    continue ;;
                $delimiter)
                    if (( ! quoted )); then
                        CSVLINE+=( "$field" ) field=
                        continue
                    fi ;;
            esac
            field+=$c
        done <<< "$line"
        [[ $field ]] && CSVLINE+=( "$field" ) ||:
    } # _____________________________________________________________________
    

提交回复
热议问题