Properties are either atomic or nonatomic, The difference has to do with multithreading. atomic is the default value.

Properties are either readonly or readwrite. readwrite is the default value.

Whenever you declare a property that points to an NSString or an NSArray, you should include the copy attribute.

Property gets an instance variable, such as _hireDate.

A instance of a subclass can stand in for an instance of the superclass without problems because it inherits everything in the superclass.

e.g. BNRPerson *mikey = [[BNREmployee alloc] init];

All objects inherit from NSObject.

// the token %@ to get an object to describe itself. The %@ token sends a description message to the object pointed to by the corresponding variable.

NSDate *now = [NSDate date]; //创建NSDate实例,并赋值为当前日期

NSLog(@"%@", now);

Categories of object instance variables:

Object-type attributes: a pointer to a simple, value-like object like an NSString or an NSDate.

To-one relationship: a pointer to a single complex object.

To-many relationships: a pointer to an instance of a collection class, such as an NSMutableArray. You will often end up explicitly creating instance variables, accessors, and methods for adding or removing objects from the relationship.

object ownership: when an object has an object instance variable, the object with the pointer is said to own the object that is being pointed to. When an object has zero owners, it figures no one needs it around anymore and deallocate itself.

If a class overrides dealloc, then this method will be executed when an instance of the class is deallocated.

Command-B: build your program without running it.

Important things about collections and ownership:

When an object is added to the collection, the collection establishes a pointer to the object, and the object gains an owner.

When an object is removed from a collection, the collection gets rid of its pointer to the object,  and the object loses an owner.

When unnecessary objects do not get deallocated, you are said to have a memory leak.

Using @class instead of #import gives the compiler less information, but makes the processing of this particular file faster.

A class extension is a set of declarations that is private. Class extensions are added to the class implementation file above the @implementation block where methods are implemented. e.g.

@interface BNREmployee ()

{

NSMutableArray *_assets;

}

@property (nonatomic) unsigned int officeAlarmCode;

@end

Objects that are not instances of BNREmployee can no longer see this property. Moving the _assets instance variable to BNREmployee’s class extension.

A subclass has no access to its superclass’s class extensions.

When a class declares a property in its header, only the accessors for this property are visible to other objects. The instance variables generated by property declarations can not be accessed.

A weak reference is a pointer that does not imply ownership.

Preventing strong reference cycle: the parent owns the child, but the child should not own the parent.

A strong reference will keep the object it points to from being deallocated. A weak reference will not.

ARC uses the autorelease pool automatically, but you must create and drain the pool.

@autoreleasepool { … }

Retain count rules:

If you create an object using a method whose name starts with alloc or new or contains copy, then you have taken ownership of it.

An object created through any other means is not owned by you.

If you do not own an object and you want to ensure its continued existence, take ownership by sending it the message retain.

When you own an object and no longer need it, give up ownership by sending it the message release or autorelease.

As long as an object has at least one owner, it will continue to exist.

A set is a collection that has no sense of order, and a particular object can only appear in a set once. Sets are faster at testing object membership than arrays are.——NSSet, NSMutableSet. NSSet method:

(BOOL) containsObject: (id) x;

NSMutableArray methods:

(NSUInteger) indexOfObject:(id) anObject;

(NSUInteger) indexOfObjectIdenticalTo:(id) anObject;

Identical objects are always equal. Equal objects are not always identical.

A dictionary is a collection of key-value pairs. The key is typically a string, and the value can be any sort of object.——NSDictionary, NSMutableDictionary.

- (void) setObject:(ObjectType)anObject  forKey:(id<NSCopying>)aKey

The keys in a dictionary are unique. If you try to add a second object under an existing key, the first key-value pair gets replaced.

NSDictionary *numberOfMoons = @{ @“Mercury” : @0, @“Venus” : @1, };

NSNumber *venusMoonCount = numberOfMoons[@‘Venus’];

@[] is an empty array, equivalent to [NSArray array];

Using an immutable collection conserves memory and improves performance because that collection never needs to be copied.

Immutable classes: NSArray, NSString, NSAttributedString, NSData, NSCharacterSet, NSDictionary, NSSet, NSIndexSet, NSURLRequest.

NSMutableArray method:

