从一个低级错误说起,动态语言和静态语言

先从一个低级错误说起

上周,在项目中使用SQLAchemy操作数据库时,代码中出现了数据修改操作无法写入数据库的bug。为简化问题,就不贴原本的代码了,代码逻辑结构大致如下:

try:
      results = db.session.query(all_record).all()
      for result in results:
            result.nmae = 'new name'
            print(result.nmae)
      db.session.commit()
except Exception as e:
      print(e)

执行结果:

new name

大家知道,SQLAchemy是Python的一个用于实现ORM库(Object Relational Mapping,即对象关系映射),我们可以藉由它通过编程语言对多种类型的数据库实现数据表定义创建和对数据的增删改查。它将数据库中的关系转化为程序语言中的对象,因此所有对数据库的操作都体现为对对象属性的修改。
然而这段代码执行完后,并没有将数据写入数据表的对应字段,这种情况一般来说是没有执行commit提交事务,导致数据修改的结果并没有真正写入数据库。
但是在这里,我是明明写了commit()语句并且用断点的方式检查却是执行了的,而且也没有抛出异常。
排除了许多可能后,我才猛然发现,犯了一个写错变量名的低级错误。比如,这里代码操作的属性是 result.nmae ,在数据库中定义的本应该是name,在对象所对应的类里的属性自然也是name,故而在写入的时候没有相应字段可以写入,所以我去数据库查看name依然是以前的值。可是它也没有抛出异常,这个大概就是SQLAchemy的坑了。
但是,问题就来了,既然类中没有nmae这个属性,为什么可以被定义和赋值?
这似乎颠覆了我对面向对象的认知。经某大牛同学Talk is cheap ,Show me the code式的点播,我才猛然认识到这是Python作为一门动态语言的特性。

大牛code如下:

In [16]: class test:
    ...:     def __init__(self):
    ...:         self.A = 1
    ...:
    ...:

In [17]: t = test()

In [18]: t.A
Out[18]: 1

In [19]: t.B = 2

In [20]: t.__dict__
Out[20]: {'A': 1, 'B': 2}

In [21]: t.B
Out[21]: 2

Python类由类变量和实例变量、方法以及可以看作特殊变量的数据成员(类变量或者实例变量, 用于处理类及其实例对象的相关的数据)构成。其中,类变量为所有类产生实例所共用,比如一个类对应的实例个数的计数器。想要修改类变量,必须通过类内部特定的修改方法来修改。而作为实例变量,直接像上面那样,在实例化后赋值即可,甚至也可以定义一个类中不存在的新的对象属性。作为一个从JAVA入门面向对象的程序员来说,却是有些难以理解,因为JAVA并没有这种常规操作,javassit这种非常规技术就不说了。
造成这种差异和误解的根本原因是JAVA是一门静态语言,而Python是一门动态语言。而恰好,JAVA又是一门编译型的语言,而Python是一个用解释器执行的解释型语言。于是有人告诉我,编译型语言就是静态语言,而解释型语言就是动态语言。
事实果真如此么?我们一起来梳理下面几个概念!

编译型语言

需通过编译器(compiler)将源代码编译成机器码,之后才能执行的语言。一般需经过编译(compile)链接(linker)这两个步骤。编译是把源代码编译成机器码,链接是把各个模块的机器码和依赖库串连起来生成可执行文件。

  • 优点:一次编译,重复可用,不需要每次执行时再编译一遍,由于编译后生成的是能直接被机器执行的机器码,所以通常情况下执行效率会更高。
  • 缺点:编译之后如果需要修改就需要整个模块重新编译。编译的时候根据对应的运行环境生成机器码,不同的操作系统之间移植就会有问题,需要根据运行的操作系统环境编译不同的可执行文件(比如Windows上的.exe文件,CentOS上的.rpm文件)。
  • 代表语言:C、C++、Pascal、Object-C以及最近很火的苹果新语言swift

解释型语言

解释性语言的程序不需要编译,相比编译型语言省了道工序,解释性语言在运行程序的时候才逐行翻译为机器码来给机器执行。

  • 优点:可以在任何安装了解释器(虚拟机)的机器上运行,不用关心用了什么操作系统。灵活,修改代码的时候直接修改就可以,可以快速部署,不用停机维护。
  • 缺点:每次运行的时候都要解释一遍,性能上不如编译型语言。
  • 代表语言:JavaScript、Python、Erlang、PHP、Perl、Ruby

混合型语言

既然编译型和解释型各有缺点就会有人想到把两种类型整合起来,取其精华去其糟粕。就出现了半编译型语言。比如C#,C#在编译的时候不是直接编译成机器码而是中间码,.NET平台提供了中间语言运行库运行中间码,中间语言运行库类似于Java虚拟机。.net在编译成IL代码后,保存在dll中,首次运行时由JIT在编译成机器码缓存在内存中,下次直接执行。Java先生成字节码再在Java虚拟机中解释执行。严格来说混合型语言属于解释型语言。C#更接近编译型语言。

