python tips

  • 可变对象与不可变对象

    在python中,可变对象有数值类型(int,float),字符串(str),元组(tuple),可变对象有列表(list),字典(dict),集合(set)。在赋值操作中 可变对象是复制的对象本身,相当于复制了对象的指针,对赋值后的修改会修改原来的对象 例如:

    In [1]: b = [1,2,3,4]
    In [2]: _b = b In [3]: id(_b)
    Out[3]: 4519417736 In [4]: id(b)
    Out[4]: 4519417736 In [5]: _b.append(6) # 修改_b,b也会跟着一起变化,因为_b和b指向的是同一个地址 In [6]: b
    Out[6]: [1, 2, 3, 4, 6]

可以看出修改了_b的值,也修改了b的值,如果想仅仅修改_b的值,需要使用 _b = copy.copy(b) 而不是简单的_b = b

对于不可变对象,对象本身的值是不能修改的,每次对其的操作都会生成一个新的对象,保存新的值。如下:

```python
In [1]: a = "sss" In [2]: id(a)
Out[2]: 4338243040 In [3]: a += "c" # 不可变对象的赋值并不会使用同一个地址,所以id会发生变化 In [4]: id(a)
Out[4]: 4338242704 ```

所以在函数中传入可变对象的时候,要小心,因为可能修改原有的对象。

  • 元编程(metaclass)

    在正常业务开发时很少使用,但在一些库中经常遇到,例如Django的ORM,其主要功能是在类初始化(不是实例化)的时候,定义一些行为和操作,代码如下

    import os
    import numpy
    import random class MetaClass(type): def __init__(self, *args, **kwargs):
    print('Meta Init') def __new__(cls, name, base, attr): print('class_name is :' , name)
    print(base,' is the Base *****')
    print(cls.v, getattr(cls, 'b', 99 )) print('==========') del cls.v
    return super(MetaClass, cls).__new__(cls, name, base, attr) class Model(metaclass=MetaClass): # python3中指定metaclass指定方式
    __metaclass__ = MetaClass # python2中指定方式
    MetaClass.v = 100
    MetaClass.b = '-------------'
    c = 9
    def __new__(cls, name, base, attrs):
    print('Model New---')
    return super(Model, cls).__new__(cls, name, base, attrs) def __init__(self, *args, **kwargs):
    self.arg = args
    self.kwargs = kwargs def __str__(self):
    return "test Models ----->>>>>>" class Test(metaclass=MetaClass):
    __metaclass__ = MetaClass
    MetaClass.v = 888 class Serial(Model):
    MetaClass.v = 777 class test(Serial):
    MetaClass.v = 666

    在import 或者直接运行的时候,我们没有做任何类的实例化操作,但还是会有下面的打印信息,会发现,继承的每个类都运行了metaclass中的__new____init__方法

    class_name is : Model
    () is the Base *****
    100 -------------
    ==========
    Meta Init
    class_name is : Test
    () is the Base *****
    888 -------------
    ==========
    Meta Init
    class_name is : Serial
    (<class '__main__.Model'>,) is the Base *****
    777 -------------
    ==========
    Meta Init
    class_name is : test
    (<class '__main__.Serial'>,) is the Base *****
    666 -------------
    ==========
    Meta Init
  • slots

    在python中,一般我们可以自由的给类,实例添加属性,然而有时候我们并不希望这样,我们需要指定的类只能有我们要求的这些属性,在这种情况下,我们引入了__slots__ 例如:

    class R(object):
    """在这个类中,我们只能添加prev,next等的属性,
    当尝试添加别的的时候,会报AttributeError"""
    __slots__ = 'prev', 'next', 'key', '__weakref__' class Link(R):
    “”“__slots__ 不会被继承,所以在这个类中可添加指定之外的属性”“”
    pass class Node(R):
    """添加了 __slots__ 之后,父类的 __slots__也会生效,此时,
    限制的属性为 value, prev, next 等
    """
    __slots__ = "value",
  • Django Queryset 切片

    我们知道Django 的queryset是lazy的,只有当真正使用的时候才会去数据库取数据,并且缓存取出来的数据。

    所以对于没有执行的queryset,切片操作之后仍然是queryset,执行sql之后的queryset,切片会变成list。

    In [2]: qs = Order.objects.all()
    
    In [3]: qs_1 = qs[10:20]
    
    In [4]: type(qs_1)
    Out[4]: django.db.models.query.QuerySet In [5]: _qs = list(qs) # 会查询数据 In [6]: qs_2 = qs[10:20] In [7]: type(qs_2)
    Out[7]: list

最新文章

  1. 白电迁移-Auto fdisk
  2. 百度地图API地理位置和坐标转换
  3. [转] 腾讯云直播OBS推流教程
  4. Oracle利用数据泵迁移用户
  5. 用通俗的例子解释OAuth和OpenID的区别【原】
  6. enter 默认搜索
  7. [视频]K8飞刀 hacking team flash0day shellcode exploit
  8. Django-website 程序案例系列-9 分页
  9. netcore webapi统一配置跨域问题
  10. log4j的基本使用方法
  11. tmux 快捷操作
  12. IPython:一种交互式计算和开发环境(魔术命令,快捷键)
  13. IDEA项目搭建五——使用JRebel插件实现IDEA热部署
  14. c# 协变和逆变的理解
  15. 错误: No API token found for service account &quot;default&quot;,
  16. RHEL 7 下内存管理小记
  17. ReactNative 打包 APK
  18. (转)USB的描述符及各种描述符之间的依赖关系
  19. 锁Lock的概念
  20. python 爬虫登录保存会话去获取只有登录能获取的数据

热门文章

  1. Wp及Windows应用商店程序Logo生成器
  2. Hibernate课堂笔记
  3. scss-嵌套属性
  4. Python爬虫《爬取get请求的页面数据》
  5. String StringBuffer StringBuilder对比
  6. Drupal中hook_theme函数用法
  7. mui 的注意事项点
  8. php基础--取默认值以及类的继承
  9. ArcGIS Engine从服务器(ArcSDE geodatabases)读取数据
  10. python里面的encode和decode函数