(void) sortUsingDescriptors:(NSArray<NSSortDescriptor *> *) sortDescriptors;

NSSortDescriptor *voa = [NSSortDescriptor sortDescriptorWithKey:@“valueOfAssets” ascending: YES];

NSSortDescriptor *eid = [NSSortDescriptor sortDescriptorWithKey:@“employeeID” ascending: YES];

[employee sortUsingDescriptors: @[voa, eid]];

- (void) filterUsingPredicate:(NSPredicate *) predicate;

NSArray method:

- (NSArray *) filteredArrayUsingPredicate:(NSPredicate *) predicate;

NSPredicate *predicate = [NSPredicate predicateWithFormat:@“holder.valueOfAssets > 70”];

NSArray *toBeReclaimed = [allAssets filteredArrayUsingPredicate: predicate];

NSSet method:

- (NSSet *) filteredSetUsingPredicate:(NSPredicate *) predicate;

NSMutableSet method:

- (void) filterUsingPredicate:(NSPredicate *) predicate;

NSValue is the superclass of NSNumber.

NSPoint somePoint = NSMakePoint(100, 100);

NSValue *pointValue = [NSValue valueWithPoint: somePoint];

instance of NSNull, representing nothingness.

NSMutableArray *hotel = [[NSMutableArray alloc] init];

[hotel addObject: [NSNull null]];

#import ensures that the preprocessor only includes a file once. #include will allow you to include the same file many times.

NSString * const NSLocaleCurrencyCode = @“currency”;

typedef enum {

BlenderSpeedStir = 1,

BlendSpeedChop = 2,

BlenderSpeedLiquify = 5,

} BlenderSpeed;

BlenderSpeed speed;

UTF-16: uses two or more bytes for every character.

UTF-8: uses one byte for the first 128 ASCII characters and two or more for other characters.

NSMutableString *str = [[NSMutableString alloc] init];

for (int i = 0; i < 10; i++) {

[str appendString:@"Aaron is cool!\n"];

}

NSError *error;

BOOl  success = [str writeToFile:@"/tmp/cool.txt"

atomically:YES

encoding:NSUTF8StringEncoding

error:&error];

Do not try to access the NSError unless the return value indicates that an error occurred; if the NSError object does not actually exist, trying to access it will crash your program.

NSError *error;

NSString *str = [[NSString alloc] initWithContentsOfFile:@"/etc/resolv.conf"

encoding:NSASCIIStringEncoding

error:&error];

if (!str) {

NSLog(@"read failed: %@", [error localizedDescription]);

} else {

NSLog(@"resolv.conf looks like this: \n%@", str);

}

NSLog(@"The file is %lu bytes", (unsigned long)[data length]);

BOOL written = [data writeToFile:@"/tmp/google.png"

options:NSDataWritingAtomic

error:&error];

NSData *readData = [NSData dataWithContentsOfFile:@"/tmp/google.png"];

NSLog(@"has %lu bytes", (unsigned long)[readData length]);

NSArray *desktops = NSSearchPathForDirectoriesInDomains(NSDesktopDirectory, NSUserDomainMask, YES); //get the path for user’s desktop directory

NSString *desktopPath = desktops[0];

NSApplicationDirectory; NSLibraryDirectory; NSUserDirectory; NSDocumentDirectory; NSDesktopDirectory; NSCachesDirectory; NSApplicationSupportDirectory; NSDownloadsDirectory; NSMoviesDirectory; NSMusicDirectory; NSPicturesDirectory; NSTrashDirectory.

A callback lets you write a piece of code and then associate that code with a particular event.

Four forms of callback:

Target-action: the object receiving the message is the target, the selector for the message is the action. “When this event happens, send this message to this object.”

Helper objects: aften known as delegates or data sources. “Here is an object that will take on a role that helps another object do its job.”

Notifications.

Blocks: just a chunk of code to be executed.

[[NSRunLoop currentRunLoop] run]; // Waiting for sth to happen

Creating a timer with a time interval, a target, and an action. After the interval has elapsed, the timer sends the action message to its target.

Action methods always take one argument - the object that is sending the action message.

static NSDateFormatter *dateFormatter = nil;

BNRLogger *logger = [[BNRLogger alloc] init];