动态语言和静态语言

动态语言

是一类在运行时可以改变其结构的语言:例如新的函数、对象、甚至代码可以被引进,已有的函数可以被删除或是其他结构上的变化。通俗点说就是在运行时代码可以根据某些条件改变自身结构。
主要动态语言:Object-C、C#、JavaScript、PHP、Python、Erlang。

静态语言

与动态语言相对应的,运行时结构不可变的语言就是静态语言。如Java、C、C++。

走出误区

很多人认为解释型语言都是动态语言,这个观点是错的!Java是解释型语言但是不是动态语言,Java不能在运行的时候改变自己结构。反之成立吗?动态语言都是解释型语言。也是错的!Object-C是编译型语言,但是他是动态语言。得益于特有的run time机制(准确说run time不是语法特性是运行时环境,这里不展开)OC代码是可以在运行的时候插入、替换方法的。

静态类型语言和动态类型语言

为了不混淆动态语言和动态类型语言,下面还补充说明以下几个概念

动态类型语言

很多网上资料把动态类型语言和动态语言混为一谈,简直是误人子弟。动态类型语言和动态语言是完全不同的两个概念。动态类型语言是指在运行期间才去做数据类型检查的语言,说的是数据类型,动态语言说的是运行是改变结构,说的是代码结构。
动态类型语言的数据类型不是在编译阶段决定的,而是把类型绑定延后到了运行阶段。
主要语言:Python、Ruby、Erlang、JavaScript、swift、PHP、Perl。

静态类型语言

静态语言的数据类型是在编译其间确定的或者说运行之前确定的,编写代码的时候要明确确定变量的数据类型。
主要语言:C、C++、C#、Java、Object-C。

注意

解释型语言并不都是动态类型语言,编译型语言也不全是静态类型语言。swift是编译型语言但是它也是动态类型语言。C#和Java是解释型语言也是静态类型语言,虽然它们看作解释型语言,但是它们也有编译过程,会在编译过程中做数据类型的检查
所以,动态类型语言是数据类型检查的动态(运行时检查),而不是像动态语言一样代码逻辑结构的在运行时可变更,静态则相反。

强类型语言和弱类型语言

说到数据类型,就不得不提从数据类型是否严格定义的角度产生的另外两个概念:强类型语言和弱类型语言

强类型语言:

强类型语言,一旦一个变量被指定了某个数据类型,如果不经过强制类型转换,那么它就永远是这个数据类型。你不能把一个整形变量当成一个字符串来处理。
主要语言:Java、C#、Python、Object-C、Ruby

弱类型语言:

数据类型可以被忽略,一个变量可以赋不同数据类型的值。一旦给一个整型变量a赋一个字符串值,那么a就变成字符类型。
主要语言:JavaScript、PHP

3、注意:

一个语言是不是强类型语言和是不是动态类型语言也没有必然联系。Python是动态类型语言,是强类型语言。JavaScript是动态类型语言,是弱类型语言。Java是静态类型语言,是强类型语言。

附录概念对照表

概念 划分角度 特点
编译型语言 执行方式 将源代码编译为机器码给机器执行
解释型语言 执行方式 将源代码交由解释器解释为机器码执行
混合型语言 执行方式 将源代码编译为“中间码”,再交由解释器(虚拟机)解释为机器码执行
动态语言 代码逻辑结构是否可以在运行时改变 可以改变
静态语言 代码逻辑结构是否可以在运行时改变 不可改变
动态类型语言 代码的数据类型何时检查 运行时检查
静态类型语言 代码的数据类型何时检查 运行时不检查
强类型语言 变量定义后是否可以随意改变数据类型 不可以
弱类型语言 变量定义后是否可以随意改变数据类型 可以
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • TITLE: 编程语言乱炖 码农最大的烦恼——编程语言太多。不是我不学习,这世界变化快! 有时候还是蛮怀念十几、二...
    码园老农阅读 5,309评论 2 35
  • 天堂的名单里,没有他的名字。他从未出生,从未死亡,从未存在。1900,海上钢琴师,一个传奇中的传奇。 《海上钢琴师...
    渚清沙阅读 900评论 3 5
  • 今天是我们芳老大亲自讲课,每一次听她讲课感觉都是一种享受!并且收获颇多、每一次都是直切主题,让我不禁恍然大悟!从现...
    FAB小太阳阅读 107评论 0 0
  • 亲爱的Chloe: 见信如面,一切安好!不知深圳近几日天气如何?九月的湖南仍是骄阳似火,写信当时窗外大风起...
    大花是个好名字阅读 375评论 0 0