首页 > 解决方案 > 派生一个打开套接字对的子例程

问题描述

我想分叉一个打开套接字的子程序。

我已经编写了打开套接字、接收数据并打印其接收数据的代码。GUI是使用编写的 Tk

下面是代码,它基本上是我想做的,除了不分叉子new_port程序。每次我单击submit按钮时,Tk窗口都会卡住。我正在寻求将 a 添加forknew_port子例程的帮助,以便它产生一个新的子进程。我个人在执行fork没有语法错误或不将套接字推入子进程时遇到了麻烦。

这个想法是我可以在表单中填写一个新端口并点击提交。窗口关闭,然后我再次按 new 插入一个新端口,现在第二个套接字与第一个套接字同时打开。例如,同时监听端口 1234 和 5678。

#!/usr/bin/perl -w

use IO::Socket::INET;
use Tk;

$myip = `ifconfig | grep -i inet | head -1 | cut -d ":" -f2 | cut -d " " -f1`;

sub new_port {

    my $socket = new IO::Socket::INET(
        LocalHost => "$myip",
        LocalPort => "$myport",
        Proto     => 'tcp' Reuse => 1
   );

    die "Cannot create socket on local host" unless $socket;
    print "Server waiting for client connection on port $myport\n";

    while ( 1 ) {

        my $client_socket  = $socket->accept();
        my $client_address = $client_socket->peerhost();
        my $client_port    = $client_socket->peerport();
        my $input_data     = "";
        my $received_data  = "";

        do {
            $client_socket->recv($received_data, 65536);
            $input_data = $input_data . $received_data;
        } while ( $received_data ne "" );

        print "INPUT----------------------------------\n";
        print "Data from $client_address on port $client_port\n";
        print $input_data;
        shutdown($client_socket, 1);
    }
}

sub new_port_window {

    my $sw = MainWindow->new;

    $sw->geometry("200x100");
    $sw->title("port opener");
    $sw->Label(
        -text "Insert port #"
    )->place(
        -anchor => 'center',
        -relx => 0.5,
        -rely => 0.2
    );
    $sw->Entry(
        -bg => 'white',
        -fg => 'black',
        -textvariable => \$myport
    )->place(
        -anchor => 'center',
        -relx => 0.5,
        -rely => 0.4
    );
    $sw->Button(
        -text "submit",
        -command => sub {new_port}
    )->place(
        width   => 100,
        -anchor => "center",
        -relx   => 0.5,
        -rely   => 0.8
    );
}

my $mw = MainWindow->new;

$mw->geometry("150x100");
$mw->title("GUI TEST NEW FUNCTION");
$mw->Label(
    -text => "click new"
)->place(
    -anchor => "center",
    -relx => 0.5,
    -rely => 0.3
);
$mw->Button(
    -text => "NEW",
    -command => sub {new_port_window}
)->place(
    -width => 50,
    -anchor => "center",
    -relx => 0.5,
    -rely => 0.8
);

MainLoop;

标签: perlforktkinet

解决方案


自从我第一次尝试这个已经很长时间了,所以我忘记了这是必须完成的方式,还是只是许多工作方式中的一种,但要点是:

  1. 在分叉之前创建一个套接字对
  2. 执行分叉
  3. 在父级中使用一个套接字,在子级中使用另一个
  4. 如果您希望套接字是单向的(从父级到子级或子级到父级),请shutdown在父级和子级中调用适当的套接字

由于您只想从父级向子级发送数据,因此看起来像

($sock_child, $sock_par) = IO::Socket->socketpair(
    Socket::AF_UNIX, Socket::SOCK_STREAM, Socket::PF_UNSPEC);
$pid = fork;
if ($pid) {
    # parent
    shutdown($sock_par, 0); # no more reading from parent
    print $sock_par $data_to_pass_to_child;
    ...
 } else {
    # child
    shutdown($sock_child, 1);  # no more writing in child
    $data_from_parent = <$sock_child>;
    ...
 }

推荐阅读