说三道四技术文摘-感悟人生的经典句子
说三道四 > 文档快照

多面编程语言Scala

HTML文档下载 WORD文档下载 PDF文档下载
Scala把近二十年间大行其道的面向对象编程与旧而有之的函数式编程有机结合起来,形成其独特的魔力。本文将带你领略Scala独特魅力,练就自己的寒冰掌、火焰刀。

如Scala官网宣称的:“Object-OrientedMeetsFunctional”,这一句当属对Scala最抽象的精准描述,它把近二十年间大行其道的面向对象编程与旧而有之的函数式编程有机结合起来,形成其独特的魔力。希望通过本文能够吸引你去了解、尝试Scala,体验一下其独特魅力,练就自己的寒冰掌、火焰刀。

回首初次接触Scala,时光已忽忽过去四五年。从当初“Scala取代Java”的争论,到今天两者的相安无事,Scala带给了我们哪些有意义的尝试呢?在我掌握的众多编程语言之中,Scala无疑是其中最让我感到舒适的,如Scala官网宣称的:“Object-OrientedMeetsFunctional”,这一句当属对Scala最抽象的精准描述,它把近二十年间大行其道的面向对象编程与旧而有之的函数式编程有机结合起来,形成其独特的魔力。不知你是否看过梁羽生的著作《绝塞传烽录》?里面白驼山主宇文博的绝学:左手“寒冰掌”、右手“火焰刀”,用来形容Scala最为合适了,能够将OOP与FP结合得如此完美的语言,我认为唯有Scala。

众所周知,Java称不上纯粹的面向对象语言,但Scala却拥有纯粹的面向对象特性,即便是1+1这么简单的事情,实际上也是执行1.+(1)。而在对象组合方面,Scala拥有比接口更加强大的武器──特质(trait)。

Scala同时作为一门函数式编程语言,理所当然地具备了函数式语言的函数为头等“公民”、方法无副作用等特性。事实上,Scala更吸引我的并不是OOP特性,而是FP特性!一边是OOP、一边是FP,这就是多面的Scala,极具魅力而且功能强大。

在多核时代,现代并发语言不断涌现出来,例如Erlang、Go、Rust,Scala当然也位列其中。Scala的并发特性,堪称Scala最吸引开发者的招牌式特性!Scala是静态类型的。许多人会把vals="ABC"这样的当作动态类型特性,而vals:String="ABC"才认为是静态类型特性。实际上,这无关类型争论,而是类型系统实现的范畴。是的,在Scala里,你可以放心大胆地使用vals="ABC",而Scala里强大的类型推断和模式匹配,绝对会让你爱不释手。

此外,Scala作为JVM语言,理所当然享有Java庞大而优质的资源,与Java间可实现无缝交互,事实上,Scala最终当然是编译为Java字节码。

本文将把重点放在Scala的特色之处。作为一门完备而日趋成熟的语言,Scala的知识点有不少,本文当然无法做到面面俱到,但希望能够带你感受Scala魅力,并理解其重要概念。

Scala的面向对象

开胃菜──类的定义

来看个开胃菜,定义一个类:


我们知道,动态语言一般都提供了REPL环境,同时,动态语言的程序代码都是以脚本方式解释运行的,这给开发带来了不少的便利。Scala虽然是静态类型系统的语言,但同样提供了这两个福利,让你倍感贴心。

因此,你可以任意采取以下运行方式:

  • 在命令行窗口或终端输入:scala,进入Scala的REPL窗口,逐行运行上述代码;
  • 此外,也可以将上述代码放入某个后缀名为.scala的文件里,如test.scala,然后通过脚本运行方式运行:scala test.scala。

测试信息“小强今年32岁,是一名程序员”结果出来了!

多么简单,类的定义就这么多,却能够做这么多事情,想想Java的实现吧,差别太大了。我们先来分析下代码。假设在上述第二种方式的test.scala文件中,注释掉后面两行并保存,运行:

  • scalac test.scala 
  • javap -p Person
我们先是把文件编译成字节码(这实际上是跟Java编译对应的第三种编译/运行方式),之后反编译并查看结果:


这个结果跟Java实现的代码类似(生成的getter和 setter跟Java实现有所不同,但在这里不是什么问题),可见,Scala帮我们做了多少简化工作。这段代码有以下值得注意的地方:

我们可以把字段定义和构造函数直接写在Scala的类定义里,其中,关键字val的含义是“不可变”,var 为“可变”,Scala的惯用法是优先考虑val,因为这更 贴近函数式编程风格; 

  • 在Scala中,语句末尾的分号是可选的;
  • Scala默认类访问修饰符为public;
  • 注意println("测试信息")这一行,将在主构造函数里执行;
  • val与var两者对应Java声明的差异性已在反编译代码中体现了。
伴生对象与伴生类

伴生对象与伴生类在Scala的面向对象编程方法中占据极其重要的位置,例如Scala中许多工具方法都是由伴 生对象提供的。

伴生对象首先是一个单例对象,单例对象用关键字object定义。在Scala中,单例对象分为两种,一种是并未自动关联到特定类上的单例对象,称为独立对象 (Standalone Object);另一种是关联到一个类上的单例对象,该单例对象与该类共有相同名字,则这种单例对象称为伴生对象(Companion Object),对应类称为伴生类。 

Java中的类,可以既有静态成员,又有实例成员。而在Scala中没有静态成员(静态字段和静态方法),因为静态成员从严格意义而言是破坏面向对象纯洁性的,因此,Scala借助伴生对象来完整支持类一级的属 性和操作。伴生类和伴生对象间可以相互访问对方的 private字段和方法。

接下来看一个伴生类和伴生对象的例子(Person. scala)。


这是一个典型的伴生类和伴生对象的例子,注意以下说明:

  • 伴生类Person的构造函数定义为private,虽然这不是必须的,却可以有效防止外部实例化Person类,使得Person类只能供对应伴生对象使用;
  • 每个类都可以有伴生对象,伴生类与伴生对象写在同一个文件中;
  • 在伴生类中,可以访问伴生对象的private字段Person.uniqueSkill;
  • 而在伴生对象中,也可以访问伴生类的private方法Person.getUniqueSkill();
  • 最后,在外部不用实例化,直接通过伴生对象访问Person.printUniqueSkill()方法。
特质(Trait)

Scala的特质类似于Java中的接口作用,专门用来解决现实编程中的横切关注点矛盾,可以在类或实例中混入(Mixin)这些特质。实际上,特质最终会被编译成Java的接口及相应的实现类。Scala的特质提供的特性远比Java的接口灵活,让我们直接来看点有趣的东西吧。



我们先是定义了一个Programmer抽象类。最后定义了四个不同程序员的Trait,且都继承自Programmer抽象类,然后,通过不同的特质排列组合,看看我们产生的结果是什么样子的:

所有程序员都至少掌握一门编程语言。

我掌握Scala。我掌握Golang。

所有程序员都至少掌握一门编程语言。

我掌握Scala。我掌握Golang。我掌握PHP。......

Wow~!有趣的事情发生了,通过混入不同的特质组合,不同的程序员都可以有合适的词来介绍自己,而每个程序员的共性就是:“所有程序员都至少掌握一门编程语言”。让我们来解释一下具体思路:

这段代码里面,特质通过with混入实例,如:new  Programmer with Scalaist。当然,特质也可以混入类中;

  • 为什么信息可以传递呢?比如我掌握Scala。我掌握Golang。我掌握PHP?答案就在super.getSkill()上。该调用不是对父类的调用,而是对其左边混入的Trait的调用,如果到左边第一个,就是调用Programmer抽象类的getSkill()方法。这是Trait的一个链式延时绑定特性,那么在现实中,这个特性就表现出极大的灵活性,可以根据需要任意搭配,大大降低代码量。

Scala的面向对象特性,暂先介绍到这里。其实还有好些内容,限于篇幅,实在是有点意犹未尽的感觉。

Scala的函数式风格

Scala的魅力之一就是其函数式编程风格实现。如果把上面介绍的面向对象特性看成是Scala的“寒冰掌”,让你感受到了迥异于Java实现的特性,那么,Scala强大而魔幻的函数式特性,就是其另一大杀招“火焰刀”,喷发的是无坚不摧的怒焰之火。

集合类型

Scala常用集合类型有Array、Set、Map、Tuple和List等。Scala提供了可变(mutable)与不可变(immutable)的集合类型版本,多线程应用中应该使用不可变版本,这很容易理解。

  • Array:数组是可变的同类对象序列;
  • Set:无序不重复集合类型,有可变和不可变实现;
  • Map:键值对的映射,有可变和不可变实现;
  • Tuple:可以包含不同类元素,不可变实现;
  • List:Scala的列表是不可变实现的同类对象序列,因应函数式编程特性的需要。
  • List大概是日常开发中使用最多的集合类型了。

这些集合类型包含了许多高阶函数,如:map、find、filter、fold、reduce等等,构建出浓郁的函数式风格用法,接下来我们就来简单了解一下:


输出如下:

JavaScript很棒~

Scala很棒~

Golang很棒~

map()函数在List上迭代,对List中的每个元素,都会调用以参数形式传入的Lambda表达式(或者叫匿名函数)。其结果是创建一个新的List,其元素内容都发生了相应改变,可以从输出结果观察到。注意,代码中有一行是速写法代码,我个人比较喜欢这种形式,但在复杂代码中可读性差一些。

最后,我们用了另一个foreach()方法来迭代输出结果。

高阶函数、Lambda表达式,都是纯正的函数式编程风格。如果你接触过Haskell,就会发现Scala函数式风格的实现,在骨子里像极了Haskell,感觉非常亲切。在编写Scala代码的过程中,将处处体现出它的函数式编程风格,高效而简洁。

限于篇幅,我们只能浅尝辄止,如果有兴趣,可以进一步参考我以前写的两篇相关博文,里面有比较详细的描述:七八个函数,两三门语言㈠和七八个函数,两三门语言㈡•完结篇。

高阶函数、柯里化、不全函数和闭包

实际上我们在前面已经见识过Scala的高阶函数(Higher-order Function)了,只不过是Scala自带的map()和foreach()。高阶函数在维基百科中的定义 是:“高阶函数是至少满足下列一个条件的函数:接 受函数作为输入;输出一个函数”。接下来,我们来实现一个自己的高阶函数──求圆周 长和圆面积:


我们定义了一个高阶函数cycle。输入参数中传入一个函数值calc,其类型是函数,接收Float输入,输出也是Float。在实现里,我们会调用calc函数。在调用时,我们分别传入求圆周长和圆面积的匿名函数,用于实现calc函数的逻辑。

这样,我们用一个高阶函数cycle,就可以满足求圆周长和圆面积的需求,不需要分别定义两个函数来处理不同任务,而且代码直观简洁。最后,我们打印结果,输出一组半径分别对应的圆周长和圆面积。在这里,我们用到了映射Map:

圆周长:Map(1.0 -> 6.28, 2.3 -> 14.444, 4.5 -> 28.26) 

圆面积:Map(1.0 -> 3.14, 2.3 -> 16.6106, 4.5 ->63.585) 

接下来,我们对上述代码稍加改动:


输出结果同上。

注意到了吗?我们把cycle函数的两个输入参数进行了拆分(如上述代码第一行),同时在调用cycle函数时,方式也有所不同(如上述代码最后两行)。这是什么意思? 

这在函数式编程中称为柯里化(Curry),柯里化可以把函数定义中原有的一个参数列表转变为接收多个参数列表。在函数式编程中,一个参数列表里含多个参数的函数都是柯里函数,可以柯里化。

要知道,在函数式编程里,函数是一等的,当然函数也可以作为参数和返回被传递。这对初次接触函数式编程的开发者而言确实比较抽象。上述代码的理解,你可以这样想象:(cacl: Float => Float)是函数cycle2(r: Array[Float])的输入参数!进一步,可以这么理解:cacl取一个参数,变成了一个不全函数(Partially Function)cycle2(r: Array[Float]),所谓不全函数就是它还有参数未确定,你想要完整用它的话,还需要继续告知它未定的参数,如(cacl: Float => Float)。 

还没完!根据上述描述,我们继续看看如何用各种Hacker的调用方式:


可以用valc21=cycle2 _、val c22 = cycle2(Array(1.0f, 2.3f, 4.5f)) _诸如此类的方式创建不全函数,并调用它。

看得出来,不全函数同样可以提升代码的简洁程度,比如本例代码中,参数Array(1.0f, 2.3f, 4.5f)是固定不变的,我们就不用每次都在调用cycle2时传入它,可以先定义c22,再用c22来处理。 

函数式崇尚的“函数是第一等公民”理念可不容小觑。函数,就是这么任性!接下来,我们来了解下闭包(Closure)的概念,依旧先看个简单的例子:


这个例子用来求圆柱体的体积。这里定义了一个caclCylinderVolume函数(因为函数式风格里函数是一等公民,所以可以用这样的函数字面量方式来定义。或者也可以称之为代码块),函数里面引用了一个自由变量high,caclCylinderVolume函数并未绑定high。而在caclCylinderVolume函数运行时,要先“闭合”函数及其所引用变量high的外部上下文,这样也就绑定了变量high,此时绑定了变量high的函数对象称为闭包。

由代码可知,由于函数绑定到了变量high本身,因此,high如果发生改变,将影响函数的运算结果;而如果在函数里更新了变量,那这种更新在函数之外也会被体现。

模式匹配(PatternMatching

Scala的模式匹配实现非常强大。模式匹配为编程过程带来了莫大便利,在Scala并发编程中也得到了广泛应用。


输出结果如下:

多面者Scala~

你的Scala版本是:2.11.6

八成是干净简洁的Go、PHP语言呢?

可见,模式匹配特性非常好用,可以灵活应对许多复杂的应用场景:

  • 第一个case表达式匹配普通的字面量; 
  • 第二个case表达式匹配正则表达式;
  • 第三个case表达式使用了if判断,这种方式称为模式护卫(Pattern Guard),可以对匹配条件加以过滤;
  • 第四个case表达式使用了“_”来处理未匹配前面几项的情况。

此外,Scala的模式匹配还有更多用法,如case类匹配、option类型匹配,同时还能带入变量,匹配各种集合类型。综合运用模式匹配,能够极大提升开发效率。

并发编程

现代语言的特性往往是随硬件环境和技术趋势演进的,多核时代的来临,互联网大规模复杂业务处理,都对传统语言提出了挑战,于是,新展现的语言几乎都非常关注并发特性,Scala亦然。

Scala语言并发设计采用Actor模型,借鉴了Erlang的Actor实现,并且在Scala2.10之后,改为使用AkkaActor模型库。Actor模型主要特征如下:

  • “一切皆是参与者”,且各个actor间是独立的;
  • 发送者与已发送消息间解耦,这是Actor模型显著特点,据此实现异步通信;
  • actor是封装状态和行为的对象,通过消息交换进行相互通信,交换的消息存放在接收方的邮箱中;actor可以有父子关系,父actor可以监管子actor,子actor唯一的监管者就是父actor;
  • 一个actor就是一个容器,它包含了状态、行为、一个邮箱(邮箱用来接受消息)、子actor和一个监管策略。

    我们先来看个例子感受下:



    在这里,Concurrency是CalcActor的父actor。在Concurrency中先要构建一个Akka系统:


    同时,这里的设置将会在线程池里初始化称为“routee”的子actor(这里是CalcActor),数量为4,也就是我们需要4个CalcActor实例参与并发计算。这一步很关键。actor是一个容器,使用actorOf来创建Actor实例时,也就意味着需指定具体Actor实例,即指定哪个actor在执行任务,该actor必然要有“身份”标识,否则怎么指定呢?!

    在Concurrency中通过以下代码向CalcActor发送序号并启动并发计算:

    for(i<-1to4)calcActor!i

    然后,在CalcActor的receive中,通过模式匹配,对接收值进行处理,直到接收值处理完成。在运行结果就会发现每次输出的顺序都是不一样的,因为我们的程序是并发计算。比如某次的运行结果如下。

    • 序号为:1。
    • 序号为:3。
    • 序号为:2。
    • 序号为:4。

    actor是异步的,因为发送者与已发送消息间实现了解耦;在整个运算过程中,我们很容易理解发送者与已发送消息间的解耦特征,发送者和接收者各种关心自己要处理的任务即可,比如状态和行为处理、发送的时机与内容、接收消息的时机与内容等。当然,actor确实是一个容器,且五脏俱全:我们用类来封装,里面也封装了必须的逻辑方法。Akka基于JVM,虽然可以穿插混合应用函数式风格,但实现模式是面向对象,天然讲究抽象与封装,其当然也能应用于Java语言。我们的Scala之旅就要告一个段落了!Scala功能丰富而具有一定挑战度,上述三块内容,每一块都值得扩展详述,但由于篇幅关系,在此无法一一展开。

    希望通过本文能够吸引你去了解、尝试Scala,体验一下其独特魅力,练就自己的寒冰掌、火焰刀。



    作者简介:卢俊祥,网名2gua,译者,书迷;关注Web技术趋势,热衷App开发、Web开发、数据分析、架构设计以及各类编程语言;陈氏太极拳五十六式爱好者;佛禅人生,缘散缘聚。微博: @2gua、 博客。


    Paul Graham:创业者如何说服投资人? 【CTO俱乐部深圳站】企业大数据建设案例分享 Gmail新设计影响邮件营销 Fcaebook间接获利 百度高德小米网易杭研讲师登场 SDCC 2013第二批议题发布(图) [开源推荐]JS-Git:一款纯JavaScript实现的Git项目 良心企业Lavabit、Slient Mail相继关闭,美国云安全已死? 一周观察:LXC真的是PaaS的未来吗? [企业开源系列]Google背后的开源力量 合作: “云”潮中淘金的金钥匙 Windows 8.1尚未发布 Build 9471版遭泄露网爆高清大图 Imangi创始人:我眼中的App Store 职业设计生涯:打造世界上最小巧最强大手电 【SDCC 2013】Coding For Fun活动火热招募中! YARN之上的大数据框架REEF:微软出品,是否值得期待? 创建大数据公司之前你需知晓5件事 前途渺茫,考虑出售的黑莓还能找到买家吗? 帮助开发者轻松管理项目的8款在线工具 现代Web应用开发者必备的六大技能 JavaScript异步编程好助手:Promise模式 功能解析:Red语言推出0.3.3新版本 可穿戴?可吞咽!UI和UX的悄然转变 华为刘成:讲述开发者联盟的故事 出尔反尔,Amazon咬文嚼字玩转私有云 创业公司SOASTA:曾帮助伦敦奥运会官网进行云负载测试 金山云进一步聚焦云存储 Facebook公布用户数据:美国移动用户占比78% Zynga开源服务器监控工具zPerfmon 上海海事大学王洁:打造物流教学科研应用云平台生态系统 15个步骤创立技术公司,并收获千万用户(二) HTML5发展慢热 开发者热情不减 NimbleBit孪生兄弟创始人:最后悔没有专注和扩大品牌 请教高手一个ado的问题. 请问一个TWebBrowser的问题 请问:系统崩溃后,找不到BOOT,系统告诉:DOS基本分区为格式化?无法引导启动! 在VB.NET的tabcontrol上放置AxMSFlexGrid控件问题,急 java3D 现在大家是用Oracle 8还是 Oracle 9??? C++问题求助! 请教一个错误的原因. 各位大侠请问Jeffrey Richter的windows核心程序设计是不是就是andvanced windows程序设计的第四版 SCJP里的那些什么300-020,020是什么意思啊? 请问一个关于document.write()方法的问题。急啊! 初学者,欢迎灌水。 Delphi6的Bug?不兼容? 为什么我用GetProfileString()却总是得不到结果 sos 请教各位大侠一个问题 delphi6中的installShield 请介绍接收端CRC校验传输内容是否正确的方法,资料或代码。谢谢! 各位大侠请问Jeffrey Richter的windows核心程序设计是不是就是andvanced windows程序设计的第四版? 谁会做软件狗?有钱赚了!!! 为什么我用QuickRep作的报表只能显示一条记录,而且columnHeaderBand上内容也显示不出来 请问在税务局开发数据库的同行么?? 大家好各位大侠请问Jeffrey Richter的windows核心程序设计是不是就是andvanced windows程序设计的第四版? 谁做过软件狗,马上来拿钱!!! 江苏省的成绩什么时候可以出来啊,我是徐州的 shockwaveflash没有hwnd,怎么办? chechy(chechy) 你在吗? 一个关于用excel做的日报的问题 谁会做软件狗,马上来拿钱!!! 给你下跪了,给你磕头了,给你鞠躬了。很简单的问题。谢谢 那位大哥能告诉我在VC中我要创建类的话应该做哪些工作步骤(要详细的,呵呵) 紧急求救,asp连接mysql问题 在使用twebbrowser的时候,怎样制止网页的自动刷新和跳转? vc编译器错了怎么办???突然死机,然后就不好好干了!!谢谢 ?如何将二进制的 值写如注册表? Delphi紧急求助! 我想问的是,返回的 ResultSet 中的某个字段(int 型),如何放到一个 int[]数组里去。因为 ResultSet 不知道怎样取 size,我又不想用 vector, ResultSet的getArray()方法也是莫名其妙。 c盘格式化后,怎样找到linux系统? 有意思 过来研究一下!!!!insert 的!!!!难道 25 不是 long integer 吗 我的程序debug版,一直调试好好的,我rebuid all 一下就出错了,错误如下:怎么回事? 位图问题? 怎样使用自定义的光标??? 我想找Hopen SDK的开发包。 为什么我的iis总是报500内部服务器错误。我先装的framework,后来升级了ie5.5,最后在w2kprofessional上装了iis,手动添加了aspx解析:指向aspnet_isapi.dll,为什么还报错? 那里有关于用组件上传图片并且显示上传图片的介绍 NT 上已安装好mysql服务器,WIN98 client连接不上,求救 程序员梦难圆,请大家给我这个非计算机专业的兄弟出出主意。 小弟是delphi初学者,师兄师姐们给我介绍一下object pascal自带的函数好么? 你知道如何禁止后退按钮吗? 请各位高高手!解决启动时ctrl+alt+del可结束任务的问题。 哪位大侠指点一下ActiveSkin怎么用的? 引用ado后,打包程序为什么如此之大 关于压力容器设计订货方给的工作参数怎么来分析?比如说排污扩容器 容积:3.5方 工作压力0.15MPA 工作温度≤250℃ 怎么分析这些数据来做CDA图纸 压力容器由锅检所检验质量,膨胀节的质量检验工作由哪个部门来做 描写春天、夏天、秋天、冬天景色的四字词语有哪些?2008-12 计算机的应用范围有哪些? 我单位欲取A2类压力容器制造证,不知哪位权威人士知道需不需要办理压力容器设计分包协议! 描写春天、夏天、秋天、冬天风景的谚语最好多一点分类我会奖励分 压力容器分为一二三类,分为高压低压中压容器、分为反应、换热、分离、储存容器,这些关于压力容器的分类,有哪些法规依据呢?麻烦亲列举一下 知道台风预警信号中哪个颜色的台风信号最弱吗? 农村春天夏天秋天和冬天各有什么景物 市场调查的应用范围是什么 春天一般是多少度之内?夏天、秋天、冬天呢?一般多少度之内代表是这个季节了? 压力容器的效验周期的多长?压力管道(蒸汽管道)的周期?压力容器(空气储罐)的效验周期? 21.根据《广东省突发气象灾害预警信号发布规定》,悬挂 ____预警信号时中小学应该停课.A.暴雨橙色 B.暴雨红色C.高温橙色 D.高温红色 气象台在发布台风预报时,除了有时间、风速、风级以外,还强调了什么? 压力容器与压力管道的特点 气象部门发布的气象灾害预警信号,危害程度最高的是哪种颜色 气象台已经发布台风橙色预警信号了.教育局怎么还不放假? 200克食物的概念是什么啊?举例说明下 上海发布过哪些突发气象灾害预警信号 你知道气象台以发布台风预报时,除了有时间,风速,风级以外,还强调什么呢?你能说出其原因吗? 200克蘑菇是多少,没概念 提纯混有甲烷的乙烯用什么方法?常用的简单方法 算盘是我国传统得计算工具是褒义词、贬义词还是中性词阎王好见,(小鬼)难求!是中性、褒义、贬义 气象台在发布台风预报时,除了有时间、风速、风级外,还强调() 沼气怎么提炼甲烷 他的小算盘打得很精是褒义词还是贬义词还是中性词 压力容器安全技术监察规程规定哪些容器需办理使用证 氨系统油冷器是压力容器吗? 中性词,褒义词,贬义词快 (固定式压力容器安全技术监察规程)属于国家标准吗 压力容器检验工程师、监检压力容器检验工程师,从事压力容器、化工机械的制造监检,工地安装监检,(一般做为第三方)需要哪些资格证?最好详细点, 算盘是我国传统的计算工具是褒义词还是贬义词还是中性词?在线回答提高财富悬赏,复制的不采纳. 中国古代测量时间的原理和玛雅人测量时间的原理是相同的吗? 压力容器是否必须由当地的质监部门的特检所检验 天气预报的预警信号的眼色如何区分?天气预报中的预警信号的颜色如何区分? 我国古代利用日影测定时间的工具称之为什么? 空冷器是什么 天气预报中的多云和晴有什么区别我们这里高达39度的温度,居然还说是多云?暗道多云不是阴天的感觉吗 春天,夏天,秋天,冬天各像什么? 有哪些气象预警警报? 天气预报中的“阴”和“多云”有什么区别? 春天的树是花儿们选美的舞台,那夏天的、秋天的、冬天的树是什么呢 古代中国时间是什么?具体时间 压力容器安全技术监察规程》的适用与不适用范围是什么? HT200材料淬火硬度可以达到多少? 中国古代的时间是如何计算的.比如一盏茶的时间算现在多少 HT200灰铸铁怎么配料及比例是多少 HT200具体介绍下,HT200是什么材料?200表示什么? 一寸光阴一寸金日晷 我国古代是怎样计量时间的 HT200铸造原材料包括什么?具体包含哪些元素?各元素的所占含量比例? 气象台向公众发布灾害性天气预警信号有什么意义 中国古代时间更指什么 HT200性能有何特点?举例说明其用途. 天气灾害分为几种? 赛尔号收集甲烷燃气的装备在哪要 计算工具的演变 气象灾害预警信号是由哪三部分组成 压力容器怎么报价 HT200成分是什么?急用 200分贝的概念是什么啊?举个例子.详细点!
    备案号:鲁ICP备13029499号-2 说三道四 www.s3d4.cn