跳转到内容

D(编程语言)/d2/指针、传引用和静态数组

25% developed
来自维基教科书,自由的教科书

第 7 课:指针、传引用和静态数组

[编辑 | 编辑源代码]

本章,你将学习如何在 D 中使用指针和静态数组。你还会学习如何在不使用指针的情况下传引用。课文不会解释指针和数组背后的概念,它假设你从 C 或 C++ 中了解这些知识。D 还有动态数组和数组切片,两者将在以后的课文中介绍。

入门代码

[编辑 | 编辑源代码]

使用指针

[编辑 | 编辑源代码]
import std.stdio;

void add_numbers(int* a, int* b)
{
    *a = *a + *b;
}

void main()
{
    int a = 50;
    int* b = &a;
    *b = 30;
    writeln(a);  // 30
    
    // int* error = &50; Error: constant 50 is not an lvalue
    
    int c = 2;
    int d = 2;
    add_numbers(&c, &d);
    writeln(c);  // 4
}

传引用

[编辑 | 编辑源代码]

如果你是一位 C++ 程序员,你读到这里,可能会认为这是比 C++ 引用退步的一步,那么恭喜你:虽然 D 支持指针,但它也支持更优的解决方案来将变量作为引用传递给函数。

import std.stdio;

void add_numbers(ref int a, int b)
{
    a += b;
}

void main()
{
    int c = 2;
    int d = 2;
    add_numbers(c, d);
    writeln(c); // 4

    // add_numbers(4, 2); Error: integer literals are not lvalues and cannot be passed by reference
}

它还专门支持 *输出参数*

import std.stdio;

void initializeNumber(out int n)
{
    n = 42;
}

void main()
{
    int a;
    initializeNumber(a);
    writeln(a); // 42
}

使用 `out`,在函数开始时,`n` 在 `initializeNumber` 中被默认初始化。

使用静态数组

[编辑 | 编辑源代码]
import std.stdio;

void main()
{
    int[5] a;
    a[0] = 1;
    writeln(a);  // [1, 0, 0, 0, 0]
    
    // a[10] = 3;  Error: Array index 10 is out of bounds
    
    int* ptr_a = &a[0];
    ptr_a[0] = 5;
    ptr_a[4] = 3;
    writeln(a); // [5, 0, 0, 0, 3]
    
    // ptr_a[10] = 3;  // Bad: This memory is not owned by 'a' as it is out of bounds
    // Compiler would allow it, but the result is undefined!

    int[3] b = 3;
    writeln(b);  // [3, 3, 3]
    b[] = 2;
    writeln(b);  // [2, 2, 2]
}

引用和解引用

[编辑 | 编辑源代码]

D 中的语法与 C 和 C++ 相同。运算符 `&` 获取对象的引用,而 `*` 解引用。

指针类型

[编辑 | 编辑源代码]

在 D 中,指向 `typeA` 的指针是

typeA*

声明一个指针的代码是

typeA* identifier;

.

静态数组

[编辑 | 编辑源代码]

静态数组具有固定长度。你可以像这样声明它们:

byte[10] array_of_ten_bytes;

你也可以这样做:

byte[2] array_of_two_bytes = [39, 0x3A];

数组索引从第一个元素的 0 开始。编译器会在你尝试访问索引大于或等于数组长度的元素时捕获你。你可以像这样访问数组元素:

my_array[index]

你也可以用单个值填充数组,使数组的所有元素都等于该值:

int[7] a = 2;
writeln(a);  // [2, 2, 2, 2, 2, 2, 2]
a[] = 7;
writeln(a);  // [7, 7, 7, 7, 7, 7, 7]

作为函数参数

[编辑 | 编辑源代码]

看看这段代码:

import std.stdio;

void addOne(int[3] arr)
{
    arr[0] += 1;
    // this means arr[0] = arr[0] + 1
}

void main()
{
    int[3] array1 = [1, 2, 3];
    addOne(array1);
    writeln(array1); // [1, 2, 3]
}

输出是 `[1,2,3]`,而不是 `[2,2,3]`。为什么?因为静态数组在作为函数参数传递时会被复制。`addOne` 函数中的 `arr` 只是 `array1` 的一个副本。将代码改写成这样:

import std.stdio;

void addOne(ref int[3] arr)
{
    arr[0] += 1;
}

void main()
{
    int[3] array1 = [1, 2, 3];
    addOne(array1);
    writeln(array1); // [2, 2, 3]
}
华夏公益教科书