Linux read语法及浅析

read命令 -n(不换行) -p(提示语句) -n(字符个数) -t(等待时间) -s(不回显)

1、基本读取
read命令接收标准输入(键盘)的输入,或其他文件描述符的输入(后面在说)。得到输入后,read命令将数据放入一个标准变量中。下面是read命令
的最简单形式::
#!/bin/bash
echo -n "Enter your name:" //参数-n的作用是不换行,echo默认是换行
read name //从键盘输入
echo "hello $name,welcome to my program" //显示信息
exit 0 //退出shell程序。
//********************************
由于read命令提供了-p参数,允许在read命令行中直接指定一个提示。
所以上面的脚本可以简写成下面的脚本::
#!/bin/bash
read -p "Enter your name:" name
echo "hello $name, welcome to my program"
exit 0
在上面read后面的变量只有name一个,也可以有多个,这时如果输入多个数据,则第一个数据给第一个变量,第二个数据给第二个变量,如果输入数据个数过多,则最后所有的值都给第一个变量。如果太少输入不会结束。
//*****************************************
在read命令行中也可以不指定变量.如果不指定变量,那么read命令会将接收到的数据放置在环境变量REPLY中。
例如::
read -p "Enter a number"
环境变量REPLY中包含输入的所有数据,可以像使用其他变量一样在shell脚本中使用环境变量REPLY.
2、计时输入.
使用read命令存在着潜在危险。脚本很可能会停下来一直等待用户的输入。如果无论是否输入数据脚本都必须继续执行,那么可以使用-t选项指定一个计时器。
-t选项指定read命令等待输入的秒数。当计时满时,read命令返回一个非零退出状态;
#!/bin/bash
if read -t 5 -p "please enter your name:" name
then
echo "hello $name ,welcome to my script"
else
echo "sorry,too slow"
fi
exit 0
除了输入时间计时,还可以设置read命令计数输入的字符。当输入的字符数目达到预定数目时,自动退出,并将输入的数据赋值给变量。
#!/bin/bash
read -n1 -p "Do you want to continue [Y/N]?" answer
case $answer in
Y | y)
echo "fine ,continue";;
N | n)
echo "ok,good bye";;
*)
echo "error choice";;
esac
exit 0
该例子使用了-n选项,后接数值1,指示read命令只要接受到一个字符就退出。只要按下一个字符进行回答,read命令立即
接受输入并将其传给变量。无需按回车键。

3、默读(输入不显示在监视器上)
有时会需要脚本用户输入,但不希望输入的数据显示在监视器上。典型的例子就是输入密码,当然还有很多其他需要隐藏的数据。
-s选项能够使read命令中输入的数据不显示在监视器上(实际上,数据是显示的,只是read命令将文本颜色设置成与背景相同的颜色)。
#!/bin/bash
read -s -p "Enter your password:" pass
echo "your password is $pass"
exit 0
4、读文件
最后,还可以使用read命令读取Linux系统上的文件。
每次调用read命令都会读取文件中的"一行"文本。当文件没有可读的行时,read命令将以非零状态退出。
读取文件的关键是如何将文本中的数据传送给read命令。
最常用的方法是对文件使用cat命令并通过管道将结果直接传送给包含read命令的while命令
例子::
#!/bin/bash
count=1 //赋值语句,不加空格
cat test | while read line //cat 命令的输出作为read命令的输入,read读到的值放在line中

 

总结一下Shell读取文件的方法

a),
#使用read命令读取一行数据
while read myline
do
    echo "LINE:"$myline
done < datafile.txt  

b),
#使用read命令读取一行数据
cat datafile.txt | while read myline
do
    echo "LINE:"$myline
Done  

c),
#读取一行数据
cat datafile.txt | while myline=$(line)
do
    echo "LINE:"$myline
Done  

d),
#读取一行数据
while myline=$(line)
do
    echo "LINE:"$myline
done < datafile.txt  

e),
#使用read命令读取变量数据
cat datafile.txt | while read paraa parab parac
do
    echo "PARAA:"$paraa
    echo "PARAB:"$parab
    echo "PARAC:"$parac
Done  

f),
#使用read命令读取变量数据
while read paraa parab parac
do
    echo "PARAA:"$paraa
    echo "PARAB:"$parab
    echo "PARAC:"$parac
done < datafile.txt

下面这个其实不能算是读取文件,应该算是从标准输入读取,代码如下:

#!/bin/sh   

ip=192.168.253.111
while read line <&3 ; do
        echo "  attempt with ($line)"
        # Try to connect and exit when done if it worked.
        $line && exit 0
done 3<<EOF
/usr/bin/rlogin -l snap-admin $ip
/usr/bin/ssh dev@$ip
/usr/bin/ssh snap-admin@$ip
/usr/bin/ssh root@$ip
EOF 

