답변:
몇 가지 방법이 있습니다.
$0
스크립트가 CWD 이하인 경우 POSIX에서 제공하는 현재 실행중인 스크립트입니다.cwd()
, getcwd()
과 abs_path()
에 의해 제공되는 Cwd
모듈과 스크립트에서 실행되는 위치를 알려FindBin
은 일반적으로 실행 스크립트의 경로 인 $Bin
& $RealBin
변수를 제공합니다 . 이 모듈은 또한 스크립트 이름 인 & 를 제공합니다.$Script
$RealScript
__FILE__
컴파일 도중 전체 경로를 포함하여 Perl 인터프리터가 처리하는 실제 파일입니다.첫 번째 세 개 ( $0
, Cwd
모듈 및 FindBin
모듈)가 놀랍게 실패 하여 빈 문자열 mod_perl
과 같은 가치없는 출력을 생성하는 것을 보았습니다 '.'
. 그러한 환경에서는 모듈 __FILE__
을 사용하여 경로를 사용하고 경로를 얻습니다 File::Basename
.
use File::Basename;
my $dirname = dirname(__FILE__);
__FILE__
)이 예상대로 작동하지 않습니까? 스크립트가 실행 된 상대 경로로 끝나고 허용 된 답변은 전체 경로를 제공합니다.
dirname(__FILE__)
당신이 확인해야 설치 위치에 다른 파일의 위치를 찾으려는 실행 파일과 장소를 연결 그렇다면, 심볼릭 링크를 따르지 않는 if( -l __FILE__)
다음과 dirname(readlink(__FILE__))
.
perl -e 'use Module::CoreList; print Module::CoreList->first_release("File::Basename");'; echo
. 그 File::Basename
이유는 90 년대 후반에 출시 된 Perl 5.0.0이었습니다. 지금 사용하기에는 비용이 적게 든다고 생각합니다.
$ 0은 일반적으로 프로그램의 이름입니다. 어떻습니까?
use Cwd 'abs_path';
print abs_path($0);
abs_path가 상대 경로 또는 절대 경로를 사용하고 있는지 알기 때문에 이것이 작동해야합니다.
업데이트 몇 년 후 읽은 사람이라면 Drew 's answer를 읽어야 합니다. 내 것보다 훨씬 낫다.
realpath
에 동의어 인, abs_path
당신이없는 밑줄 이름을 선호 넣다,
perl -e 'use Cwd "abs_path";print abs_path($0);'
인쇄/tmp/-e
-e
) 을 실행하면 perl이 인라인 스크립트를 저장할 임시 파일을 생성한다고 생각합니다. 귀하의 경우 위치는 /tmp
입니다. 결과는 어땠습니까?
당신은 사용할 수 FindBin , CWD , 파일 : 기본 이름 또는 이들의 조합. 그들은 모두 Perl IIRC의 기본 배포판에 있습니다.
나는 과거에 Cwd를 사용했다.
Cwd :
use Cwd qw(abs_path);
my $path = abs_path($0);
print "$path\n";
$0
또는 __FILE__
원하는 경로를 얻는 것입니다. 누군가를 한 경우에만 문제는 chdir()
하고는 $0
당신이에 절대 경로를 얻을 필요가 - 친척 BEGIN{}
어떤 놀라움을 방지 할 수 있습니다.
FindBin
에서 $PATH
일치하는 무언가를 위해 더 잘되고 그루브를 시도 basename($0)
하지만, 너무 놀라운 일을하는 경우가 있습니다 (특히 : 파일이 cwd에서 "바로 앞에 있음").
File::Fu
가 File::Fu->program_name
및 File::Fu->program_dir
이에 대한.
chdir()
컴파일 타임에 누군가가 (영구적으로) 어리석은 것일 가능성이 있습니까?
일부 짧은 배경 :
불행히도 Unix API는 실행 파일에 대한 전체 경로를 가진 실행중인 프로그램을 제공하지 않습니다. 실제로, 당신을 실행하는 프로그램은 일반적으로 프로그램이 무엇인지 알려주는 필드에서 원하는 것을 제공 할 수 있습니다. 모든 답변에서 알 수 있듯이, 가능한 후보를 찾기위한 다양한 휴리스틱이 있습니다. 그러나 전체 파일 시스템을 검색하는 데 항상 부족한 것은 없으며 실행 파일을 이동하거나 제거해도 실패합니다.
그러나 실제로 실행중인 Perl 실행 파일은 원하지 않지만 실행중인 스크립트는 원하지 않습니다. 그리고 Perl은 스크립트가 어디에 있는지 알아야합니다. 유닉스 API에서 온 __FILE__
동안 이것을에 저장합니다 $0
. 이것은 여전히 상대 경로 일 수 있으므로 Mark의 제안을 받아이를 표준화하십시오.File::Spec->rel2abs( __FILE__ );
__FILE__
여전히 상대 경로를 제공합니다. 즉 '.'.
시도해 보셨습니까?
$ENV{'SCRIPT_NAME'}
또는
use FindBin '$Bin';
print "The script is located in $Bin.\n";
실제로 호출되는 방식과 CGI인지 일반 쉘에서 실행 중인지 등에 달려 있습니다.
#!/usr/bin/perl -w
use strict;
my $path = $0;
$path =~ s/\.\///g;
if ($path =~ /\//){
if ($path =~ /^\//){
$path =~ /^((\/[^\/]+){1,}\/)[^\/]+$/;
$path = $1;
}
else {
$path =~ /^(([^\/]+\/){1,})[^\/]+$/;
my $path_b = $1;
my $path_a = `pwd`;
chop($path_a);
$path = $path_a."/".$path_b;
}
}
else{
$path = `pwd`;
chop($path);
$path.="/";
}
$path =~ s/\/\//\//g;
print "\n$path\n";
: DD
use strict ; use warnings ; use Cwd 'abs_path';
sub ResolveMyProductBaseDir {
# Start - Resolve the ProductBaseDir
#resolve the run dir where this scripts is placed
my $ScriptAbsolutPath = abs_path($0) ;
#debug print "\$ScriptAbsolutPath is $ScriptAbsolutPath \n" ;
$ScriptAbsolutPath =~ m/^(.*)(\\|\/)(.*)\.([a-z]*)/;
$RunDir = $1 ;
#debug print "\$1 is $1 \n" ;
#change the \'s to /'s if we are on Windows
$RunDir =~s/\\/\//gi ;
my @DirParts = split ('/' , $RunDir) ;
for (my $count=0; $count < 4; $count++) { pop @DirParts ; }
my $ProductBaseDir = join ( '/' , @DirParts ) ;
# Stop - Resolve the ProductBaseDir
#debug print "ResolveMyProductBaseDir $ProductBaseDir is $ProductBaseDir \n" ;
return $ProductBaseDir ;
} #eof sub
문제 __FILE__
는 코어 모듈 ".pm"경로가 반드시 실행중인 ".cgi"또는 ".pl"스크립트 경로 일 필요는 없다는 것입니다. 나는 그것이 당신의 목표가 무엇인지에 달려 있다고 생각합니다.
Cwd
mod_perl에 대해서만 업데이트 해야하는 것으로 보입니다 . 내 제안은 다음과 같습니다.
my $path;
use File::Basename;
my $file = basename($ENV{SCRIPT_NAME});
if (exists $ENV{MOD_PERL} && ($ENV{MOD_PERL_API_VERSION} < 2)) {
if ($^O =~/Win/) {
$path = `echo %cd%`;
chop $path;
$path =~ s!\\!/!g;
$path .= $ENV{SCRIPT_NAME};
}
else {
$path = `pwd`;
$path .= "/$file";
}
# add support for other operating systems
}
else {
require Cwd;
$path = Cwd::getcwd()."/$file";
}
print $path;
제안 사항을 추가하십시오.
쉘에 유효한 외부 모듈이 없으면 '../'에서도 잘 작동합니다.
my $self = `pwd`;
chomp $self;
$self .='/'.$1 if $0 =~/([^\/]*)$/; #keep the filename only
print "self=$self\n";
테스트:
$ /my/temp/Host$ perl ./host-mod.pl
self=/my/temp/Host/host-mod.pl
$ /my/temp/Host$ ./host-mod.pl
self=/my/temp/Host/host-mod.pl
$ /my/temp/Host$ ../Host/./host-mod.pl
self=/my/temp/Host/host-mod.pl
라이브러리가없는 모든 솔루션은 실제로 경로를 작성하는 몇 가지 방법 이상으로 작동하지 않습니다 (../ 또는 /bla/x/../bin/./x/../ 등). 내 솔루션은 다음과 같습니다. 한 가지 단점이 있습니다. 왜 교체를 두 번 실행해야하는지에 대한 가장 확실한 아이디어가 없습니다. 그렇지 않은 경우, 의심스러운 "./"또는 "../"를 얻습니다. 나에게 꽤 강해 보인다.
my $callpath = $0;
my $pwd = `pwd`; chomp($pwd);
# if called relative -> add pwd in front
if ($callpath !~ /^\//) { $callpath = $pwd."/".$callpath; }
# do the cleanup
$callpath =~ s!^\./!!; # starts with ./ -> drop
$callpath =~ s!/\./!/!g; # /./ -> /
$callpath =~ s!/\./!/!g; # /./ -> / (twice)
$callpath =~ s!/[^/]+/\.\./!/!g; # /xxx/../ -> /
$callpath =~ s!/[^/]+/\.\./!/!g; # /xxx/../ -> / (twice)
my $calldir = $callpath;
$calldir =~ s/(.*)\/([^\/]+)/$1/;
"정상적인"답변 중 어느 것도 나에게 맞지 않았습니다. FindBin '$ Bin'또는 Cwd를 사용할 때의 문제점은 모든 심볼릭 링크가 해결 된 상태에서 절대 경로를 반환한다는 것입니다. 제 경우에는 심볼릭 링크가있는 정확한 경로가 필요했습니다. "pwd -P"가 아닌 Unix 명령 "pwd"를 반환하는 것과 같습니다. 다음 기능이 솔루션을 제공합니다.
sub get_script_full_path {
use File::Basename;
use File::Spec;
use Cwd qw(chdir cwd);
my $curr_dir = cwd();
chdir(dirname($0));
my $dir = $ENV{PWD};
chdir( $curr_dir);
return File::Spec->catfile($dir, basename($0));
}
Windows에서 사용 dirname
하고 abs_path
함께 사용하면 가장 효과적이었습니다.
use File::Basename;
use Cwd qw(abs_path);
# absolute path of the directory containing the executing script
my $abs_dirname = dirname(abs_path($0));
print "\ndirname(abs_path(\$0)) -> $abs_dirname\n";
이유는 다음과 같습니다.
# this gives the answer I want in relative path form, not absolute
my $rel_dirname = dirname(__FILE__);
print "dirname(__FILE__) -> $rel_dirname\n";
# this gives the slightly wrong answer, but in the form I want
my $full_filepath = abs_path($0);
print "abs_path(\$0) -> $full_filepath\n";
무슨 일이야 $^X
?
#!/usr/bin/env perl<br>
print "This is executed by $^X\n";
사용중인 Perl 바이너리의 전체 경로를 알려줍니다.
뒤집다