Python和C++赋值共享内存、Python函数传址传值、一些其他的遇到的bug

2024-06-07 11:36

本文主要是介绍Python和C++赋值共享内存、Python函数传址传值、一些其他的遇到的bug,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、Numpy共享内存的情况:

array1 = np.array([1, 2, 3])
array2 = array1
array2[0] = 0       # array1也会跟着改变,就地操作
array2 = array2 * 2 # array2不会跟着改变,属于非就地操作,会创建一个新的地址给array2
array2 = array1[:]
array2 = array1.view()
array2 = array1.reshape((3, 1))

使用array.copy() 创建深拷贝以避免这种问题

非就地操作:会创建一个新的数组,并将其赋值给 array2(指向的地址发生改变)。在这种情况下,array2 将引用一个新的数组,原来的 array2 不再共享原始的内存。如下,都不会影响到array1:

array2 = -array1 # 有运算时不共享(会创建一个新的数组给array2)
array2[0] = 0    # array1不会跟着改变
array2 = array1		# 共享内存
array2 = array2 + 1 # array1不会改变(此时创建一个新的数组给array2)。但array2 += 1会影响array1,自增是就地(in-place)操作

2、pytorch共享内存的情况:

和Numpy完全一样

tensor1 = torch.tensor([1, 2, 3])
tensor2 = tensor1
tensor2[0] = 0 # tensor1也会跟着改变
tensor2 = tensor1.view(-1)
tensor2 = tensor1.detach()

使用tensor.clone()创建深拷贝以避免这种问题

3、Eigen中共享内存的情况:

Eigen::MatrixXd matrix1 = Eigen::MatrixXd::Random(3, 3);
Eigen::MatrixXd matrix2 = matrix1.block(0, 0, 2, 2);
Eigen::ArrayXd array1 = Eigen::ArrayXd::Random(5);
Eigen::ArrayXd array2 = array1.segment(1, 3);

直接进行赋值不会共享内存:

Eigen::VectorXd vector1 = Eigen::VectorXd::Random(5);
Eigen::VectorXd vector2 = vector1; // 不共享内存,vector2更改不影响vector1

4、List中共享内存的情况

Python当向列表中添加一个元素时,列表会存储对该元素的引用

a = np.array([1, 2, 3])
list_a = [a]
a[0] = 0 # list_a也会改变

C++中std::vector不会有这种情况:

int main() {Eigen::Vector3d A(0,0,0);std::vector<Eigen::Vector3d> vector_A;vector_A.push_back(A);A(0) = 1;	// vector_A不改std::cout << vector_A[0].transpose() << std::endl;return 0;
}

4、默认拷贝构造函数

C++是浅拷贝,如果有指针会有问题,但没指针的话没啥问题:

#include <iostream>class MyClass {
public:int value;MyClass(int v) : value(v) {}
};int main() {MyClass A(5);MyClass B = A; B.value = 10;std::cout << "修改后的 A 的值:" << A.value << std::endl; // 输出 5std::cout << "修改后的 B 的值:" << B.value << std::endl; // 输出 10return 0;
}

Python当将一个对象赋值给另一个对象时,实际上是创建了对同一个对象的引用

import numpy as npclass MyClass:def __init__(self, num):self.int = numA = MyClass(1)
B = A
B.int = 10 	# A也会被改print(id(A))
print(id(B)) # 地址是一样的!
print("A.array:", A.int)
print("B.array:", B.int)

5、Python函数的传址和传值

不可变对象(如数字、字符串、元组):
当你传递不可变对象给函数时,函数内部对参数的任何修改都不会影响原始对象,类似于值传递的行为。
可变对象(如列表、字典、集合、np.array、torch.tensor):
当你传递可变对象给函数时,函数内部对参数的修改会影响原始对象,类似于引用传递的行为。