下面这个格式:
<<EOF
(内容)
EOF

把EOF替换成其他东西

意思是把内容当作标准输入传给程序

在这个例子中这么写 3<<EOF 应该是把它重定向到一个文件描述符中,大家都知道文件描述符都是一个整形,这里的3就是作为一个文件描述符来用。

这里再简要回顾一下<<的用法。
当shell看到<<的时候,它就会知道下一个词是一个分界符
该分界符以后的内容都被当作输入,直到shell又看到该分界符(位于单独的一行)。
这个分界符可以是你所定义的任何字符串

下面是对常见的文件描述符命令的整理:

command > filename  把标准输出重定向到一个新文件中
 command >> filename  把标准输出重定向到一个文件中(追加)
 command 1 > filename  把标准输出重定向到一个文件中
 command > filename 2 >&1 把标准输出和标准错误一起重定向到一个文件中
 command 2 >filename  把标准错误重定向到一个文件中
 command 2 >> filename  把标准错误重定向到一个文件中(追加)
 command >> filename 2 >&1 把标准输出和标准错误一起重定向到一个文件中(追加)
 command < filename > filename2 command命令以filename文件作为标准输入,  以filename2文件作为标准输出
 command < filename  command命令以filename文件作为标准输入
 command << delimiter  从标准输出中读入,直至遇到delimiter分界符
 command <&m   把文件描述符m作为标准输出
 command >&m   把标准输出重定向到文件描述符m中
 command <&-   关闭标准输入  

2 ,就read命令的使用方法整理如下:

read命令从标准输入读取一行,并把输入行的每个字段(以指定的分隔符分隔)的值赋给命令行上的变量。 

read [-ers] [-u fd] [-t timeout] [-p prompt] [-a array] [-n nchars] [-d delim] [name ...]  

参数解析:
-r
指定读取命令把“\”(反斜杠)做为输入行的一个普通字符,而非控制字符。
-s
安静模式。如果指定该参数且从终端读入数据,那么输入的时候将不回显在屏幕上。
-u <fd>
指定读入数据的文件描述符,不再使用默认的标准输入。
-t <timeout>
等待标准输入的超时时间,单位为秒。如果在指定的时间内没有输入,即时返回。
-p <prompt>
打印提示符,等待输入,并将输入赋值给REPLY变量或者指定的变量。
-a <array>
读入一组词,依次赋值给数组array。
-n <nchars>
读取指定的字符数。如果已读取n个字符,马上返回,下次从返回点继续读取;如果已到行结束,无论满不满足n个字符都返回。
-d <delim>
指定行结束符,读到该字符就当做一行的结束。
name ...
指定read变量。read读取一行数据后,分隔行数据为各个字段,然后将字段依次赋给指定的变量。如果分隔后的字段数目比指定的变量多,那么将把剩余的全部字段值都赋给最后的那个变量;反之,剩余的变量被赋值为空字符串。如果read没有指定变量,系统使用默认的REPLY作为缺省变量名。
使用重定向读取数据

exec 6< datafile.txt
while read -u 6 myline
do
    echo "LINE:"$myline
done

变量分隔符 
read命令默认的分隔符是空格,多个空格被当做一个空格处理。我们也可以使用IFS(内部字段分隔符)指定的的字符作为分隔符。假如有如下内容的一个文件,它以“$”来分隔变量,希望把每个变量区别开来,可以使用如下脚本: 

baidu$google$tencnt$sina

123456789

#使用read命令读取变量数据
while read paraa parab parac parad
do
    echo "PARAA:"$paraa
    echo "PARAB:"$parab
    echo "PARAC:"$parac
    echo "PARAD:"$parad
done < datafile.txt
执行脚本的输出如下:
PARAA:baidu
PARAB:google
PARAC:tencent
PARAD:sina
PARAA:123456789
PARAB:
PARAC:  

http://blog.csdn.net/xj178926426/article/details/6925770

 

 

时间: 2024-12-16 07:15:04

Linux read语法及浅析的相关文章

LINUX 中的mmap浅析

原创LINUX系统编程水平有限,参考UNIX系统编程手册 LINUX 中的mmap浅析 一.mmap基本原理和分类 在LINUX中我们可以使用mmap用来在进程虚拟地址空间中分配创建一片虚拟内存地址映射 其可以是 1.文件映射    使用文件内容初始化内存 2.匿名映射    初始化全为0的内存空间(calloc也可以) 下面配图来自UNIX系统编程手册 而对于是否共享又分为 1.私有映射(MAP_PRIVATE)    多进程间数据共享,修改不反应到磁盘实际文件,    私有写时复制实现 2.

linux IPv4报文处理浅析

