Как я могу получить букву диска в Perl?
Мне нужно получить букву диска в Perl. Может ли кто-нибудь любезно помочь мне? $ENV{SYSTEMDRIVE} не работает; это дает мне реальную логическую букву, а не скрытую букву.
3 ответа
Спасибо всем, наконец, я решил это намного более простым способом - используя команду getcwd для получения текущего рабочего каталога, а затем я использовал первые две буквы из его вывода - так просто:-)
use Cwd;
my $driveletter = substr(getcwd, 0, 2);
Если вы хотите сделать это самостоятельно, вы можете захватить выходные данные команды 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: