最近接手一个小项目,要求使用谷歌的aapt.exe获取apk软件包中的信息。依稀记得去年年中时,有个同事也问过我如何获取被调用进程的输出结果,当时还研究了一番,只是没有做整理。...这个问题,从微软以为为我们考虑过了,我们可以从一个API中可以找到一些端倪——CreateProcess。...它是我们启动子进程时,控制子进程启动方式的参数。...这三个参数似乎就点中了标题中的两个关键字“标准输出”、“标准错误输出”。是的!我们正是靠这几个参数来解决我们所遇到的问题。那么如何使用这些参数呢? 我们选用的还是老方法——管道。...我们之后将hWrite交给我们创建的子进程,让它去将信息写入管道。而我们父进程,则使用hRead去读取子进程写入管道的内容。
使用管道 while p.poll() is None: line=p.stdout.readline().decode("utf8") print(line) shell =false...意思是command 使用的不是linux shell指令,如果要用shell 指令如ls 要将false 变成true, 通过指定stderr=subprocess.STDOUT,将子程序的标准错误输出重定向到了标准输出...,以使我们可以直接从标准输出中同时获取标准输出和标准错误的信息。...p.poll() 返回子进程的返回值,如果为None 表示 c++子进程还未结束. p.stdout.readline() 从 c++的标准输出里获取一行....参考文章1 python中的subprocess.Popen()使用 参考文章 2 python 从subprocess运行的子进程中实时获取输出
本文实例讲述了PHP swoole的process模块创建和使用子进程操作。...swoole提供了2种进程间的通信: 1、基于 unix socket 的管道 pipe。 2、基于 sysvmsg 的消息队列。...我们可以通过 new swoole_process() 快速的创建一个进程,默认会创建一个 SOCK_DGRAM 类型的管道,用于进程间的通信,当然可以设置成其他类型,也可以不创建。...//默认为每个子进程创建一个管道,如果不想创建设置$pipe_type参数为false //注意管道默认是同步阻塞,半双工,如果读取不到数据就会阻塞 $worker = new swoole_process...: {$worker- pid} 计算 {$task['start']} - {$task['end']} \n"; //子进程把计算的结果,写入管道 $worker- write
父子进程管道通信 Linux进程通信的几种方式 管道通信 中断信号 共享内存、消息队列 Unix Socket 我们PHP中所使用的workman、swoole 或者其他语言当中的进行通信也是无非以上的几种方式...fprintf(STDOUT,"子进程退出成功 pid=%d\n",$pid); } ?...当通过运行代码时,我们可以发现的,当父进程写入数据后,子进程也会读到父进程写入的数据,但以上管道通信是以阻塞方式运行的,当没有数据时,进程则会阻塞不执行 非阻塞方式 $file = 'pipe_file...($status); if($pid > 0) { fprintf(STDOUT,"子进程退出成功 pid=%d\n",$pid); } ?...当加上函数stream_set_blocking以非阻塞方式运行后,会发现写进程并没有写进去,并且报了一个警告的错误,这个就是因为非阻塞模式,不管有没有接受到数据,都执行完毕退出导致的,下面我们再修改一下代码
Swoole是有自己的一个进程管理模块,用来替代PHP的pcntl扩展,需要注意Process进程在系统是非常昂贵的资源,创建进程消耗很大,另外创建的进程过多会导致进程切换开销大幅上升。...提供了基于unixsock的进程间通信,使用很简单只需调用write/read或者push/pop即可 2.swoole_process支持重定向标准输入和输出,在子进程内echo不会打印屏幕,而是写入管道...$redirect_stdin_stdout,重定向子进程的标准输入和输出。启用此选项后,在子进程内输出内容将不是打印屏幕,而是写入到主进程管道。读取键盘输入将变为从管道中读取数据。默认为阻塞读取。...会发现,每次创建一个进程后,就会随之创建一个管道,主进程想和哪一个进程通信,就向那个进程的管道写入/读取数据。...->push('hello 子进程');#推送到子进程,不能当做管道使用 // echo '主进程消息:' .
父进程调用fork创建子进程,那么子进程也有两个文件描述符指向同一管道。 父进程关闭管道读端,子进程关闭管道写端。父进程可以向管道中写入数据,子进程将管道中的数据读出。...write(STDOUT_FILENO, buf, len); (gdb) p len $1 = 0 PIPE读写行为总结 如果所有指向管道写端的文件描述符都关闭了(管道写端引用计数为0),而仍然有进程从管道的读端读数据...如果有指向管道写端的文件描述符没关闭(管道写端引用计数大于0),而持有管道写端的进程也没有向管道中写数据,这时有进程从管道读端读数据,那么管道中剩余的数据都被读取后,再次read会阻塞,直到管道中有数据可读了才读取数据并返回...如果所有指向管道读端的文件描述符都关闭了(管道读端引用计数为0),这时有进程向管道的写端write,那么该进程会收到信号SIGPIPE,通常会导致进程异常终止。...管道的原理: 管道实为内核使用环形队列机制,借助内核缓冲区(4k)实现。 管道的局限性: 数据自己读不能自己写。 数据一旦被读走,便不在管道中存在,不可反复读取。 由于管道采用半双工通信方式。
重定向的原理是: 首先声明两个概念:主程序(重定向的操纵者)、子进程(被重定向的子进程) 如果要重定位stdout的话,先生成一个管道, 管道的写入端交给子进程去写,主程序从管道的读出端读数据,然后可以把数据写成文件...重定向stderr和stdout是相同的。 同理,要重定向stdin的话,生成一个管道, 管道的写入端由主程序写,子进程从管道的读出端读数据。...,那么还必须在父进程中创建一个子进程,同时,这个子进程必须能够继承和使用父进程的一些公开的句柄,因为在子进程中必须要使用父进程创建的匿名管道的读写句柄,通过这个匿名管道才能实现父子进程的通信,所以必须继承父进程的公开句柄...当父进程向子进程发送数据时,用SetStdHandle()将 管道的读句柄赋予标准输入句柄(这样就不会从标准输入读入数据,而从读句柄所表示的位置读取数据);在从子进程接收数据时,则用SetStdHandle...如果父进程要发送数据到子进程,父进程可调用WriteFile()将数据写入到管道(传 递管道写句柄给函数),子进程则调用GetStdHandle()取得管道的读句柄,将该句柄传入ReadFile()后从管道读取数据
PIPE表示应该创建一个新的管道给孩子。随着无,则不会发生重定向; 孩子的文件句柄将从父类继承。...在Popen对象中,可以设值subprocess.stdout=PIPE 即通过管道 p.stdout.read()取出 该进程的标准输出 preexec_fn 如果将preexec_fn设置为可调用对象...返回returncode Popen.communicate(input=None) 将信息输入到进程,从stdout和stderr中读取数据,直到达到文件结尾。等待进程终止。...可选的stdin参数应该是要发送到子进程的字符串,如果没有数据应发送给子进程,则为None。...返回一个元组(stdout,stderr) 但是读取的数据缓存在内存中,所以如果数据量很大或者无限,就不要使用这种方法 Popen.pid 返回子进程的pid Popen.returncode 读取进程的状态码
不能读取自己发出去的 echo $rec = $process->read();//同步阻塞读取管道数据}//子进程创建成功后要执行的函数function my_process(swoole_process...原因是父进程读取子进程返回的数据的时候,是同步阻塞读取: echo $rec = $process->read();//同步阻塞读取管道数据 导致的后果就是父进程依次等待每个进程处理完并返回了内容,才走下一次循环...解决方案1: 使用 swoole_event_add将管道加入到事件循环中,变为异步模式: // echo $rec = $process->read();//同步阻塞读取管道数据//使用swoole_event_add...$redirect_stdin_stdout,重定向子进程的标准输入和输出。启用此选项后,在子进程内输出内容将不是打印屏幕,而是写入到主进程管道(例如用echo打印的内容也写入管道)。...读取键盘输入将变为从管道中读取数据。默认为阻塞读取。 $create_pipe,是否创建管道,启用 $redirect_stdin_stdout后,此选项将忽略用户参数,强制为true。
因为不是从当前进程中读取管道(pipe),如果子进程没有生成足够的输出来填充OS的管道缓冲区,可能会阻塞子进程。...因为不是从当前进程中读取管道(pipe),如果子进程没有生成足够的输出来填充OS的管道缓冲区,可能会阻塞子进程。...因为不是从当前进程中读取管道(pipe),如果子进程没有生成足够的输出来填充OS的管道缓冲区,可能会阻塞子进程。...PIPE表示应该创建通往子进程的管道。DEVNULL表示应该使用指定文件os.devnull。默认参数None则表示无进行重定向,子进程文件句柄从父进程继承。...PIPE表示应该创建通往子进程的管道。DEVNULL表示应该使用指定文件os.devnull。默认参数None则表示无进行重定向,子进程文件句柄从父进程继承。
如果异常未捕获,则会一直从底向事件循环冒泡。如是冒泡到事件循环的异常没被处理,那么就会导致当前进程异常退出。...不推荐直接使用 process.exit(),这会导致事件循环中的任务直接不被处理,以及可能导致数据的截断和丢失(例如 stdout 的写入)。...本文从以下几个方面介绍 child_process 模块的使用:创建子进程父子进程通信独立子进程进程管道创建子进程nodejs 的 child_process 模块创建子进程的方法:spawn, fork...options.stdio 选项用于配置在父进程和子进程之间建立的管道。..."pipe", // 把子进程的 stdout 通过管道传到父进程 。 fs.openSync("err.out", "w") // 把子进程的 stderr 定向到一个文件。
程序通常执行序列或字符串的第一项,但可以通过使用明确的参数进行设置。 在UNIX上,shell = False(默认):在这种情况下,Popen类使用os.execvp()来执行程序的子进程。...PIPE创建一个新的子管道。None,没有重定向;子管道将会继承父管道的文件句柄。此外,标准错误可以用STDOUT来定义,表明应用程序应该从STDOUT捕获到相同的文件句柄的标准错误数据。...communicate(input=None) 与进程相互作用: 发送数据到标准输入。从标准输出、标准错误读取数据, 直到到达文件尾。等待进程终止。...可选的input参数应该是发送给子进程的字符串,或者如果没有要发送给子进程的数据那就用None communicate() 返回一个元组 (stdout, stderr). ...注意:读取的数据是保存在缓冲区中,因此,如果数据太大或没有限制则不要使用这个方法 下面的属性也是有效的: ===================== stdin 如果stdin参数是PIPE,
PIPE 创建管道/文件对象/文件描述符(整数)/stderr 还可以设置为 STDOUT 后面会给出常见的用法 shell 是否使用shell来执行程序。...Popen.communicate(input=None) 与子进程进行交互。向stdin发送数据,或从stdout和stderr中读取数据。可选参数input指定发送到子进程的参数。...即当stdout/stdin设置为PIPE时,使用wait()可能会导致死锁。...那么坑爹的问题来了:当你要使用Python的subprocess.Popen实现命令行之间的管道传输,同时数据源又非常大(比如读取上GB的文本或者无尽的网络流)时,官方文档不建议用wait,同时communicate...是无任何返回输出的,使用,child.communicate()或者读取stdout 则会持续等待。
多进程多线程 由之前的进程处理事务,改成使用线程处理事务,解决了开销大,资源浪费的问题,还可以使用线程池,预先创建就绪线程,减少创建和销毁线程的开销。 但是一个cpu某一时刻只能处理一个事务。...像这样: 单进程单线程(多进程架构) node提供了cluster和child_process两个模块进行进程的创建,也就是我们常说的主(Master)从(Worker)模式。...} console.log(`stdout: ${stdout}`); }); console.log('用户事务处理'); // 子进程 #!...管道实际上是在内核中开辟一块缓冲区,它有一个读端一个写端,并传给用户程序两个文件描述符,一个指向读端,一个指向写端口,然后该缓存区存储不同进程间写入的内容,并供不同进程读取内容,进而达到通信的目的。...管道又分为匿名管道和命名管道,匿名管道常见于一个进程fork出子进程,只能亲缘进程通信,而命名管道可以让非亲缘进程进行通信。
接收传入的系统命令,使用 subprocess.Popen 函数创建子进程来执行指定的命令。...设置了一些参数来配置子进程的执行环境,具体包括: stdout=subprocess.PIPE:将子进程的标准输出连接到管道,以便后续读取输出。...shell=True:表示通过系统的 shell 来执行命令,可以使用命令的通配符、管道等功能。...cwd=cwd:设置子进程的工作目录。 如果调用的shell命令本身在执行之后会突然出现很多输出,则这个时候可能会导致hang在那里,表现就是卡死了,程序也不往下走,也不会报错。。。...综上所述,subprocess.run 适合简单地执行外部命令并获取输出;subprocess.Popen 适合更灵活地控制子进程,以及处理更复杂的子进程交互,注意不要使用 stdout=subprocess.PIPE
它关闭管道的另一端,然后在fork出的子进程中将另一端重定向到子进程的标准输入、输出。...之后不断从console读入用户输入的两个整数,创建一个临时文件(get_temp_fd)并将用户输入写入文件, 之后通过管道将此临时文件传递给子进程,然后在管道上等待子进程返回的另一个临时文件句柄,...这就奇怪了,读取管道返回这个错误的唯一原因只能是管道被关闭,而此管道在子进程端已经被重定向到了标准输入、标准输出, 当标准输入输出关闭时,唯一的可能性是进程已经退出。难道子进程已经不在了么?...else fprintf (stderr, "no more data\n"); 再运行 demo,果然发现多了一句: no more data 看来确实是因为子进程退出导致管道关闭了...recv fd 3, position 4 这下原因清楚了,原来是接收进程与发送进程共享了文件句柄的偏移,导致再读取的过程中直接读到了文件尾。
有时候我们需要在程序里执行一些cmd命令,使用os或者其它模块中的popen方法去执行 这个问题一般是程序内有输入导致的,这个输入可以是input(),也可以是其它的一些stdin操作(如os.popen...如果是PIPE,则表示需要创建一个新的管道,如果是 None,不会做任何重定向工作,子进程的文件描述符会继承父进程的。...另外,stderr的值还可以是STDOUT,表示子进程的标准错误也输出到标准输出。 如果把preexec_fn设置为一个可调用的对象(比如函数),就会在子进程被执行前被调用。...如果指定了startupinfo和creationflags,它们将会被传递给后面的CreateProcess()函数,用于指定子程序的各种其他属性,比如主窗口样式或者是子进程的优先级等。...现在回到我们将要解决的问题 已知: 用pyinstaller的-w参数打包导致python无法处理输入值(stdin) os.popen 打开的管道却需要处理输入值(stdin) 所以,我们不使用os.popen
根据官方文档的解释,该函数会执行 fork 一个子进程执行 command 这个命令,同时将子进程的标准输出通过管道连接到父进程; 也就该方法返回的文件描述符。...这里画个图能更好地理解其中的原理: 在这里的使用场景中并没有获取 popen() 的返回值,所以 command 的执行本质上是异步的; 也就是说当 task.py 执行完毕后会自动关闭读取端的管道。...如图所示,关闭之后子进程会向 pipe 中输出 print '1000'*1024,由于这里输出的内容较多会一下子填满管道的缓冲区; 于是写入端会收到 SIGPIPE 信号,从而导致 Broken pipe...解决办法 既然知道了问题原因,那解决起来就比较简单了,主要有以下几个方案: 使用 read() 函数读取管道中的数据,全部读取之后再关闭。...父子进程是通过匿名管道进行通信的,当读取端关闭时,写入端输出到达管道最大缓存时会收到 SIGPIPE 信号,从而抛出 Broken pipe 异常。 子进程会继承父进程的文件描述符。
超时处理 在实际应用中,我们可能希望设置子进程的最长运行时间,以避免因子进程无法正常退出而导致父进程一直等待。...使用管道进行进程间通信 Subprocess库允许你使用管道(pipes)进行进程间通信。这在需要将一个进程的输出传递给另一个进程时非常有用。...参数将子进程的标准输出和标准错误输出捕获到父进程,然后通过循环实时读取输出。...子进程间的数据传递 除了使用管道进行进程间通信,Subprocess库还支持使用subprocess.send_signal()和subprocess.terminate()等方法向子进程发送信号。...,从文件中读取输入并传递给子进程。
领取专属 10元无门槛券
手把手带您无忧上云