AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / computer / 问题 / 1560809
Accepted
Christian Hick
Christian Hick
Asked: 2020-06-15 17:33:00 +0800 CST2020-06-15 17:33:00 +0800 CST 2020-06-15 17:33:00 +0800 CST

在 for 循环中使用 awk 命令的 Bash case 语句不起作用

  • 772

我有以下 bash 脚本:

#!/bin/bash

OUTPUT=NULL,2,3,NULL,5,6,7,8

pvar1=$(echo $OUTPUT | awk -F ',' '{print $1}')
pvar2=$(echo $OUTPUT | awk -F ',' '{print $2}')
pvar3=$(echo $OUTPUT | awk -F ',' '{print $3}')
pvar4=$(echo $OUTPUT | awk -F ',' '{print $4}')
pvar5=$(echo $OUTPUT | awk -F ',' '{print $5}')
pvar6=$(echo $OUTPUT | awk -F ',' '{print $6}')
pvar7=$(echo $OUTPUT | awk -F ',' '{print $7}')
pvar8=$(echo $OUTPUT | awk -F ',' '{print $8}')

for i in {$pvar1..$pvar8}
do

case "$i" in

        "NULL")
                pOUT="$(awk '{ $* = $* }1' FS=, OFS=, <<<"$OUTPUT")"
                ;;
        "")
                echo "particulars field is not allowed to be empty. Run the program again and enter NULL or choose a value!" && exit 0
                ;;
        *)
                pOUT="$(awk '{ $* = q $* q }1' FS=, OFS=, q="'"  <<<"$OUTPUT")"
                ;;

esac

done

echo "$pOUT"

输出应该是:

pOUT=NULL,'2','3',NULL,'5','6','7','8'

如果没有提供值,脚本应该退出,例如:

OUTPUT=NULL,2,3,NULL,5,6,,8

我不确定如何在 for 循环中编写 awk 命令。有人可以帮忙吗?

awk bash
  • 3 3 个回答
  • 546 Views

