-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
93 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
# Imports | ||
|
||
Jou's import system is super simple. | ||
There are two kinds of imports: | ||
- **Imports starting with a dot:** e.g. `import "./foo.jou"`, `import "../foo.jou"` or `import "./foo/bar.jou"` | ||
- **Imports starting with `stdlib`:** e.g. `import "stdlib/foo.jou"` or `import "stdlib/foo/bar.jou"` | ||
|
||
Any other kind of import is a compiler error. | ||
|
||
Imports starting with a dot are resolved relative to where the Jou file is. | ||
For example, if file `/home/akuli/folder/foo.jou` contains `import "./bar/baz.jou"`, | ||
that means `/home/akuli/folder/bar/baz.jou`. | ||
As usual, `..` goes up to the parent directory, | ||
so `import "../test.jou"` would be `/home/akuli/test.jou`. | ||
|
||
Imports starting with `stdlib` are resolved relative to where the Jou compiler is. | ||
You usually don't need to care about how exactly the compiler finds the `stdlib` folder. | ||
Look at [compiler/paths.jou](../compiler/paths.jou) | ||
or create an issue on GitHub if you want to know more about this. | ||
|
||
Jou's `import` does slightly more than copy/pasting file contents (like C's `#include`), | ||
but not very much more. | ||
Practically, here's what you need to know: | ||
- The same file can be imported multiple times by different files, | ||
and it will be compiled just once. | ||
For example, if `bar.jou` imports `foo.jou`, and `baz.jou` imports `foo.jou`, | ||
the compiler will compile `foo.jou` only once. | ||
- Importing does not affect other files, only the file that contains the import. | ||
This means that if `utils.jou` defines `a_really_useful_function()`, | ||
then you need to `import "./utils.jou"` in all files that use `a_really_useful_function()`. | ||
|
||
|
||
## Conflicting Names | ||
|
||
TODO: This will need to be updated once [#84](https://github.com/Akuli/jou/issues/84) is implemented. | ||
|
||
**You cannot have multiple public things with the same name, even if they are in different Jou files.** | ||
|
||
This means that if `bar.jou` defines a function `foo()` | ||
and `baz.jou` also defines a function `foo()`, | ||
then you cannot use `bar.jou` and `baz.jou` in the same project. | ||
To work around this, rename the functions. | ||
|
||
The same applies to basically anything public (that is, accessible from other files). | ||
However, it's fine to have multiple methods with the same name in different classes | ||
as long as the class names are different. | ||
For example, [the `ast.jou` file in the Jou compiler](../compiler/ast.jou) | ||
has many classes whose name starts with `Ast` to prevent conflicts with other files, | ||
and most of the classes have a method named `free()`. | ||
|
||
Let me explain why Jou has this limitation. | ||
When the Jou compiler has compiled all Jou files individually, | ||
it runs the **linker**, which is a program that combines them into an executable file | ||
(e.g. `jou_compiled/my_program/my_program.exe`). | ||
The linker complains if multiple functions have the same name, | ||
because it doesn't know which function to use. | ||
|
||
You might be thinking that other low-level languages allow having multiple functions with the same name. | ||
In reality, they just rename the functions automatically. | ||
For example, consider the following C++ code. | ||
It has two functions named `foo`, | ||
but the code compiles because they are in different C++ namespaces. | ||
|
||
```c++ | ||
#include <iostream> | ||
|
||
namespace bar { | ||
void foo() { | ||
std::cout << "foo from bar namespace" << std::endl; | ||
} | ||
} | ||
|
||
namespace baz { | ||
void foo() { | ||
std::cout << "foo from baz namespace" << std::endl; | ||
} | ||
} | ||
``` | ||
I compiled this on Linux with a C++ compiler, and the compiler created the following functions: | ||
- `_ZN3bar3fooEv` | ||
- `_ZN3baz3fooEv` | ||
Jou is a simple language that avoids surprising and "magical" things. | ||
If you want two differently named functions in Jou, then **you** need to name them differently: | ||
a Jou function named `foo()` is actually named `foo()`. | ||
Methods are entirely a Jou concept, and they become plain old functions when the Jou code is compiled. | ||
If you have a Jou method `bark()` in Jou class `Dog`, | ||
the compiler generates a function named `Dog.bark`. | ||
This means that debugging tools (e.g. [valgrind](doc.md#crashing-and-valgrind)) | ||
show `Dog.bark` as the function name when they tell you something about the Jou method. |