我正在读一本叫做《21天自学C语言》的书(我已经学会了Java和c#,所以我正在以更快的速度前进)。我正在阅读关于指针的章节,->(箭头)操作符没有解释就出现了。我认为它是用来调用成员和函数的(类似于。(点)操作符,但用于指针而不是成员)。但我不完全确定。
我能得到一个解释和一个代码样本吗?
我正在读一本叫做《21天自学C语言》的书(我已经学会了Java和c#,所以我正在以更快的速度前进)。我正在阅读关于指针的章节,->(箭头)操作符没有解释就出现了。我认为它是用来调用成员和函数的(类似于。(点)操作符,但用于指针而不是成员)。但我不完全确定。
我能得到一个解释和一个代码样本吗?
当前回答
Foo ->bar只是(* Foo).bar的简写。这就是它的全部。
其他回答
Foo ->bar只是(* Foo).bar的简写。这就是它的全部。
在某些情况下,->操作符使代码比*操作符更具可读性。
如:(引自EDK II项目)
typedef
EFI_STATUS
(EFIAPI *EFI_BLOCK_READ)(
IN EFI_BLOCK_IO_PROTOCOL *This,
IN UINT32 MediaId,
IN EFI_LBA Lba,
IN UINTN BufferSize,
OUT VOID *Buffer
);
struct _EFI_BLOCK_IO_PROTOCOL {
///
/// The revision to which the block IO interface adheres. All future
/// revisions must be backwards compatible. If a future version is not
/// back wards compatible, it is not the same GUID.
///
UINT64 Revision;
///
/// Pointer to the EFI_BLOCK_IO_MEDIA data for this device.
///
EFI_BLOCK_IO_MEDIA *Media;
EFI_BLOCK_RESET Reset;
EFI_BLOCK_READ ReadBlocks;
EFI_BLOCK_WRITE WriteBlocks;
EFI_BLOCK_FLUSH FlushBlocks;
};
_EFI_BLOCK_IO_PROTOCOL结构体包含4个函数指针成员。
假设你有一个变量结构_EFI_BLOCK_IO_PROTOCOL * pStruct,你想使用老的*操作符来调用它的成员函数指针。你最终会得到这样的代码:
(* pStruct) .ReadBlocks参数(……)
但是使用->操作符,你可以这样写:
pStruct - > ReadBlocks参数(……)。
哪个看起来更好?
我还得补充点什么。结构与数组有点不同,因为数组是指针,而结构不是。所以要小心!
假设我写了这段无用的代码:
#include <stdio.h>
typedef struct{
int km;
int kph;
int kg;
} car;
int main(void){
car audi = {12000, 230, 760};
car *ptr = &audi;
}
这里的指针ptr指向结构变量audi的地址(!),但在地址结构旁边也有一大块数据(!)!数据块的第一个成员具有与结构本身相同的地址,您可以通过对*ptr(不带大括号)这样的指针进行解引用来获得它的数据。
但是如果你想访问除第一个成员之外的任何其他成员,你必须添加一个指示符,如.km, .kph, .kg,这只不过是对数据块的基址的偏移量…
但是因为优先级,你不能写*ptr。Kg作为访问操作员。在解引用操作符*之前求值,您将得到*(ptr.kg),这是不可能的,因为指针没有成员!编译器知道这一点,因此会发出一个错误,例如:
error: ‘ptr’ is a pointer; did you mean to use ‘->’?
printf("%d\n", *ptr.km);
相反,您可以使用this (*ptr)。Kg和你强迫编译器首先解引用指针,并允许访问数据块和第二你添加一个偏移量(指示符)选择成员。
看看我做的这张图:
但是如果你有嵌套的成员,这个语法将变得不可读,因此->被引入。我认为可读性是使用它的唯一合理的原因,因为这个ptr->kg比(*ptr).kg更容易编写。
现在让我们用不同的方式来写,这样你就能更清楚地看到其中的联系。(* ptr)。Kg⟹(*&奥迪)。Kg⟹audi.kg。这里我首先使用了ptr是“audi的地址”,即&audi,以及“引用”&和“解引用”*操作符相互抵消的事实。
#include<stdio.h>
int main()
{
struct foo
{
int x;
float y;
} var1;
struct foo var;
struct foo* pvar;
pvar = &var1;
/* if pvar = &var; it directly
takes values stored in var, and if give
new > values like pvar->x = 6; pvar->y = 22.4;
it modifies the values of var
object..so better to give new reference. */
var.x = 5;
(&var)->y = 14.3;
printf("%i - %.02f\n", var.x, (&var)->y);
pvar->x = 6;
pvar->y = 22.4;
printf("%i - %.02f\n", pvar->x, pvar->y);
return 0;
}
struct Node {
int i;
int j;
};
struct Node a, *p = &a;
在这里要访问i和j的值,我们可以使用变量a和指针p,如下所示:a.i, (*p)。I和p-> I都是一样的。
在这里。是“直接选择器”,->是“间接选择器”。