linux管道学习笔记
linux学习笔记31--命令route和routetrace

linux学习笔记31--命令route和routetraceLinux系统的route命令⽤于显⽰和操作IP路由表(show / manipulate the IP routing table)。
要实现两个不同的⼦⽹之间的通信,需要⼀台连接两个⽹络的路由器,或者同时位于两个⽹络的⽹关来实现。
在Linux系统中,设置路由通常是为了解决以下问题:该Linux系统在⼀个局域⽹中,局域⽹中有⼀个⽹关,能够让机器访问Internet,那么就需要将这台机器的IP地址设置为Linux机器的默认路由。
要注意的是,直接在命令⾏下执⾏route命令来添加路由,不会永久保存,当⽹卡重启或者机器重启之后,该路由就失效了;可以在/etc/rc.local中添加route命令来保证该路由设置永久有效。
1.命令格式:route [-f] [-p] [Command [Destination] [mask Netmask] [Gateway] [metric Metric]] [if Interface]]2.命令功能:Route命令是⽤于操作基于内核ip路由表,它的主要作⽤是创建⼀个静态路由让指定⼀个主机或者⼀个⽹络通过⼀个⽹络接⼝,如eth0。
当使⽤"add"或者"del"参数时,路由表被修改,如果没有参数,则显⽰路由表当前的内容。
3.命令参数:-c 显⽰更多信息-n 不解析名字-v 显⽰详细的处理信息-F 显⽰发送信息-C 显⽰路由缓存-f 清除所有⽹关⼊⼝的路由表。
-p 与 add 命令⼀起使⽤时使路由具有永久性。
add:添加⼀条新路由。
del:删除⼀条路由。
-net:⽬标地址是⼀个⽹络。
-host:⽬标地址是⼀个主机。
netmask:当添加⼀个⽹络路由时,需要使⽤⽹络掩码。
gw:路由数据包通过⽹关。
注意,你指定的⽹关必须能够达到。
metric:设置路由跳数。
Command 指定您想运⾏的命令 (Add/Change/Delete/Print)。
linux操作系统课程学习笔记,我的Linux学习笔记·Linux操作系统基础

linux操作系统课程学习笔记,我的Linux学习笔记·Linux操作系统基础今天的笔记主要是关于Linux操作系统根底的相关学问。
那就从我⾯前的电脑开端讲起。
计算机和操作系统计算机主要包括五个部分:运算器,控制器,存储器,输⼊设备和输出设备。
通常,运算器,控制器再加上其他⼀些部件如寄存器等构成了我们通常所说的CPU(central processing unit),存储器则主要是内存。
运算器,控制器和存储器可以实现数据的处理.但是数据从何⽽来,运算之后的结果去往哪⾥?这就需要输⼊设备和输出设备(I/O设备)。
我们通常⽤到的输⼊设备包括键盘⿏标等,输出设备为屏幕,打印机等。
值得⼀提的是,计算机中有个叫做硬盘的东西,它并不是存储器,⽽是⼀个I/O设备。
在将数据读取到内存时,它是⼀个输⼊设备;⽽将结果保存到磁盘时,它就变成了⼀个输出设备。
这么多设备整合在⼀起,就成了⼀台计算机。
它可以接收我们的指令(键盘⿏标),通过运算(CPU),把结果展⽰给我们(屏幕,硬盘等)。
但是这么多硬件是如何协调作⽤,共同完成⼀个任务⽽不会我⾏我素地乱来呢?我们需要⼀个东西,它可以控制硬件有序地⼯作,各⾃执⾏⾃⼰的任务,这个东西就是操作系统(Operating System)。
操作系统是⼀个特殊的软件,它的任务就是硬件管理—控制CPU的运算,控制内存的分配,控制计算机的⼏乎⼀切。
假如⼀台电脑没有操作系统,它可能只是⼀个艺术品,或者⼀堆废铁。
⼀个完整的操作系统包括内核和⼀些辅助软件。
内核的主要任务就是进⾏硬件管理,它是⼀个操作系统最基础最底层的东西。
内核若想很好地控制硬件并使其发挥相应的功能,需要和硬件相识相知相爱,他俩可以成为完美的⼀对,全都仰仗于驱动的帮忙。
驱动是硬件的灵魂,它向操作系统提供了访问和使⽤硬件的接⼝,以便在某项任务中最⾼效地调⽤硬件。
什么是LinuxLinux就是⼀个操作系统,它可以管理整个计算机硬件,并且可以接收我们的指令,来指挥硬件完成相应的任务,并把结果反馈给我们。
linux学习笔记5--命令rmdir和rm

linux学习笔记5--命令rmdir和rm昨天学习了创建⽬录的命令mkdir ,接下来学习⼀下linux中删除⽂件和⽬录的命令: rm命令。
rm是⼀个危险的命令,使⽤的时候要特别当⼼,尤其对于新⼿,否则整个系统就会毁在这个命令(⽐如在/(根⽬录)下执⾏rm * -rf)。
所以,我们在执⾏rm之前最好先确认⼀下在哪个⽬录,到底要删除什么东西,操作时保持⾼度清醒的头脑。
rm命令可以删除⼀个⽬录中的⼀个或多个⽂件或⽬录,也可以将某个⽬录及其下属的所有⽂件及其⼦⽬录均删除掉。
对于链接⽂件,只是删除整个链接⽂件,⽽原有⽂件保持不变。
注意:使⽤rm命令要格外⼩⼼。
因为⼀旦删除了⼀个⽂件,就⽆法再恢复它。
所以,在删除⽂件之前,最好再看⼀下⽂件的内容,确定是否真要删除。
rm命令可以⽤-i选项,这个选项在使⽤⽂件扩展名字符删除多个⽂件时特别有⽤。
使⽤这个选项,系统会要求你逐⼀确定是否要删除。
这时,必须输⼊y并按Enter键,才能删除⽂件。
如果仅按Enter键或其他字符,⽂件不会被删除。
rmdir命令⽤来删除空⽬录。
当⽬录不再被使⽤时,或者磁盘空间已到达使⽤限定值,就需要删除失去使⽤价值的⽬录。
利⽤rmdir命令可以从⼀个⽬录中删除⼀个或多个空的⼦⽬录。
该命令从⼀个⽬录中删除⼀个或多个⼦⽬录,其中dirname表⽰⽬录名。
如果dirname中没有指定路径,则删除当前⽬录下由dirname指定的⽬录;如dirname中包含路径,则删除指定位置的⽬录。
删除⽬录时,必须具有对其⽗⽬录的写权限。
1.命令格式:rm [选项] ⽂件…2.命令功能:删除⼀个⽬录中的⼀个或多个⽂件或⽬录,如果没有使⽤- r选项,则rm不会删除⽬录。
如果使⽤ rm 来删除⽂件,通常仍可以将该⽂件恢复原状。
3.命令参数:-f, --force 忽略不存在的⽂件,从不给出提⽰。
-i, --interactive 进⾏交互式删除-r, -R, --recursive 指⽰rm将参数中列出的全部⽬录和⼦⽬录均递归地删除。
linux中管道命令的作用和用法

在Linux中,管道命令的作用是将一个命令的输出作为另一个命令的输入,实现两个或多个命令之间的数据传递和处理。
通过使用管道命令,我们可以实现数据传递、数据处理以及提高效率等功能。
管道命令的符号是“|”,它将一个命令的输出直接连接到另一个命令的输入。
第一个命令的输出作为第二个命令的输入,第二个命令的输出又可以作为第三个命令的输入,依此类推。
下面是一些例子来说明管道命令的用法:
1. 数据传递:使用管道命令可以将一个命令的输出传递给另一个命令进行处理。
例如,我们可以使用`ls -l | grep "filename"`命令来查找包含特定文件名的文件列表。
2. 数据处理:通过将多个命令组合起来,可以实现复杂的数据处理逻辑。
例如,我们可以使用`cat file1.txt | grep "pattern" | sort | uniq`命令来从一个文本文件中提取出包含特定模式的行,并对结果进行排序和去重。
3. 提高效率:使用管道命令可以避免中间文件的产生,减少磁盘IO的开销,提高命令行操作的效率。
例如,我们可以使用`grep "pattern" file1.txt | cat > newfile.txt`命令来将包含特定模式的行输出到一个新文件中,而不需要创建中间文件。
总之,管道命令是Linux中非常有用的功能,可以帮助我们实现更高效和灵活的命令行操作。
linux学习笔记

目录第一章 LINUX简介及安装 (2)一、LINUX介绍 (2)二、LINUX安装 (2)三、LINUX目录 (2)四、总结来说: (4)第二章常用命令及帐户管理 (4)一、linux命令格式 (4)二、常用命令 (4)三、用户管理命令 (5)使用技巧: (7)第三章 vi 编辑器 (8)一、Linux 系统中的编辑器知识 (8)二、vi 的模式 (8)三、命令: (8)第四章 SHELL的使用 (8)一:Shell的环境 (8)二、Bash的主要功能 (9)三、Shell变量 (9)第五章、应用程序安装与管理 (11)一、Linux应用程序组成 (11)二、RPM(Redhat Package Manager)包管理 (11)三、应用程序的编译安装 (11)第六章Linux系统管理 (12)一、启动过程: (12)二、运行级别: (12)三、系统服务的启动状态: (13)四、磁盘空间配额: (14)五、压缩命令: (14)第7章Linux基本网络配置 (15)第八章 NFS文件系统: (15)一、NFS的概述和安装 (15)二、NFS服务器的配置 (16)三、图形界面的NFS服务器配置工具: (16)第一章 LINUX简介及安装一、LINUX介绍1Linux:Linux是由芬兰大学的Linus Torvalds 李納斯发起创建的开源软件项目。
2版本号xx.yy.zz :①.xx表为主版本号,yy为次版本号,zz为修订的版本号。
②.次版本号中,单数代表测试版,双数代表正式发行版3开源软件:①源代码开放。
②GPL协议:主要是以源代码形式发布,任何人都可以得到源代码,但是不提供任何的担保,不限制商业性质的发行和包装。
③LGPL许可协议:允许在使用者自己的应用程序中使用程序库,即使不公开自己的源代码。
二、LINUX安装1RHEL4是由RED HAT公司发布的2Linux硬盘概念:①可以分为主分区、扩展分区、逻辑分区。
linux中管道的作用

linux中管道的作用
管道是Linux中一个重要的概念,它可以将一个命令的输出作为另一个命令的输入。
这个过程可以被看作是一种数据流的传输,其中第一个命令的输出被传输到第二个命令的输入,然后第二个命令会将处理后的数据输出到屏幕或者文件中。
管道的作用在于可以将多个命令的功能组合起来,从而实现复杂的操作。
例如,我们可以使用管道将一个文本文件中的所有单词提取出来,并按照字母顺序排序:
cat file.txt | tr -cs A-Za-z '
' | tr A-Z a-z | sort | uniq
上述命令中,首先使用cat命令将文件内容输出到屏幕上,然后使用tr命令将所有非字母字符替换成换行符,接着使用tr命令将所有大写字母转换成小写字母,使用sort命令按照字母顺序排序,最后使用uniq命令去除重复单词。
除了用于组合命令的功能外,管道还有一个重要的作用就是可以节约系统资源。
在Linux中,每次执行一个命令都会启动一个新的进程,而进程的启动和关闭都需要消耗一定的时间和系统资源。
使用管道可以将多个命令组合到一个进程中执行,从而节约了系统资源。
总的来说,管道在Linux中具有非常重要的作用,它可以方便地组合多个命令,实现复杂的操作,并且可以节约系统资源。
学习和掌握管道的使用方法对于Linux系统的使用和管理非常重要。
- 1 -。
Linux学习笔记一(文件和目录操作)

Linux学习笔记⼀(⽂件和⽬录操作)1.基本shell操作命令解析器--根据命令的名字,调⽤对应的可执⾏程序shell--unix操作系统bash--Linux操作系统(⽂件系统)Linux⽂件系统的存储单元是块在磁盘上存储的时候每个⽂件都有⼀个inode--i节点,保存了⼀些⽂件信息,通过iNode找到对应的⽂件a.快捷键b.虚拟终端history--查询命令的历史记录ctrl+p == 向上的箭头,查询上⼀个命令ctrl+n == 向下的箭头,查询下⼀个命令ctrl+b 向左移动backctrl+f 向右移动forwardctrl+a 移动到⾏⾸ctrl+e 移动到⾏尾ctrl+h 刪除光标前⾯的字符ctrl+d 刪除光标覆盖的字符ctrl+u 刪除光标前⾯的所有字符ctrl+k 删除光标位置到⾏尾的字符ctrl+l或者clear命令清理屏幕c.命令和路径补齐tab智能提⽰键,按⼀次没反应说明有很多符合条件的命令,再按⼀次出现符合条件的命令列表cd ⽬录 + 连续两次tab,显⽰⽬录下⼀级的所有路径d.centos7防⽕墙操作systemctl start firewalld.service 启动systemctl enable firewalld.service 开机启动systemctl stop firewalld.service 停⽌systemctl disable firewalld.service 禁⽌开机启动systemctl status firewalld.service 查看状态firewall-cmd --state 查看状态2.Linux系统⽬录结构ls 路径(查询当前路径下的所有⽂件)/根⽬录下的⽬录说明:/bin bin是binary的缩写,这个⽬录存放着经常使⽤的命令可执⾏程序/boot 存放的是启动Linux时的⼀些核⼼⽂件,包括⼀些连接⽂件以及镜像⽂件(开机启动项)/dev 是Device(设备)的缩写,该⽬录存放的是Linux的外部设备,在Linux中访问设备的⽅式和访问⽂件的⽅式是相同的(Linux奉⾏⼀切皆⽂件,它会把所有硬件外设抽象成设备⽂件存到dev⽬录之下,⽐如⿏标键盘)/etc ⽤来存放所有系统管理所需要的配置⽂件和⼦⽬录/home ⽤户的主⽬录,在Linux中,每个⽤户都有⼀个⾃⼰的⽬录,⼀般该⽬录名是以⽤户的账号命名。
Linux学习笔记之常用命令

