如何在打破foreach循环后重新启动“do-while”循环?

时间:2011-12-22 00:05:54

标签: perl

我的配置脚本中有一小段代码,想法是加载配置然后检查每个密钥是否输入了主机名。但是,如果发现配置包含相同的主机名,则会被拒绝,并显示一条警告消息,指出已存在具有该主机名的配置。

问题是我需要foreach循环检查散列键是否存在以重新启动do-while循环,以便可以尝试使用另一个主机名,或者用户可以^C退出脚本。

这是片段;

my $host;
do {
    print "Enter the hostname or IP of the ESXi server: ";
    chomp($host = <STDIN>);

    if ($host eq '') {
        print "You must enter a hostname or IP address!\n";
    } elsif ($host ne '') {

        # We need to catch duplicate configurations for we don't do the same work twice
        foreach (keys %config) {
            if ($config{$_}{host} ne $host) {
                last;
            } elsif ($config{$_}{host} eq $host) {
                warn "Configuration for $host already exists!\n";
            }
        }

        if ($ping_obj->ping($host)) {
            $config{$config_tag}{host} = $host;
        } elsif (! $ping_obj->ping($host)) {
            print RED . "Ping test for \'$host\' failed" . RESET . "\n";
        }

        $ping_obj->close();
    }
} while ($config{$config_tag}{host} eq 'undef');

这就是模板哈希的样子。

my %template = (
    host => 'undef',
    port => 'undef',
    login => {
        user => 'undef',
        password => 'undef',
    },
    options => {
        snapshots => "0",
        compress => "0",

        # This is expressed as an array
        exclude => 'undef',
    },
);

3 个答案:

答案 0 :(得分:5)

如果在Perl中使用goto LABEL语句,就是这样。

do {
    START:     # could also go right before the "do"
    ...
    if (...) {
        warn "Configuration exists. Start over.\n";
        goto START;
    }
} while (...);

答案 1 :(得分:2)

为什么简单elsif会有3 else个? 我的意思是,他们只测试与相关的if测试的完全相反。

if ($host eq '') {
    ...
} elsif ($host ne '') {
    ...
}
if ($config{$_}{host} ne $host) {
    ...
} elsif ($config{$_}{host} eq $host) {
    ...
}
if ($ping_obj->ping($host)) {
    ...
} elsif (! $ping_obj->ping($host)) {
    ...
}

我会使用普通的while循环,而不是do{...}while(...)循环。

do{
  RESTART:
  if(...){
    goto RESTART;
  }
}while(...);

VS

while(...){
  if(...){
    redo;
  }
}

在此循环中,您只使用%config的键来查找关联值,那么为什么不使用values %config代替。

foreach (keys %config) {
    if ($config{$_}{host} ne $host) {
        last;
    } elsif ($config{$_}{host} eq $host) {
        warn "Configuration for $host already exists!\n";
    }
}

VS

for( values %config ){
  if( $_->{host} ne $host ){
    ...
  } else {
    ...
  }
}

如果您使用的是5.10.0或更高版本,则可以使用smart match (~~),这样可以更清楚地了解您正在测试的内容。

my @hosts = map{ $_->{host} } values %config;
if( $host ~~ @hosts ){
  ...
}

答案 2 :(得分:1)

do ... while看起来更自然时,我不确定您为什么要使用while

一些注意事项:

  • 您无需仔细检查if语句。如果是$host eq ''为真,那么$host ne ''必须为假。根据定义。
  • 如果您不打算在循环外使用$host,我假设 你不是,因为你把它存储在哈希中,你应该把my $host放在循环中,以限制范围。

一些提示:

  • 您可以使用redo重新启动循环。
  • 您可以使用smart matching取消for循环。

while ($config{$config_tag}{host} eq 'undef') {
    print "Enter the hostname or IP of the ESXi server: ";
    chomp(my $host = <STDIN>);
    if ($host eq '') {
        print "You must enter a hostname or IP address!\n";
        redo;
    } else {
        # We need to catch duplicate configurations 
        my @host_list = map { $_->{host} } values %config 
        if ($host ~~ @host_list) {
            warn "Configuration for $host already exists!\n";
            redo;
        }
    }
    if ($ping_obj->ping($host)) {
        $config{$config_tag}{host} = $host;
    } else {
        print RED . "Ping test for \'$host\' failed" . RESET . "\n";
    }
    $ping_obj->close();
}