百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

python静态方法和类方法之内置函数

off999 2024-09-20 22:49 34 浏览 0 评论

python类方法分为实例方法、类方法、静态方法。

(1) 实例方法,不用修饰,第1个参数为实例对象,默认为self。

通过实例调用时,自动将当前实例传给self;

通过类调用时,需要显式将实例传给self。

(2) 类方法,用@classmethod修饰,第1个参数为类对象,默认为cls。

也可以通过内置函数classmethod(cmeth)将cmeth转为类方法。

通过实例调用时,自动将当前类传递给第1个参数;

通过类调用时,自动将当前类传递给第1个参数。

(3) 静态方法,用@staticmethod修饰,第1个参数不需要默认,无self和cls。

也可以通过内置函数staticmethod(smeth)将smeth转为静态方法。

通过实例调用时,不会自动将当前实例传给第1个参数。

通过类调用时,不需要传送实例给第1个参数。


python2.2版本新增类方法和静态方法,对经典类有效,对新式类无效。


1.1 python类方法

python类方法通过@classmethod修饰,或通过内置函数classmethod()转换。类方法第1个参数为类对象,默认为cls。通过实例调用时,自动将当前类传递给第1个参数,通过类调用时,自动将当前类传递给第1个参数。

类方法适合处理每个类中不同的数据,通过第1个入参cls完成。

1.2 python静态方法

python静态方法通过@staticmethod修饰,或通过内置函数staticmehod()转换。静态方法入参无self和cls。通过实例调用时,不会自动将当前实例传给第1个参数,通过类调用时,不需要显式传递实例给第1个参数。


python静态方法用于处理与类而不是与实例相关的数据。

比如,记录类创建的实例数。

把计数器作为类属性,每次创建实例对象时,构造函数对计数器加1.

类属性是所有实例共享的,可以被所有实例使用。

1.2.1 类内未使用静态方法的无参方法

描述

python2.x和3.x,类的方法未定义第1个入参,通过类和实例调用结果不同。

 class NoStaticMed:
     def printNumOfIns():
         pass


NO

调用方式

调用举例

python2.x

python3.x

1

类调用

NoStaticMed.printNumOfIns()

报错

成功

2

实例调用

NoStaticMed ().printNumOfIns()

报错

报错

示例

staticmedcls.py

 # coding:utf-8
 import sys
 print('python版本为:python{}'.format(sys.version.split(' ')[0]))
 class NoStaticMed:
     numOfInstances=0
     def __init__(self):
         NoStaticMed.numOfInstances+=1
     def printNumOfIns():
         print('创建的实例数为:{}'.format(NoStaticMed.numOfInstances))

python2.x在idle执行结果

 >>> import os
 >>> os.chdir(r'E:\documents\F盘')
 >>> from staticmedcls import NoStaticMed
 python版本为:python2.7.18
 >>> sm1=NoStaticMed()
 >>> sm2=NoStaticMed()
 >>> sm3=NoStaticMed()
 >>> NoStaticMed.printNumOfIns()
 # python 2.x 通过类调用无入参类方法,报 无绑定方法 必须传实例作为第1个入参。
 Traceback (most recent call last):
   File "<pyshell#6>", line 1, in <module>
     NoStaticMed.printNumOfIns()
 TypeError: unbound method printNumOfIns() must be called with NoStaticMed instance as first argument (got nothing instead)
 >>> sm1.printNumOfIns()
 # python 2.x 通过实例调用无入参类方法,报 收到1个入参。即会自动传入一个实例。
 Traceback (most recent call last):
   File "<pyshell#7>", line 1, in <module>
     sm1.printNumOfIns()
 TypeError: printNumOfIns() takes no arguments (1 given)

