Python __init__()方法

定义类的时候,若是添加__init__方法,那么在创建类的实例的时候,实例会自动调用这个方法,一般用来对实例的属性进行初使化。比如:
class testClass:
def init(self, name, gender):
#定义 __init__方法,这里有三个参数,这个self指的是一会创建类的实例的时候这个被创建的实例本身(例中的self),你也可以写成其他的东西,比如写成me也是可以的,这样的话下面的self.Name就要写成me.Name。 self.Name=name #通常会写成self.name=name,这里为了区分前后两个是不同的东东,把前面那个大写了,等号左边的那个Name(或name)是实例的属性,后面那个是方法__init__的参数,两个是不同的)

__init__还是有个特殊之处,那就是它不允许有返回值。如果你的__init__过于复杂有可能要提前结束的话,使用单独的return就好,不要带返回值。


class  testClass:         
    def __init__(self, name, gender): 
        self.Name=name
        self.Gender=gender #通常会写成	self.gender=gender 
        print('hello')    #这个print('hello')是为了说明在创建类的实例的时候,__init__方法就立马被调用了。
        
        
t1=testClass('aa','bb')
print(t1.Name)
print(t1.Gender)

##输出:
#hello
#aa
#bb

参考:

init_()方法意义重大的原因有两个。第一个原因是在对象生命周期中初始化是最重要的一步;每个对象必须正确初始化后才能正常工作。第二个原因是__init__()参数值可以有多种形式。

因为有很多种方式为__init__()提供参数值,对于对象创建有大量的用例,我们可以看看其中的几个。我们想尽可能的弄清楚,因此我们需要定义一个初始化来正确的描述问题区域。

在我们接触__init__()方法之前,无论如何,我们都需要粗略、简单地看看在Python中隐含的object类的层次结构。

在这一章,我们看看不同形式的简单对象的初始化(例如:打牌)。在这之后,我们还可以看看更复杂的对象,就像包含集合的hands对象以及包含策略和状态的players。
隐含的超类——object

每一个Python类都隐含了一个超类:object。它是一个非常简单的类定义,几乎不做任何事情。我们可以创建object的实例,但是我们不能用它做太多,因为许多特殊的方法容易抛出异常。

当我们自定义一个类,object则为超类。下面是一个类定义示例,它使用新的名称简单的继承了object:

class X:
pass
下面是和自定义类的一些交互:

X.class
<class ‘type’>

X.class.base
<class ‘object’>
我们可以看到该类是type类的一个对象,且它的基类为object。

就像在每个方法中看到的那样,我们也看看从object继承的默认行为。在某些情况下,超类特殊方法的行为是我们所想要的。在其他情况下,我们需要覆盖这个特殊方法。
基类对象的init()方法

对象生命周期的基础是创建、初始化和销毁。我们将创建和销毁的高级特殊方法推迟到后面的章节中,目前只关注初始化。

所有类的超类object,有一个默认包含pass的__init__()实现,我们不需要去实现__init__()。如果不实现它,则在对象创建后就不会创建实例变量。在某些情况下,这种默认行为是可以接受的。

我们总是给对象添加属性,该对象为基类object的子类。思考以下类,需要两个实例变量但不初始化它们:

class Rectangle:
def area(self):
return self.length * self.width
Rectangle类有一个使用两个属性来返回一个值的方法。这些属性没有初始化。这是合法的Python代码。它可以有效的避免专门设置属性,虽然感觉有点奇怪,但是有效。

下面是于Rectangle类的交互:

r = Rectangle()
r.length, r.width = 13, 8
r.area()
104
显然这是合法的,但也是容易混淆的根源,所以也是我们需要避免的原因。

无论如何,这个设计给予了很大的灵活性,这样有时候我们不用在__init__()方法中设置所有属性。至此我们走的很顺利。一个可选属性其实就是一个子类,只是没有真正的正式声明为子类。我们创建多态在某种程度上可能会引起混乱以及if语句的不恰当使用所造成的盘绕。虽然未初始化的属性可能是有用的,但很有可能是糟糕设计的前兆。

《Python之禅》中的建议:

"显式比隐式更好。"

一个__init__()方法应该让实例变量显式。

可怜的多态

灵活和愚蠢就在一念之间。

当我们觉得需要像下面这样写的时候,我们正从灵活的边缘走向愚蠢:

1
if ‘x’ in self.dict:
或者:

try:
self.x
except AttributeError:
是时候重新考虑API并添加一个通用的方法或属性。重构比添加if语句更明智。
在超类中实现init()

我们通过实现__init__()方法来初始化对象。当一个对象被创建,Python首先创建一个空对象,然后为那个新对象调用__init__()方法。这个方法函数通常用来创建对象的实例变量并执行任何其他一次性处理。

下面是Card类示例定义的层次结构。我们将定义Card超类和三个子类,这三个子类是Card的变种。两个实例变量直接由参数值设置,两个变量通过初始化方法计算:

class Card:
def init(self, rank, suit):
self.suit = suit
self.rank = rank
self.hard, self.soft = self._points()

