Как я могу получить букву диска в Perl?

Мне нужно получить букву диска в Perl. Может ли кто-нибудь любезно помочь мне? $ENV{SYSTEMDRIVE} не работает; это дает мне реальную логическую букву, а не скрытую букву.

3 ответа

Решение

Спасибо всем, наконец, я решил это намного более простым способом - используя команду getcwd для получения текущего рабочего каталога, а затем я использовал первые две буквы из его вывода - так просто:-)

use Cwd;

my $driveletter = substr(getcwd, 0, 2); 

Вы ищете Win32:: FileOp?

Если вы хотите сделать это самостоятельно, вы можете захватить выходные данные команды subst и обработать ее, поскольку она выводит все текущие замещенные диски.

SUBST [drive1: [drive2:]path]
SUBST drive1: /D
    drive1:        Specifies a virtual drive to which you want to assign a path.
    [drive2:]path  Specifies a physical drive and path you want to assign to
                   a virtual drive.
    /D             Deletes a substituted (virtual) drive.
Type SUBST with no parameters to display a list of current virtual drives.

C:\Documents and Settings\Administrator\My Documents>subst r: c:\bin

C:\Documents and Settings\Administrator\My Documents>subst
    R:\: => C:\bin

Для того, чтобы сделать это, вам нужна функция для возврата вывода с подстановкой, как показано ниже:

sub get_drive {
    my $drv = shift;
    my $ln;
    $drv = substr($drv,0,1);
    open (IN, "subst |");
    while ($ln = <IN>) {
            chomp ($ln);
            if ((substr($ln,0,1) eq $drv) && (substr($ln,1,6) eq ":\\: =>")) {
                    close (IN);
                    return substr($ln,8);
            }
    }
    close (IN);
    return $drv . ":\\";
}

print get_drive ("R:") . "\n";
print get_drive ("S:") . "\n";

Это выводит:

C:\bin
S:\

в моей системе, которая имеет только один диск.

(Ответ действительно запоздалый, я знаю), но только сегодня мне нужно что-то подобное, и Win32::FileOp не скомпилируется в моей системе. Поэтому я вызвал суб и заменил виртуальные диски на "настоящие"; фрагмент следует:

use strict;
use Data::Dumper;
use feature 'say';

my $DB=1;

$Data::Dumper::Indent = 1;
$Data::Dumper::Terse  = 1;
my %Virt;

exit main();

sub main
{
    my $rtn;
    my (@args) = @_;
    open CMD,"subst|" or die "can't run subst command";
    while (<CMD>) {
        chomp;
        my ($drv, $path) = split(/:\\: => /);
        $Virt{$drv} = $path;
    }

    my %rset; # result set
    while (my ($d,$p) = each %Virt) {
        $rset{$d} = expand($p);
    }
    #D say Dumper rset => \%rset;
    return $rtn;
}

# recursive call if expanded path has another 'virtual' drive
sub expand
{
    my ($loc) = @_;
    my $rtn = undef;
    my ($drv, $path) =  split(/:\\/, $loc);
    if ($a = $Virt{$drv}) {
        #D say "$a $path";
        $rtn = "$a\\$path";
        $rtn = expand($rtn);
    } else {
        #D say "$drv $path";
        $rtn = "$drv:\\$path";
    }
    return $rtn;
}

Примечания: я использую #D для быстрых отладочных операторов

Я проверил это на трех уровнях, то есть w: sub для x:, x: sub для y: и y: subst для c:

Другие вопросы по тегам