Python中的私有构造函数

     2023-02-25     24

关键词:

【中文标题】Python中的私有构造函数【英文标题】:Private Constructor in Python 【发布时间】:2012-01-02 23:11:18 【问题描述】:

如何创建一个只能由类的静态函数调用而不是从其他地方调用的私有构造函数?

【问题讨论】:

Python 中没有私有与公共方法的概念。 Python is not Java. @Tadeck,请停止错误地纠正人们。修改名称不会使其私有化。 @DanielRoseman:larsmans 说“Python 中没有私有与公共方法的概念”。我说他错了。 Documentation 说:“由于类私有成员(...)有一个有效的用例,因此对这种机制的支持有限,称为名称修改。”。请不要在没有证明你是对的情况下争论。 @BasicWolf:请先完整阅读我的评论。名称修改为您提供了私有类的(部分)概念。请不要说在ClassName 之外调用obj._ClassName__Method() 是正确的行为。 【参考方案1】:

首先,术语“构造函数”不适用于 Python,因为虽然__init__() 方法扮演了一个角色,但它只是一个在对象已经创建并需要初始化时调用的方法。

Python 中类的每个方法都是公共的。通常,程序员在方法名称中使用___ 标记“私有”方法,例如:

# inheriting from object is relevant for Python 2.x only
class MyClass(object): 
    # kinda "constructor"
    def __init__(self):
        pass

    # here is a "private" method
    def _some_method(self):
        pass

    # ... and a public one
    def another_method(self):
        pass

【讨论】:

@Tadeck,这不是真的,BasicWolf 是正确的。以双下划线为前缀的方法不是私有的 - 它们是名称混乱的,但仍然可以从类外部访问它们。 @Tadeck,试试这个:用__my() 方法创建一个类MyClass。考虑mc_obj = MyClass()。您将能够通过mc_obj._MyClass__my() 访问__my() @BasicWolf:我知道你可以做到,但这并不意味着你应该这样做,如果你想遵循 Python 的规则的话。如果您不在乎,您也可以成为eval() 的粉丝,或者使用vars()['a'] = 10 等作业代替a = 10 @BasicWolf:你说“用单个下划线标记方法意味着完全相同”,我完全不同意。请阅读 gimel 的答案并自己检查,这样您就知道以单下划线和双下划线开头的方法之间的区别。 @BasicWolf:请不要说这完全一样或证明是一样的。你最后的评论是题外话。当我想要它们提供的功能时,我会使用 __ 方法。我想指出的是,以_ 开头的方法名称与__ 不同,如果您还没有注意到,请参阅@gimel 或@mac 的答案。【参考方案2】:

引用Python style guide (PEP 8):

此外,以下特殊形式使用前导或尾随 下划线被识别(这些通常可以与任何大小写组合 约定):

_single_leading_underscore:弱“内部使用”指标。例如。 “from M import *”不导入名称以下划线开头的对象。

single_trailing_underscore_:按惯例使用,以避免与 Python 关键字,例如 Tkinter.Toplevel(master, class_='ClassName')

__double_leading_underscore:命名类属性时,调用名称 mangling(在 FooBar 类中,__boo 变为 _FooBar__boo;见下文)。

__double_leading_and_trailing_underscore__:“魔法”对象或 存在于用户控制的命名空间中的属性。例如。 __init__, __import____file__。永远不要发明这样的名字;只使用它们 记录在案。

【讨论】:

我不明白这是如何回答这个问题的。它没有说明构造函数。 @larsmans - 通过样式指南回答“私人”和“未从其他地方调用”。 但是构造函数有固定的名字,所以你不能按照惯例只附加一个下划线。【参考方案3】:

如何创建私有构造函数?

从本质上讲,不可能既是因为 python 不会像您使用其他 OOP 语言时所认为的那样使用构造函数,而且因为 python 不会强制隐私,它只是有一个特定的语法来建议给定的方法/属性应该被认为是私有的。让我详细说明...