class NumberCard(Card):
def _points(self):
return int(self.rank), int(self.rank)

class AceCard(Card):
def _points(self):
return 1, 11

class FaceCard(Card):
def points(self):
return 10, 10
在这个示例中,我们提取__init
_()方法到超类,这样在Card超类中的通用初始化可以适用于三个子类NumberCard、AceCard和FaceCard。

这是一种常见的多态设计。每一个子类都提供一个唯一的_points()方法实现。所有子类都有相同的签名:有相同的方法和属性。这三个子类的对象在一个应用程序中可以交替使用。

如果我们为花色使用简单的字符,我们可以创建Card实例,如下所示:

1
cards = [AceCard(‘A’, ‘?’), NumberCard(‘2’,’?’), NumberCard(‘3’,’?’),]
我们在列表中枚举出一些牌的类、牌值和花色。从长远来说,我们需要更智能的工厂函数来创建Card实例;用这个方法枚举52张牌无聊且容易出错。在我们接触工厂函数之前,我们看一些其他问题。
使用init()创建显式常量

可以给牌定义花色类。在二十一点中,花色无关紧要,简单的字符串就可以。

我们使用花色构造函数作为创建常量对象的示例。在许多情况下,我们应用中小部分对象可以通过常量集合来定义。小部分的静态对象可能是实现策略模式或状态模式的一部分。

在某些情况下,我们会有一个在初始化或配置文件中创建的常量对象池,或者我们可以基于命令行参数创建常量对象。我们会在第十六章《通过命令进行复制》中获取初始化设计和启动设计的详细信息。

Python没有简单正式的机制来定义一个不可变对象,我们将在第三章《属性访问、方法属性和描述符》看看保证不可变性的相关技术。在本示例中,花色不可变是有道理的。

下面这个类,我们将用于创建四个显式常量:
class Suit:
def init(self, name, symbol):
self.name= name
self.symbol= symbol
下面是通过这个类创建的常量:

1
Club, Diamond, Heart, Spade = Suit(‘Club’,’?’), Suit(‘Diamond’,’?’), Suit(‘Heart’,’?’), Suit(‘Spade’,’?’)
现在我们可以通过下面展示的代码片段创建cards:

1
cards = [AceCard(‘A’, Spade), NumberCard(‘2’, Spade), NumberCard(‘3’, Spade),]
这个小示例,这种方法对于单个特性的花色代码来说并不是一个巨大的进步。在更复杂的情况下,会有一些策略或状态对象通过这个方式创建。通过从小的、静态的常量对象中复用可以使策略或状态设计模式更有效率。

我们必须承认,在Python中这些对象并不是技术上一成不变的,它是可变的。进行额外的编码使得这些对象真正不变可能会有一些好处。

无关紧要的不变性

不变性很有吸引力但却容易带来麻烦。有时候被神话般的“恶意程序员”在他们的应用程序中通过修改常量值进行调整。从设计上考虑,这是非常愚蠢的。这些神话般的、恶意的程序员不会停止这样做,因为已经没有更好的方法去更简洁简单的在Python中编码。恶意程序员访问到源码并且修改它仅仅是希望尽可能轻松地编写代码来修改一个常数。

在定义不可变对象的类的时候最好不要挣扎太久。在第三章《属性访问、方法属性和描述符》中,我们将通过在有bug的程序中提供合适的诊断信息来展示如何实现不变性。

  • 12
    点赞
  • 72
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中,`__init__`方法是一个特殊的方法,用于在创建对象时进行初始化操作。该方法实例化时自动调用,并且通常用于设置对象的初始属性值。引用提供了一个`__init__`方法的定义格式示例: ``` class Studeng: def __init__(self, name, score): self.name = name self.score = score ``` 该示例中,`__init__`方法接受两个参数`name`和`score`,并将其赋值给对象的属性`self.name`和`self.score`。通过在实例化时传入参数,可以在实例化对象时直接初始化对象的属性值。 另外,`__init__`方法还支持多态的特性,即不同的子可以通过定义自己的`__init__`方法来覆盖父的`__init__`方法,以实现不同的初始化行为。引用提供了一个示例: ``` class Person: def __init__(self, name, age): self.name = name self.age = age class Student(Person): def __init__(self, name, age, major): Person.__init__(self, name, age) self.major = major ``` 在该示例中,`Student`继承自`Person`,并定义了自己的`__init__`方法。在`Student`的`__init__`方法中,首先调用父`Person`的`__init__`方法,然后再新增一个属性`self.major`。通过这种方式,`Student`可以在初始化时同时设置`name`、`age`和`major`三个属性。 综上所述,`__init__`方法Python中用于对象的初始化操作,可以设置对象的初始属性值,支持多态特性,允许子覆盖父的`__init__`方法来实现不同的初始化行为。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [python:__init__( )方法](https://blog.csdn.net/qq_39314932/article/details/80721822)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [【知识点】python的__init__详解](https://blog.csdn.net/weixin_39888010/article/details/129012009)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值