Python进阶 函数式编程和面向对象编程等

函数式编程

函数:function

函数式:functional,一种编程范式。函数式编程是一种抽象计算机的编程模式。

函数!= 函数式(如计算!=计算机)

如下是不同语言的抽象 层次不同

高阶函数:能接收函数做参数的函数

  1. 变量可以指向函数
  2. 函数的参数可以接收变量
  3. 一个函数可以接收另一个函数作为参数

例子:

接收abs函数,

定义一个函数,接收x,y,z三个参数。其中x,y是数值,z是函数 。


1

2

3

def add(x,y,z):

    return z(x)+z(y)

print add(-2,-3,abs)

其他高阶函数:map()函数、reduce()函数、filter()函数、

PS:Python的函数不但可以返回int、str、list、dict等数据类型,还可以返回函数!

注意:

闭包

像这种内层函数引用了外层函数的变量(参数也算变量),然后返回内层函数的情况,称为闭包(Closure)。

特点 是返回的函数还引用了外层函数的局部变量,所以,要正确使用闭包,就要确保引用的局部变量在函数返回后不能变。举例如下:

# 希望一次返回3个函数,分别计算1x1,2x2,3x3:
def count():
    fs = []
    for i in range(1, 4):
        def f():
             return i*i
        fs.append(f)
    return fs

f1, f2, f3 = count()

你可能认为调用f1(),f2()和f3()结果应该是1,4,9,但实际结果全部都是 9(请自己动手验证)。

原因就是当count()函数返回了3个函数时,这3个函数所引用的变量 i 的值已经变成了3。由于f1、f2、f3并没有被调用,所以,此时他们并未计算 i*i,当 f1 被调用时:

>>> f1()
9     # 因为f1现在才计算i*i,但现在i的值已经变为3

因此,返回函数不要引用任何循环变量,或者后续会发生变化的变量。

匿名函数(lambda )

>>> map(lambda x: x * x, [1, 2, 3, 4, 5, 6, 7, 8, 9])
[1, 4, 9, 16, 25, 36, 49, 64, 81]

通过对比可以看出,匿名函数lambda x: x * x 实际上就是:

def f(x):
    return x * x

关键字lambda 表示匿名函数,冒号前面的 x 表示函数参数

(匿名函数有个限制,就是只能有一个表达式,不写return,返回值就是该表达式的结果,返回函数的时候,也可以返回匿名函数:)

装饰器 decorator

Python的 decorator 本质上就是一个高阶函数,它接收一个函数作为参数,然后,返回一个新函数。

使用 decorator 用Python提供的 @ 语法,这样可以避免手动编写 f = decorate(f) 这样的代码,极大简化Python代码。

模块

导入系统自带的模块 math:

import math

如果我们只希望导入用到的math模块的某几个函数,而不是所有函数,可以用下面的语句:

from math import pow, sin, log

如果遇到名字冲突怎么办?

如果使用import导入模块名,由于必须通过模块名引用函数名,因此不存在冲突;

如果使用 from...import 导入 log 函数,势必引起冲突。这时,可以给函数起个“别名”来避免冲突:

from math import log

from logging import log as logger   # logging的log现在变成了logger

print log(10)   # 调用的是math的log

logger(10, ‘import from logging‘)   # 调用的是logging的log

动态导入模块

try:

from cStringIO import StringIO

except ImportError:

from StringIO import StringIO

上述代码先尝试从cStringIO导入,如果失败了(比如cStringIO没有被安装),再尝试从StringIO导入

利用import ... as ...,还可以动态导入不同名称的模块

try:

import json

except ImportError:

import simplejson as json

PS:

1. Python的新版本会引入新的功能,但是实际上这些功能在上一个老版本中就已经存在了。要“试用”某一新的特性,就可以通过导入__future__模块的某些功能来实现。

2. 如何区分包和普通目录:包下面有个__init__.py 每层都有。

安装第三方模块

1. 使用easy_install

2. pip(推荐,内置到Python2.7.9)


1

2

3

4

5

wget "https://pypi.python.org/packages/source/p/pip/pip-1.5.4.tar.gz#md5=834b2904f92d46aaa333267fb1c922bb" --no-check-certificate  #下载pip

tar -xzvf pip-1.5.4.tar.gz

cd pip-1.5.4 #(会提示安装setuptools,yum install python-setuptools 即可)

python setup.py install

pip install pymongo #可以使用pip安装第三方模块了

也可以直接 yum install python-pip

面向对象编程基础

定义类并创建实例

定义一个Person类如下


1

2

class Person(object):

    pass

(object),表示该类是从哪个类继承下来的

创建实例

xiaoming = Person()

如何让每个实例拥有各自不同的属性?

