Attention: Here be dragons

This is the latest (unstable) version of this documentation, which may document features not available in or compatible with released stable versions of Godot.

C# 风格指南

对于每个项目而言, 拥有定义良好且一致的编码约定非常重要,Godot也不例外.

本页面包含一份编码风格指南,Godot 本身的开发人员和贡献者都遵循该指南。因此,它的目标读者是希望为该项目做出贡献的人员,但是由于本文中提到的约定和规范被该语言用户最广泛采用,所以我们建议你也这样做,尤其是如果你还没有这样的指南。

备注

本文绝不是关于如何遵循标准编码约定或最佳实践的详尽指南。如果您不确定此处未涉及的方面,请参阅更全面的文档,例如 C# 编码约定框架设计规范

语言规范

Godot当前在其引擎和示例源代码中使用 C# 7.0 版本 . 因此, 在我们迁移使用较新版本之前, 必须注意避免混合仅在 C# 7.1 或更高版本中可用的语言功能.

有关不同版本的C#功能的详细信息, 请参阅 C #中的新功能 .

格式

总体规范

  • 使用换行符( LF )来换行, 而不是 CRLFCR.

  • 在每个文件末尾使用一个换行符, 但 csproj 文件除外.

  • 使用不带 字节顺序标记(BOM)UTF-8 编码.

  • 使用 4空格 代替制表符进行缩进(称为 "软制表符").

  • 如果长度超过100个字符, 请考虑将其分成几行.

换行符和空白行

对于一般缩进规则, 请遵循 Allman 风格, 它建议将与控制语句关联的大括号放在下一行, 缩进到同一级别:

// Use this style:
if (x > 0)
{
    DoSomething();
}

// NOT this:
if (x > 0) {
    DoSomething();
}

但是, 您可以选择省略括号内的换行符:

  • 对于简单的属性访问者.

  • 对于简单对象, 数组, 或集合初始化.

  • 对于抽象的自动属性, 索引器, 或事件声明.

// You may put the brackets in a single line in following cases:
public interface MyInterface
{
    int MyProperty { get; set; }
}

public class MyClass : ParentClass
{
    public int Value
    {
        get { return 0; }
        set
        {
            ArrayValue = new [] {value};
        }
    }
}

插入一个空行:

  • 在一列 using 语句之后.

  • 在方法, 属性, 和内部类型声明之间.

  • 在每个文件的末尾.

字段声明和常量声明可以根据相关性编组在一起. 在这种情况下, 请考虑在编组之间插入空白行以便于阅读.

避免插入空白行:

  • 在开括号 { 之后。

  • 在闭合括号 } 之前。

  • 在注释块或单行注释之后.

  • 与另一个空白行相邻.

using System;
using Godot;
                                          // Blank line after `using` list.
public class MyClass
{                                         // No blank line after `{`.
    public enum MyEnum
    {
        Value,
        AnotherValue                      // No blank line before `}`.
    }
                                          // Blank line around inner types.
    public const int SomeConstant = 1;
    public const int AnotherConstant = 2;

    private Vector3 _x;                  // Related constants or fields can be
    private Vector3 _y;                  // grouped together.

    private float _width;
    private float _height;

    public int MyProperty { get; set; }
                                          // Blank line around properties.
    public void MyMethod()
    {
        // Some comment.
        AnotherMethod();                  // No blank line after a comment.
    }
                                          // Blank line around methods.
    public void AnotherMethod()
    {
    }
}

使用空格

