不要求第一的论点承担任何有关的可执行文件的名称:
int main(void)
{
char *args[3] = { "rip van winkle", "30", 0 };
execv("/bin/sleep", args);
return 1;
}
试试吧 - 在Mac上(后三项测试):
make x; ./x & sleep 1; ps
在第三运行的输出是:
MiniMac JL: make x; ./x & sleep 1; ps
make: `x' is up to date.
[3] 5557
PID TTY TIME CMD
5532 ttys000 0:00.04 -bash
5549 ttys000 0:00.00 rip van winkle 30
5553 ttys000 0:00.00 rip van winkle 30
5557 ttys000 0:00.00 rip van winkle 30
MiniMac JL:
EBM评论:
呀,这使它更加怪异。在我的测试bash脚本(execve的目标)中,我没有看到execve在arg [0]中的任何地方的价值 - 而不是在环境中,而不是$ 0。
修改实验 - 一个名为'bash的脚本。脚本':
#!/bin/bash
echo "bash script at sleep (0: $0; *: $*)"
sleep 30
而修订的计划:
int main(void)
{
char *args[3] = { "rip van winkle", "30", 0 };
execv("./bash.script", args);
return 1;
}
这就产生ps输出:
bash script at sleep (0: ./bash.script; *: 30)
PID TTY TIME CMD
7804 ttys000 0:00.11 -bash
7829 ttys000 0:00.00 /bin/bash ./bash.script 30
7832 ttys000 0:00.00 sleep 30
有两种可能性,因为我看到它:
- 内核在通过脚本执行脚本时会对命令行进行操作shebang('
#!/bin/bash
')行,或
- Bash本身与其参数列表沾在一起。
如何建立区别?我想复制壳的替代名称,然后使用在家当替代名称会告诉我们一些东西:
$ cp /bin/bash jiminy.cricket
$ sed "s%/bin/bash%$PWD/jiminy.cricket%" bash.script > tmp
$ mv tmp bash.script
$ chmod +w bash.script
$ ./x & sleep 1; ps
[1] 7851
bash script at sleep (0: ./bash.script; *: 30)
PID TTY TIME CMD
7804 ttys000 0:00.12 -bash
7851 ttys000 0:00.01 /Users/jleffler/tmp/soq/jiminy.cricket ./bash.script 30
7854 ttys000 0:00.00 sleep 30
$
此,我认为,表示所使用的家当机制,当内核重写argv[0]
。
解决因nategoose评论:“#!/路径/到/程序”
MiniMac JL: pwd
/Users/jleffler/tmp/soq
MiniMac JL: cat al.c
#include <stdio.h>
int main(int argc, char **argv)
{
while (*argv)
puts(*argv++);
return 0;
}
MiniMac JL: make al.c
cc al.c -o al
MiniMac JL: ./al a b 'c d' e
./al
a
b
c d
e
MiniMac JL: cat bash.script
#!/Users/jleffler/tmp/soq/al
echo "bash script at sleep (0: $0; *: $*)"
sleep 30
MiniMac JL: ./x
/Users/jleffler/tmp/soq/al
./bash.script
30
MiniMac JL:
这表明它是家当机制,而不是任何程序,如猛砸,即调整argv[0]
的值。所以,执行二进制时,argv[0]
的值不会被调整;当通过shebang执行脚本时,参数列表由内核调整; argv[0]
是shebang上列出的二进制文件;如果在shebang之后有争论,那就变成argv[1]
;下一个参数是脚本文件的名称,后跟execv()
或同等调用的其余参数。
MiniMac JL: cat bash.script
#!/Users/jleffler/tmp/soq/al -arg0
#!/bin/bash
#!/Users/jleffler/tmp/soq/jiminy.cricket
echo "bash script at sleep (0: $0; *: $*)"
sleep 30
MiniMac JL: ./x
/Users/jleffler/tmp/soq/al
-arg0
./bash.script
30
MiniMac JL:
您需要双引号;并可能是一个空char *来终止参数列表,假设您使用的是execl()。 – 2010-06-12 03:28:26
我会接受你的话,但有趣的是,我创建了一个测试,执行一个简单的bash脚本,吐出$ 0,甚至当我使用“美化”程序名时,$ 0仍然是完整路径execve()的第一个参数,那么,我假设)。 – EBM 2010-06-12 03:30:32
@Johnathan:怪罪Python。 – 2010-06-12 03:31:19