由于Python是动态语言,对每一个实例,都可以直接给他们的属性赋值,例如,给xiaoming这个实例加上name、gender和birth属性:

xiaoming = Person()

xiaoming.name = ‘Xiao Ming‘

xiaoming.gender = ‘Male‘

xiaoming.birth = ‘1990-1-1‘

给xiaohong加上的属性不一定要和xiaoming相同:

xiaohong = Person()

xiaohong.name = ‘Xiao Hong‘

xiaohong.school = ‘No. 1 High School‘

xiaohong.grade = 2

实例的属性可以像普通变量一样进行操作:

xiaohong.grade = xiaohong.grade + 1

初始化实例属性


1

2

3

4

5

class Person(object):

    def __init__(self, name, gender, birth):

        self.name = name

        self.gender = gender

        self.birth = birth

__init__() 方法的第一个参数必须是 self(也可以用别的名字,但建议使用习惯用法)。后续参数则可以自由指定,和定义函数没有任何区别。

相应地,创建实例时,就必须要提供除 self 以外的参数:

xiaoming = Person(‘Xiao Ming‘, ‘Male‘, ‘1991-1-1‘)

xiaohong = Person(‘Xiao Hong‘, ‘Female‘, ‘1992-2-2‘)

定义实例方法


1

2

3

4

5

6

class Person(object):

    def __init__(self, name):

        self.__name = name

 

    def get_name(self): #它的第一个参数永远是 self,指向调用该方法的实例本身

        return self.__name

定义类方法(类似Java的静态方法)


1

2

3

4

5

6

7

8

9

10

11

12

class Person(object):

    count = 0

    @classmethod

    def how_many(cls): #类方法,不用self?

        return cls.count

    def __init__(self, name):

        self.name = name

        Person.count = Person.count + 1

print Person.how_many()

p1 = Person(‘Bob‘)

print Person.how_many()

访问限制

如果一个属性由双下划线开头(__),该属性就无法被外部访问(相当于private).

但是,如果一个属性以"__xxx__"的形式定义,那它又可以被外部访问了,以"__xxx__"定义的属性在Python的类中被称为特殊属性,有很多预定义的特殊属性可以使用,通常我们不要把普通属性用"__xxx__"定义。

题目

请定义Person类的__init__方法,除了接受 name、gender 和 birth 外,还可接受任意关键字参数,并把他们都作为属性赋值给实例。

要定义关键字参数,使用 **kw;

除了可以直接使用self.name = ‘xxx‘设置一个属性外,还可以通过 setattr(self, ‘name‘, ‘xxx‘) 设置属性。

参考代码:

class Person(object):
    def __init__(self, name, gender, birth, **kw):
        self.name = name
        self.gender = gender
        self.birth = birth
        for k, v in kw.iteritems():
            setattr(self, k, v)
xiaoming = Person(‘Xiao Ming‘, ‘Male‘, ‘1990-1-1‘, job=‘Student‘)
print xiaoming.name
print xiaoming.job

类的继承

继承一个类


1

2

3

4

class Person(object):

    def __init__(self, name, gender):

        self.name = name

        self.gender = gender

定义Student类时,只需要把额外的属性加上,例如score:


1

2

3

4

class Student(Person):

    def __init__(self, name, gender, score):

        super(Student, self).__init__(name, gender)

        self.score = score

一定要用 super(Student, self).__init__(name, gender) 去初始化父类,否则,继承自 Person 的 Student 将没有 name 和 gender。函数super(Student, self)将返回当前类继承的父类,即 Person ,然后调用__init__()方法,注意self参数已在super()中传入,在__init__()中将隐式传递,不需要写出(也不能写)

判断类型

函数isinstance()可以判断一个变量的类型

>>> isinstance(p, Person)

获取对象信息

除了用 isinstance() 判断它是否是某种类型的实例外

type() 函数获取变量的类型,它返回一个 Type 对象

可以用 dir() 函数获取变量的所有属性: dir(s)

参考:慕课网,python进阶

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-05 11:16:08

Python进阶 函数式编程和面向对象编程等的相关文章

Python 进阶(三)面向对象编程基础

定义类并创建实例 在Python中,类通过 class 关键字定义.以 Person 为例,定义一个Person类如下: class Person(object): pass 按照 Python 的编程习惯,类名以大写字母开头,紧接着是(object),表示该类是从哪个类继承下来的.类的继承将在后面的章节讲解,现在我们只需要简单地从object类继承. 有了Person类的定义,就可以创建出具体的xiaoming.xiaohong等实例.创建实例使用 类名+(),类似函数调用的形式创建: xia

Python基础-第七天-面向对象编程进阶和Socket编程简介

