4
.
├── gen.py
├── lexer
│   ├── engine.py
|   └── ...
└── parser
    ├── engine.py
    └── ...

我正在编写我的编译器项目,但现在我遇到了 python 导入冲突。在 gen.py 中,我想导入一些代码生成函数,比如

import lexer.engine   # OK
import parser.engine  # ModuleNotFoundError: No module named 'parser.engine'; 'parser' is not a package

经过一番调查,我了解到“解析器”是为 python 内部解析器保留的。但我无法更改目录名称“解析器”,因为它已在任何地方使用。

我该如何解决这个问题?

4

3 回答 3

2

要被接受为 Python 包,目录必须有一个__init__.py文件。
可以像包是常规模块一样访问此文件中声明的变量。它也可以是空的。

tl; dr:将一个空__init__.py文件添加到目录中,它应该可以工作。

于 2018-06-15T13:09:02.877 回答
1

假设您的项目实际上包含在这样的项目目录中,...

my_package
| 
├── gen.py
├── lexer
│   ├── __init__.py
│   ├── engine.py
|   └── ...
├── parser
|   ├── __init__.py
|   ├── engine.py
|   └── ...
├── tests  
|   ├── test_thingy.py

在 gen.py 中:

import my_package.lexer.engine
import my_package.parser.engine

在 my_package 的父目录中,您可以运行python -m my_package.gen. 这应该完全按预期运行,没有名称冲突。在您的测试中使用类似的导入语句,如果您以相同的方式运行测试模块,它应该可以正常工作。

我用以下方法对此进行了测试。在E/work/temp/我有一个名为my_package. 它具有以下结构。

my_package
    | 
    ├── __init__.py  # needed in python 2, but not 3
    ├── import_test_b.py
    ├── parser
    |   ├── __init__.py
    |   └── import_test_a.py
    └── tests  
        ├── __init__.py  # needed in python 2 but not python 3
        └── test_imports.py

进口测试a:

def test(num):
    return num+3

进口测试b:

from my_package.parser.import_test_a import tst

print(tst(4))

test_imports.py:

from my_package.parser.import_test_a import tst

import unittest

class TestTst(unittest.TestCase):
    def test_one(self):
        self.assertEqual(tst(4), 7)

if __name__ == '__main__':
    unittest.main()

E/work/temp

运行:python -m my_package.import_test_b-输出=7

运行:python -m my_package.tests.test_imports 输出:。

----------------------------------------------------------------------
Ran 1 test in 0.000s

OK
于 2018-06-12T16:09:37.510 回答
1

python有一个模块解析器,你想导入engine它没有的,如果你想尝试的话

import parser
print(dir(parser))
print(parser.__file__)

你会看到那个parser模块不是你的parser模块。

简单地重命名您的parser文件夹,一切都会好起来的

更新

您可以尝试使用from例如导入:

from parser import engine
于 2018-06-11T10:18:14.313 回答