首先:你可以在 python 中找到最接近构造函数的是__new__ method,但这很少使用(你通常使用__init__,它修改了刚刚创建的对象(实际上它已经有@987654324 @ 作为第一个参数)。

无论如何,python 是基于每个人都是同意的成年人的假设,因此私有/公共不像其他语言那样强制执行。

正如其他响应者所提到的,“私有”的方法通常在前面加上一个或两个下划线:_private__private。两者的区别在于后者会打乱方法的名称,所以你将无法从对象实例化之外调用它,而前者则不会。

例如,如果您的班级 A 同时定义了 _private(self)__private(self)

>>> a = A()
>>> a._private()   # will work
>>> a.__private()  # will raise an exception

您通常希望使用单下划线,因为 - 特别是对于单元测试 - 使用双下划线会使事情变得非常棘手......

HTH!

【讨论】:

+1 那是正确的答案! :) 其实你可以从类外调用私有方法,但你不应该这样做,所以你的答案是可以的。谢谢。 问题是关于私有构造函数,而不是私有成员函数。你只是在解释如何拥有私有函数。 @Syncopated - 正如该问题的所有答案和 cmets 所指出的,Python 中没有“私有构造函数”之类的东西 @mac 如果是这样,那么你的答案应该是“不可能。”,而不是你写的,因为问题显然是关于构造函数,而不是函数。或者,如果您能想出一种达到预期效果的方法,那就是答案。 整个“同意的成年人”模因需要消亡。存在私有修饰符的原因是因为对象旨在成为一种状态机。在 OOP 中,方法和成员是紧密耦合的。创建私有成员的原因是因为在公共接口之外更改它可能会产生不明显的副作用,导致难以追踪错误。【参考方案4】:

目前还没有人提到这一点——您可以相当程度地控制哪些名称在哪些范围内可见——并且有很多个范围可用。以下是两个 三个 将类的构造限制为工厂方法的其他方法:

#Define the class within the factory method
def factory():
  class Foo:
    pass
  return Foo()

#Assign the class as an attribute of the factory method
def factory():
  return factory.Foo()
class Foo:
  pass
factory.Foo = Foo
del Foo

(注意:这仍然允许从外部引用该类(例如,用于isinstance 检查),但很明显您不应该直接实例化它。)

#Assign the class to a local variable of an outer function
class Foo:
  pass
def factory_maker():
  inner_Foo=Foo
  def factory():
    return inner_Foo()
  return factory
factory = factory_maker()
del Foo
del factory_maker