python3.x在idle执行结果

 >>> import os
 >>> os.chdir(r'E:\documents\F盘')
 >>> from staticmedcls import NoStaticMed
 python版本为:python3.7.8
 >>> sm1=NoStaticMed()
 >>> sm2=NoStaticMed()
 >>> sm3=NoStaticMed()
 # python 3.x 通过类调用无入参类方法,成功。
 >>> NoStaticMed.printNumOfIns()
 创建的实例数为:3
 >>> sm1.printNumOfIns()
 # python 3.x 通过实例调用无入参类方法,报 收到1个入参。即会自动传入一个实例。
 Traceback (most recent call last):
   File "<pyshell#7>", line 1, in <module>
     sm1.printNumOfIns()
 TypeError: printNumOfIns() takes 0 positional arguments but 1 was given

1.2.2 类外无参方法

描述

在类外定义一个函数,用于统计类创建的实例数量。

示例

 # coding:utf-8
 import sys
 print('python版本为:python{}'.format(sys.version.split(' ')[0]))
 class OutClassMed:
     numOfInstances=0
     def __init__(self):
         OutClassMed.numOfInstances+=1
 def printNumOfIns():
     print('从 OutClassMed 创建的实例数为:{}'.format(OutClassMed.numOfInstances))
 

python2.x在idle执行结果

 >>> import os;os.chdir(r'E:\documents\F盘')
 >>> from staticmedcls import OutClassMed,printNumOfIns
 python版本为:python2.7.18
 >>> ocm1,ocm2,ocm3=OutClassMed(),OutClassMed(),OutClassMed()
 >>> printNumOfIns()
 从 OutClassMed 创建的实例数为:3

python3.x在idle执行结果

 >>> import os;os.chdir(r'E:\documents\F盘')
 >>> from staticmedcls import OutClassMed,printNumOfIns
 python版本为:python3.7.8
 >>> ocm1=OutClassMed();ocm2=OutClassMed();ocm3=OutClassMed()
 >>> printNumOfIns()
 从 OutClassMed 创建的实例数为:3

1.2.3 内置函数staticmethod和classmethod

描述

使用内置函数staticmethod()转为静态方法,

使用内置函数classmethod()转为类方法。

示例

Python2.x在idle执行结果

 >>> import sys
 >>> print('python版本为:python{}'.format(sys.version.split(' ')[0]))
 python版本为:python2.7.15
 >>> class BuiltInSCMed:
     def instanceMed(self,x):
         print(self,x)
     def staticMed(x):
         print(x)
     def clsMed(cls,x):
         print(cls,x)
     # 通过内置函数 staticmethod 将 staticMed 转为静态方法
     staticMed=staticmethod(staticMed)
     # 通过内置函数 classmethod 将 clsMed 转为类方法
     staticMed=classmethod(clsMed)
 >>> biscm1=BuiltInSCMed()
 # 通过实例调用实例方法
 >>> biscm1.instanceMed(1)
 (<__main__.BuiltInSCMed instance at 0x03B71620>, 1)
 # 通过类调用实例方法
 >>> BuiltInSCMed.instanceMed(biscm1,2)
 (<__main__.BuiltInSCMed instance at 0x03B71620>, 2)
 # 通过类调用静态方法
 >>> BuiltInSCMed.staticMed(3)
 3
 # 通过实例调用静态方法
 >>> biscm1.staticMed('梯阅线条')
 梯阅线条
 # 通过类调用类方法
 >>> BuiltInSCMed.clsMed('tyxt.work')
 (<class __main__.BuiltInSCMed at 0x03CD6650>, 'tyxt.work')
 # 通过实例调用类方法
 >>> biscm1.clsMed('tyxt.work')
 (<class __main__.BuiltInSCMed at 0x03CD6650>, 'tyxt.work')

