zoukankan      html  css  js  c++  java
  • PHPunit 测试私有方法

    This article is part of a series on testing untestable code:

    No, not those privates. If you need help with those, this book might help.

    One question I get over and over again when talking about Unit Testing is this:

    "How do I test the private attributes and methods of my objects?"

    Lets assume we have a class Foo:

    <?php
    class Foo
    {
        private $bar = 'baz';
     
        public function doSomething()
        {
            return $this->bar = $this->doSomethingPrivate();
        }
     
        private function doSomethingPrivate()
        {
            return 'blah';
        }
    }
    ?>

    Before we explore how protected and private attributes and methods can be tested directly, lets have a look at how they can be tested indirectly.

    The following test calls the testDoSomething() method which in turn calls thedoSomethingPrivate() method:

    <?php
    class FooTest extends PHPUnit_Framework_TestCase
    {
        /**
         * @covers Foo::doSomething
         * @covers Foo::doSomethingPrivate
         */
        public function testDoSomething()
        {
            $foo = new Foo;
            $this->assertEquals('blah', $foo->doSomething());
        }
    }
    ?>

    The test above assumes that testDoSomething() only works correctly whentestDoSomethingPrivate() works correctly. This means that we have indirectly testedtestDoSomethingPrivate(). The problem with this approach is that when the test fails we do not know directly where the root cause for the failure is. It could be in eithertestDoSomething() or testDoSomethingPrivate(). This makes the test less valuable.

    PHPUnit supports reading protected and private attributes through thePHPUnit_Framework_Assert::readAttribute() method. Convenience wrappers such asPHPUnit_Framework_TestCase::assertAttributeEquals() exist to express assertions onprotected and private attributes:

    <?php
    class FooTest extends PHPUnit_Framework_TestCase
    {
        public function testPrivateAttribute()
        {
            $this->assertAttributeEquals(
              'baz',  /* expected value */
              'bar',  /* attribute name */
              new Foo /* object         */
            );
        }
    }
    ?>

    PHP 5.3.2 introduces the ReflectionMethod::setAccessible() method to allow the invocation of protected and private methods through the Reflection API:

    <?php
    class FooTest extends PHPUnit_Framework_TestCase
    {
        /**
         * @covers Foo::doSomethingPrivate
         */
        public function testPrivateMethod()
        {
            $method = new ReflectionMethod(
              'Foo', 'doSomethingPrivate'
            );
     
            $method->setAccessible(TRUE);
     
            $this->assertEquals(
              'blah', $method->invoke(new Foo)
            );
        }
    }
    ?>

    In the test above we directly test testDoSomethingPrivate(). When it fails we immediately know where to look for the root cause.

    I agree with Dave Thomas and Andy Hunt, who write in their book "Pragmatic Unit Testing":

    "In general, you don't want to break any encapsulation for the sake of testing (or as Mom used to say, "don't expose your privates!"). Most of the time, you should be able to test a class by exercising its public methods. If there is significant functionality that is hidden behind private or protected access, that might be a warning sign that there's another class in there struggling to get out."

    So: Just because the testing of protected and private attributes and methods is possible does not mean that this is a "good thing".

  • 相关阅读:
    TransactionScop事务机制的使用
    MVC无刷新上传图片并显示
    WebClient和WebRequest获取html代码
    Web.config配置详解
    分类和扩展有什么区别?可以分别用来做什么?分类有哪些局限性?分类的结构体里面有哪些成员?
    有序二维数组的查找
    生成Excel.xlsx文件 iOS
    charles Https抓包
    https 通信流程和Charles 抓包原理
    fastlane自动化打包ipa并发布到firim或者蒲公英
  • 原文地址:https://www.cnblogs.com/liuguanghuiyes/p/2272429.html
Copyright © 2011-2022 走看看