Python包?

Jon*_*anb 9 python unit-testing packages

好吧,我认为无论我做错了什么,它可能都是显而易见的,但我无法弄清楚.我已阅读并重新阅读有关软件包的教程部分,我唯一可以想到的是,这不起作用,因为我正在直接执行它.这是目录设置:

eulerproject/
  __init__.py
  euler1.py
  euler2.py
  ...
  eulern.py
  tests/
    __init__.py
    testeulern.py
Run Code Online (Sandbox Code Playgroud)

以下是testeuler12.py(我编写的第一个测试模块)的内容:

import unittest
from .. import euler12

class Euler12UnitTests(unittest.TestCase):


    def testtriangle(self):
        """
        Ensure that the triangle number generator returns the first 10
        triangle numbers.

        """
        self.seq = [1,3,6,10,15,21,28,36,45,55]
        self.generator = euler12.trianglegenerator()
        self.results = []
        while len(self.results) != 10:
            self.results.append(self.generator.next())
        self.assertEqual(self.seq, self.results)

    def testdivisors(self):
        """
        Ensure that the divisors function can properly factor the number 28.

        """
        self.number = 28
        self.answer = [1,2,4,7,14,28]
        self.assertEqual(self.answer, euler12.divisors(self.number))


if __name__ == '__main__':

    unittest.main()
Run Code Online (Sandbox Code Playgroud)

现在,当我从IDLE和目录中的命令行执行此操作时,我收到以下错误:

Traceback (most recent call last):
  File "C:\Documents and Settings\jbennet\My Documents\Python\eulerproject\tests\testeuler12.py", line 2, in <module>
    from .. import euler12
ValueError: Attempted relative import in non-package
Run Code Online (Sandbox Code Playgroud)

我认为问题是,因为我直接运行它,我不能做相对导入(因为__name__更改,我对包描述的模糊理解是它__name__如何告诉它所在的包的一部分),但在那里case你们有什么建议如何导入从测试代码中存储1级的"生产"代码?

Vir*_*ras 10

我有同样的问题.我现在使用nose来运行我的测试,并正确处理相对导入.

是的,这整个相对重要的东西令人困惑.

  • 我接受另一个答案只是因为它让我做了我一直试图用相对进口做的事情.然而,我也在下载鼻子作为我的测试安全带,因为它看起来比我自己可以整合的任何东西都强大得多. (3认同)

Jas*_*n S 8

通常,您的PYTHONPATH上会有一个目录,其名称是您的包名.例如:

eulerproject/
    euler/
        __init__.py
        euler1.py
        ...
        tests/
            ...
    setup.py
Run Code Online (Sandbox Code Playgroud)

然后,您可以在系统范围内安装,也可以确保PYTHONPATH=/path/to/eulerproject/:$PYTHONPATH在调用脚本时进行设置.

这样的绝对导入将起作用:

from euler import euler1
Run Code Online (Sandbox Code Playgroud)

编辑:

根据Python文档,"用作Python应用程序主模块的模块应始终使用绝对导入." (引用)

因此nose,另一个答案提到的测试工具可以工作,因为它导入包而不是从命令行运行它们.

如果您想手动执行操作,则可运行脚本需要位于包层次结构之外,如下所示:

eulerproject/
    runtests.py
    euler/
        __init__.py
        euler1.py
        ...
        tests/
            __init__.py
           testeulern.py
Run Code Online (Sandbox Code Playgroud)

现在,runtests.py可以做from euler.tests.testeulern import TestCasetesteulern.py可以做from .. import euler1

  • 是的,但是你的项目不再利用相对进口了.关于相对导入的好处是它使你的包自包含.它不依赖于它的主文件夹名称,并且它不能错误地导入单元,例如,包含在PYTHONPATH中某处的过时版本的包. (2认同)