python 的深浅拷贝问题

深浅拷贝概念

基本类型和引用类型数据拷贝的问题。因为基本类型的数据大小是固定的,所以他保存在栈内存中;而引用类型的数据大小不固定,因而保存在堆内存中,单引用类型在栈内存中只保存一个指向堆内存的指针。
浅拷贝:对于浅拷贝来说,如果拷贝基本类型,那么就等于赋值一样,会直接拷贝其本身;但如果拷贝的是引用类型,就只会拷贝一层,如果 原对象发生改变,那么拷贝对象也会发生改变。
深拷贝:深拷贝的话就会拷贝多层,嵌套的对象也会被拷贝出来,相当于开辟一个新的内存地址用于存放拷贝的对象。
在python语言中没有明显的指出,例如,操作指针或对于指针的操作。
在java或者go中是可以的。

业务需求

抓取交易所黄金的k线,进行数据分析。

问题代码

import datetime


def judgment(info):
    minute = datetime.timedelta(minutes=15)
    count = 0
    while True:
        Long = len(info)
        if count + 1 == Long:
            break
        if info[count]["TS"] + minute == info[count + 1]["TS"]:
            pass
        else:
            date = info[count]
            date["TS"] = info[count]["TS"] + minute
            info.insert(count + 1, date)
        count += 1
    return info


info = [{'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 2, 45), 'P': '312.10', 'L': '312.10',
         'H': '312.10', 'O': '312.10'},
        {'C': 'AUTD', 'T': '1561338900', 'TS': datetime.datetime(2019, 6, 22, 5, 15), 'P': '313.88', 'L': '313.03',
         'H': '314.97', 'O': '313.30'}]
print(len(info))
date = judgment(info)
print(date)
date1 = [{'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561338900', 'TS': datetime.datetime(2019, 6, 22, 5, 15), 'P': '313.88', 'L': '313.03',
          'H': '314.97', 'O': '313.30'}]

可以发现代码逻辑是没有问题的,问题出在深浅拷贝的问题。按照正常的逻辑他的值会进行替换,而不是出现错误。一个简单的问题弄了一下午,汗颜。

import datetime
import copy


def judgment(info):
    minute = datetime.timedelta(minutes=15)
    count = 0
    info_date = []
    while True:
        Long = len(info)
        if count + 1 == Long:
            break
        if info[count]["TS"] + minute == info[count + 1]["TS"]:
            pass
        else:
            date = info[count]
            date["TS"] = info[count]["TS"] + minute
            info.insert(count + 1, date)
        # print(info[count])
        info_date.append(copy.deepcopy(info[count]))

        count += 1
    return info_date


info = [{'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 2, 45), 'P': '312.10', 'L': '312.10',
         'H': '312.10', 'O': '312.10'},
        {'C': 'AUTD', 'T': '1561338900', 'TS': datetime.datetime(2019, 6, 22, 5, 15), 'P': '313.88', 'L': '313.03',
         'H': '314.97', 'O': '313.30'}]
print(len(info))
date = judgment(info)
print(date)

date1 = [{'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 3, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 3, 15), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 3, 30), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 3, 45), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 4, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 4, 15), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 4, 30), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 4, 45), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'},
         {'C': 'AUTD', 'T': '1561142700', 'TS': datetime.datetime(2019, 6, 22, 5, 0), 'P': '312.10', 'L': '312.10',
          'H': '312.10', 'O': '312.10'}]

总结:Python是没有基础数据类型(primitive value type),全部都是对象.也就意味着变量和属性全部都是引用类型.
为了提醒大家所以讲问题整理出来。在写程序的时候需要注意,深浅拷贝问题,有时候我们往往忽略最简单的问题。

原文地址:https://www.cnblogs.com/xiaogongzi/p/11215125.html