这使得 不可能(至少,不使用至少一个魔法(双下划线)属性)访问 Foo 类,但仍然允许多个函数使用它(通过定义它们在删除全局 Foo 名称之前。

【讨论】:

+1。我想补充一点,在这种情况下,isinstance 检查将是不可能的,因为该类不可访问(或者至少不应该被访问)。这可以通过让类从可见的抽象基类继承来解决。 第二个选项允许 isinstance 检查,将类称为 factory.Foo 没错,但我认为客户端代码不允许从外部查看课程? 我打算让它更容易调用工厂而不是直接实例化类,而不是让它不可能(嗯,主要是) 直接实例化(为此,使用第一种形式)。 太棒了!感谢分享!这应该是公认的答案!【参考方案5】:
class MongoConn(object):
    @classmethod
    def instance(cls):
        if not hasattr(cls, '_instance'):
            cls._instance = cls()
        return cls._instance

    def __init__(self):
        assert not hasattr(self.__class__, '_instance'), 'Do not call constructor directly!'

如果你想要一个实例。

【讨论】:

我可以打电话给MongoConn()这里没问题【参考方案6】:

___ 前缀没有提供将对象实例化限制为特定“工厂”的解决方案,但是 Python 是一个强大的工具箱,并且可以通过多种方式实现所需的行为(正如 Z 的@Jesse W 所证明的那样)。 这是一个可能的解决方案,可以让类公开可见(允许isinstance 等),但确保只能通过类方法进行构造:

class OnlyCreatable(object):

    __create_key = object()

    @classmethod
    def create(cls, value):
        return OnlyCreatable(cls.__create_key, value)

    def __init__(self, create_key, value):
        assert(create_key == OnlyCreatable.__create_key), \
            "OnlyCreatable objects must be created using OnlyCreatable.create"
        self.value = value

create类方法构造一个对象:

>>> OnlyCreatable.create("I'm a test") 
<__main__.OnlyCreatable object at 0x1023a6f60>

当尝试在不使用create 类方法的情况下构造对象时,由于断言失败:

>>> OnlyCreatable(0, "I'm a test")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 11, in __init__
AssertionError: OnlyCreatable objects can only be created using OnlyCreatable.create

如果试图通过模仿 create 类方法来创建对象 由于OnlyCreatable.__createKey 的编译器错误,创建失败。

>>> OnlyCreatable(OnlyCreatable.__createKey, "I'm a test")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: type object 'OnlyCreatable' has no attribute '__createKey'

在类方法之外构造OnlyCreatable 的唯一方法是知道OnlyCreatable.__create_key 的值。由于此类属性的值是在运行时生成的,并且它的名称以 __ 为前缀,将其标记为不可访问,因此实际上“不可能”获取该值和/或构造对象。

【讨论】:

不是不能访问,是OnlyCreatable._OnlyCreatable__create_key。或者,您可以使用 -O 标志或 PYTHONOPTIMIZE env var 禁用断言来运行 Python。 这与 python 拥有“强大的工具箱”无关。任何语言都可以实现这一点。在任何真正的 OOP 中,它被认为是一种可怕的做法的原因是,在您实际运行代码之前,无法识别实例化一个抛出异常的类。因此,在这个用例中,最小惊讶原则被严重破坏了。【参考方案7】:

如果模块级函数可以代替静态方法,请参阅我的other answer

您可以使用抽象类来实现这种效果。任何需要在“私有构造函数”中定义的实例属性都可以是抽象属性。然后,您的工厂类方法通过填充这些抽象属性以及执行任何其他初始化工作(例如数据验证)来构建自己的具体类。

from abc import ABC, abstractmethod

class Foo(ABC):
    @property
    @abstractmethod
    def _a(self) -> int:
        pass

    def bar(self) -> int:
        return self._a + 1

    @classmethod
    def from_values(cls, a: int) -> 'Foo':
        class _Foo(cls):
            def __init__(self, __a):
                self.__a = __a

            @property
            def _a(self):
                return self.__a
        
        return _Foo(a)

Foo()  # TypeError: Can't instantiate abstract class ...
Foo.from_values(1).bar()  # 1

如果您发现Foo 上不需要抽象属性,则在调用Foo() 时将不会得到TypeError。在这种情况下,您可以依赖 ABC 的继承作为文档,或者定义一个虚拟属性以确保安全。

可选调整

需要可变实例属性?添加设置器。 不关心类和实例属性的区别?简化为
class _Foo(cls):
    _a = a

return _Foo()

【讨论】:

【参考方案8】:

虽然 Python 中不存在严格的私有属性,但您可以使用元类来防止使用 MyClass() 语法来创建 MyClass 对象。

这是一个改编自 Trio 项目的示例:

from typing import Type, Any, TypeVar


T = TypeVar("T")


class NoPublicConstructor(type):
    """Metaclass that ensures a private constructor

    If a class uses this metaclass like this:

        class SomeClass(metaclass=NoPublicConstructor):
            pass

    If you try to instantiate your class (`SomeClass()`),
    a `TypeError` will be thrown.
    """

    def __call__(cls, *args, **kwargs):
        raise TypeError(
            f"cls.__module__.cls.__qualname__ has no public constructor"
        )

    def _create(cls: Type[T], *args: Any, **kwargs: Any) -> T:
        return super().__call__(*args, **kwargs)  # type: ignore

这是一个使用示例:

from math import cos, sin


class Point(metaclass=NoPublicConstructor):
     def __init__(self, x, y):
         self.x = x
         self.y = y

     @classmethod
     def from_cartesian(cls, x, y):
         return cls._create(x, y)
     
     @classmethod
     def from_polar(cls, rho, phi):
         return cls._create(rho * cos(phi), rho * sin(phi))

Point(1, 2) # raises a type error
Point.from_cartesian(1, 2) # OK
Point.from_polar(1, 2) # OK

【讨论】:

这个类的消费者在运行代码之前怎么知道不存在公共构造函数? @Tarynn 添加我猜的文档字符串【参考方案9】:

其他答案提到 Python 中没有“私有”构造函数(或一般的访问修饰符)之类的东西。虽然这是真的,但您仍然可以利用 __ 前缀行为通过在具有双下划线前缀的嵌套类中“隐藏”实际实现来实现相同的最终效果:

class MyPublicClassWithPrivateConstructor:

    class __Implementation:
        def __init__(self, *args, **kwargs):
            # do the things
            self.args = args
            self.kwargs = kwargs

    @classmethod
    def make_my_thing(cls, *args, **kwargs)
        return cls.__Implementation(*args, **kwargs)

    def __new__(cls, *args, **kwargs):
        raise Exception("MyPublicClassWithPrivateConstructor cannot be initialized directly, use MyPublicClassWithPrivateConstructor.make_my_thing")

# elsewhere
from whateva import MyPublicClassWithPrivateConstructor

MyPublicClassWithPrivateConstructor.__Implementation  # type object 'MyPublicClassWithPrivateConstructor' has no attribute '__Implementation'
MyPublicClassWithPrivateConstructor.make_my_thing  # <bound method MyPublicClassWithPrivateConstructor.make_my_thing of <class 'whateva. MyPublicClassWithPrivateConstructor'>>
           

【讨论】:

【参考方案10】:

如果模块级函数可以代替静态方法...

将整个类设为私有,将 API 公开为抽象类,并使用函数实例化您的私有类

class Foo(ABC):
    @abstractmethod
    def bar(self) -> int:
        ...


class _SpecificFoo(Foo):
    def __init__(self, bar: int):
        self._bar = bar

    def bar(self) -> int:
        return self._bar


def specific_foo(bar: int) -> Foo:
    return _SpecificFoo(bar)

注意

_SpecificFoo.__init__specific_foo 可以有不同的签名 specific_foo 返回 Foo。 API 没有提及_SpecificFoo

这与 Jesse 的第一个选项非常相似,但不会在每次调用 specific_foo 时重新定义类

【讨论】:

从类型注解的角度来看,这是一个很好的解决方案。我们仍然可以在类型注释中使用Foo,但_SpecificFoo 的受保护状态将强烈鼓励我们使用specific_foo 来建立新实例。如果我们尝试使用构造函数,这也比实际抛出错误的解决方案更 Pythonic。最好使用下划线来阻止它。我很惊讶这没有更多的赞成票。【参考方案11】:

如果您只是想要一种简单的方式来表明不应直接使用构造函数,则可以按照以下模式使构造函数的默认用法引发异常:

class Foo:
    def __init__(self, arg, __private=False):
        assert __private, "Constructor shouldn't be called directly"
        self.arg = arg

    @classmethod
    def from_bar(cls, bar: Bar):
        return cls(bar.arg, __private=True)

    @classmethod
    def from_baz(cls, baz: Baz):
         return cls(baz.arg, __private=True)

当然,任何人仍然可以使用Foo('abc', __private=True),但它会发出一个明确的信号,即不支持遵循 Python 的一般设计的这种行为。

【讨论】:

请注意,当然在某些配置中(即PYTHONOPTIMIZE=TRUE)Python 会跳过检查断言。如果在这种情况下您仍需要此检查,您可以考虑将assert 更改为if not __private: raise RuntimeError("...")【参考方案12】:

我知道,我迟到了,但我们可以做这样的事情来创建私有构造函数。

class MyClass:
    __instance = None
    __private = False

    @staticmethod
    def get_instance():

        MyClass.__private = True
        __instance = MyClass()
        MyClass.__private = False
        return MyClass.__instance
     
    def __init__(self):

        if MyClass.__private == False:
             raise Exception("Its a private method. Please use get_instance method")
        

【讨论】:

这对单身人士来说非常棒。其余的用例呢? @Tarynn 对于正常用例,只需删除 if not MyClass.__instance: 条件并将所有内容置于条件之外,这样就可以了。让我为一般用例编辑它。

我们啥时候需要 C++ 中的私有构造函数?

】我们啥时候需要C++中的私有构造函数?【英文标题】:WhendoweneedaprivateconstructorinC++?我们什么时候需要C++中的私有构造函数?【发布时间】:2011-09-2722:48:06【问题描述】:我对C++中的私有构造函数有疑问。如果构造函数是私有... 查看详情

Kotlin 中的私有构造函数

】Kotlin中的私有构造函数【英文标题】:PrivateconstructorinKotlin【发布时间】:2018-04-0113:04:11【问题描述】:在Java中,可以隐藏一个类的主构造函数,方法是将其设为private,然后通过该类中的publicstatic方法访问它:publicfinalclassFoo/... 查看详情

抽象类中的私有构造函数

】抽象类中的私有构造函数【英文标题】:Privateconstructorinabstractclass【发布时间】:2012-07-2211:05:46【问题描述】:在Java中,在抽象类中使用私有构造函数的目的是什么?在评论中我得到了这个问题,我很好奇,在什么情况下我... 查看详情

Java中的构造函数可以是私有的吗?

】Java中的构造函数可以是私有的吗?【英文标题】:CanaconstructorbeprivateinJava[duplicate]【发布时间】:2011-08-1611:01:53【问题描述】:可能的重复:QuestionaboutconstructorsinJavaCanaconstructorinJavabeprivate?Java中的构造函数可以是私有的吗?我... 查看详情

私有类语法中的复制构造函数

】私有类语法中的复制构造函数【英文标题】:CopyconstructorinprivateclassSyntax【发布时间】:2015-04-3011:35:00【问题描述】:2小时后我将参加C++考试,但我仍然想知道复制构造函数的语法。到目前为止,我所理解的是,您将CopyConstruc... 查看详情

如何限制开发人员使用反射访问Java中的私有方法和构造函数?

】如何限制开发人员使用反射访问Java中的私有方法和构造函数?【英文标题】:HowtorestrictdeveloperstousereflectiontoaccessprivatemethodsandconstructorsinJava?【发布时间】:2011-11-2520:50:00【问题描述】:如何限制开发者使用反射访问Java中的私... 查看详情

内部类中的私有构造函数在外部类中初始化

】内部类中的私有构造函数在外部类中初始化【英文标题】:Privateconstrutorininnerclassinitializedwithinouterclass【发布时间】:2014-12-1209:07:27【问题描述】:是否可以初始化内部类的实例,该实例具有从外部类中标记为私有的构造函数... 查看详情

javajava中的单元测试私有构造函数(代码片段)

查看详情

构造函数可以访问 PHP 中的父私有属性吗?

】构造函数可以访问PHP中的父私有属性吗?【英文标题】:ConstructorcanaccessparentprivatepropertiesinPHP?【发布时间】:2014-12-1020:29:16【问题描述】:在处理PHP中的继承时,我发现缺乏一些知识,主要是关于constructors和private属性。我们... 查看详情

继承公共/受保护/私有构造函数

...*...*/;classDerived:publicBasepublic:usingBase::Base;将强制继承Derived中的所有Base构造函数。但是公共/受保护/私有构造函数呢?classB 查看详情

TypeScript 构造函数中的私有变量声明以诱导 DI

】TypeScript构造函数中的私有变量声明以诱导DI【英文标题】:privatevariabledeclarationinTypeScriptconstructortoinduceDI【发布时间】:2017-11-1301:50:42【问题描述】:为了在Angular2中使用TypeScript引入依赖注入,我们使用如下代码:constructor(priv... 查看详情

Java中私有构造函数有啥用?

...013-06-2422:26:04【问题描述】:我想知道私有构造函数在Java中的用途。在Java中使用私有构造函数有哪些不同的方法?【问题讨论】:当你不想让别人创建你的类的实例时(至少通过调用构造函数)。可能重复-***.com/questions/2816123 查看详情

访问从基类构造函数调用的重写方法中的私有方法

】访问从基类构造函数调用的重写方法中的私有方法【英文标题】:Accessprivatemethodinanoverridenmethodcalledfromthebaseclassconstructor【发布时间】:2020-07-2822:04:48【问题描述】:考虑以下代码:classMyBaseconstructor(b)this.myOverrideMethod(b);myOverri... 查看详情

为啥我可以在复制构造函数中访问私有变量?

...描述】:我了解到我永远无法访问私有变量,只能使用类中的get函数。但是为什么我可以在复制构造函数中访问它呢?例子:Field::Field(constField&f)pFirst 查看详情

Scala案例类私有构造函数但公共应用方法

...下带有私有构造函数的案例类,并且我无法访问伴随对象中的应用方法。caseclassMeterprivate(m:Int)valm=Meter(10)//constructorMeterinclassMet 查看详情

javascript之构造函数

js中构造函数://构造函数functionDog(sex){  //公有属性this.sex=sex;this.name=‘blog‘;  //私有属性 varsound=‘汪汪汪‘;varthat=this;  //私有方法functionshowName(){console.log(that.name);}showName();}/**向构造函数添加公有属性/公有方法*需要用... 查看详情

java构造函数私有化

构造函数私有化的意义在于声明多个对象却只占用一个内存。写一个类classSY{//构造函数私有privatestaticSYinstance=newSY();//通过私有的静态的方式来声明一个属性,这个属性的名字是instance,类型是SY,指向一个新生成的内存privateSY(){... 查看详情

为啥我们需要私有构造函数?

】为啥我们需要私有构造函数?【英文标题】:Whydoweneedaprivateconstructor?为什么我们需要私有构造函数?【发布时间】:2011-02-0420:42:51【问题描述】:如果一个类有一个私有构造函数,那么它就不能被实例化。因此,如果我不希... 查看详情