- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我读过 @synthesize
will automatically create corresponding instance variables for @property
并且默认情况下,ivars 是 @protected
。但是,如果我使用类扩展(如下所示)来指示 @property
方法是私有(private)的呢?
// Photo.m
@interface Photo ()
@property (nonatomic, retain) NSMutableData *urlData;
@end
那么对应的ivar会不会是@private
?或者我应该像这样将它显式声明为 @private
吗?
// Photo.h
@interface Photo : Resource {
@private
NSMutableData *urlData;
}
最佳答案
详细阐述凯文的回答:
当你声明一个类时,例如:
@interface SomeClass : NSObject {
@public
id publicIvar;
@protected
id protectedIvar;
@private
id privateIvar;
}
@end
编译器1 决定该类的实例变量布局。此布局确定实例变量相对于该类实例地址的偏移量。一种可能的布局是:
+--> publicIvar address = instance address + offsetOfPublicIvar
|
|
+-----+------------+-----+---------------+-----+-------------+-----+
| ... | publicIvar | ... | protectedIvar | ... | privateIvar | ... |
+-----+------------+-----+---------------+-----+-------------+-----+
|
|
+--> instance address
当在代码中引用实例变量时——无论是在类的实现中还是在代码库的其他部分,编译器都会用实例变量相对于相应地址的偏移量替换该引用实例。
比如在SomeClass的实现中,
privateIvar = someObject;
或
self->privateIvar = someValue;
翻译成这样:
*(self + offsetOfPrivateIvar) = someObject;
同样,课外,
SomeClass *obj = [SomeClass new];
obj->publicIvar = someObject;
翻译成这样:
SomeClass *obj = [SomeClass new];
*(obj + offsetOfPublicIvar) = someObject;
但是,编译器只根据实例变量的可见性允许这样做:
在类扩展中声明实例变量时,例如
@interface SomeClass () {
id extensionIvar;
}
@end
编译器将其添加到实例变量布局中:
+-----+------------+-----+---------------+
| ... | otherIvars | ... | extensionIvar |
+-----+------------+-----+---------------+
并且对该实例变量的任何引用都将替换为其相对于该实例的相应偏移量。但是,由于该实例变量只为声明类扩展的实现文件所知,编译器不允许其他文件引用它。任意源文件只能引用它知道的实例变量(遵守可见性规则)。如果实例变量在由源文件导入的头文件中声明,则源文件(或者更准确地说,编译器在翻译该单元时)会知道它们。
另一方面,扩展变量只有声明它的源文件才知道。因此我们可以说在类扩展中声明的实例变量对其他文件是隐藏的。同样的推理适用于在类扩展中声明的属性的支持实例变量。它类似于 @private
,但限制性更强。
但是请注意,在运行时可见性规则不会强制执行。使用键值编码,任意源文件有时可以(规则描述 here )访问实例变量:
SomeClass *obj = [SomeClass new];
id privateValue = [obj valueForKey:@"privateIvar"];
包括在扩展中声明的实例变量:
id extensionValue = [obj valueForKey:@"extensionIvar"];
不管 KVC,对实例变量的访问都是通过 Objective-C 运行时 API 完成的:
Ivar privateIvar = class_getInstanceVariable([SomeClass class],
"privateIvar");
Ivar extensionIvar = class_getInstanceVariable([SomeClass class],
"extensionIvar");
id privateValue = object_getIvar(obj, privateIvar);
id extensionValue = object_getIvar(obj, extensionIvar);
请注意,一个类可以有多个类扩展。但是,一个类扩展不能声明与另一个实例变量同名的实例变量,包括在其他类扩展中声明的实例变量。由于编译器发出如下符号:
_OBJC_IVAR_$_SomeClass.extensionIvar
对于每个实例变量,使用不同的扩展名声明具有相同名称的实例变量不会产生编译器错误,因为给定的源文件不知道另一个源文件,但它确实会产生链接器错误。
1此布局可由 Objective-C 运行时更改。事实上,偏移量由编译器计算并存储为变量,运行时可以根据需要更改它们。
PS: 并非此答案中的所有内容都适用于所有编译器/运行时版本。我只考虑了具有非脆弱 ABI 和最新版本的 Clang/LLVM 的 Objective-C 2.0。
关于objective-c - 私有(private)@property 是否创建@private 实例变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5784875/
据我所知,在 C# 中,所有字段默认都是私有(private)的,如果没有另外标记的话。 class Foo { private string bar; } class Foo { strin
这个问题在这里已经有了答案: Why is it allowed to access Java private fields via reflection? (7 个答案) 关闭 6 年前。 使用反
在 C#(和许多其他语言)中,访问相同类型的其他实例的私有(private)字段是完全合法的。例如: public class Foo { private bool aBool; pu
使用私有(private)方法通过将一些决策点重构为单独的方法来降低 CC 会降低实际方法的 CC 并易于阅读,但不会减少在测试中获得完整分支覆盖的工作量。 这合理吗?你有什么现场经验? 最佳答案 好
在下面的例子中,模块outer有一个私有(private)类型Private和一个私有(private)内部模块inner。 inner 能够访问Private(因为子模块可以访问其父级的私有(pri
class Person def one @var = 99 self.two end private def two p @var end end p=P
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我正在尝试获得良好的 Ruby 编码风格。为防止意外调用具有相同名称的局部变量,我总是在适当的地方使用 self.。但是现在我偶然发现了这个: class MyClass "method" a
今天遇到一个案例类构造函数的奇怪问题。我想将构造函数设为私有(private),看来这不是问题。所以我已经在我的一个项目中尝试过它并且它有效。但在另一个项目中,我可以调用私有(private)构造函数
我想坚持使用记录,并且不想返回对象。所以我想知道是否可以将记录的字段设置为私有(private)?或者创建记录的私有(private)成员。其他具体类型(例如可区分联合)怎么样? 或者,这个要求是否违
我正在通过 Flickr API 进行经过身份验证的调用来访问照片。但我只得到我的公开照片,而没有任何私有(private)照片。 下面给出的是我正在使用的代码, Flickr f; Request
这两个类的行为不同;原因似乎与使用 private[this] 声明而不是 private 有关。有人可以解释一下为什么吗? 私有(private): class Person( private
在 private 中的 1000 秒 private 之后,我想到可能不需要它 public class Outer { private static class Inner { // yo
我有以下代码: class C { private enum E { // ... } } private extension C { func f(e: E)
OOP 语言中是否有object-private 的概念??我的意思是比经典的私有(private)访问限制更多? Private (or class-private) restricts the a
swift 3.0 我知道fileprivate访问级别修饰符将函数/属性的使用限制在声明它的源文件和 private - 仅限于声明的词法范围。但似乎这条规则不适用于扩展。例如。此代码有效: cla
即将推出的 Delphi 版本中启用该功能的功能怎么样? 也许它可能是一个编译器开关,促进所有 ** private **s to ** strict private **小号。 ... 或者它可能是
我可以通过将函数放入类的私有(private)扩展中来创建私有(private)函数,而不是通过不断调用 private func functionName(){} 来创建新的私有(private)函
部署专用端点并需要专用 IP 地址作为输出,但似乎无法正确获取值查询。下面的结果是“模板输出'主机名'无效:语言表达式属性|'privateIPAddress'具有无效的数组索引..(代码:Deplo
我是一名优秀的程序员,十分优秀!