通过C++学习Python
我会随便说,C++近年来开始"抄袭"Python么?我只会说,我在用C++来学习Python.
不信?来跟着我学?
字面量
Python早在2.6版本中就支持将二进制作为字面量了1,最近C++14逐步成熟,刚刚支持这么干2:
staticconstintprimes=0b10100000100010100010100010101100;
更不用说Python在1.5时代就有了rawstringliterals的概念3,咱们C++也不算晚,C++11里也有了类似做法:
constchar*path=r"C:\Python27\Doc"; RangeLoop
Python写for循环是一件非常舒畅的事情:
forxinmylist: print(x);
大家都知道了,C++11里我总算也能做同样的事情了:
for(intx:mylist) std::cout<<x;
类型自动推导
Python中真的有类型的概念吗?(笑
x="HelloWorld" print(x)
C++11也学会了这招,只不过保留了老太太的裹脚布(auto)。
autox="HelloWorld"; std::cout<<x;
元组
Python里的元组(tuple)让人羡慕已久,这玩意Python从一开始就有了。
triple=(5,"Hello",True) print(triple[0])
好嘛,我来用C++11照猫画虎:
autotriple=std::make_tuple(5,"hello",true); std::cout<<std::get<0>(triple);
有人说了,Python大法好,还能逆向解析成变量呢
x,y,z=triple
哼,C++难道不行?
std::tie(x,y,z)=triple;
Lists
Python里,Lists是内置类型4,创建一个list无比简单:
mylist=[1,2,3,4] mylist.append(5);
以前我们可以说,这有啥,std::vector差不多也能干这事。可Python粉较真了,您能像上面那样初始化吗?这话让BjarneStroustrup老爹听到了,暗自羞愧,于是在C++11里整出了个initializer_list做出回应5。
automylist=std::vector<int>{1,2,3,4}; mylist.push_back(5);
可人又说了,Python里创造个Dictionary简单的跟什么一样6。
myDict={5:"foo",6:"bar"} print(myDict[5])
切,C++本身就有map类型,现在又多了个哈希表unordered_map,更像了:
automyDict=std::unordered_map<int,constchar*>{{5,"foo"},{6,"bar"}}; std::cout<<myDict[5];
Lambda表达式
Python祭出大神器,1994年就有的Lambda表达式:
mylist.sort(key=lambdax:abs(x))
C++11开始了拙劣的模仿:
std::sort(mylist.begin(),mylist.end(),[](intx,inty){returnstd::abs(x)<std::abs(y);});
而Python在2001年加了一把力,引入了NestedScopes的技术7:
defadder(amount): returnlambdax:x+amount ... print(adder(5)(5))
C++11不甘示弱,整出了capture-list的概念8。
autoadder(intamount){ return[=](intx){returnx+amount;}; } ... std::cout<<adder(5)(5);
内置算法
Python里有诸多内置的强大算法函数,如filter:
result=filter(mylist,lambdax:x>=0)
C++11倒也可以用std::copy_if干同样的事情:
autoresult=std::vector<int>{}; std::copy_if(mylist.begin(),mylist.end(),std::back_inserter(result),[](intx){returnx>=0;});
这样的函数在<algorithm>中屡见不鲜,而且都在与Python中的某种功能遥相呼应:transform,any_of,all_of,min,max.
可变参数
Python从一开始就支持可变参数了。你可以定义一个变参的函数,个数可以不确定,类型也可以不一样。
deffoo(*args): forxinargs: print(x); foo(5,"hello",True)
C++11里initializer_list可以支持同类型个数可变的参数(C++Primer5th6.2.6)。
voidfoo(std::initializer_list<int>il){ for(autox:il) std::cout<<x; }
foo({4,5,6});