Python3.x在idle执行结果

 >>> import sys
 >>> print('python版本为:python{}'.format(sys.version.split(' ')[0]))
 python版本为:python3.9.0
 >>> class BuiltInSCMed:
     def instanceMed(self,x):
         print(self,x)
     def staticMed(x):
         print(x)
     def clsMed(cls,x):
         print(cls,x)
     staticMed=staticmethod(staticMed)
     clsMed=classmethod(clsMed)
 >>> biscm1=BuiltInSCMed()
 >>> biscm1.instanceMed(1)
 <__main__.BuiltInSCMed object at 0x000001B16B6FEBB0> 1
 >>> BuiltInSCMed.instanceMed(biscm1,2)
 <__main__.BuiltInSCMed object at 0x000001B16B6FEBB0> 2
 >>> BuiltInSCMed.staticMed(3)
 3
 >>> biscm1.staticMed('梯阅线条')
 梯阅线条
 >>> BuiltInSCMed.clsMed('tyxt.work')
 <class '__main__.BuiltInSCMed'> tyxt.work
 >>> biscm1.clsMed('tyxt.work')
 <class '__main__.BuiltInSCMed'> tyxt.work

1.2.4 内置函数staticmethod转换的静态方法统计实例

python2.x 和3.x 在idle 执行结果 相同

 >>> class CountInsBISM:
     numOfInstances=0
     def __init__(self):
         CountInsBISM.numOfInstances+=1
     def printNumOfIns():
         print('创建的实例数为:{}'.format(CountInsBISM.numOfInstances))
     printNumOfIns=staticmethod(printNumOfIns)
 >>> cibs1,cibs2,cibs3=CountInsBISM(),CountInsBISM(),CountInsBISM()
 >>> CountInsBISM.printNumOfIns()    # 通过类调用
 创建的实例数为:3
 >>> cibs1.printNumOfIns()   # 通过实例调用
 创建的实例数为:3

1.2.5 内置函数classmethod转换的类方法统计实例

python2.x 和3.x 在idle 执行结果 相同

 >>> class CountInsBICM:
     numOfInstances=0
     def __init__(self):
         CountInsBICM.numOfInstances+=1
     def printNumOfIns(cls):
         print('创建的实例数为:{}'.format(cls.numOfInstances))
     printNumOfIns=classmethod(printNumOfIns)
 
 >>> cibc1,cibc2,cibc3=CountInsBICM(),CountInsBICM(),CountInsBICM()
 >>> CountInsBICM.printNumOfIns()    # 通过类调用
 创建的实例数为:3
 >>> cibc1.printNumOfIns()   # 通过实例调用
 创建的实例数为:3

1.2.6 统计每个类的实例

通过类方法统计继承中每个类的实例。

需要在继承中,每个类各自维护一个实例数属性,用于存放各自数据。

示例

 >>> class CountInsEC:
     numOfInstances=0
     def countcls(cls):
         cls.numOfInstances+=1
     def __init__(self):
         self.countcls()
     countcls=classmethod(countcls)
 
     
 >>> class SubA(CountInsEC):
     numOfInstances=0
     def __init__(self):
         CountInsEC.__init__(self)
 
         
 >>> class SubB(CountInsEC):
     numOfInstances=0
 
     
 >>> ciec1,ciec2,ciec3=CountInsEC(),CountInsEC(),CountInsEC()
 >>> suba1,suba2=SubA(),SubA()
 >>> subb1=SubB()
 >>> ciec1.numOfInstances,suba1.numOfInstances,subb1.numOfInstances
 (3, 2, 1)
 >>> CountInsEC.numOfInstances,SubA.numOfInstances,SubB.numOfInstances
 (3, 2, 1)

本文首发微信公众号:梯阅线条

更多内容参考python知识分享或软件测试开发目录。

相关推荐

Linux 网络协议栈_linux网络协议栈

前言;更多学习资料(包含视频、技术学习路线图谱、文档等)后台私信《资料》免费领取技术点包含了C/C++,Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,Z...

揭秘 BPF map 前生今世_bpfdm

1.前言众所周知,map可用于内核BPF程序和用户应用程序之间实现双向的数据交换,为BPF技术中的重要基础数据结构。在BPF程序中可以通过声明structbpf_map_def...

教你简单 提取fmpeg 视频,音频,字幕 方法