3 个回答

  • Voted
  1. Best Answer
    dave_thompson_085
    2020-06-16T00:08:42+08:002020-06-16T00:08:42+08:00

    运行 awk 来拆分一个(!)字段或连接字符是浪费且不必要的;shell 可以很好地做到这一点。目前尚不清楚您想要什么,因为您的某些代码根本没有意义,但根据您的单个示例,这应该很接近:

    #!/bin/bash
    OUTPUT=NULL,2,3,NULL,5,6,7,8
    
    IFS=,; set -f; set -- $OUTPUT; out=
    for i in "$@"; do case "$i" in # you _can_ omit in "$@" as the default, but this is clearer
    "") echo empty input not allowed; exit 0;; # may want nonzero status for error
    NULL) out="$out,$i";; *) out="$out,'$i'";; 
    esac; done
    echo "${out:1}" # bash only; or "${out#,}" on any Bourne/POSIX shell
    
    # this leaves IFS and noglob set; for a standalone script this doesn't matter,
    # but if source'd or entered manually you probably need to either reset these 
    # explicitly or (more easily) put the whole thing in parentheses (a subshell)
    

    但是如果您确实想使用 awk 它可以轻松地一次完成整个工作

    echo NULL,2,3,NULL,5,6,7,8 |\
    awk -F, -vq=\' '{for(i=1;i<=NF;i++)if($i==""){print "empty not allowed";exit 0}
      else if($i=="NULL")out=out","$i; else out=out","q$(i)q; print substr(out,2)}'
    # linebreak in script for clarity, may be omitted
    # since there's only one line=record input, you _could_ 
    # replace the for(...) with while(++i<=NF) but I don't like it
    

    甚至更简单,perl:

    echo NULL,2,3,NULL,5,6,7,8 |\
    perl -lne 'print join ",",map{ length or die "empty not allowed"; $_=="NULL"?$_:"\047".$_."\047"}split(",")'
    
    • 1
  2. Ljm Dullaart
    2020-06-15T23:05:13+08:002020-06-15T23:05:13+08:00

    不知道您要在这里实现什么,但是循环中的 awk 可能不是出了什么问题。

    for i in {$pvar1..$pvar8}
    do
        echo $i
    done
    

    会给你

    {NULL..8}
    

    这可能不是你想要的。

    一个选项是:

    f=0
    while [ $f -lt 8 ] ; do 
        f=$((f+1))
        fv=$(echo $OUTPUT | awk -F ',' "{print \$$f}")
        if [ "$fv" = "" ] ; then
            echo "$f is empty; run the program again"
            exit 0
        fi
    done
    echo "pOUT=$pOUT"
    

    或多或少的行为与您描述的一样。除了awk,您还可以查看cut,但这取决于您对脚本的进一步处理。

    • 0
  3. Kamil Maciorowski
    2020-06-16T00:09:25+08:002020-06-16T00:09:25+08:00

    {$pvar1..$pvar8} 不按预期工作。

    代替填充pvar1等pvar2,将信息存储在位置参数中:

    # parse the original $@ first and save what you want from it
    
    OUTPUT=NULL,2,3,NULL,5,6,7,8
    
    old_IFS="$IFS"
    IFS=,
    set -f
    set -- $OUTPUT
    IFS="$old_IFS"
    

    然后很容易迭代。例子:

    for i
    do
       printf '%s\n' "$i"
    done
    

    此外,您还$#可以免费获得字段数 ( )。代码是可移植的。

    不过有一个缺陷。UnsetIFS不等于空IFS。变量将被设置后IFS="$old_IFS",即使原始变量未设置。希望您知道IFS整个脚本应该使用什么,以便您可以调整代码。添加一些逻辑来测试相关变量是否在将其值分配给另一个变量之前被设置也是一种可能性。

    在 Bash 中,您可以通过使用命名数组变量来避免覆盖原始位置参数数组。

    #!/bin/bash
    
    OUTPUT=NULL,2,3,NULL,5,6,7,8
    
    IFS=, read -ra myarray <<<"$OUTPUT"
    
    for i in "${myarray[@]}"
    do
       printf '%s\n' "$i"
    done
    

    在您的循环中,如果awk命令正在工作,pOUT=…则会将某些内容分配给变量而不保存已经完成的内容;以前pOUT的不使用。这意味着最终pOUT将仅来自最后一次迭代。我认为您不想pOUT依赖 ; 的最终字段OUTPUT。您希望 的每个部分都pOUT依赖于OUTPUT.

    pOUT按顺序构建:

    #!/bin/bash
    
    OUTPUT=NULL,2,3,NULL,5,6,7,8
    
    IFS=, read -ra myarray <<<"$OUTPUT"
    
    pOUT=""
    for i in "${myarray[@]}"
    do
       case "$i" in
            "NULL")
                    pOUT="$pOUT,$i"
                    ;;
            "")
                    echo "particulars field is not allowed to be empty. Run the program again and enter NULL or choose a value!" && exit 0
                    ;;
            *)
                    pOUT="$pOUT,'$i'"
                    ;;
       esac
    done
    pOUT="${pOUT#,}"
    printf 'pOUT=%s\n' "$pOUT"
    

    请注意,此方法会生成一个前导,。pOUT在循环之后,我们用${pOUT#,}.

    • 0

相关问题

  • 在 macOS High Sierra 的终端中设置环境变量时遇到问题

  • 对于 cp 或 mv,是否有等同于 cd - 的东西?

  • Notify-发送窗口下出现的通知

  • 如何从 WSL 打开 office 文件

  • awk 根据变量匹配提取文件中的特定模式

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何减少“vmmem”进程的消耗?

    • 11 个回答
  • Marko Smith

    从 Microsoft Stream 下载视频

    • 4 个回答
  • Marko Smith

    Google Chrome DevTools 无法解析 SourceMap:chrome-extension

    • 6 个回答
  • Marko Smith

    Windows 照片查看器因为内存不足而无法运行?

    • 5 个回答
  • Marko Smith

    支持结束后如何激活 WindowsXP?

    • 6 个回答
  • Marko Smith

    远程桌面间歇性冻结

    • 7 个回答
  • Marko Smith

    子网掩码 /32 是什么意思?

    • 6 个回答
  • Marko Smith

    鼠标指针在 Windows 中按下的箭头键上移动?

    • 1 个回答
  • Marko Smith

    VirtualBox 无法以 VERR_NEM_VM_CREATE_FAILED 启动

    • 8 个回答
  • Marko Smith

    应用程序不会出现在 MacBook 的摄像头和麦克风隐私设置中

    • 5 个回答
  • Martin Hope
    CiaranWelsh 如何减少“vmmem”进程的消耗? 2020-06-10 02:06:58 +0800 CST
  • Martin Hope
    Jim Windows 10 搜索未加载,显示空白窗口 2020-02-06 03:28:26 +0800 CST
  • Martin Hope
    v15 为什么通过电缆(同轴电缆)的千兆位/秒 Internet 连接不能像光纤一样提供对称速度? 2020-01-25 08:53:31 +0800 CST
  • Martin Hope
    fixer1234 “HTTPS Everywhere”仍然相关吗? 2019-10-27 18:06:25 +0800 CST
  • Martin Hope
    andre_ss6 远程桌面间歇性冻结 2019-09-11 12:56:40 +0800 CST
  • Martin Hope
    Riley Carney 为什么在 URL 后面加一个点会删除登录信息? 2019-08-06 10:59:24 +0800 CST
  • Martin Hope
    zdimension 鼠标指针在 Windows 中按下的箭头键上移动? 2019-08-04 06:39:57 +0800 CST
  • Martin Hope
    jonsca 我所有的 Firefox 附加组件突然被禁用了,我该如何重新启用它们? 2019-05-04 17:58:52 +0800 CST
  • Martin Hope
    MCK 是否可以使用文本创建二维码? 2019-04-02 06:32:14 +0800 CST
  • Martin Hope
    SoniEx2 更改 git init 默认分支名称 2019-04-01 06:16:56 +0800 CST

热门标签

windows-10 linux windows microsoft-excel networking ubuntu worksheet-function bash command-line hard-drive

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve