- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这是我为自己的 shell 找到的代码。它工作正常,但我无法理解的是代码的管道部分。
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
char* cmndtkn[256];
char buffer[256];
char* path=NULL;
char pwd[128];
int main(){
//setting path variable
char *env;
env=getenv("PATH");
putenv(env);
system("clear");
printf("\t MY OWN SHELL !!!!!!!!!!\n ");
printf("_______________________________________\n\n");
while(1){
fflush(stdin);
getcwd(pwd,128);
printf("[MOSH~%s]$",pwd);
fgets(buffer,sizeof(buffer),stdin);
buffer[sizeof(buffer)-1] = '\0';
//tokenize the input command line
char* tkn = strtok(buffer," \t\n");
int i=0;
int indictr=0;
// loop for every part of the command
while(tkn!=NULL)
{
if(strcoll(tkn,"exit")==0 ){
exit(0);
}
else if(strcoll(buffer,"cd")==0){
path = buffer;
chdir(path+=3);
}
else if(strcoll(tkn,"|")==0){
indictr=i;
}
cmndtkn[i++] = tkn;
tkn = strtok(NULL," \t\n");
}cmndtkn[i]='\0';
// execute when command has pipe. when | command is found indictr is greater than 0.
if(indictr>0){
char* leftcmnd[indictr+1];
char* rightcmnd[i-indictr];
int a,b;
for(b=0;b<indictr;b++)
leftcmnd[b]=cmndtkn[b];
leftcmnd[indictr]=NULL;
for(a=0;a<i-indictr-1;a++)
rightcmnd[a]=cmndtkn[a+indictr+1];
rightcmnd[i-indictr]=NULL;
if(!fork())
{
fflush(stdout);
int pfds[2];
pipe(pfds);
if(!fork()){
close(1);
dup(pfds[1]);
close(pfds[0]);
execvp(leftcmnd[0],leftcmnd);
}
else{
close(0);
dup(pfds[0]);
close(pfds[1]);
execvp(rightcmnd[0],rightcmnd);
}
}else
wait(NULL);
//command not include pipe
}else{
if(!fork()){
fflush(stdout);
execvp(cmndtkn[0],cmndtkn);
}else
wait(NULL);
}
}
}
调用参数为 0 和 1 的 close() 的目的是什么?调用 dup() 的作用是什么?
最佳答案
在 Unix 上,dup()
调用使用编号最低的未使用文件描述符。因此,调用 dup()
之前的 close(1)
是强制 dup()
使用文件描述符 1。对于 关闭(0)
。
因此,别名是让进程将管道的写入端用于 stdout
(文件描述符 1 用于控制台输出),并将管道的读取端用于 >stdin
(文件描述符0用于控制台输入)。
使用 dup2()
代替可以更清楚地表达代码。
dup2(fd[1], 1); /* alias fd[1] to 1 */
<小时/>
根据您关于ls | 的问题sort
有效,您的问题不限于为什么进行 dup()
系统调用。您的问题实际上是 Unix 中的管道如何工作,以及 shell 命令管道如何工作。
Unix 中的管道是一对相关的文件描述符,在可写描述符上写入数据允许从可读描述符中读取该数据。 pipe()
调用以数组形式返回该对,其中第一个数组元素可读,第二个数组元素可写。
在 Unix 中,fork()
后跟某种 exec()
是生成新进程的唯一方法(还有其他库调用,例如system()
或 popen()
创建进程,但它们调用 fork()
并执行 exec()
> 在引擎盖下)。 fork()
产生一个子进程。子进程看到调用的返回值 0
,而父进程看到非零返回值,该值要么是子进程的 PID,要么是 -1
> 表示发生错误。
子进程是父进程的副本。这意味着当子进程修改变量时,它正在修改驻留在其自己进程中的变量的副本。父级看不到修改发生,因为父级拥有原始副本)。但是,形成管道的一对重复的文件描述符可用于允许子进程与其父进程相互通信。
所以,ls | sort
意味着生成了两个进程,并且 ls
写入的输出被 sort
读取为输入。两个进程意味着两次调用 fork()
来创建两个子进程。一个子进程将exec()
执行ls
命令,另一个子进程将exec()
执行sort
命令。它们之间使用管道来允许进程相互通信。 ls
进程写入管道的可写端,sort
进程从管道的可读端读取。
在发出 close(1)
后,通过 dup()
调用强制 ls
进程写入管道的可写端。 sort
进程通过 close(0)
之后的 dup()
调用强制读取管道的可读端。
此外,关闭管道文件描述符的 close()
调用用于确保 ls
进程是唯一拥有打开引用的进程对于可写的 fd,sort
进程是唯一拥有对可读 fd 的开放引用的进程。该步骤很重要,因为 ls 退出后,它将关闭 fd 的可写端,并且 sort 进程将期望看到 EOF 作为结果。但是,如果其他进程仍然打开可写 fd,则不会发生这种情况。
关于无法理解我自己的 shell 中的 pipeline(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16535562/
我相信我在子 shell 中调用 exit 会导致我的程序继续: #!/bin/bash grep str file | while read line do exit 0 done
我相信我在子 shell 中调用 exit 会导致我的程序继续: #!/bin/bash grep str file | while read line do exit 0 done
我有几个脚本,它们的第一部分看起来是一样的。这部分的功能是识别脚本在哪台机器上运行并相应地设置几个变量。它看起来像这样: ENV=`echo $LOGNAME | cut -c1-8` if
这是我正在尝试做的事情。我有 4 个 shell 脚本。脚本 1 需要先运行,然后是 2,然后是 3,然后是 4,并且它们必须按此顺序运行。脚本 1 需要运行(并在后台等待)2 才能正常运行,但是脚本
我有一个名为 a.sh 的脚本,其中的内容是: //a.sh: #!/bin/bash temp=0 while [ "$temp" -ne 500 ] do echo `date`
在snakemake中,使用shell()函数执行多个命令的推荐方式是什么? 最佳答案 您可以调用shell()多次内run规则块(规则可以指定 run: 而不是 shell: ): rule pro
我有一个 shell 脚本,我向其中传递了一些参数。Test1.sh -a 1 -b 2 -c“一二三” 在 Test1.sh 中,我按以下方式调用另一个 shell 脚本。Test2.sh $* 我
我有 2 个 shell 脚本。 第二个shell脚本包含以下函数第二个.sh func1 func2 first.sh 将使用一些参数调用第二个 shell 脚本, 将使用特定于该函数的一些其他参数
我有一个 Unix shell 脚本 test.sh。在脚本中,我想调用另一个 shell,然后从子 shell 执行 shell 脚本中的其余命令并退出 说清楚: test.sh #! /bin/b
我想在 shell 脚本中更改路径环境变量。路径变量需要在shell脚本执行后修改。 最佳答案 我知道有两种方法可以做到这一点。第一种是在当前 shell 的上下文中运行脚本: . myscript.
此 shell 脚本按预期运行。 trap 'echo exit' EXIT foo() { exit } echo begin foo echo end 这是输出。 $ sh foo.sh
我正在使用 vimshell在 vim 中执行命令 nnoremap vs :VimShellPop 使用此键映射,我可以打开 vim shell 并执行诸如“捆绑安装”之类的命令,然后 输入 exi
我想连接到不同的 shell(csh、ksh 等)并在每个切换的 shell 中执行命令。 下面是反射(reflect)我的意图的示例程序: #!/bin/bash echo $SHELL csh e
我目前正在尝试使用 BNF 和 LL 解析器在 C 中重新编写 shell。 否则,我需要知道 shell 运算符的优先级是什么| , > , > , & , ; ? 有没有人可以提供给我? 谢谢 最
不幸的是,我没有suspend 命令(busybox/ash)。但是我可以使用 kill -STOP $$ 从后台 shell (sh &) 返回到父 shell(以及 fg 之后)。 但是我不想输入
我需要知道,当用户切换到另一个 shell 时,通过单击它。 我试过 shellListener.shellDeactivated()但是当 shell 失去对它自己的控件的焦点时,会触发此事件,这意
file1.txt aaaa bbbb cccc dddd eeee file2.txt DDDD cccc aaaa 结果 bbbb eeee 如果能不区分大小写就更好了! 谢谢! 最佳答案 gre
我见过解压缩目录中所有 zip 文件的循环。但是,在运行此之前,我宁愿确保我将要运行的内容正常工作: for i in dir; do cd $i; unzip '*.zip'; rm -rf *.z
我对编程还很陌生,但我想知道 vim、emacs、nano 等 shell 文本编辑器如何能够控制命令行窗口。我主要是一名 Windows 程序员,所以可能在 *nix 上有所不同。据我所知,只能将文
我有一个包含第 7 列日期的文件,我的要求是将它与今天的日期进行比较,如果小于它,则删除该完整行。 此外,如果第 7 列中提到的任何日期超过 15 天,则将其修改为最多 15 天 下面的例子- now
我是一名优秀的程序员,十分优秀!