本篇内容: 1.面向对象编程进阶-静态方法 2.面向对象编程进阶-类方法 3.面向对象编程进阶-属性方法 4.面向对象编程进阶-特殊成员(内置方法) 5.面向对象编程进阶-反射 6.异常处理.断言 7.Socket编程简介 一.面向对象编程进阶-静态方法 1.静态方法的实现 通过@staticmethod装饰器可以把其装饰的方法变为一个静态方法: 变成静态方法后,形参中可以不用写self了.如果写了self,默认是不会把对象本身传递给self,需要手动传递: class Dog(object):

python编程:函数式编程和面向对象编程的对比

言而简之,面向对象编程就是把函数编程外面包装一个class类,然后再把这个class类指向一个对象 而class类中的函数在这里称为方法 举个例子: 用函数式编程写出一键发送邮件的脚本: def e_mail(email,message): print('邮件发送中...') return True e_mail('[email protected]','邮件内容') 用面向对象编程写出一键发送邮件的脚本: class Foo: #e_mail称为方法 def e_mail(self,email

python学习笔记day6【面向对象编程】

面向对象编程 一.编程范式:编程的方法论.程序员编程的“套路”及“特点”特点总结,抽象成方法.规范. 二.面向对象编程介绍: 1.描述 世界万物,皆可分类:世间万物,皆为对象:只要是对象,就肯定属于某种品类:只要是对象,就肯定有一定的属性. opp编程(面向对象编程)的抽象机制是将待解问题抽象为面向对象的程序中的对象.利用封装使每个对象都拥有个体的身份.程序便是成堆的对象,彼此通过消息的传递,请求其它对象 进行工作.也可以说opp编程是利用“类”和“对象”来创建各种模型来实现对真实世界的描述.

python_way.day7 模块(configparser,xml,shutil,subprocess)、面向对象(上)(创建类,类的构成,函数式编程与面向对象编程的选择,类的继承)

python_way.day7 1.模块 configparser,xml,shutil,subprocess 2.面向对象(上) 创建类,类的构成,函数式编程与面向对象编程的选择,类的继承 1.模块 configparser 用于处理特定格式的文件,其本职上使用open来操作,只能是  [test1] 特定的格式 [test1] k1 = 123 k2 = True [test2] k1 = 123 k2 = v1 文件内容 1.获取 import configparser #打开文件找到文件

函数式编程与面向对象编程的比较

函数式编程作为结构化编程的一种,正在受到越来越多的重视.工程中不在只是面向对象编程,更多的人尝试着开始使用函数式编程来解决软件工程中遇到的问题. 什么是函数式编程?在维基百科中给出了详细的定义,函数式编程(英语:functional programming)或称函数程序设计,又称泛函编程,是一种编程范型,它将电脑运算视为数学上的函数计算,并且避免使用程序状态以及易变对象.函数编程语言最重要的基础是λ演算(lambda calculus).而且λ演算的函数可以接受函数当作输入(引数)和输出(传出值

[从玩游戏来理解编程]关于面向对象编程的浅解(1)

我最近在玩星际争霸2,感觉到面向对象真是太重要了.有三个种族,每个种族有自己的兵种,一个兵种就是一个类,而且他们的父类都是一个类. 每个单位都有自己的属性和技能,之前学习的c语言是面向过程的,慢慢的也理解到了面向对象在某些地方的重要性. 之前都是在学习算法,实现上用的c语言,看不出来到底有什么区别,反而在用c来的更简洁和更快,最近慢慢的接触到小项目之类的什么的就慢慢感觉到面向对象的重要性. 最近在看一本书叫<making games with python and pygame>,这本书很好,

基于对象编程与面向对象编程(表达式计算器3)

基于对象编程与面向对象编程 我们的最终目的是用C++设计一个面向对象的表达式计算器,所以非常有必要弄清楚,什么是基于对象编程和面向对象的编程.而要弄清楚这一点,又要先弄明白什么是值语言,什么是对象语义 值语义:对象的拷贝与原对象无关,拷贝后与原对象脱离关系,互不影响.这种拷贝叫深拷贝.拷贝之后脱离关系,只要在拷贝的时候都为对象分配内存空间就行了.某种些情况下算是一种对资源的浪费 值语义例子 class Test { private: int * pNum_; public: Test(int n

Python 函数式编程和面向对象编程

函数式编程 函数:function.函数式:functional,一种编程范式.函数式编程是一种抽象计算机的编程模式.函数!= 函数式(如计算!=计算机) 如下是不同语言的抽象 层次不同 高阶函数 能接收函数做参数的函数: 1.变量可以指向函数2.函数的参数可以接收变量3.一个函数可以接收另一个函数作为参数 例子 接收abs函数,定义一个函数,接收x,y,z三个参数.其中x,y是数值,z是函数 . def add(x,y,z): return z(x)+z(y) print add(-2,-3,