Reputation: 5067
I'm looking for a way to set the scope of require_once()
to the global scope, when require_once()
is used inside a function. Something like the following code should work:
file `foo.php':
<?php
$foo = 42;
actual code:
<?php
function includeFooFile() {
require_once("foo.php"); // scope of "foo.php" will be the function scope
}
$foo = 23;
includeFooFile();
echo($foo."\n"); // will print 23, but I want it to print 42.
Is there a way to explicitly set the scope of require_once()
? Is there a nice workaround?
Upvotes: 12
Views: 6765
Reputation: 1
You can use eval() to get your require_once to run in a global context. This method would require changing just two lines in your main program file: your function call, and the return value from the function. No changes to foo.php would be required, or other global iteration tricks.
function includeFooFile() {
# don't do the require in the fn, let the caller do it
return 'require_once("foo.php");';
}
$foo = 23;
eval(includeFooFile());
echo($foo."\n"); # will print 42.
There are a couple of catches to the eval() approach though:
Do not include any user-supplied or untrusted strings in the returned & eval'd code: see the eval() documentation as to why. This shouldn't be an issue for require_once's because in general you will be including only known source files.
Your includeFooFile function cannot use for itself any of the variables or functions declared in foo.php, because they do not exist until after the return. If you wanted includeFooFile to manipulate foo.php's changes further, you'd need to define a follow-up function and call that after the eval, or change the return value to include the follow-up code after the require_once. However, the latter approach may get tricky with nested quoting, so a heredoc is recommended. Double-quote style heredocs, shown below allow you to use variables, perhaps parameters passed to includeFooFile. Single-quote heredocs, like <<<'EOINCLUDE', would make quoting much easier if you know for sure you do not require variable interpolation.
function includeFooFile() {
return
<<<EOINCLUDE
require_once('foo.php');
\$foo++;
EOINCLUDE;
}
Prints 43.
If you wanted super simple, you could do away with the eval and simply return the name of the file to require_once, and then have the caller just require that:
function includeFooFile() {
return 'foo.php';
}
$foo=23;
require_once(includeFooFile());
echo($foo."\n"); # will print 42.
But then you cannot add any follow-up code, nor require_once two or more files within includeFooFile. However, you do eliminate the always-uncomfortable eval().
Note: if you were to have to further nest the call to includeFooFile itself inside another function, you would have to create a bubbling-up chain of these evals somehow. What I've demonstrated would not be adequate as is.
Ideally, PHP devs will eventually give us an optional parameter to require_once to specify we want it done in a global scope.
Upvotes: 0
Reputation: 283213
You can use this hacky function I wrote:
/**
* Extracts all global variables as references and includes the file.
* Useful for including legacy plugins.
*
* @param string $__filename__ File to include
* @param array $__vars__ Extra variables to extract into local scope
* @throws Exception
* @return void
*/
function GlobalInclude($__filename__, &$__vars__ = null) {
if(!is_file($__filename__)) throw new Exception('File ' . $__filename__ . ' does not exist');
extract($GLOBALS, EXTR_REFS | EXTR_SKIP);
if($__vars__ !== null) extract($__vars__, EXTR_REFS);
unset($__vars__);
include $__filename__;
unset($__filename__);
foreach(array_diff_key(get_defined_vars(), $GLOBALS) as $key => $val) {
$GLOBALS[$key] = $val;
}
}
It moves any newly defined vars back into global space when the include file returns. There's a caveat that if the included file includes another file, it won't be able to access any variables defined in the parent file via $GLOBALS
because they haven't been globalized yet.
Upvotes: 1
Reputation: 14808
Pending on your exact requirements, you could use constants. Require your file in the global scope, but inside it set a constant.
IE file.php:
define('MY_CONSTANT', 42);
Then anywhere in your script just use MY_CONSTANT
to refer to the value, you won't be able to edit once it's been set though. Other than that, you could globalize your variable as the other answer says, but it's not 100% clear what you're trying to achieve other than simply retrieving a value from the included file? In which case constants should be fine.
Update: Now you've explained that you want an objects properties to be available everywhere, I suggest you look into creating a static class, which once instantiated in your global scope can be used anywhere in your app. Read the linked manual page, it has a bare-bones example.
Upvotes: 1
Reputation: 88707
This is definitely not a "nice" work around but it would work:
function includeFooFile() {
require_once("foo.php");
foreach (get_defined_vars() as $key => $value) {
// Ignore superglobals
if (!in_array($key, array('GLOBALS','_SERVER','_GET','_POST','_FILES','_COOKIE','_SESSION','_REQUEST','_ENV'))) {
$GLOBALS[$key] = $value;
}
}
}
However, your included file cannot define any functions or classes (and possibly some other things as well that I cannot currently think of) because it will result in a parse error, since you cannot nest classes or functions.
EDIT apparently you can include functions in your file. I had always thought you couldn't but after testing it seems that you can.
Upvotes: 3
Reputation: 198204
As the scope is explicitly defined where you use require
and the like, you would need to specify what to do with the variables inside the scope of the function:
function includeFooFile() {
require_once("foo.php"); // scope of "foo.php" will be the function scope
foreach (get_defined_vars() as $k => $v)
{
$GLOBALS[$k] = &$v;
}
}
This example takes care of both, variables and references which might be what you're looking for. Demo. Please note that require_once
would only work once and would only define the variables once.
Upvotes: 2
Reputation: 2825
I haven't tried it (since using global vars is a bad idea tbh) but this could potentially work:
require_once '...';
$GLOBALS = array_merge($GLOBALS, get_defined_vars());
Alternatively you can just do it manually:
foreach (get_defined_vars() as $k => $v) {
$GLOBALS[$k] = $v;
}
Upvotes: 1
Reputation: 12244
Apart from "globalizing" your variable, there is no way to do this:
global $foo;
$foo = 42;
OR
$GLOBALS['foo'] = 42;
Then your value should be 42 when you print it out.
UPDATE
Regarding the inclusion of classes or functions, note that all functions and classes are always considered global unless we are talking about a class method. At that point, the method in a class is only available from the class definition itself and not as a global function.
Upvotes: 5
Reputation: 14792
You will need to declare global in your foo.php
:
<?php
global $foo;
$foo = 42;
?>
Otherwise it's probably not possible.
You could try to play around with extract()
, get_defined_vars()
, global
and $GLOBALS
in various combinations maybe... like iterating through all defined variables and calling global on them before requiring a file...
$vars = get_defined_vars();
foreach($vars as $varname => $value)
{
global $$varname; //$$ is no mistake here
}
require...
But i'm not quite sure if you get to where you want to go...
Upvotes: 3