插入一个空格:

  • Around a binary and ternary operator.

  • 在左括号和 ifforforeachcatchwhilelockusing 关键字之间。

  • 在单行访问器块之前和之内.

  • 在单行访问器块中的访问器之间.

  • 在不是在行尾的逗号之后.

  • for 语句中的分号之后.

  • 在单行 case 语句中的冒号之后.

  • 在类型声明中的冒号周围.

  • 围绕一个lambda箭头.

  • 在单行注释符号(//)之后,并且如果在行末使用,则在它之前。

不要使用空格:

  • 在类型转换括号后.

  • 在单行初始化括号内侧.

下面的示例根据上述的一些约定显示了对空格的正确使用:

public class MyClass<A, B> : Parent<A, B>
{
    public float MyProperty { get; set; }

    public float AnotherProperty
    {
        get { return MyProperty; }
    }

    public void MyMethod()
    {
        int[] values = {1, 2, 3, 4}; // No space within initializer brackets.
        int sum = 0;

        // Single line comment.
        for (int i = 0; i < values.Length; i++)
        {
            switch (i)
            {
                case 3: return;
                default:
                    sum += i > 2 ? 0 : 1;
                    break;
            }
        }

        i += (int)MyProperty; // No space after a type cast.
    }
}

命名约定

对所有命名空间、类型名称、成员级别标识符(即方法、属性、常量、事件)使用 PascalCase,私有字段除外:

namespace ExampleProject
{
    public class PlayerCharacter
    {
        public const float DefaultSpeed = 10f;

        public float CurrentSpeed { get; set; }

        protected int HitPoints;

        private void CalculateWeaponDamage()
        {
        }
    }
}

camelCase 用于所有其他标识符(即局部变量、方法参数),并使用下划线(_)作为私有字段的前缀(但不能用于方法或属性,如上所述):

private Vector3 _aimingAt; // Use a `_` prefix for private fields.

private void Attack(float attackStrength)
{
    Enemy targetFound = FindTarget(_aimingAt);

    targetFound?.Hit(attackStrength);
}

类似 UI 这种只有两个字母的首字母缩写应特殊处理,使用 PascalCase 时都应写作大写字母,否则都应写作小写字母。

请注意,id 不是首字母缩写,因此应将其视为普通标识符:

public string Id { get; }

public UIManager UI
{
    get { return uiManager; }
}

通常不建议将类型名称用作标识符的前缀,例如 string strTextfloat fPower。但是,对于接口来说是个例外,实际上,接口应该在其名称前加上大写字母 I,例如 IInventoryHolderIDamageable

最后,请考虑有意义的名称,请勿对名称进行过度缩写,以免影响可读性。

例如,如果您想编写代码来查找附近的敌人并用武器击中它,请选择:

FindNearbyEnemy()?.Damage(weaponDamage);

而不是:

FindNode()?.Change(wpnDmg);

成员变量

如果变量只在方法中使用, 勿声明其为成员变量, 因为我们难以定位在何处使用了该变量. 相反, 你应该将它们在方法内部定义为局部变量.

局部变量

声明局部变量的位置离首次使用它的位置越近越好. 这让人更容易跟上代码的思路, 而不需要上下翻找该变量的声明位置.

隐式类型的局部变量

考虑使用隐式类型化(var)声明局部变量,但是请只在赋值右侧能够推出该类型时使用:

// You can use `var` for these cases:

var direction = new Vector2(1, 0);

var value = (int)speed;

var text = "Some value";

for (var i = 0; i < 10; i++)
{
}

// But not for these:

var value = GetValue();

var velocity = direction * 1.5;

// It's generally a better idea to use explicit typing for numeric values, especially with
// the existence of the `real_t` alias in Godot, which can either be double or float
// depending on the build configuration.

var value = 1.5;

其他注意事项

  • 使用显式访问修饰符。

  • 使用属性而不是非私有字段。

  • 按此顺序使用修饰符:public/protected/private/internal/virtual/override/abstract/new/static/readonly

  • 避免在不必要时,为成员使用完全限定的名称或 this. 前缀。

  • 删除未使用的 using 语句和不必要的括号。

  • 考虑省略类型的默认初始值。

  • 考虑使用空条件运算符或类型初始化器来使代码更紧凑。

  • 当值可能会成为另一种不同的类型,请使用安全类型转换,否则使用直接类型转换。