Nor*_*sey 499

强类型语言和静态类型语言有什么区别?

静态类型语言具有类型系统,该类型系统在编译时由实现(编译器或解释器)检查.类型检查拒绝某些程序,通过检查的程序通常会带来一些保证; 例如,编译器保证不对浮点数使用整数算术指令.

虽然在专业文献中使用最广泛的定义是,在"强类型"语言中,程序员不可能解决类型系统所施加的限制,但对"强类型"的含义没有真正的一致意见. .该术语几乎总是用于描述静态类型语言.

静态与动态

与静态类型相反的是"动态类型",这意味着

  1. 运行时使用的值分为类型.
  2. 如何使用这些值存在限制.
  3. 违反这些限制时,违规将报告为(动态)类型错误.

例如,Lua是一种动态类型语言,具有字符串类型,数字类型和布尔类型等.在Lua中的每个值属于正好一个类型,但是这并不适用于所有动态类型语言的要求.在Lua中,允许连接两个字符串,但不允许连接字符串和布尔值.

强与弱

"强类型"的反义词是"弱类型",这意味着你可以解决类型系统.C是众所周知的弱类型,因为任何指针类型只需通过强制转换就可以转换为任何其他指针类型.Pascal的目的是强类型,但设计中的疏忽(未标记的变体记录)在类型系统中引入了一个漏洞,因此从技术上讲它是弱类型的.真正强类型语言的示例包括CLU,Standard ML和Haskell.事实上,标准ML经历了几次修订,以消除在语言被广泛部署后发现的类型系统中的漏洞.

这里到底发生了什么?

总的来说,谈论"强"和"弱"并没有那么有用.类型系统是否存在漏洞的重要性不如漏洞的确切数量和性质,它们在实践中出现的可能性,以及利用漏洞的后果是什么.在实践中,最好完全避免使用"强"和"弱"这两个术语,因为

  • 业余爱好者经常将他们与"静态"和"动态"混为一谈.

  • 显然,某些人使用"弱类型"来讨论隐式转换的相对普遍性或缺失.

  • 专业人士无法就这些术语的含义达成一致.

  • 总的来说,您不太可能通知或启发您的观众.

可悲的事实是,当涉及到类型系统时,"强"和"弱"没有普遍认同的技术含义. 如果您想讨论类型系统的相对强度,最好准确讨论哪些保证是什么,哪些不提供.例如,一个很好的问题是:"是否保证通过调用该类型的构造函数之一来创建给定类型(或类)的每个值?" 在C中答案是否定的.在CLU,F#和Haskell中,它是肯定的.对于C++,我不确定 - 我想知道.

相比之下,静态类型意味着在执行之前检查程序,并且程序在启动之前可能会被拒绝. 动态类型意味着执行期间检查的类型,并且类型不佳的操作可能导致程序在运行时停止或以其他方式发出错误信号.静态类型的主要原因是排除可能具有此类"动态类型错误"的程序.

一个人是否意味着另一个?

在学究层面,不,因为"强"这个词并不真正意味着什么.但在实践中,人们几乎总是做两件事之一:

  • 他们(错误地)使用"强"和"弱"来表示"静态"和"动态",在这种情况下,他们(错误地)使用"强类型"和"静态类型"可互换使用.

  • 他们使用"强"和"弱"来比较静态类型系统的属性.听到有人谈论"强"或"弱"动态类型系统是非常罕见的.除了没有任何类型系统的FORTH之外,我无法想到可以颠覆类型系统的动态类型语言.根据定义排序,这些检查是执行引擎,并且在执行之前检查每个操作的健全性.

无论哪种方式,如果一个人称一种语言为"强类型",那么这个人很可能会谈论一种静态类型的语言.

  • @Adam:显然不够正确,不能被投票:)因为Cletus的答案包含了很多误解(尽管我编辑了最糟糕的一些),我觉得有必要用一个音节的单词拼出所有内容...... (3认同)
  • 好吧,我给你投了一票:) 对于当今运行动态语言的虚拟机来说,甚至“编译”这个词也不是一个明确的词。从技术上讲,Java 和 C# 都经过两次编译 (JIT),并且都进行一些类型分析。由于 VM,在 .NET vm 中运行的像 Javascript 这样的语言可能更加类型安全。 (2认同)
  • 我现在很困惑!好吧,亲爱的竞技场的伟大角斗士,像我这样的可怜的灵魂能够遵循以下简单的理解吗?1.Static:值在编译期间与类型相关联而不是运行时间.2.动态:值在运行时与类型相关联,因此值的类型可以在运行时更改〜因此更容易出现与类型转换相关的问题在运行时.3.强弱:忘了!这些不是技术术语,只是不好的命名法.这取决于人们在谈论的背景.我能用这种简单的理解继续我的生活吗?:( (2认同)

cle*_*tus 236

这经常被误解,所以让我澄清一下.

静态/动态打字

静态类型是类型绑定到变量的位置.在编译时检查类型.

动态类型是类型绑定到值的位置.在运行时检查类型.

所以在Java中例如:

String s = "abcd";
Run Code Online (Sandbox Code Playgroud)

s将"永远"成为一个String.在它的生命中它可能指向不同的Strings(因为它s是Java中的参考).它可能有一个null值但它永远不会引用一个Integer或一个List.那是静态打字.

在PHP中:

$s = "abcd";          // $s is a string
$s = 123;             // $s is now an integer
$s = array(1, 2, 3);  // $s is now an array
$s = new DOMDocument; // $s is an instance of the DOMDocument class
Run Code Online (Sandbox Code Playgroud)

这是动态打字.

强/弱打字

(编辑提醒!)

强类型是一个没有广泛认同意义的短语.大多数程序员使用这个术语来表示静态类型以外的东西,使用它来暗示存在由编译器强制执行的类型规则.例如,CLU有一个强类型系统,除了使用类型提供的构造函数外,不允许客户端代码创建抽象类型的值.C有一个强大的类型系统,但它可以被"颠覆"到一定程度,因为程序总是可以将一个指针类型的值转换为另一个指针类型的值.因此,例如,在C中你可以获取一个返回的值,malloc()并愉快地将其转换为FILE*,并且编译器不会试图阻止你 - 甚至警告你你正在做任何狡猾的事情.

(最初的答案说的是"在运行时不改变类型"这个值.我认识很多语言设计师和编译器编写者,并且不知道在运行时谈论值改变类型的人,除了可能是一些非常先进的类型研究系统,这被称为"强烈更新问题".)

弱类型意味着编译器不强制执行输入discpline,或者可能很容易破坏强制执行.

这个答案的原始内容将弱类型与隐式转换(有时也称为"隐式提升")混为一谈.例如,在Java中:

String s = "abc" + 123; // "abc123";
Run Code Online (Sandbox Code Playgroud)

这是代码是隐式提升的示例:123在连接之前被隐式转换为字符串"abc".可以说Java编译器将该代码重写为:

String s = "abc" + new Integer(123).toString();
Run Code Online (Sandbox Code Playgroud)

考虑一个经典的PHP"开头"问题:

if (strpos('abcdef', 'abc') == false) {
  // not found
}
Run Code Online (Sandbox Code Playgroud)

这里的错误是strpos()返回匹配的索引,为0. 0被强制转换为布尔值false,因此条件实际为真.解决方案是使用===而不是==避免隐式转换.

这个例子说明了隐式转换和动态类型的组合如何导致程序员误入歧途.

与Ruby相比:

val = "abc" + 123
Run Code Online (Sandbox Code Playgroud)

这是一个运行时错误,因为在Ruby中,对象 123 没有被隐式转换,只是因为它碰巧被传递给一个+方法.在Ruby中,程序员必须明确转换:

val = "abc" + 123.to_s
Run Code Online (Sandbox Code Playgroud)

比较PHP和Ruby是一个很好的例子.两者都是动态类型语言,但PHP有很多隐式转换和Ruby(如果你不熟悉它可能会令人惊讶)不会.

静态/动态与强/弱

这里的要点是静态/动态轴独立于强/弱轴.人们可能会混淆他们,部分原因是因为强类型和弱类型不仅没有明确定义,对于确切的强弱意味着什么也没有真正的共识.因此,强/弱打字更像是灰色而不是黑色或白色.

因此,要回答你的问题:另一种方式来看待这个这主要是正确的是说,静态类型的编译时类型安全性和强大的打字运行时类型安全.

原因是静态类型语言中的变量具有必须声明的类型,并且可以在编译时检查.强类型语言的值在运行时具有类型,并且程序员很难在没有动态检查的情况下破坏类型系统.

但重要的是要理解语言可以是静态/强,静态/弱,动态/强或动态/弱.


Dar*_*mas 15

两者都是两个不同轴上的两极:

  • 强类型与弱类型
  • 静态类型与动态类型

强类型意味着,不会自动从一种类型转换为另一种类型.弱类型是相反的:Perl可以使用像"123"数字上下文中的字符串,通过自动将其转换为int 123.像python这样的强类型语言不会这样做.

静态类型意味着,编译器在编译时计算出每个变量的类型.动态类型语言仅在运行时计算出变量的类型.

  • 我不同意.强类型语言是在运行时知道类型是什么的语言.弱类型语言是不喜欢Assembly的语言.您的示例位于第三个轴上,即"隐式转换与显式转换". (5认同)
  • Python 是动态类型和强类型语言的示例 (5认同)
  • 实际上很正常,我同意乔纳森,但如果您进行完整的静态分析并且不允许强制转换,则不必在运行时拥有可用的类型来进行强类型化。(请参阅我编辑的答案)。 (2认同)

Bal*_*hna 13

上面已经给出了答案。试图区分强与周以及静态与动态概念。

什么是强类型 VS 弱类型?

强类型:不会自动从一种类型转换为另一种类型

在 Go 或 Python 之类的强类型语言中,“2”+8 会引发类型错误,因为它们不允许“类型强制”。

弱(松散)类型:将自动转换为一种类型到另一种类型: 弱类型语言(如 JavaScript 或 Perl)不会抛出错误,在这种情况下,JavaScript 的结果是“28”,而 perl 的结果是 10。

Perl 示例:

my $a = "2" + 8;
print $a,"\n";
Run Code Online (Sandbox Code Playgroud)

将其保存到 main.pl 并运行perl main.pl,您将获得输出 10。

什么是静态 VS 动态类型?

在编程中,程序员根据检查变量类型的点来定义静态类型和动态类型。静态类型语言是在编译时进行类型检查的语言,而动态类型语言是在运行时进行类型检查的语言。

  • 静态:在运行前检查的类型
  • 动态:在执行期间即时检查类型

这是什么意思?

在 Go 中,它会在运行时之前检查键入的内容(静态检查)。这意味着它不仅会翻译和类型检查正在执行的代码,还会扫描所有代码,甚至在代码运行之前就会抛出类型错误。例如,

package main

import "fmt"

func foo(a int) {
    if (a > 0) {
        fmt.Println("I am feeling lucky (maybe).")
    } else {
        fmt.Println("2" + 8)
    }
}

func main() {
    foo(2)
}
Run Code Online (Sandbox Code Playgroud)

将此文件保存在 main.go 中并运行它,您将收到编译失败消息。

go run main.go
# command-line-arguments
./main.go:9:25: cannot convert "2" (type untyped string) to type int
./main.go:9:25: invalid operation: "2" + 8 (mismatched types string and int)
Run Code Online (Sandbox Code Playgroud)

但这种情况对 Python 无效。例如,以下代码块将在第一次 foo(2) 调用时执行,而在第二次 foo(0) 调用时将失败。这是因为 Python 是动态类型的,它只对正在执行的代码进行翻译和类型检查。else 块永远不会为 foo(2) 执行,因此“2” + 8 甚至从未被查看过,对于 foo(0) 调用,它将尝试执行该块并失败。

def foo(a):
    if a > 0:
        print 'I am feeling lucky.'
    else:
        print "2" + 8
foo(2)
foo(0)
Run Code Online (Sandbox Code Playgroud)

您将看到以下输出

python main.py
I am feeling lucky.
Traceback (most recent call last):
  File "pyth.py", line 7, in <module>
    foo(0)
  File "pyth.py", line 5, in foo
    print "2" + 8
TypeError: cannot concatenate 'str' and 'int' objects
Run Code Online (Sandbox Code Playgroud)


Sve*_*lov 12

强类型意味着类型之间的转换之间存在限制.静态类型意味着类型不是动态的 - 一旦创建变量,就无法更改变量的类型.

  • 为了证明这一点:在强类型语言中,您无法比较“5”== 5并使其为真:字符串不是整数。如果我没记错的话,大多数现代脚本语言都是强动态类型的。然而,Tcl/Tk 是弱类型的——一切都可以被视为字符串。 (2认同)
  • 我必须不同意你们俩的观点。以Lua为例;您可以比较“5”== 5,它将返回 false,但是可以通过“5”+ 0 来完成快速转换。 (2认同)

Ada*_*ent 11

数据强制并不一定意味着弱类型,因为有时它的合成糖:

以上Java的例子因为弱的类型而被输入

String s = "abc" + 123;
Run Code Online (Sandbox Code Playgroud)

不是弱类型的例子,因为它真的在做:

String s = "abc" + new Integer(123).toString()
Run Code Online (Sandbox Code Playgroud)

如果要构造新对象,则数据强制也不是弱类型.Java是弱类型的一个非常糟糕的例子(并且任何具有良好反射的语言很可能不会被弱类型化).因为语言的运行时总是知道类型是什么(异常可能是本机类型).

这与C不同.C是弱类型的最好例子之一.运行时不知道4个字节是整数,结构,指针还是4个字符.

语言的运行时确实定义了它的弱类型,否则它的真正意义.

编辑: 经过进一步思考后,这不一定是正确的,因为运行时不必将运行时系统中所有类型都设置为强类型系统.Haskell和ML有这样完整的静态分析,他们可以从运行时省略类型信息.


Joe*_*tti 8

一个并不意味着另一个.对于静态类型的语言,这意味着所有变量的类型在编译时是已知的或推断的.

一个类型语言不允许你使用一种类型作为另一个.C是一种弱类型语言,是强类型语言不允许的一个很好的例子.在C中,您可以传递错误类型的数据元素,它不会抱怨.在强类型语言中,你不能.


Hig*_*ark 7

强类型可能意味着变量具有明确定义的类型,并且存在关于在表达式中组合不同类型的变量的严格规则.例如,如果A是整数而B是浮点数,那么关于A + B的严格规则可能是A被强制转换为浮点数并且结果作为浮点数返回.如果A是整数而B是字符串,则严格规则可能是A + B无效.

静态类型可能意味着类型在编译时分配(或其等效于非编译语言),并且在程序执行期间不能更改.

请注意,这些分类并不是相互排斥的,实际上我希望它们经常一起出现.许多强类型语言也是静态类型的.

请注意,当我使用"可能"这个词时,这是因为这些术语没有普遍接受的定义.正如您已经从目前为止的答案中看到的那样.