重启命令立刻重启(root用户使用)rebootshutdown -r nowinit 610分钟后自动重启(root用户使用)shutdown -r 10在时间为20:35时候重启(root用户使用)shutdown -r 20:35如果是通过shutdown命令设置重启的话,可以取消重启shutdown -c关机命令立刻关机(root用户使用)haltpoweroffshutdown -h nowinit 010分钟后自动关机shutdown -h 10hostname命令查看主机名hostname设置主机名临时修改hostname 主机名永久修改vi /etc/hostname网络服务查看IP信息ip a网络连通性测试ping [选项] 目标主机设置网络信息vi /etc/sysconfig/network-scripts/ifcfg-ens32重启network网络服务service network restart防火墙设置查看防火墙状态systemctl status firewalld关闭防火墙systemctl start firewalld禁止开机启动systemctl disable firewalld主机映射文件修改主机名与IP映射关系vi /etc/hosts目录操作命令查看工作目录(Print Working Directory)pwd切换工作目录(Change Directory)cd [目录位置]列表(List)显示目录内容ls [选项]... [目录或文件名]常用命令选项-l :详细信息显示-a:显示所有子目录和文件的信息,包括隐藏文件-A:类似于“-a”,但不显示“.”和“…”目录的信息-R:递归显示内容创建新的目录(Make Directory)mkdir [-p] [/路径/]目录名统计目录及文件的空间占用情况(estimate file space usage)du [选项]... [目录或文件名]常用命令选项-a:统计时包括所有的文件,而不仅仅只统计目录-h:以更易读的字节单位(K、M等)显示信息-s:只统计每个参数所占用空间总的大小文件操作命令新建空文件,或更新文件时间标记touch 文件名查看文件类型file 文件名复制(copy)文件或目录cp [选项] 源文件或目录… 目标文件或目录常用命令选项-r:递归复制整个目录树-p:保持源文件的属性不变-f:强制覆盖目标同名文件或目录-i:需要覆盖文件或目录时进行提醒删除(Remove)文件或目录rm [选项] 文件或目录常用命令选项-f:强行删除文件,不进行提醒-i:删除文件时提醒用户确认-r:递归删除整个目录树移动(Move)文件或目录mv [选项]... 源文件或目录… 目标文件或目录如果目标位置与源位置相同,则相当于改名显示系统命令所在目录which <选项> command(命令名称)常用命令选项-a:将所有由PATH路径中可以找到的指令均列出,而不止第一个被找到的指令名称文件内容操作命令显示出文件的全部内容cat全屏方式分页显示文件内容more交互操作方法按Enter键向下逐行滚动按空格键向下翻一屏、按b键向上翻一屏按q键退出与more命令相同less查看文件开头的一部分内容(默认为10行)head -n 文件名查看文件结尾的少部分内容(默认为10行)tail -n 文件名统计文件中的单词数量(Word Count)等信息wc [选项] 目标文件常用命令选项-l:统计行数-w:统计单词个数-c:统计字节数查找文件里符合条件的字符串grep [选项] <关键字> <文件…>常用选项-c:计算匹配关键字的行数-i:忽略字符大小写的差别-n:显示匹配的行及其行号-s:不显示不存在或不匹配文本的错误信息-h: 查询多个文件时不显示文件名-l:查询文件时只显示匹配字符所在的文件名–color=auto:将找到的关键字部分加上颜色显示压缩命令压缩(解压)文件或目录,压缩文件后缀为gzgzip [选项] 压缩(解压缩)的文件名常用选项-d将压缩文件解压(decompress)-l显示压缩文件的大小,未压缩文件的大小,压缩比(list)-v显示文件名和压缩比(verbose)-num用指定的数字num调整压缩的速度,-1或–fast表示最快压缩方法(低压缩比),-9或–best表示最慢压缩方法(高压缩比)。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
声明:本文档所有内容均为网上整理所得,不保证所有内容正确性.版权属作者本人所有。
不用做任何商业用途,仅供广大网友学习交流之用,如有侵权,请联系本人删除,若发现文档中内容有错误或者有个人见解,欢迎指教与相互讨论.Email:.weiming999@。
管道无名管道管道包括无名管道和有名管道两种,前者在父子进程中流行,后者由于可以独立成为磁盘文件而存在,因为能够被无血缘关系的进程共享.无名管道通常直接称为管道,它占用两个文件描述符,不能被非血缘关系的进程共享,一般应用于父子进程.UNIX中一切皆为文件,管道也是一种文件,称为管道文件.当系统中创建一个管道时,它返回两个文件描述符:一个文件以只写打开,作为管道的输入端;另一个文件以只读打开,作为管道的输出端.#include<unistd.h>int pipe(int fildes[2]);函数pipe在内核中创建一个管道,并分配两个文件描述符标识管道的两端,这两个文件描述符存储与fildes[0]和fildes[1]中.一般约定fildes[0]描述管道和输出端,进程向此文件描述符中读取数据,fildes[1]描述管道的输入端,进程向此文件描述符写入数据.fildes[0]:只读文件描述符.fildes[1]:只写文件描述符.Pipe调用成功返回0,否则返回-1.单向管道流模型管道的两端(输入端和输出端)被一个进程控制没有太大的意义,如果管道的两端分别控制在不同的进程中,这两个进程之间就能够进行通信.拥有管道输入端的进程,可以向管道发送数据,拥有管道输出端的进程,可以从管道中接受前一个进程发送来的消息.1)从父进程流向子进程的管道在父进程创建无名管道并产生子进程后,父子进程均拥有管道两端的访问权.此时关闭父进程的管道输出端,关闭子进程的管道输入端,就形成一个从父进程到子进程的管道流,数据由父进程写入,从子进程读出.2)从子进程流向父进程的管道在父进程中创建无名管道并产生子进程后,父子进程均拥有两端的访问权.此时关闭父进程的管道输入端,关闭子进程的管道输出端,就形成了一个从子进程到父进程的管道流.数据由子进程写入,从父进程读出.#i n c l u d e<u n i s t d.h>#i n c l u d e<s t r i n g>#i n c l u d e<s t r i n g.h>#i n c l u d e<s t d i o.h>i n t m a i n(){p i d_t p i d;i n t f i l d e s[2];c h a r b u f[256];i n t i,j;i f(p i p e(f i l d e s)<0){p r i n t f("p i p e e r r o r\n");r e t u r n-1;}i f((p i d=f o r k())<0){p r i n t f("f o r k e r r o r!\n");r e t u r n-1;}i f(0==p i d){//子进程c l o s e(f i lde s[1]);j=r e a d(f i l d e s[0],b u f,s i z e o f(b u f));b u f[j]='\0';p r i n t f("[c h i l d]l e n g t h=[%d]b u f=[%s]\n",j,b u f);r e t u r n0;}//父进程c l o s e(f i lde s[0]);w r i t e(f i l d e s[1],"H e l l o",s t r l e n("H e l l o"));r e t u r n0;}在进程的通信中,我们无法判断每次通信中报文的字节数,即无法对数据流进行自动拆分,从而发生了子进程一次性读取父进程两次通信的报文,为了能正常拆分发送报文,常常采用以下几种方法:a)固定长度.b)显式长度.每条报文由长度域和数据域组成.长度域大小固定,储存了数据域长度.分为字符串型和整形两种.数据域是传输的实际报文数据.接收进程先获取长度域的数据,转换成数据域的长度,再读取相应长度的信息即为数据域内容.c)短连接.每当进程间需要通信时,创建一个通信线路,发生一条报文后立即废弃这调通线路.这种方式在socket通信中很常用.#i n c l u d e<u n i s t d.h>#i n c l u d e<s t r i n g>#i n c l u d e<s t r i n g.h>#i n c l u d e<s t d i o.h>i n t m a i n(){p i d_t p i d;i n t f i l d e s[2];c h a r b u f[256];i n t i,j;i f(p i p e(f i l d e s)<0){p r i n t f("p i p e e r r o r\n");r e t u r n-1;}i f((p i d=f o r k())<0){p r i n t f("f o r k e r r o r!\n");r e t u r n-1;}i f(0==p i d){//子进程c l o s e(f i lde s[1]);j=r e a d(f i l d e s[0],b u f,5);b u f[j]='\0';p r i n t f("[c h i l d]l e n g t h=[%d]b u f=[%s]\n",j,b u f);m e m s e t(b u f,0,s i z e o f(b u f));j=r e a d(f i l d e s[0],b u f,5);b u f[j]='\0';p r i n t f("[c h i l d]l e n g t h=[%d]b u f=[%s]\n",j,b u f);m e m s e t(b u f,0,s i z e o f(b u f));j=r e a d(f i l d e s[0],b u f,5);b u f[j]='\0';p r i n t f("[c h i l d]l e n g t h=[%d]b u f=[%s]\n",j,b u f);r e t u r n0;}//父进程c l o s e(f i lde s[0]);w r i t e(f i l d e s[1],"12345",5);w r i t e(f i l d e s[1],"67890",5);w r i t e(f i l d e s[1],"a b c d e",5);r e t u r n0;}双向管道流模型管道是进程之间的一种单向交流方式,要实现进程间的双向交流,就必须通过两个管道来完成.1)创建管道,返回两个无名管道文件描述符fildes1和fildes2,fildes1为管道1,fildes2为管道2.int fildes1[2],fildes2[2];pipe(fildes1);pipe(fildes2);2)创建子进程,子进程继承管道1和管道2.3)管道1从父进程流程子进程,管道2从子进程流程父进程.连接标准I/O的管道模型dup2(fd0,0); //复制fd0到文件描述符0中,更改标准输入为fd0dup2(fd1,1); //复制fd1到文件描述符1中,更改标准输出为fd1dup2(fd2,2); //复制fd2到文件描述符2中,更改标准错误输出为fd2示例一:#i n c l u d e<u n i s t d.h>#i n c l u d e<f c n t l.h>#i n c l u d e<i o s t r e a m>u s i n g n a m e s p a c e s t d;i n t m a i n(i n t a r g c,c h a r*a r g v[]){i n t f d;f d=o p e n("./o u t l o g",O_R D W R|O_C R E A T,0755);d u p2(f d,1);//更改标准输出为o u t l o gc o u t<<"H o l l o W o r l d\n";r e t u r n0;}示例二:父进程的输出连接子进程的输入通信实例#i n c l u d e<u n i s t d.h>#i n c l u d e<f c n t l.h>#i n c l u d e<s t r i n g.h>#i n c l u d e<i o s t r e a m>u s i n g n a m e s p a c e s t d;i n t m a i n(i n t a r g c,c h a r*a r g v[]){i n t f i l d e s[2];p i d_t p i d;i n t i,j;c h a r b u f[256];i f(p i p e(f i l d e s)<0||(p i d=f o r k())<0){c o u t<<"e r r o r\n";r e t u r n-1;}i f(0==p i d){//子进程c l o s e(f i lde s[1]);d u p2(f i l de s[0],0);c l o s e(f i lde s[0]);g e t s(b u f);c o u t<<"[c h i l d]b u f="<<b u f<<e nd l;r e t u r n0;}//父进程c l o s e(f i lde s[0]);d u p2(f i l de s[1],1);c l o s e(f i lde s[1]);c o u t<<"H e l l o W o r l d"<<e nd l;r e t u r n0;}Popen模型创建连接标准I/O的管道需要多个步骤,需要使用大量的代码,UNIX提供了一组函数简化了这个复杂的过程.#include<stdio.h>FILE *popen(const char *command,char *type);Int pclose(FIEL *stream);函数popen类似于函数system,它首先fork一个子进程,然后调用exec执行参数command中给定的shell命令.不同的是,函数popen自动在父进程和exec创建的子进程之间建立了一个管道,这个管道可以连接子进程的标准输入,也可以连接子进程的标准输出,参数type决定了管道的I/O类型参数type的取值情况r 创建与子进程的标准输出连接的管道(管道数据由子进程流程父进程)w 创建与子进程的标准输入连接的管道(管道数据由父进程流程子进程)#i n c l u d e<s t d i o.h>i n t m a i n(i n t a r g c,c h a r*a r g v[]){F I L E*i n,*o u t;c h a r b u f[256];i f(N U L L==(o u t=p o p e n("g r e p i n i t","w"))){p r i n t f("e r r o r\n");r e t u r n-1;}i f(N U L L==(i n=p o p e n("p s-e f","r"))){p r i n t f("e r r o r\n");r e t u r n-1;}w h i l e(f g e t s(b u f,s i z e o f(b u f),i n)){f p u t s(b u f,o u t);}p c l o s e(i n);p c l o s e(o u t);r e t u r n0;}有名管道管道如果无名,只能在共同血缘进程中使用,管道如果有名,就可以在整个系统中使用.FIFO管道,有名的管道,它以一种特殊的文件类型储存与文件系统中,以供无血缘的关系进程访问. Shell命令和C程序都可以创建有名管道.1)Shell命令创建有名管道mknod name [b|c] major minor //创建块设备或字符设备文件mknod name p //创建管道文件mknod name s //创建信号量mknod name m //创建共享内存参数name为创建的文件名称,参数major和minor分别代表主次设备号.2)命令mkfifo创建管道mkfifo [–m mode] FILE其中mode是管道文件创建后的访问权限,FILE是管道文件创建后的名称eg.创建一个用户本身可读写,其他任何用户只读的管道文件k2.#mkfifo –m 644 k2;3)C函数mkfifo创建管道UNIX的C语言中,也提供了创建有名管道的函数#include<sys/types.h>#include<sys/stat.h>Int mkfifo(char *path,mode_t mode);函数mkfifo创建有名管道,字符串path指定了管道的文件路径和名称,参数mode决定了管道文件的访问权限,它的取值类似于open函数的第三个参数,并且自带了O_CREAT和O_EXCL选项,因此本函数只能创建一个不存在的管道文件,或者返回”文件以存在”错误.如果只是希望打开而不创建文件,请使用函数open或者函数fopen.有名管道的应用管道本身就是文件,因此对普通文件的操作也适合于管道文件,可以按照以下步骤应用管道.1)创建管道文件(应用命令mknod或者mkfifo,或者函数mkfifo)2)读进程a)只读打开管道文件(应用函数open或fopen)b)读管道(应用函数read或者fread)3)写进程a)只写打开管道b)写管道(应用函数write或者fwrite)4)关闭管道.低级文件编程库和标准文件编程库都可以操作管道,在打开管道文件前请务必先确认该管道是否存在和是否具备访问权限.管道在执行读写操作前,两端必须同时打开,否则执行打开管道某端操作的进程将一直阻塞直到某个进程以相反方向打开管道为止.示例:写有名管道#i n c l u d e<s t d i o.h>#i n c l u d e<i o s t r e a m>#i n c l u d e<s y s/t y p e s.h>#i n c l u d e<s y s/s t a t.h>u s i n g n a m e s p a c e s t d;i n t m a i n(i n t a r g c,c h a r*a r g v[]){F I L E*f p=N U L L;c h a r b u f[256];i f(m k f i f o("./f i f o d a t e",S_I F I F O|0666)<0){c o u t<<"s y s t e m e r r o r"<<e nd l;r e t u r n-1;}w h i l e(1){//打开和关闭管道操作,要在循环里面i f((f p=f o p e n("./f i f o d a t e","w"))==N U L L){c o u t<<"f o p e n f i l e f a i l e d"<<e nd l;r e t u r n-1;}c o u t<<"p l e a s e i n p u t:";m e m s e t(b u f,0,s i z e o f(b u f));c i n>>b u f;f p u t s(b u f,f p);f c l o s e(f p);f p=N U L L;i f(0==s t r c m p(b u f,"q u i t")){b r e a k;}}r e t u r n0;}读有名管道#i n c l u d e<s t d i o.h>#i n c l u d e<i o s t r e a m>#i n c l u d e<s y s/t y p e s.h>#i n c l u d e<s y s/s t a t.h>u s i n g n a m e s p a c e s t d;i n t m a i n(i n t a r g c,c h a r*a r g v[]){c h a r b u f[256];F I L E*f p=N U L L;w h i l e(1){//打开和关闭管道操作,要在循环里面i f((f p=f o p e n("./f i f o d a t e","r"))==N U L L){c o u t<<"f o p e n f a i l e d"<<e nd l;r e t u r n-1;}m e m s e t(b u f,0,s i z e o f(b u f));f g e t s(b u f,s i z e o f(b u f),f p);c o u t<<"r e ad f i f o m s g:"<<b u f<<e n d l;f c l o s e(f p);f p=N U L L;i f(0==s t r c m p(b u f,"q u i t")){b r e a k;}}r e t u r n0;}管道的模型1)“1-1”模型本模型应用于两个进程之间的双向通信,设置两个FIFO.进程A的数据从管道1流程流程B,进程B的数据通过管道2流程进程A.2)“N-1”模型本进程适用于非交互式服务系统,客户端掌握了公共FIFO的输入端,将消息写入管道,后台服务进程掌握了公共FIFO的输出口,它读取管道中得信息.3)“N-1-N”模型本进程适合于交互式服务系统,客户进程掌握了一个总所周知的可以向后台服务进程传递消息的有名管道外,每个客户进程均还拥有一个私有的FIFO.为了使服务进程正确找到客户进程的私有管道,客户进程务必在其发送的请求消息中增加专用FIFO标识.管道小结:管道是进程之间最古老的通信方式,它在UNIX系统中是以一种特殊的文件形式---管道文件.它占用磁盘i节点块和数据块,在目录中记载了文件和i节点对应关系的管道时有名管道,没有记载的是无名管道.在UNIX中,创建无名管道有pipe,popen等,关闭无名管道的函数有pclose等.创建有名管道的命令有mknod,mkfifo等,创建有名管道的函数有mkfifo.无名管道应用于父子进程间,实现从父进程到子进程或者从子进程到父进程之间的单向交流.常常实现进程之间的输入输出重定向.有名管道以管道文件的形式存储与磁盘等外部设备中,可再任意两个进程之间应用,常常实现进程之间的数据交换.管道还有以下特性1)虽然无名管道没有路径,不在任何目录文件中记录该目录项,但是它仍然是文件,占用物理上的i节点和数据块.2)无名管道没有路径,因此无法实现open调用,从而导致无名管道只能在血缘进程中以继承的方式获取访问所需的文件描述符.3)无名管道文件一旦关闭,就不能再次使用,即使是管道的创建进程也一样.4)如果管道一次性写入小于PIPE_BUF字节的数据,该写入操作是原子操作,否则数据将先拆分再写入,此时只能保证各个拆分块的原子操作,拆分了之间的写操作不再具备原子性.例如当多个进程同时写超过PIPE_BUF字节的数据到管道时,写入的数据将被拆分成块,这时进程写入到管道的数据中可能插有其他进程的数据.5)管道最大能够存储PIPE_BUF字节的数据,一旦管道达到最大容量,写管道操作将阻塞,直到管道中数据被读出为止.6)管道以FIFO(先进先出)方式处理数据.7)管道数据一旦读出,就从管道中删除,具有不可再现性.Example:客户端发送客户端id消息域,服务端接收并放回结果//服务端接收客户端通过有名管道发送的消息//消息格式"11+2""1":表示客户端1,固定一个字节,"1+2":数据域,固定3个字节//服务端返回结果"3"到特定的有名管道(./d a t a/d a t a_c l i e n t1)#i n c l u d e<s y s/t y p e s.h>#i n c l u d e<s y s/s t a t.h>#i n c l u d e<u n i s t d.h>#i n c l u d e<s t r i n g>#i n c l u d e<f c n t l.h>#i n c l u d e<i o s t r e a m>u s i n g n a m e s p a c e s t d;v o i d O p e r a t i o n(c h a r*_d a t a,c h a r*C h i l d){s t r i n g_1,_2;s t r i n g s t r I n=_d a t a;i n t p o s=s t r I n.f i n d("+");i f(p o s!=s t r i n g::n p o s){_1=s t r I n.s u b s t r(0,p o s);_2=s t r I n.s u b s t r(p o s+1,1);}i n t i R e t=a t o i(_1.c_s t r())+a t o i(_2.c_s t r());c h a r s t r R e t[3];s p r i n t f(s t r R e t,"%d",i R e t);i n t f C l i e n t=o p e n(C h i l d,O_R D W R|O_C R E A T,0755);w r i t e(f C l i e n t,s t r R e t,s t r l e n(s t r R e t));c l o s e(f C l i e n t);c o u t<<"s e nd m s g["<<s t r Re t<<"]"<<C h i l d<<e n d l;r e t u r n;}i n t m a i n(){i n t f S r v;//管道d a t a_s e r v,服务端读取i f(-1==m k f i f o("./d a t a/d a t a_s e r v",S_I F I F O|0666)){c o u t<<"m k f i f od a t a_se r v e r r o r!"<<e n d l;r e t u r n-1;}//管道d a t a_c l i e n t1,返回给客户端1的管道i f(-1==m k f i f o("./d a t a/d a t a_c l i e n t1",S_I F I F O|0666)){c o u t<<"m k f i f od a t a_c l ie n t1e r r o r!"<<e n d l;r e t u r n-1;}//管道d a t a_c l i e n t2,返回给客户端2的管道i f(-1==m k f i f o("./d a t a/d a t a_c l i e n t2",S_I F I F O|0666)){c o u t<<"m k f i f od a t a_c l ie n t2e r r o r!"<<e n d l;r e t u r n-1;}w h i l e(1){c h a r w h i c h C l i e n t[2];//用于储存客户端编号,客户端发送的编号"1"固定1个字节c h a rd a t a[12];//用于存储数据域,客户端发送的数据域"1+2"固定3个字节c h a r C l i e n t[50];//服务器端读取的内容格式为"11+2"表示"客户端编号数据域":"1"客户端编号"1+2"数据域f S r v=o p e n("./d a t a/d a t a_s e r v",O_R D W R|O_C R E A T,0755);r e a d(f S r v,w h i c h C l i e n t,1);//读取客户端进程编号,占一个字节"1"r e a d(f S r v,d a t a,3);//读取该客户端对应的数据域e g"1+2"w h i c h C l i e n t[1]='\0';d a t a[3]='\0';c o u t<<"r e c v m s g["<<w h i c h C l i e n t<<"]["<<d a t a<<"]"<<e n d l;i f(s t r c m p(w h i c h C l i e n t,"1")==0){s t r c p y(C l i e n t,"./d a t a/d a t a_c l i e n t1");}e l s e i f(s t r c m p(w h i c h C l i e n t,"2")==0){s t r c p y(C l i e n t,"./d a t a/d a t a_c l i e n t2");}E l s e{c o n t i n u e;}O p e r a t i o n(d a t a,C l i e n t);c l o s e(f S r v);}}/****************************************************///客户端代码#i n c l u d e<s y s/t y p e s.h>#i n c l u d e<s y s/s t a t.h>#i n c l u d e<f c n t l.h>#i n c l u d e<s t r i n g>#i n c l u d e<i o s t r e a m>u s i n g n a m e s p a c e s t d;c o n s t s t r i n g C l i e n t I D="1";i n t m a i n(i n t a r g c,c h a r*a r g v[]){i n t f i n,f o u t;s t r i n g_i n;w h i l e(1){c o u t<<"P l e a s e i n p u t:";c i n>>_i n;f i n=o p e n("./d a t a/d a t a_s e r v",O_R D W R|O_C R E A T,0755);s t r i n g_s e n d=C l i e n t I D;_s e n d+=_i n;w r i t e(f i n,_s e n d.c_s t r(),s i z e o f(_s e n d.c_s t r()));c o u t<<"s e nd m s g["<<_se n d<<"]"<<e n d l;f o u t=o p e n("./d a t a/d a t a_c l i e n t1",O_R D W R|O_C R E A T,0755);c h a r r e c v[10];r e a d(f o u t,r e c v,s i z e o f(r e c v));c o u t<<"r e c e i v e m s g["<<r e c v<<"]"<<e nd l;c o u t<<e nd l;c l o s e(f o u t);c l o s e(f i n);}r e t u r n0;}。