- 论坛徽章:
- 0
|
本帖最后由 gecedu 于 2012-05-26 15:14 编辑
Shell高级应用一:.CD管理系统设计
1 求假如我们有一个扩展的CD集合.要使得我们的生活更为简单,我们设计和实现在一个管理CD的程序.用我们所学的Linux编程的知识来实现一个电子目录似乎是一个很不错的主意.我们希望,至少最初是这样的,我们的程序可以存储基本的CD信息,例如标签,音乐种类或是艺术家.我们还希望可以保存一些跟踪信息.我们希望可以在每一个CD项目上进行搜索,但是却不会在跟踪细节上进行搜索.为了使得我们的程序完整,我们希望可以输入,更新,删除任何信息.2 设计我们在上面所说的三个需求---更新,搜索,显示数据---指明了我们要实现一个简单的菜单程序.所有我们要存储的内容都是文本的,而且我们在这里假设我们的CD集合并不是很大.我们并不需要一个完整的数据库,我们只需要一些简单的文本就可以了.将信息存放在一个文本文件中可以使得我们的程序保持简单,而且如果我们的需求发生变化,处理文本文件总是要比其他类型的文件简单得多.作为最后的一个手段,我们可以手工使用文本编辑器来输入和删除数据,而并需要编写一个程序来完成这样的工作.我们必须为我们的数据存储做出一个重要的设计决定:是不是单一的一个文件就足够了呢?如果是这样,那么他应是什么格式的文件呢?我们所希望存储的大多数的数据信息对于每一个CD来说只有一次,当然这样排除跟踪信息.所有的CD将会有多于一个跟踪信息.我们是否应在我们要存储的每一个CD上设置一个数字的极限呢?这看起来是任意的和不必需的限制,所以我们立刻否决了这样的想法.如果我们允许灵活的跟踪数目,我们有下面的三个选项:1 使用单一的文件,使用一行来存放CD的标题类信息并使用N行来存放CD的跟踪信息.2 将CD的所有信息放在一行,允许行继续直到没有跟踪信息需要存入为止.3 将标题信息与跟踪信息相分离并使用另一个不同的文件进行存储.只有第三个选项可以允许我们灵活的来修正文件的格式,如果曾经希望将我们的数据库转换为相关的格式我们可以做出这样的选择,所以在这里我们会选择第三个选项.下一个决定就是我们要在我们的文件中存放什么内容:最初,对于每一个CD标题,我们会选择存放:1 CD目录标号2 标题3 类型4 作曲家或艺术家对于轨迹内容,我们会简单的存放:1 轨迹标号2 轨迹名字为了能够合并这两个文件,我们必须将轨迹信息与CD的其他信息相关联.为了这样做,我们会使用CD目录编号.因为这对于每一个CD来说是唯一的,这样他就会在标题文件和轨迹文件中只出现一次.下面让我们来看一下一个简单的标题文件的例子:目录 标题 类型 作曲家CD123 Cool sax Jazz BixCD234 Classic violin Classical Bach CD345 Hits99 Pop Various他们所对应的轨迹信息如下:目录 轨迹标号 标题CD123 1 Some jazzCD123 2 More jazzCD345 1 DizzyCD234 1 Sonata in D minor这两个文件使用目录域进行联合.在这里我们要记住,对于标题文件中的每一个实体在轨迹文件中会有多行与之对应.我们要做的最后一件事就是如何来区分这些实体.在关系数据库中通常采用确定宽度的域,但是并不总是合适的.另一个常用的办法就是逗号,而这也是我们这里所采用的方法.在我们下面的部分中,为了使得我们不至于迷惑,我们会使用下面的一些函数:get_return()get_confirm()set_menu_choice()insert_title()insert_track()add_record_tracks()add_records()find_cd()update_cd()count_cds()remove_records()list_tracks()一个CD程序1 在我们的例子程序中,第一行总是要保证这个程序要作为一个Shell脚本来执行,接下来的是一些版权信息:#!/bin/sh2 我们要做的第一件事就是保证我们在整个脚本中使用的全局变量已经进行设置.我们要设置标题文件,一个轨迹文件以及一个临时文件.我们同时要跟踪Ctrl+C操作,这样如果用户中断了脚本,我们可以保证删除临时文件.menu_choice=””current_cd=””title_file=”title.cdb”tracks_file=”tracks.cdb”temp_file=/tmp/cdb.$$trap ‘rm -f $temp_file’ EXIT3 现在我们要定义我们需要的一些函数,这样脚本从顶行开始执行,在我们试着第一次调用这些函数时可以找到这些函数的定义.为了避免在一些重写一些相同的代码,最初的两个函数是简单的实用程序.get_return() {echo -e “Press return /c”read xreturn 0}get_confirm() {echo -e “Are you sure? /c”while truedo read x case “$x” in y | yes | Y | Yes | YES ) return 0;; n | no | N | No | NO ) echo echo “Cancelled” return 1;; *) echo “Please enter yes or no” ;; esacdone}4 现在我们来定义主要的菜单函数,set_menu_choise.菜单的内容可以动态的变化,如果一个CD实体被选择可以增加一些其他的选项.set_menu_choice() {clearecho “Options :-”echoecho “ a) Add new CD”echo “ f) Find CD”echo “ c) Count the CDs and tracks in the catalog”if [ “$cdcatnum” != “” ]; then echo “ l) List tracks on $cdtitle” echo “ r) Remove $cdtitle” echo “ u) Update track information for $cdtitle”fiecho “ q) Quit”echoecho -e “Please enter choice then press return /c”read menu_choicereturn}5 现在是两个非常短小的函数,insert_title和insert_track,这样可以增加数据库文件.虽然一些讨厌这样的内容,但是他们却可以使得其他的函数看起来更为简洁.紧随着他们的是一个较大的函数定义,add_record_track,在这其中使用前面的两个函数.这个函数使用模式匹配从而保证用户没有输入逗号(因为我们要使用逗号作为区域分隔符),并且当输入了轨迹信息后使用算村运算来增加轨迹数.insert_title() {echo $* >> $title_filereturn}insert_track() {echo $* >> $tracks_filereturn}add_record_tracks() {echo “Enter track information for this CD”echo “When no more tracks enter q”cdtrack=1cdttitle=””while [ “$cdttitle” != “q” ]do echo -e “Track $cdtrack, track title? /c” read tmp cdttitle=${tmp%%,*} if [ “$tmp” != “$cdttitle” ]; then echo “Sorry, no commas allowed” continue fi if [ -n “$cdttitle” ] ; then if [ “$cdttitle” != “q” ]; then insert_track $cdcatnum,$cdtrack,$cdttitle fi else cdtrack=$((cdtrack-1)) fi cdtrack=$((cdtrack+1))done}6 add_records函数允许将一个主CD的信息实体作为一个新的CDadd_records() {# Prompt for the initial informationecho -e “Enter catalog name /c”read tmpcdcatnum=${tmp%%,*}echo -e “Enter title /c”read tmpcdtitle=${tmp%%,*}echo -e “Enter type /c”read tmpcdtype=${tmp%%,*}echo -e “Enter artist/composer /c”read tmpcdac=${tmp%%,*}# Check that they want to enter the informationecho About to add new entryecho “$cdcatnum $cdtitle $cdtype $cdac”# If confirmed then append it to the titles fileif get_confirm ; then insert_title $cdcatnum,$cdtitle,$cdtype,$cdac add_record_trackselse remove_recordsfireturn}7 find_cd函数使用grep命令在CD标题文件中查找指定的目录名字的文本.我们需要知道查找到多少次字符串,但是grep命令只会在他匹配了零次或是多次时才返回一个值.因为这样的情况,我们在一个文件中存储这些输出,其中每一个行匹配一个,然后我们可以计算这个文件中的行数.字数统计命令wc,在他的输出中使用空格来分隔行数,字数以及文件中的字符数.我们使用$(wc -l $temp_file)命令来从这个命令的输出中得到第一个参数来设置linesfound变量.如果我们希望得到其他的后面的参数,我们可以使用set命令来设置命令输出的Shell参数变量.我们将IFS设置为逗号,这样我们就可以使用逗号来进行分隔了.另一个办法就是使用cut命令.find_cd() {if [ “$1” = “n” ]; then asklist=nelse asklist=yficdcatnum=””echo -e “Enter a string to search for in the CD titles /c”read searchstrif [ “$searchstr” = “” ]; then return 0figrep “$searchstr” $title_file > $temp_fileset $(wc -l $temp_file)linesfound=$lcase “$linesfound” in0) echo “Sorry, nothing found” get_return return 0 ;;1) ;;2) echo “Sorry, not unique.” echo “Found the following” cat $temp_file get_return return 0esacIFS=”,”read cdcatnum cdtitle cdtype cdac < $temp_fileIFS=” “if [ -z “$cdcatnum” ]; then echo “Sorry, could not extract catalog field from $temp_file” get_return return 0fiechoecho Catalog number: $cdcatnumecho Title: $cdtitleecho Type: $cdtypeecho Artist/Composer: $cdacechoget_returnif [ “$asklist” = “y” ]; then echo -e “View tracks for this CD? /c” read x if [ “$x” = “y” ]; then echo list_tracks echo fifireturn 1}8 update_cd可以允许我们重新输入一个CD的信息.在这里我们要注意的就是我们要查找(使用grep命令)以$cdcatnum开头(^)并且后跟逗号的行,而且我们需要将$cdcatnum用{}作来一个整体,这样我们就可以在他与目录标号之间没有空格的情况下查找逗号.如果get_confirm函数返回值为真,则这个函数同时还会使用{}括起多条语句来执行.update_cd() {if [ -z “$cdcatnum” ]; then echo “You must select a CD first” find_cd nfiif [ -n “$cdcatnum” ]; then echo “Current tracks are :-” list_tracks echo echo “This will re-enter the tracks for $cdtitle” get_confirm && { grep -v “^${cdcatnum},” $tracks_file > $temp_file mv $temp_file $tracks_file echo add_record_tracks }fireturn}9 count_cds将会返回我们数据库内容的一个快速计数.count_cds() {set $(wc -l $title_file)num_titles=$lset $(wc -l $tracks_file)num_tracks=$lecho found $num_titles CDs, with a total of $num_tracks tracksget_returnreturn}10 remove_records将断开数据库文件与实体之间的联系,使用grep -v命令来移除所有匹配的字符串.在这里我们要注意的是我们必须使用临时文件.如果我们试着使用下面的命令:grep -v “^$cdcatnum” > $title_file在grep命令有机会执行之前$title_file已经使用>输出重定向设置为一个空文件,所以grep命令将会读取一个空文件.remove_records() {if [ -z “$cdcatnum” ]; then echo You must select a CD first find_cd nfiif [ -n “$cdcatnum” ]; then echo “You are about to delete $cdtitle” get_confirm && { grep -v “^${cdcatnum},” $title_file > $temp_file mv $temp_file $title_file grep -v “^${cdcatnum},” $tracks_file > $temp_file mv $temp_file $tracks_file cdcatnum=”” echo Entry removed } get_returnfireturn}11 list_tracks现一次使用grep命令来解压我们需要的行,并且截取我们所需要部分而且会提供一个已标记的输出.如果我们来考虑一下要将这些20几行的古怪代码使用C代码重新进行实现会需要多少行代码,这时我们就会体会Shell是一个多么强大的工具了.list_tracks() {if [ “$cdcatnum” = “” ]; then echo no CD selected yet returnelse grep “^${cdcatnum},” $tracks_file > $temp_file num_tracks=$(wc -l $temp_file) if [ “$num_tracks” = “0” ]; then echo no tracks found for $cdtitle else { echo echo “$cdtitle :-” echo cut -f 2- -d , $temp_file echo } | ${PAGER:-more} fifiget_returnreturn}12 现在所有的函数都已经进行定义,我们可以进入我们的主要部分了.最初的几行可以简单的将文件读入到一个已知的状态,然后我们调用菜单函数,set_menu_choice并且反应在输出.当选择了quit,我们删除这个临时文件,输出信息,并且已成功条件返回.rm -f $temp_fileif [ ! -f $title_file ]; then touch $title_filefiif [ ! -f $tracks_file ]; then touch $tracks_filefi# Now the application properclearechoechoecho “Mini CD manager”sleep 1quit=nwhile [ “$quit” != “y” ];do set_menu_choice case “$menu_choice” in a) add_records;; r) remove_records;; f) find_cd y;; u) update_cd;; c) count_cds;; l) list_tracks;; b) echo more $title_file echo get_return;; q | Q ) quit=y;; *) echo “Sorry, choice not recognized”;;esacdone#Tidy up and leaverm -f $temp_fileecho “Finished”exit 0这样,我们就完成了我们这个简单但是实用的一个程序了~~
Shell 应用二:输出彩色文字
对于在SHELL下面编写屏幕程序,大家使用echo命令比较多,而且此命令也比较快。下面介绍的是另一方法,只是速度比较慢。要实现这一点,系统上需要一个彩色监视器和tput命令。
一.使用
在使用tput前,需要在脚本或命令行中使用tput命令初始化终端。
$ tput inittput产生三种不同的输出:字符型、数字型和布尔型(真/假)。/字符串输出/
下面是大部分常用字符串:名字含义bel 警铃blink 闪烁模式bold 粗体civis 隐藏光标clear 清屏cnorm 不隐藏光标cup 移动光标到屏幕位置( x,y)el 清除到行尾ell 清除到行首smso 启动突出模式rmso 停止突出模式smul 开始下划线模式rmul 结束下划线模式sc 保存当前光标位置rc 恢复光标到最后保存位置sgr0 正常屏幕rev 逆转视图/数字输出/以下是大部分常用数字输出。名字含义cols 列数目ittab设置宽度lines 屏幕行数/布尔输出/在tput中只有两种布尔操作符。名字含义chts 光标不可见hs 具有状态行例如:tput clear 清除屏幕二.光标位置
可以用tput将光标放在屏幕任意位置。格式为:cup r cr为从上至下屏幕行数, c为穿过屏幕列数。最好将之编成函数,这样就可以把行和列的值传递给它。例子:tput cup 2 3命令将光标放到第2行、第3列。
三.菜单中使用颜色
对域使用颜色可以使数据输入屏幕看起来更加专业。下面将使用的颜色是A N S I标准颜色,并不是所有颜色都适合于所有系统。下面列出了大部分常用颜色。
1. 前景色:数字颜色数字颜色3 0 黑色3 4 蓝色3 1 红色3 5 紫色3 2 绿色3 6 青色3 3 黄(或棕)色3 7 白(或灰)色
2. 背景色:数字颜色数字颜色4 0 黑色4 4 青色4 1 红色4 5 蓝色4 2 绿色4 6 青色4 3 黄(或棕)色4 7 白(或灰)色例子:黑底绿字echo "33[40;32m"使用tput命令可以增强应用外观及脚本的控制。颜色设置可以增加应用的专业性。注意使用颜色不要太过火,否则速度会非常的慢。 |
|