import numpy as np
import torchdef modify_immutable(x):x = 100def modify_mutable(lst):lst.append(4)def modify_array(arr):arr[0] = 100def modify_tensor(tensor):tensor[0] = 100a = 3
print("函数调用前的值:", a)
modify_immutable(a)
print("函数调用后的值:", a)  # 不会改变lst = [1, 2, 3]
print("函数调用前的列表:", lst)
modify_mutable(lst)
print("函数调用后的列表:", lst)	# 会改变original_array = np.array([1, 2, 3, 4, 5])
print("调用函数前的原始数组:", original_array)
modify_array(original_array)
print("调用函数后的原始数组:", original_array)	# 会改变original_tensor = torch.tensor([1, 2, 3, 4, 5])
print("调用函数前的原始 Tensor:", original_tensor)
modify_tensor(original_tensor)
print("调用函数后的原始 Tensor:", original_tensor)	# 会改变

6、std::vector<>访问溢出时不会报错,只是会返回垃圾值
RPG代码访问相机类型和个数时,ID错了,但访问却没出错,导致一直没注意他的bug

这篇关于Python和C++赋值共享内存、Python函数传址传值、一些其他的遇到的bug的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1039067

相关文章

Python如何实现读取csv文件时忽略文件的编码格式

《Python如何实现读取csv文件时忽略文件的编码格式》我们再日常读取csv文件的时候经常会发现csv文件的格式有多种,所以这篇文章为大家介绍了Python如何实现读取csv文件时忽略文件的编码格式... 目录1、背景介绍2、库的安装3、核心代码4、完整代码1、背景介绍我们再日常读取csv文件的时候经常

基于Python实现多语言朗读与单词选择测验

《基于Python实现多语言朗读与单词选择测验》在数字化教育日益普及的今天,开发一款能够支持多语言朗读和单词选择测验的程序,对于语言学习者来说无疑是一个巨大的福音,下面我们就来用Python实现一个这... 目录一、项目概述二、环境准备三、实现朗读功能四、实现单词选择测验五、创建图形用户界面六、运行程序七、

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

浅析Python中的绝对导入与相对导入

《浅析Python中的绝对导入与相对导入》这篇文章主要为大家详细介绍了Python中的绝对导入与相对导入的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1 Imports快速介绍2 import语句的语法2.1 基本使用2.2 导入声明的样式3 绝对import和相对i

Python中配置文件的全面解析与使用

《Python中配置文件的全面解析与使用》在Python开发中,配置文件扮演着举足轻重的角色,它们允许开发者在不修改代码的情况下调整应用程序的行为,下面我们就来看看常见Python配置文件格式的使用吧... 目录一、INI配置文件二、YAML配置文件三、jsON配置文件四、TOML配置文件五、XML配置文件

C++实现回文串判断的两种高效方法

《C++实现回文串判断的两种高效方法》文章介绍了两种判断回文串的方法:解法一通过创建新字符串来处理,解法二在原字符串上直接筛选判断,两种方法都使用了双指针法,文中通过代码示例讲解的非常详细,需要的朋友... 目录一、问题描述示例二、解法一:将字母数字连接到新的 string思路代码实现代码解释复杂度分析三、

Python中conda虚拟环境创建及使用小结

《Python中conda虚拟环境创建及使用小结》本文主要介绍了Python中conda虚拟环境创建及使用小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录0.前言1.Miniconda安装2.conda本地基本操作3.创建conda虚拟环境4.激活c

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

使用Python创建一个能够筛选文件的PDF合并工具

《使用Python创建一个能够筛选文件的PDF合并工具》这篇文章主要为大家详细介绍了如何使用Python创建一个能够筛选文件的PDF合并工具,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录背景主要功能全部代码代码解析1. 初始化 wx.Frame 窗口2. 创建工具栏3. 创建布局和界面控件4

一文详解如何在Python中使用Requests库

《一文详解如何在Python中使用Requests库》:本文主要介绍如何在Python中使用Requests库的相关资料,Requests库是Python中常用的第三方库,用于简化HTTP请求的发... 目录前言1. 安装Requests库2. 发起GET请求3. 发送带有查询参数的GET请求4. 发起PO