ffmpeg提取视频,音频,字幕方法(HowtoExtractVideo,Audio,SubtitlefromOriginalVideo?)1.提取视频(ExtractVi...

Linux内核原理到代码详解《内核视频教程》

Linux内核原理-进程入门进程进程不仅仅是一段可执行程序的代码,通常进程还包括其他资源,比如打开的文件,挂起的信号,内核内部的数据结构,处理器状态,内存地址空间,或多个执行线程,存放全局变量的数据段...

Linux C Socket UDP编程详解及实例分享

1、UDP网络编程主要流程UDP协议的程序设计框架,客户端和服务器之间的差别在于服务器必须使用bind()函数来绑定侦听的本地UDP端口,而客户端则可以不进行绑定,直接发送到服务器地址的某个端口地址。...

libevent源码分析之bufferevent使用详解

libevent的bufferevent在event的基础上自己维护了一个buffer,这样的话,就不需要再自己管理一个buffer了。先看看structbufferevent这个结构体struct...

一次解决Linux内核内存泄漏实战全过程

什么是内存泄漏:程序向系统申请内存,使用完不需要之后,不释放内存还给系统回收,造成申请的内存被浪费.发现系统中内存使用量随着时间的流逝,消耗的越来越多,例如下图所示:接下来的排查思路是:1.监控系统中...

彻底搞清楚内存泄漏的原因,如何避免内存泄漏,如何定位内存泄漏

作为C/C++开发人员,内存泄漏是最容易遇到的问题之一,这是由C/C++语言的特性引起的。C/C++语言与其他语言不同,需要开发者去申请和释放内存,即需要开发者去管理内存,如果内存使用不当,就容易造成...

linux网络编程常见API详解_linux网络编程视频教程

Linux网络编程API函数初步剖析今天我们来分析一下前几篇博文中提到的网络编程中几个核心的API,探究一下当我们调用每个API时,内核中具体做了哪些准备和初始化工作。1、socket(family...

Linux下C++访问web—使用libcurl库调用http接口发送解析json数据

一、背景这两天由于一些原因研究了研究如何在客户端C++代码中调用web服务端接口,需要访问url,并传入json数据,拿到返回值,并解析。 现在的情形是远程服务端的接口参数和返回类型都是json的字符...

平衡感知调节:“系统如人” 视角下的架构设计与业务稳定之道

在今天这个到处都是数字化的时代,系统可不是一堆冷冰冰的代码。它就像一个活生生的“数字人”,没了它,业务根本转不起来。总说“技术要为业务服务”,但实际操作起来问题不少:系统怎么才能快速响应业务需求?...

谈谈分布式文件系统下的本地缓存_什么是分布式文件存储

在分布式文件系统中,为了提高系统的性能,常常会引入不同类型的缓存存储系统(算法优化所带来的的效果可能远远不如缓存带来的优化效果)。在软件中缓存存储系统一般可分为了两类:一、分布式缓存,例如:Memca...

进程间通信之信号量semaphore--linux内核剖析

什么是信号量信号量的使用主要是用来保护共享资源,使得资源在一个时刻只有一个进程(线程)所拥有。信号量的值为正的时候,说明它空闲。所测试的线程可以锁定而使用它。若为0,说明它被占用,测试的线程要进入睡眠...

Qt编写推流程序/支持webrtc265/从此不用再转码/打开新世界的大门

一、前言在推流领域,尤其是监控行业,现在主流设备基本上都是265格式的视频流,想要在网页上直接显示监控流,之前的方案是,要么转成hls,要么魔改支持265格式的flv,要么265转成264,如果要追求...

30 分钟搞定 SpringBoot 视频推拉流!实战避坑指南

30分钟搞定SpringBoot视频推拉流!实战避坑指南在音视频开发领域,SpringBoot凭借其快速开发特性,成为很多开发者实现视频推拉流功能的首选框架。但实际开发中,从环境搭建到流处理优...

取消回复欢迎 发表评论: