This document lists some basic differences between Nemerle and C# in a terse form. If you know Java or C++ it should still be fairly helpful.

Changes In Expressions

C# Nemerle Remarks
const int x = 3;
const string y = "foo";
readonly Object obj = getObject();
def x : int = 3;
def y : string = "foo";
def obj : Object = getObject();
Variables defined with def cannot be changed once defined. This is similar to readonlyor const in C# orfinal in Java. Most variables in Nemerle aren't explicitly typed like this.
int x = 3;
string y = "foo";
Object obj = getObject();
mutable x : int = 3;
mutable y : string = "foo";
mutable obj : Object = getObject();
Variables defined with mutable can be changed once defined. Most variables in Nemerle aren't explicitly typed like this.
var = 3; //Will compile.
var y; y = "foo";//Won't compile.
def = 3;//Will compile!
mutable y; y = "foo";//Will compile!
Nemerle's type inference is lightyears ahead of C#'s. If there is clear evidence of a variable's type, there's a 99% chance Nemerle will infer it.
int a = b = c;
def a = c;
def b = c;
The type of the assignment operator is void.
value = cond ? var1 : var2;
value = if(cond) var1 else var2
No ternary operator is needed as everything is an expression in Nemerle. The 'else' branch is mandatory here! (Don't panic! if-without-else has its own keyword.)
Class myClass = new Class(parms);
def myClass = Class(parms);
Nemerle doesn't require new when calling a constructor.
Book[] books = new Book[size];
def books = array(size) : array[Book];
Often the array type can be inferred and this is simplified; as in the next example.
Book[] books = new Book[size];
books[0] = new Book();
def books = array(size);
books[0] = Book();
When the type can be inferred from context or later use (which is most of the time), you can drop the type declaration
int[] numbers = {1, 2, 3};
def numbers = array[1, 2, 3];
Initializing an array. Without the arraykeyword this would create a list.
int[,] numbers = new int[2,3];
def numbers = array(2,3) : array.[2][int];
Multidimensional array constructor. The type can usually be inferred from use and not declared.
int[,] numbers = { {1,2,3}, {1,4,9} };
def numbers = [ [1,2,3], [1,4,9] ];
Multidimensional array initialization.
new {Prop1 = 1; Prop2 = "string"}
using Nemerle.Extensions;
new (Prop1 = 1, Prop2 = "string")
Nemerle anonymous typesare a bit more flexible (e. g. can be generic or returned from a method). They must be imported from Nemerle.Extensions however.
new Class {
Property1 = 1;
Property2 = "string"
}
Class()
The Nemerle Object Modifier macro is more powerful.
if(cond)
answer = 42;
...
when(cond)
answer = 42;
...
if without else is called when. Nemerle requiresif statements to be paired with elsefor clarity.
if(!cond)
answer = 42;
...
unless(cond)
answer = 42;
...
In Nemerle,if(!cond) can use the clearerunless(cond)syntax. Of course,when(!cond) can also always be used.
if (cond)
return foo;
do_something ();
return bar;
match(cond){
| true => foo
| _ => {doSomething(); foo}
}
Pattern Matchingprovides a clearer way of delegating control flow.
if (cond)
return foo;
do_something ();
return bar;
using Nemerle.Imperative;
when(cond)
return foo
do_something ();
return bar;
Alternately the Imperative namespace may be imported. This isdiscouragedhowever.
try {...}
catch (FooException e) { ... }
catch (BarException e) { ... }
try {...}
catch {
| e is FooException => ...
| e is BarException => ...
}
Nemerle's somewhat differenttry ... catchsyntax is consistent with its pattern matching syntax.
(type) expr
expr :> type
Runtime type cast, allows for downcasts and upcasts.
(type) expr
expr : type
Static cast, only upcasts are allowed.
using System;
using SWF = System.Windows.Forms;
using System.Xml;
...
Console.WriteLine ("foo");
SWF.Form x = new SWF.Form();
XmlDocument doc = new XmlDocument();
using System;
using System.Console;
using SWF = System.Windows.Forms;
...
WriteLine("foo");
def x = SWF.Form();
def doc = Xml.XmlDocument();
In Nemerle, you can apply the usingdirective to classes as well as namespaces. Opened namespaces allow you to drop the prefix of other namespaces, likeSystem inSystem.XmlMore info.
using System.Windows.Forms;

Button button = control as Button;

if (button != null)
...
else
...
match (control) {
| button is Button => ...
| listv is ListView => ...
| _ => ...//something else
}
as can be simulated withmatch. It is a bit more to type up in simple cases, but in general Nemerle's construct is more powerful.
int y = x++;
++x;
def y = x;
x++;
++x;
The ++ and -- operators return void, just like assignment. So, both prefix and postfix versions are equivalent.

Changes In Type Definitions

C# Nemerle Remarks
static int foo (int x, string y)
{ ... }
static foo (x : int, y : string) : int
{ ... }
Types are written after variable names.
class Foo {
public Foo (int x)
{ ... }
}
class Foo {
public this (x : int)
{ ... }
}
The constructor's name is alwaysthis.
class Foo {
~Foo ()
{ ... }
}
class Foo {
protected override Finalize () : void
{ ... }
}
There is no special syntax for the destructor, you just override theFinalizemethod.
class Foo : Bar {
public Foo (int x) : base (x)
{ ... }
}
class Foo : Bar {
public this (x : int) {
base (x);
...
}
}
The base constructor is called in the constructor's function body.
class Foo {
int x;
}
class Foo {
mutable x : int;
}
Fields which will be changed outside of the constructor need to be marked asmutable.
class Foo {
readonly int x;
const int y = 10;
}
class Foo {
x : int;
y : int = 10;
}
Read-only/const are used by default.
class Foo {
static int x = 1;
}
class Foo {
static mutable x : int = 1;
}
Static variable.
class Foo {
static readonly int x;
static int method() { ... }
}
module Foo {
x : int;
method() : int { ... }
}
A module is a class in which all members are static.
using System.Runtime.CompilerServices.CSharp;

class C {
public object this [int i]
{ ... } [IndexerName("MyItem")]
public int this [string name]
{ ... }
}
class C {
public Item [i : int] : object
{ ... } public MyItem [name : string] : int
{ ... }
}
Indexers.
C# Nemerle
When two interfaces use the same method to perform different functions, different names can be given to each method.
interface SpeaksEnglish{
void Speak();
} interface SpeaksGerman{
void Speak();
} class GermanTransfer : SpeaksEnglish, SpeaksGerman{
public void SpeaksEnglish.Speak() {}
public void SpeaksGerman.Speak() {}
}
interface SpeaksEnglish{
Speak() : void;
} interface SpeaksGerman{
Speak() : void;
} class GermanTransfer : SpeaksEnglish, SpeaksGerman{
public Speak() : void implements SpeaksEnglish.Speak{}
public Sprechen() : void implements SpeaksGerman.Speak{}
}

Generics

C# Nemerle Remarks
class A  { T x; }
class A [T] { x : T; }
Type parameters are written in square brackets [...].
typeof(A);
typeof(A[_,_]);
typeof expression

New Stuff

Nemerle contains many constructs which are not present in C#. Unfortunately, most of them don't really fit into a side-by-side comparison format:

Other Minor Differences

Ambiguity Isn't Tolerated

namespace YourAttributes{
class Serializable : System.Attribute { }
}
namespace MyAttributes{
using YourAttributes;
class Serializable : System.Attribute { } [Serializable] class SomeClass { }
}

C# compilers will choose MyAttributes.Serializable or, if its definition is commented out, YourAttributes.Serializable. Nemerle will raise an error telling you to be more specific about which attribute you want to use.

Exclusion of Overridden Methods

 class BaseClass
{
public virtual AddItem (val : string) : void { }
} class TestClass : BaseClass
{
public AddItem (val : object) : void { }
public override AddItem (val : string) : void { }
}
...
TestClass().AddItem ("a"); // C# will choose TestClass.AddItem (object)
// Nemerle will choose TestClass.AddItem (string)

This behaviour comes from section 7.6.5.1 of the C# specification, which states "...methods in a base class are not candidates [for overload resolution] if any method in a derived class is applicable (§7.6.5.1)." Unfortunately, this rule is patently absurd in situations like the above. The Nemerle compiler always chooses the method whose signature best matches the given arguments.

CSharp Similarities and Differences的更多相关文章

  1. The Similarities and Differences Between C# and Java -- Part 1(译)

    原文地址 目录 介绍(Introduction) 相似点(Similarities) 编译单位(Compiled Units) 命名空间(Namespaces) 顶层成员(类型)(Top Level ...

  2. Comparing the MSTest and Nunit Frameworks

    I haven't seen much information online comparing the similarities and differences between the Nunit ...

  3. A Brief Review of Supervised Learning

    There are a number of algorithms that are typically used for system identification, adaptive control ...

  4. TIJ——Chapter One:Introduction to Objects

    ///:~容我对这个系列美其名曰"读书笔记",其实shi在练习英文哈:-) Introduction to Objects Object-oriented programming( ...

  5. scala vs java 相同点和差异

    本贴是我摘抄自国外网站,用作备忘,也作为分享! Similarities between Scala and Java Following are some of the major similari ...

  6. Gestures_Article_4_0

    Technical Article Windows Phone 7™ Gestures Compared Lab version: 1.0.0 Last updated: August 25, 201 ...

  7. 15 things to talk about in a healthy relationship

    15 things to talk about in a healthy relationship男女交往中可以谈论的15个话题 1. Your Daily Activities 1. 你的日常活动 ...

  8. .net程序员必须知道的知识

    A while back, I posted a list of ASP.NET Interview Questions. Conventional wisdom was split, with ab ...

  9. SICP阅读笔记(一)

    2015-08-25   008   Foreword    QUOTE: I think that it's extraordinarily important that we in compute ...

随机推荐

  1. 忘记windows的登陆密码

    http://user.qzone.qq.com/372806800/blog/1342261571

  2. Android开发的进阶之路

    客户端开发工程师,简单地从某几个方面描述一下个人理解里不同的等级: 1.初级的可以熟练使用系统框架提供的组件,搭建所需应用程序: 2.中级的,会对系统框架中如view绘制.broadcast机制.内存 ...

  3. 关于JDK,tomcat,MyEclipse的配置

    1.下载安装JDK 在自定义安装路径时,jdk和之后的jre文件夹是属于平行结构,我的安装路径为:D:\jdk\jdk1.6.0_43和D:\jdk\jre6 然后是对环境变量的配置, 计算机→属性→ ...

  4. new和malloc的区别

    1.malloc与free是C++/C语言的标准库函数,new/delete是C++的运算符 2.new出来的指针是直接带类型信息的,而malloc返回的都是void*指针. 3.new 建立的是一个 ...

  5. & jobs fg Ctrl+z bg

    -l选项,jobs命令可以显示后台正在运行的任务的进程号信息: ctrl+l组合键:将放在前台的任务挂起: bg命令将挂起的任务放在后台继续运行 [xiluhua@vm-xiluhua][~]$ sl ...

  6. function adapter(函数适配器)和迭代器适配器

    所谓function adapter(函数适配器)是指能够将不同的函数对象(或是和某值或某寻常函数)结合起来的东西,它自身也是个函数对象. 迭代器适配器  运用STL中的迭代器适配器,可以使得算法能够 ...

  7. 【转】学习总结--Cookie & Session总结

    转载地址:http://www.phperzone.cn/portal.php?aid=718&mod=view 一.状态管理 1)什么是状态管理?   将浏览器与web服务器之间多次交互过程 ...

  8. File和URL的getPath()方法区别

    java.io.File对象的getPath()方法返回文件的全路径名.如果是目录返回目录路径且结尾没有"\".如果是文件包含文件名. java.io.File对象的getName ...

  9. [团队项目3.0]Scrum团队成立

    Scrum团队成立 5.Scrum团队成立 5.1 团队名称,团队目标.团队口号.团队照: 5.2 角色分配 产品负责人: 决定开发内容和优先级排序,最大化产品以及开发团队工作的价值. Scrum M ...

  10. 去掉DataTable中重复的行

    //DataView dv = dt3.DefaultView;     //dt3默认的虚拟视图 //dv.Sort = "wmid asc"; //排序 ///dv.ToTab ...