设为首页 收藏本站
查看: 1454|回复: 0

[经验分享] python 中的赋值操作,与c/c++的对比

[复制链接]

尚未签到

发表于 2015-4-28 09:40:33 | 显示全部楼层 |阅读模式
def foo(l):  
      l += 'b'


l = 'abc'
foo(l)
print(l)        #result   'abc' not 'abcb'


l = ['abc']
foo(l)
print(l)       #result ['abc','b']


作个总结,网上已经有相关的内容了,这里方便记忆。也许有理解错误的地方:)
先看后一种情况,python中的所有变量传递都是传递引用(感觉类似c中的指针,即传递对象的地址),类似java中的非内置类型。所以引用本身不可变,但是被引用的内容是可以变化的。
所以list l中的内容在执行foo(l)后变化了。
个人暂时理解,python,java中不支持引用的引用吧,类似c中的二级指针,例如如何实现下面的函数呢
createTree(Node *&root)
感觉不能实现,虽然createTree可以利用返回值用其它的方式实现但是上面的形式似乎无法在python,java中实现。


那么对于上面的代码,第一种情况,为什么内容不变呢,
个人认为可以暂时这么理解,内存中有个地址x,存储'abc‘
l 是'abc'的一个引用(类似c中 int *l; l = x),在foo(l)的时候产生一个临时变量 它也指向 'abc'
但问题在于python中的string,int等是不可变的,
也就是说 'abc' + 'b'会在内存中新开一个地址,存储 'abcb' ,然后 foo作用域的l被重新赋值,作为'abcb'的引用,但是foo执行完,glbal域的l依然指向'abc'依然是 'abc'的引用,同时脱离foo作用域后,'abcb'所占用的空间应该已经被系统回收了。
这就是所谓对于函数的中变量赋值所带来的引用所指的内容可以变,引用本身不变,而偏偏string 是不可变的 'abc' + 'b'并没有改变'abc'而是新开空间存储了'abcb'同时赋值给local l 这个赋值并不改变global
中的l.


所以这也就解释了,为什么python,java中要实现类似c++中的swap
void swap(int &a, int &b)
需要采取一些work around的方法了。


所以
python,java中一切参数传递都是引用的传递,也可理解为传地址,对象的内容可以变,地址却不会变。
所以传过去的变量,其内容可以变化,但是地址不变。

def foo(a):
     a.left = 3
     a.right = 4
     b = A()
     b.left = 5
     b.right = 6
     a = b

#初始
a.left = 1
a.right = 1
#调用foo(a)
foo(a)
#之后a.left = 3 , a.right = 4


如果传过去的变量a的类型为A,可以理解为
struct A{
   int left;
   int right;
}
void foo(A *a)   

void foo(A *a)
    a->left = 3
    a->right = 4
    A *b = new A;
    b->left = 5
    b->right = 6
    a = b
运行完 foo(a)
a->left = 3, a->right = 4

那么考虑c++引用
void foo(A &a)
        a.left = 3
        a.right = 4
        A b;
        b.left = 5
        b.right = 6
        a = b

运行完 foo(a) a.left = 5, a.right = 6
这里体现的c++与python的不同在 a = b
python中只是引用的赋值,可以理解为对象地址的赋值,而C++这里 a = b是内容的赋值,是内容拷贝了(浅拷贝,严格复制内容)。
所以完成后a.left = 5,a.right = 6.而foo中的b运行完 foo就已经不存在了。
而python 只是引用赋值,并且利用引用计数的技术,一个对象对它的引用为0后会被回收。


如果要实现c++的引用效果,
swap(int &a, int &b)

只能采用 外包一个类如s s.a s.b
swap(s)
   temp = s.a
   s.a = s.b
   s.b = temp
s.a = a
s.b = b
swap(s)
a = s.b
b = s.a

或者用list 封装 a, b
最简单的写法 [a, b] = [b, a]

对于createTree(Node *&root)
createTree(Node *&root)
    root = new Node()
    createTree(root->left())
    createTree(root->right())

将 root 作为函数返回值

如果有多个 &,如 &num, &sum,那么也应该按list封装,或者封装到类中传类对象即可。
嗯,感觉在这种情况,c++写起来更方便写,不过python,java的处理也有它的道理,所有的传递都是传对象地址,统一处理,
传过去的变量,只会使其的内部内容变化,不会是它本身改变,本质上传递的参数还是要作为右值的,可读不可写(写无意义),
不会莫名其妙的运行一个函数后原来函数作用域的变量莫名其妙变化了。我们传递参数为了使用它而不是改变它。
当然代价是失去c++在这里处理方式的灵活性了。


09.9.14
另外补充下,
1. 在C++中其实大部分传递引用的函数例如下面,其实应该是只读的,如下  
void TestSkipList(DeterminSkipList &skip_list, ifstream &in, ofstream &out)   
函数里面会有skip_list.insert()等,也可能会改变skip_list所指向的对象的
内容,所以不可加 const 限制符。这里用引用是和python中传递引用是完全一致的概念情形,如果不传递引用就好有临时对象拷贝的产生,但是注意c++唯一的不同是
理论上skip_list = another_list 是对象拷贝,而对于python 是引用拷贝。所以会出现不同情形,注意区分即可。

1 #include
2 using namespace std;
3
4 class list {
5     public:
6         list(int a = 3): data(a){}
7         
8         int data;
9 };
10
11 void foo(list &l)
12 {
13     l.data = 4;
14     
15     list b(5);
16
17     l = b;
18 }
19
20
21 int main(int argc, char *argv[])
22 {
23     list a;
24
25     list* pl = &a;
26
27     foo(a);
28
29     list* pl2 = &a;
30
31     cout

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-61446-1-1.html 上篇帖子: python sample code 下篇帖子: windows下 python的安装配置
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表