D(编程语言)/d2/指针、传引用和静态数组
外观
< D(编程语言)
(从 D(编程语言)/d2/第 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]
}