Python学习记录

一、Python调用C++动态链接库示例

首先用C++来实现一个数学加法的动态库,然后用python来调用这个库

编写mymath.h

class MyMath
{
public:
        int myadd(int a, int b);

};

编写mymath.cpp

#include "mymath.h"

int MyMath::myadd(int a, int b)
{
        return a + b;
}

编写文件main.cpp

#include <iostream>
#include "mymath.h"
using namespace std;

//如果没有extern "C"的话就在被Python调用的时候报错
extern "C"
{
        int myfoo(int a, int b)
        {
                cout << "In Myfoo Function..." << endl;
                MyMath cmm;
                int n = cmm.myadd(a, b);
                cout << "a = " << a << " ;b = " << b << " ;a + b = " << n << endl;
                return n;
        }
}

int main()
{
        cout << "In Main Function..." << endl;
        return 0;
}

然后编译为动态库

g++ main.cpp mymath.h mymath.cpp -fPIC -shared -o libtest.so  

打开python终端或者编写python脚本(注意文件夹位置)

import ctypes;
so = ctypes.CDLL("./libtest.so")
so.myfoo(2, 3)

结果输出:

In Myfoo Function…
a = 2 ;b = 3 ;a + b = 5

其他调用方式参考网址




二、import math和from math import *的区别

import math 只添加math到当前命名空间
比如说使用pi的时候要:

print math.pi

from math import * 会把math模块所有名字导入到当前命名空间,很容易产生冲突,建议少用,或者使用“_all_”来限制出口
比如在使用pi的时候可以直接使用

print pi




以下内容摘抄自廖雪峰的个人网站,若是有不明白的地方请查看原网址上下文

三、“可变的tuple”

来看一个“可变的”tuple:

>>> t = ('a', 'b', ['A', 'B'])
>>> t[2][0] = 'X'
>>> t[2][1] = 'Y'
>>> t
('a', 'b', ['X', 'Y'])

这个tuple定义的时候有3个元素,分别是’a’,’b’和一个list。不是说tuple一旦定义后就不可变了吗?怎么后来又变了?
别急,我们先看看定义的时候tuple包含的3个元素:
这里写图片描述

当我们把list的元素’A’和’B’修改为’X’和’Y’后,tuple变为:

这里写图片描述

表面上看,tuple的元素确实变了,但其实变的不是tuple的元素,而是list的元素。tuple一开始指向的list并没有改成别的list,所以,tuple所谓的“不变”是说,tuple的每个元素,指向永远不变。即指向’a’,就不能改成指向’b’,指向一个list,就不能改成指向其他对象,但指向的这个list本身是可变的!

理解了“指向不变”后,要创建一个内容也不变的tuple怎么做?那就必须保证tuple的每一个元素本身也不能变。

个人添加注解:
t[1] = ‘D’ 错误,因为tuple不可修改
t[2] = [‘C’, ‘D’] 错误, 原因同上
但是t[2][0] = ‘C’ 可以执行

原网址




四、默认参数的坑

默认参数很有用,但使用不当,也会掉坑里。默认参数有个最大的坑,演示如下:
先定义一个函数,传入一个list,添加一个END再返回:

defadd_end(L=[]):
    L.append('END')
    return L

当你正常调用时,结果似乎不错:

>>> add_end([1, 2, 3])
[1, 2, 3, 'END']
>>> add_end(['x', 'y', 'z'])
['x', 'y', 'z', 'END']

当你使用默认参数调用时,一开始结果也是对的:

>>> add_end()
['END']

但是,再次调用add_end()时,结果就不对了:

>>> add_end()
['END', 'END']
>>> add_end()
['END', 'END', 'END']

很多初学者很疑惑,默认参数是[],但是函数似乎每次都“记住了”上次添加了’END’后的list。

原因解释如下:
Python函数在定义的时候,默认参数L的值就被计算出来了,即[],因为默认参数L也是一个变量,它指向对象[],每次调用该函数,如果改变了L的内容,则下次调用时,默认参数的内容就变了,不再是函数定义时的[]了。

所以,定义默认参数要牢记一点:默认参数必须指向不变对象!
要修改上面的例子,我们可以用None这个不变对象来实现:

defadd_end(L=None):
    if L isNone:
        L = []
    L.append('END')
    return L

现在,无论调用多少次,都不会有问题:

>>> add_end()
['END']
>>> add_end()
['END']
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值