NSTimer *timer = [[NSTimer scheduledTimerWithInterval: 2.8

target: logger

selector: @selector(updateLastTime:)

userInfo.nil

repeats: YES]

[[NSRunloop currentRunLoop] run]

Problems with a synchronous connection:

It blocks the main thread while waiting for all the data to arrive.

It has no way to call back

In an asynchronous connection, the data comes in chunks rather than all at once. In the helper object, you implement the methods to be executed in response to different connection-related events.

A protocol is a list of method declarations.

NSURL *url = [NSURL URLWithString:@“http://www.baidu.com/besy.png”;

NSURLRequest *request = [NSURLRequest requestWithURL:url];

NSURLConnection *fetchConn = [[NSURLConnection alloc] initWithRequest:request

delegate:logger

startImmediately:YES];

Declaring that BNRLogger will implement methods from the NSURLConnectionDelegate and NSURLConnectionDataDelegate protocol.

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSdata *)data;

- (void)connectionDidFinishLoading:(NSURLConnection *)connection;

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;

In an asynchronous connection, you need an instance of NSMutableData. As the chunks of data arrive, you will add them to this object.

When sending one callback to one object, Apple uses target-action. When sending an assortment of callbacks to one object, Apple uses a helper object with a protocol. Objects that might need to trigger callbacks in several other objects use notification.

Each of objects can register as an observer with the notification center. When the time zone is changed, the notification NSSystemTimeZoneDidChangeNotification will be posted to the center, and the center will forward it to all the relevant observers.

[[NSNotificationCenter defaultCenter]

addObserver:logger

selector:@selector(zoneChange:)

name:NSSystemTimeZoneDidChangeNotification

object:nil];

Notification centers do not own their observers. If an object is an observer, it will typically remove itself from the notification center in its dealloc method.

[[NSNotificationCenter defaultCenter] removeObserver: self];

Objects do not own their delegates or data sources.

Objects do not own their targets.

To speed things up, the compiler assigns a unique number to each method name it encounters. At runtime, it uses that number instead of the method name.

A selector is the unique number that represents a particular method name. Using the @selector compiler directive to tell the compiler to look up the selector for the given method name.

^{

NSLog(@“This is an instruction whiten a block.”);

}

double (^divBlock)(double, double) = ^(double dividend, double divisor) {

double quotient = dividend / divisor;

return quotient;

}

double myQuotient = divBlock(42.0, 12.5);

void (^devowelizer)(id, NSUInteger, BOOL *);

NSArray *originalStrings = @[@"Sauerkraut", @"Raygun", @"Big Nerd Ranch", @"Mississippi"];

NSMutableArray *devowelizedStrings = [NSMutableArray array];

NSArray *vowels = @[@"a", @"e", @"i", @"o", @"u"];

void (^devowelizer)(id, NSUInteger, BOOL *);

devowelizer = ^(id string, NSUInteger i, BOOL *stop) {

NSRange yRange = [string rangeOfString:@"y" options:NSCaseInsensitiveSearch];

if (yRange.location != NSNotFound) {

*stop = YES;

return;     // End this iteration

}

NSMutableString *newString = [NSMutableString stringWithString:string];

for (NSString *s in vowels) {

NSRange fullRange = NSMakeRange(0, [newString length]);

[newString replaceOccurrencesOfString:s

withString:@""

options:NSCaseInsensitiveSearch

range:fullRange];

}

[devowelizedStrings addObject:newString];

};

[originalStrings enumerateObjectsUsingBlock:devowelizer];

NSLog(@"devewelized strings: %@", devowelizedStrings);

typedef void (^ArrayEnumerationBlock)(id, NSUInteger, BOOL *);

ArrayEnumerationBlock devowelizer;

An anonymous block is a block that you pass directly to a method without assigning it to a block variable first.

For primitive variables, the values are copied and stored as local variables within the block. For pointers, the block will keep a strong reference to the objects it references.

__weak BNREmployee *weakSelf = self;

myBlock = ^{

BNREmployee *innerSelf = weakSelf;

NSLog(@“Employee: %@”, innerSelf);

NSLog(@“Employee ID: %@”, innerSelf.employeeID);

};

By default, variables captured by a block are constant within the block, and you cannot change their values.

__block int counter = 0;

void (^counterBlock)() = ^{ counter++; };

counterBlock();

The class of an object is different from its role in a working system.

Frequently display data in an instance of UITableView, The UITableView object does not contain the data that it displays; it has to get data from a helper object. Creating the UITableView class specified the role of UITableView’s data source by creating the UITableViewDataSource protocol.

The UITableView class has dataSource property,

@property(nonatomic, assign) id<UITableViewDataSource> dataSource;

respondsToSelector: ——Implemented in NSObject. You pass in the selector for the method that you are asking about. The return will be YES if the object has that method.

A property list is a combination of any of the following things: NSArray, NSDictionary, NSString, NSData, NSDate, NSNumber( integer, float, or Boolean)。

NSMutableArray *stocks = [[NSMutableArray alloc] init];

NSMutableDictionary *stock;

stock = [NSMutableDictionary dictionary];

[stock setObject:@"AAPL" forKey:@"symbol"];

[stock setObject:[NSNumber numberWithInt:200] forKey:@"shares"];

[stocks addObject:stock];

stock = [NSMutableDictionary dictionary];

[stock setObject:@"GOOG" forKey:@"symbol"];

[stock setObject:[NSNumber numberWithInt:160] forKey:@"shares"];

[stocks addObject:stock];

[stocks writeToFile:@"/tmp/stocks.plist" atomically:YES];

NSArray *stockList = [NSArray arrayWithContentsOfFile:@"/tmp/stocks.plist"];

for (NSDictionary *d in stockList) {

NSLog(@"I have %@ shares of %@", [d objectForKey:@"shares"], [d objectForKey:@“symbol"]);

Output:

2015-10-05 14:40:44.893 OCDemo01[1462:114612] I have 200 shares of AAPL

2015-10-05 14:40:44.894 OCDemo01[1462:114612] I have 160 shares of GOOG

Cocoa is the collection of frameworks written by Apple that you use to write applications on the Mac. To write iOS apps, you use another set of frameworks called Cocoa Touch.

A GUI-based application is event-driven.

Models are responsible for storing data and making it available to other objects. Their sole purpose is holding and managing data. NSString, NSDate and NSArray are traditional model objects.

Views are the visual elements of an application. Views have no knowledge of the actual data that they display or how it is structured and stored. Like UIView, UITableView, UITextField and UIButton. If you can see it, is is a view.

Controllers perform the logic necessary to connect and drive the different parts of your application. Controllers are the real workhorses of any application.

#pragma mark - Application delegate callbacks group methods within a class.

When the application becomes ready, the UIApplication instance sends its delegate the message application:didFinishLaunchingWithOptions:. It is where you put everything that needs to done before the user interacts with the application.

When a Cocoa Touch application quits or is sent to the background, it sends its delegate the applicationDidEnterBackground: method from the UIApplicationDelegate protocol.

UIApplicationMain function creates the necessary objects for your application to run. It creates a single instance of the UIApplication class. Then, it creates an instance of the application’s delegate, so that it can send its delegate messages when memory gets low, when the application is quit or backgrounded, or when it finishes launching.

alloc creates the space for an object, and init makes the object ready to work.

The instancetype keyword tells the compiler to expect an instance of the class to which a method belongs. Any initializers you write or override should always return instancetype. Using instancetype ensures that initializers can be safely inherited.

- (instancetype) initWithProductName:(NSString *)pn {

// self = [super init];

if  (self = [super init]) {

_productName = [pn copy];

_voltage =120;

}

}

When you create a subclass, you typically only need to initialize the instance variables that the subclass introduced.

Add an implementation of the superclass’s initializer initWithProductName: that calls initWithProductName:firstOwnedName: and passes a default value for firstOwnedName. There is no need to implement init in subclass.

Stylish of writing initializer:

A class has only one designated initializer method. If the class has other initializers, then the implementation of those initializers must call (directly or indirectly) the designated initializer.

The designated initializer will call the superclass’s designated initializer before initializing its instance variables.

If the designated initializer of your class has a different name than the designated initializer of its superclass, you must override the superclass’s designated initializer so that it calls the new designated initializer.

If you have several initializers, clearly document which is the designated initializer in the header.

Overriding init to call the new class’s designated initializer with default values is not acceptable. The best thing to do is override the superclass’s designated initializer in a way that lets developers know that they have made a mistake and tells them how to fix it. e.g.

- (instancetype) init {

[NSException raise:@“BNRWallSafeInitialization”

format:@“Use initWithSecretCode:, not nil”];

}

assign is the default for non-object types and the simplest; it just assigns the passed-in value to the property.

strong will ensure that a strong reference is kept to passed-in object; it is the default for object pointers, and that is usually what you want.

weak does not imply ownership of the object pointed to, it keeps you safe from dangling pointers. If this object is deallocated, then the property will be set to nil.

unsafe_unretained properties do not imply ownership, it is not automatically set to nil when the object that it pointes to is deallocated.

copy option makes a copy of an object and then changes the pointer to refer to this copy. It is most common with object types that have mutable subclasses. e.g. @property (copy) NSString *lastName;

The generated setter method is:

- (void) setLastName:(NSString *)d {

_lastName = [d copy];

}

If you wish for your setter to set the property to be a mutable copy of an object, you must implement the setter yourself so that it calls the mutableCopy method on the incoming object.

The copyWithZone: and mutableCopyWithZone: methods are declared in the NSCopying and NSMutableCopying protocols, respectively. If you want your classes to be compatible with the copy property lifetime specifier, then you must ensure that they conform to the NSCopying protocol.

The nonatomic option will make your setter method run a tiny bit faster. You should always make your readwrite properties nonatomic too.

Two reasonable cases to implement an accessor yourself:

need to update the app’s user interface when the change occurs;

need to update some cached info when the change occurs.

If you declare a property and implement both accessors yourself, the compiler will not synthesize an instance variable. If you still want an instance variable, you must create it yourself by adding an @synthesize mushroom = _mushroom; statement to the class’s implementation.

The @synthesize statement tells the compiler that an instance variable named _mushroom is the backing variable for the mushroom and setMushroom: methods.

Key-Value coding:

[a setValue:@“Washing Machine” forKey:@“productName”];

[a setValue:[NSNumber numberWithInt:240] forKey:@“voltage”];

// go looking for a setter method named setProductName:. If the object does not have a setProductName: method, it will access the instance variable directly.

[a valueForKey:@“productName”];

// go looking for an accessor named productName.

Anytime a standard framework wants to push data into your objects, it will use setValue:forKey:. Anytime a standard framework wants to read data from your objects, it will use valueForKey:.

NSString *number = [sales valueForKey:@“manager.emergencyContact.phoneNumber”];

[sales setValue:@“555-666” forKey:@“manager.emergencyContact.phoneNumber”

Key-value observing (KVO): lets you get a notification when a particular property of an object changes.

BNRObserver *observer = [[BNRObserver alloc] init];

[logger addObserver:observer

forKeyPath:@“lastTime”

options: NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld

context:&contextForKVO];

- (void) observeValueForKeyPath:(NSString *)keyPath

ofObject:(id)object

change:(NSDictionary *)change

context:(void *)context {

if (context != &contextForKVO) {

… } else {

NSString *oldValue = [change objectForKey:NSKeyValueChangeOldKey];

NSString *newValue = [change objectForKey:NSKeyValueChangeNewKey];

}

}

- (void) updateLastTime:(NSTimer *)t {

NSDate *date = [NSdate date];

[self willChangeValueForKey:@“lastTime”];

_lastTime = now;

[self didChangeValueForKey:@“lastTime”];

}

+ (NSSet *)keyPathForValuesAffectingLastTimeString {

return [NSSet setWithObject:@“lastTime”];

}

Categories let a programmer add methods to any existing class.

@interface NSString (BNRVowelCounting)

- (int)bnr_vowelCount;

@end

@implementation NSString (BNRVowelCounting)

// implementing

@end

Objective-C Programming The Big Nerd Ranch Guide 笔记 19-37的更多相关文章

  1. ios Programming:The Big Nerd Ranch Guid(6th Edition) (Joe Conway & AARON HILLEGASS 著)

    Introduction (已看) Prerequisites What Has Changed in the Sixth Edition? Our Teaching Philosophy How t ...

  2. 《Android权威编程指南(The Big Nerd Ranch Guide)(第二版)》12.4挑战练习

    本书第12章是讲解Dialog.12.4挑战练习是在CriminalIntent项目中,再增加一个TimePickerFragment的对话框fragment.通过在CrimeFragment用户界面 ...

  3. 阅读 Device Driver Programmer Guide 笔记

    阅读 Device Driver Programmer Guide 笔记 xilinx驱动命名规则 以X开头 源文件命名规则 以x打头 底层头文件与高级头文件 重点来了,关于指针的使用 其中 XDev ...

  4. "Principles of Reactive Programming" 之 <Persistent Actor State>学习笔记

    这是<Pinciples of Reactive Programming>week6的最后一课. 为什么需要把actor的状态持久化? 如果actor没有状态,那么在任何实时,这个acto ...

  5. Udemy - Angular 2 - The Complete Guide 笔记

    1. install > npm install -g angular-cli 2. create app > ng new first-app 3. build app > cd ...

  6. HTB Linux queuing discipline manual - user guide笔记

    1. Introduction HTB is meant as a more understandable, intuitive and faster replacement for the CBQ ...

  7. hbase definitive guide 笔记

    ext3 file system 优化 ext3 在用在hbase上可以做如下优化: 1. mount的时候加上noatime选项.这可以减少管理开销 2. 用命令tune2fs -m 0 /dev/ ...

  8. IOS中文版资源库

    Swift 语言写成的项目会被标记为  ★ ,AppleWatch 的项目则会被标记为 ▲. [转自]https://github.com/jobbole/awesome-ios-cn#librari ...

  9. 3000本IT书籍下载地址

    http://www.shouce.ren/post/d/id/112300    黑客攻防实战入门与提高.pdfhttp://www.shouce.ren/post/d/id/112299    黑 ...

随机推荐

  1. Windows Zip/CentOS/Radhat系统安装Mysql5.7.x方法

    CentOS/Redhat 安装: wget http://dev.mysql.com/get/mysql57-community-release-el7-9.noarch.rpm rpm -Uvh ...

  2. mysql数据表的基本操作:表结构操作,字段操作

    本节介绍: 表结构操作 创建数据表. 查看数据表和查看字段. 修改数据表结构 删除数据表 字段操作 新增字段. 修改字段数据类型.位置或属性. 重命名字段 删除字段 首发时间:2018-02-18  ...

  3. vertical-align属性探究

    在前端开发中我们经常需要将元素垂直居中对齐,我们很自然的想到使用vertical-align属性,但是使用后却发现有时候能起作用,有时候却又不起作用.究其原因还是因为我们没有将vertical-ali ...

  4. Docker Compose 安装 on centos7

    本文演示如何在CentOS7上安装Docker Compose. 1 在线安装 1.1 下载安装包 $ curl -L https://github.com/docker/compose/releas ...

  5. ERROR 3009 (HY000): Column count of mysql.user is wrong…..

    在测试备份还原时,使用XtraBackup还原数据库后,创建一个测试账号时遇到了下面错误: mysql> grant all on house.* to test@'192.168.%' ide ...

  6. java笔记----面试题总结(一)【转】

    1.面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: - 抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为抽象两方面.抽象只关注对象有哪些属性和行为,并不关注 ...

  7. jsp include 报错:illegal to have multiple occurrences of contentType with different values (old: text/html; charset=UTF-8, new: text/html; carset=UTF-8)

    严重: Servlet.service() for servlet jsp threw exception org.apache.jasper.JasperException: /jsp.jsp(1, ...

  8. MySQL 5.7安装指南

    1.下载 1)进入官网下载5.7.23压缩包 下载地址:https://dev.mysql.com/downloads/mysql/5.7.html#downloads 2.安装与配置 1)将下载的压 ...

  9. MySQL 系列(四) 主从复制、读写分离、模拟宕机、备份恢复方案生产环境实战

    本章内容: 主从复制 简介原理 备份主库及恢复从库,配置从库生效 读写分离 如果主宕机了,怎么办? 双主的情况 MySQL 备份及恢复方案 备份单个及多个数据库 mysqldump 的常用参数 如何增 ...

  10. 南邮ctf-web的writeup

    WEB 签到题 nctf{flag_admiaanaaaaaaaaaaa} ctrl+u或右键查看源代码即可.在CTF比赛中,代码注释.页面隐藏元素.超链接指向的其他页面.HTTP响应头部都可能隐藏f ...