在<linux网络报文接收发送浅析>一文中介绍了数据链路层关于网络报文的处理. 对于接收到的报文,如果不被丢弃.不被网桥转发,会调用netif_receive_skb()提交给IP层: 而对于IP层向外发送的报文,则通过调用dev_queue_xmit()提交给数据链路层. 本文就以netif_receive_skb()和dev_queue_xmit()为起始,简要介绍一下报文在IP层的处理过程. 先来一张图: 报文接收(图中橙色箭头所指) netif_receive_skb()对每一种已注册

linux内核tmpfs/shmem浅析

说起共享内存,一般来说会让人想起下面一些方法: 1.多线程.线程之间的内存都是共享的.更确切的说,属于同一进程的线程使用的是同一个地址空间,而不是在不同地址空间之间进行内存共享: 2.父子进程间的内存共享.父进程以MAP_SHARED|MAP_ANONYMOUS选项mmap一块匿名内存,fork之后,其子孙进程之间就能共享这块内存.这种共享内存由于受到进程父子关系的限制,一般较少使用: 3.mmap文件.多个进程mmap到同一个文件,实际上就是大家在共享文件page cache中的内存.不过文件

linux异步信号handle浅析

在初学linux编程的时候,一直觉得异步信号handle是个很神奇的东西,用户程序可以使用singal之类的系统调用为某某信号注册一个信号处理函数(handle函数). 程序的二进制代码在内存中都有着确定的执行流程,为什么收到异步信号以后,程序会被"中断",然后跳转到这个handle函数里面去运行呢?内核怎么有能力让程序做这样的跳转呢,总不可能临时修改程序的可执行代码吧? 后来学习了一些内核知识,才知道原来进程收到信号以后,并不是立即就被"中断"的,而是先在进程的控

linux文件读写浅析

在<linux内核虚拟文件系统浅析>这篇文章中,我们看到文件是如何被打开.文件的读写是如何被触发的. 对一个已打开的文件fd进行read/write系统调用时,内核中该文件所对应的file结构的f_op->read/f_op->write被调用. 本文将顺着这条路走下去,大致看看普通磁盘文件的读写是怎样实现的. linux内核响应一个块设备文件读写的层次结构如图(摘自ULK3): 1.VFS,虚拟文件系统. 之前我们已经看到f_op->read/f_op->write如

linux进程状态浅析

众所周知,现在的分时操作系统能够在一个CPU上运行多个程序,让这些程序表面上看起来是在同时运行的.linux就是这样的一个操作系统. 在linux系统中,每个被运行的程序实例对应一个或多个进程.linux内核需要对这些进程进行管理,以使它们在系统中"同时"运行.linux内核对进程的这种管理分两个方面:进程状态管理,和进程调度.本文主要介绍进程状态管理,进程调度见<linux进程调度浅析>. 进程状态 在linux下,通过ps命令我们能够查看到系统中存在的进程,以及它们的状

linux内存管理浅析

[地址映射](图:左中) linux内核使用页式内存管理,应用程序给出的内存地址是虚拟地址,它需要经过若干级页表一级一级的变换,才变成真正的物理地址. 想一下,地址映射还是一件很恐怖的事情.当访问一个由虚拟地址表示的内存空间时,需要先经过若干次的内存访问,得到每一级页表中用于转换的页表项(页表是存放在内存里面的),才能完成映射.也就是说,要实现一次内存访问,实际上内存被访问了N+1次(N=页表级数),并且还需要做N次加法运算. 所以,地址映射必须要有硬件支持,mmu(内存管理单元)就是这个硬件.

linux虚拟文件系统浅析

虚拟文件系统(VFS) 在我看来, "虚拟"二字主要有两层含义: 1, 在同一个目录结构中, 可以挂载着若干种不同的文件系统. VFS隐藏了它们的实现细节, 为使用者提供统一的接口; 2, 目录结构本身并不是绝对的, 每个进程可能会看到不一样的目录结构. 目录结构是由"地址空间(namespace)"来描述的, 不同的进程可能拥有不同的namespace, 不同的namespace可能有着不同的目录结构(因为它们可能挂载了不同的文件系统). 操作已打开的文件 VFS

linux页面回收浅析

关于页面的使用 在之前的一些文章中,我们了解到linux内核会在很多情况下分配页面. 1.内核代码可能调用alloc_pages之类的函数,从管理物理页面的伙伴系统(管理区zone上的free_area空闲链表)上直接分配页面(见<linux内核内存管理浅析>).比如:驱动程序可能用这种方式来分配缓存:创建进程时,内核也是通过这种方式分配连续的两个页面,作为进程的thread_info结构和内核栈:等等.从伙伴系统分配页面是最基本的页面分配方式,其他的内存分配都是基于这种方式的: 2.内核中的