Recently a colleague asked: How is a weak object released? When will he be released?

Set the referent pointer in the Weak_table hash of SideTables to nil, and the weak object will be released. The release time is when the object dealloc calls objc_destructInstance to check for hash cleaning.

And then I thought, this is what I’ve seen before, so let’s go back.

The principle of weak

Create class Objc_Weak_Class, simple implementation:

//  Objc_Weak_Class.h

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface Objc_Weak_Class : NSObject

- (void)weakMethod;

@end

NS_ASSUME_NONNULL_END


//  Objc_Weak_Class.m

#import "Objc_Weak_Class.h"

@implementation Objc_Weak_Class

- (void)weakMethod
{
    Objc_Weak_Class *o = [[Objc_Weak_Class alloc] init];
    __weak id weakPtr = o;
    
    NSLog(@"weakMethod = %@",o);
}
@end
Copy the code

Compile it every time you don’t know what to do, in case you see something important. (This is very important and I hope to give readers of this article a way to solve the problem.)

For those of you who don’t know how to read compilations look here, open this up.

Okay, so that’s the end of the prep, so let’s make a break point here. (When the program runs here, open the view assembly)

The following assembly, where call XXX appears, indicates that a XXX method has been called.

Here we see three important method calls, so let’s break them down a little bit. (Objc_alloc_init is not clear.)

  • objc_alloc_initInitialize an object[[cls alloc] init];
  • objc_initWeakInitialize theweakSomething of;
  • objc_destroyWeakThe destructionweakSome of the things.

1, objc_initWeak

Command + shift + O search for objc_initWeak (this is recommended, sometimes not found in the upper left corner of Xcode)

/** * This function IS NOT thread-safe with respect to concurrent * modifications to the weak variable. (Concurrent weak  clear is safe.) * * @param location Address of __weak ptr. * @param newObj Object ptr. */ id objc_initWeak(id *location, id newObj) { if (! newObj) { *location = nil; return nil; } return storeWeak<DontHaveOld, DoHaveNew, DoCrashIfDeallocating> (location, (objc_object*)newObj); }Copy the code

And here apple has a little code that tells us how to start this method call.

/** * Initialize a fresh weak pointer to some object location. * It would be used for code like: * * (The nil case) * __weak id weakPtr; * (The non-nil case) * NSObject *o = ... ; * __weak id weakPtr = o; * /Copy the code

See a comment at the top: This function IS NOT thread-safe with respect to concurrent modifications to the weak variable. (Concurrent weak clear Is safe.), which means that changing the weak variable is not thread-safe, but cleaning up the weak variable is thread-safe.

After looking at these analyses, it becomes clear why threads are unsafe. (Written at the bottom)

OK, proceed to the storeWeak method:

2, storeWeak

Since it is initWeak, there is no old value, so it does not affect our analysis.

  • haveOld = flase
  • haveNew = ture
Static id storeWeak (id * the location, objc_object * newObj) {/ / check the old and new objects must exist a ASSERT (haveOld | | haveNew); // If haveNew == NO, check whether newObj! = nil crashes if (! haveNew) ASSERT(newObj == nil); Class previouslyInitializedClass = nil; id oldObj; SideTable *oldTable; SideTable *newTable; // Acquire locks for old and new values. // Order by lock address to prevent lock ordering problems. // Retry if the old Value changes Underneath. // Lock new and old values // Lock addresses to prevent lock sort problems. // If the old value changes below us, please try again. Retry: if (haveOld) {oldObj = *location; // StripedMap is a table where void *p is the key and PaddedT is the value. oldTable = &SideTables()[oldObj]; } else { oldTable = nil; } if (haveNew) {newTable = &SideTables()[newObj]; } else { newTable = nil; SideTable::lockTwo<haveOld, haveNew>(oldTable, newTable); SideTable::lockTwo<haveOld, haveNew>(oldTable, newTable); // If there is an old value, but *location! If (haveOld && *location! = oldObj) { SideTable::unlockTwo<haveOld, haveNew>(oldTable, newTable); goto retry; } // Prevent a deadlock between the weak reference machinery // and the +initialize machinery by ensuring that no // Weakly referenced object has an UN -+initialized isa. // We can prevent deadlocks between weakly referenced and +initialize by ensuring that weakly referenced objects don't have an uninitialized ISA. if (haveNew && newObj) { Class cls = newObj->getIsa(); // If the class is not initialized, it is initialized, and fault tolerant code if (CLS! = previouslyInitializedClass && ! ((objc_class *)cls)->isInitialized()) { SideTable::unlockTwo<haveOld, haveNew>(oldTable, newTable); class_initialize(cls, (id)newObj); // If this class is finished with +initialize then we're good. // If this class is still running +initialize on this thread // (i.e. +initialize called storeWeak on an instance of itself) // then we may proceed but it will appear initializing and // not yet initialized to the check above. // Instead set previouslyInitializedClass to recognize it on  retry. previouslyInitializedClass = cls; // Goto retry; If (haveOld) {if (haveOld) {if (haveOld) {if (haveOld) {if (haveOld) { weak_unregister_no_lock(&oldTable->weak_table, oldObj, location); } // Assign new value, NewObj = (objc_object *) Weak_register_no_lock (&newTable-> Weak_table, (id)newObj, location, crashIfDeallocating ? CrashIfDeallocating : ReturnNilIfDeallocating); // weak_register_no_lock returns nil if weak store should be rejected // Set is-weakly-referenced bit in refcount table.  if (! NewObj ->isTaggedPointerOrNil()) {// Set isa.weakly_referenced = true in obj; NewObj ->setWeaklyReferenced_nolock(); } // Do not set *location anywhere else. That would introduce a race. *location = (id)newObj; } else { // No new value. The storage is not changed. } SideTable::unlockTwo<haveOld, haveNew>(oldTable, newTable); // This must be called without the locks held, as it can invoke // arbitrary code. In particular, even if _setWeaklyReferenced // is not implemented, resolveInstanceMethod: May be, and may // call back into the weak reference machinery. // In particular, even if _setWeaklyReferenced is not implemented, resolveInstanceMethod: may also be implemented and may also call back the weak reference mechanism. callSetWeaklyReferenced((id)newObj); return (id)newObj; }Copy the code

Take a look at the code and comments above to comb through the flow:

  • 1. Get hash table objects with old and new values as keys
  • 2. After obtaining the hash table, determine if*location ! = oldObj, thread contention may occur and start all over again
  • 3. If there is an old value, release the old value pointer
  • Insert new value pointer address into new value hash table
  • 5. Set weak referencesisa.weakly_referenced = ture

3. Weak_entry_t structure description

In order to better understand the following content, weak_entry_t structure is explained first.

Struct weak_entry_t {// Weak reference object DisguisedPtr<objc_object> Referent; Weak_referrer_t = 0; weak_referrer_t = 0; weak_referrer_t = 0; weak_referrer_t = 0; uintptr_t out_of_line_ness : 2; uintptr_t num_refs : PTR_MINUS_2; // Uintptr_t mask; uintptr_t max_hash_displacement; // Maximum hash conflict value}; //WEAK_INLINE_COUNT : Struct {// out_of_line_ness field is low bits of inline_referrers[1] weak_referrer_t inline_referrers[WEAK_INLINE_COUNT]; }; }; Bool out_of_line() {return (out_of_line_ness == REFERRERS_OUT_OF_LINE); Weak_entry_t & operator=(const weak_entry_t& other) {memcpy(this, &other, sizeof(other));} //memcpy copy memory pointer method // overwrite old data. return *this; Weak_entry_t (objc_object *newReferent, objc_object **newReferrer) : weak_entry_t(objc_object ** newReferent); referent(newReferent) { inline_referrers[0] = newReferrer; for (int i = 1; i < WEAK_INLINE_COUNT; i++) { inline_referrers[i] = nil; }}};Copy the code

Weak_entry_t stores all weak reference Pointers of an object. If the number of weak reference objects is not more than four, an error is reported in the structure array inline_referrers, otherwise, it is stored in the referrers. In addition, the memory used in the Inline_Referrers array will be allocated when weak_entry_t is initialized, rather than applying for new memory space when it is needed, so as to achieve the purpose of improving operation efficiency.

4, weak_unregister_no_lock

Delete a registered weak reference.

** * Unregister a registered weak reference. * Used when the referrer's storage is about to disappear, but the referent is not dead. (Otherwise, zeroing out the referrer later would be an incorrect memory access.) * If the referent/referrer is not a weak reference currently activated, nothing is done. * References that are not 0. * * FIXME currently requires the old reference value to be passed in (lame) * If the referrer is collected, FIXME will automatically unregister * * @param Weak_table Globally weak table. * @param referent Referent object. * @param referrer weak references. */ void weak_unregister_no_lock(weak_table_t *weak_table, id referent_id, id *referrer_id) { objc_object *referent = (objc_object *)referent_id; objc_object **referrer = (objc_object **)referrer_id; weak_entry_t *entry; // If the object passed is empty, return if (! referent) return; If ((entry = weak_entry_for_referent(weak_table, Referent))) {// Remove the remove_referrer(entry, referrer) from the weak reference table; Bool empty = true; Weak_referrer_t *referrers and num_refs! If (entry->out_of_line() && Entry ->num_refs! = 0) { empty = false; For (size_t I = 0; // inline_referrers = 0; i < WEAK_INLINE_COUNT; i++) { if (entry->inline_referrers[i]) { empty = false; break; Weak_entry_remove (weak_table, entry); weak_entry_remove(weak_table, entry); } } // Do not set *referrer = nil. objc_storeWeak() requires that the // value not change. }Copy the code

1, weak_entry_for_referent

Weak_entry_for_referent from the above code see weak_entry_for_referent is to find the entry of the object in the weak reference table, and then analyze, the source code is as follows:

/** * Return the weak reference table entry for the given referent. * If there is no entry for referent, return NULL. // Perform lookup * * @param weak_table * @param referent The object. Must not be nil. * * @return The table of weak referrers to this object. */ static weak_entry_t * weak_entry_for_referent(weak_table_t *weak_table, objc_object *referent) { ASSERT(referent); Weak_entry_t 'Weak_entry_t * Weak_entries = Weak_table -> Weak_entries; // Return if (! If there are no weak reference objects in the global weak reference table. weak_entries) return nil; Weak_table ->mask and max_hash_displacement are copied in append_referrer. size_t begin = hash_pointer(referent) & weak_table->mask; size_t index = begin; size_t hash_displacement = 0; Weak_table -> Weak_entries [index]. Referent! Weak_table ->weak_entries[index]. Referent! Weak_table ->mask = (index+1) &weak_table ->mask; if (index == begin) bad_weak_table(weak_table->weak_entries); hash_displacement++; If (hash_displacement > weak_table->max_hash_displacement) {return nil; Weak_table -> Weak_entries [index]; weak_table->weak_entries[index]; }Copy the code

2, remove_referrer

Find a weak reference object to delete the last saved value, source code as follows:

/** * Remove old_referrer from set of referrers, if it's present. * Does not remove duplicates, because duplicates should not exist. * * @todo this is slow if old_referrer is not present. Is this ever the case? * * @param entry The entry holding the referrers. * @param old_referrer The referrer to remove. */ static void Weak_entry_t *entry, objc_object **old_referrer) {weak_entry_t *entry, objc_object **old_referrer) {if (! entry->out_of_line()) { for (size_t i = 0; i < WEAK_INLINE_COUNT; I ++) {if (entry->inline_referrers[I] == old_referrer) {entry->inline_referrers[I] = nil; return; } // Since calling remove_referrer means that the entry was found, Runtime: runtime: runtime (" runtime: failed ") runtime: runtime: failed (" runtime: failed ") runtime: failed (" runtime: failed ") runtime: failed (" runtime: failed ") runtime: failed (" runtime: failed" of " "objc_storeWeak() and objc_loadWeak(). " "Break on objc_weak_error to debug.\n", old_referrer); objc_weak_error(); return; Size_t begin = w_hash_pointer(old_referrer) & (entry->mask); size_t index = begin; size_t hash_displacement = 0; while (entry->referrers[index] ! = old_referrer) { index = (index+1) & entry->mask; if (index == begin) bad_weak_table(entry); hash_displacement++; // Since calling remove_referrer means that the entry was found, Crash if (hash_displacement > entry->max_hash_displacement) {_objc_inform(" runtime to unregister unknown __weak variable " "at %p. This is probably incorrect use of " "objc_storeWeak() and objc_loadWeak(). " "Break on objc_weak_error to debug.\n", old_referrer); objc_weak_error(); return; } // Referrers [index] = nil; // Number of references minus one entry->num_refs--; }Copy the code

3, weak_entry_remove

Delete the weak-reference object from the global weak-reference table.

/** * Remove entry from the zone's table of weak references. */ static void weak_entry_remove(weak_table_t *weak_table, Weak_entry_t *entry) {// Remove entry // If using the referrers release the referrers if (entry->out_of_line()) free(entry->referrers); // Clear bzero(entry, sizeof(*entry)); Weak_table ->num_entries--; weak_table->num_entries--; Weak_compact_maybe (weak_table); weak_compact_maybe(weak_table); }Copy the code

4, weak_compact_maybe

Determine whether to shrink the table only when old_size > 1024 is the original global hash and num_entries < old_size /16 is used to shrink the table.

// Shrink the table if it is mostly empty. static void weak_compact_maybe(weak_table_t *weak_table) { size_t old_size = TABLE_SIZE(weak_table); Shrink if larger than 1024 buckets and at most 1/16 full. // old_size > 1024 and num_entries < If (old_size >= 1024 && old_size /16 >= Weak_table ->num_entries) {// Shrink to half of the original table weak_resize(weak_table, old_size / 8); // leaves new table no more than 1/2 full } }Copy the code

5, weak_resize

Weak reference table generation method, source code as follows:

static void weak_resize(weak_table_t *weak_table, size_t new_size) { size_t old_size = TABLE_SIZE(weak_table); // weak_entry_t *old_entries = weak_table->weak_entries; Weak_entry_t *new_entries = (weak_entry_t *) calloc(new_size, sizeof(Weak_entry_t)); Weak_table ->mask = new_size - 1; weak_table->mask = new_size - 1; weak_table->mask = new_size - 1; weak_table->weak_entries = new_entries; weak_table->max_hash_displacement = 0; Weak_table ->num_entries = 0; weak_table->num_entries = 0; // restore by weak_entry_insert below // loop insert if (old_entries) {weak_entry_t *entry; weak_entry_t *end = old_entries + old_size; for (entry = old_entries; entry < end; entry++) { if (entry->referent) { weak_entry_insert(weak_table, entry); }} // Free old table free(old_entries); }}Copy the code

5, weak_register_no_lock

Insert the current weak reference object

/** * Registers a new (object, Creates a new weak * object entry if it does not exist Create a new weak object entry (if it doesn't exist) * * @param weak_table The global weak table. * @param referent The object pointed to by The weak reference. * @param referrer The weak pointer address. */ id weak_register_no_lock(weak_table_t *weak_table, id referent_id, id *referrer_id, WeakRegisterDeallocatingOptions deallocatingOptions) {/ / to insert objects objc_object * referent = (objc_object *) referent_id; objc_object **referrer = (objc_object **)referrer_id; If (referent->isTaggedPointerOrNil()) return referent_id; // ensure that the referenced object is viable if (deallocatingOptions == ReturnNilIfDeallocating || deallocatingOptions  == CrashIfDeallocating) { bool deallocating; ISA->hasCustomRR(), // This bit is used when the class or parent overrides the following methods / / retain/release/autorelease/retainCount _tryRetain / _isDeallocating retainWeakReference/allowsWeakReference returns true // Normally we don't override these methods, so return false and true if (! referent->ISA()->hasCustomRR()) { deallocating = referent->rootIsDeallocating(); } else { // Use lookUpImpOrForward so we can avoid the assert in // class_getInstanceMethod, Since we intentionally make this // callout with the lock held. // use lookupimportforward, // This way we can avoid assert in class_getInstanceMethod because we deliberately call this callout when the lock is held. auto allowsWeakReference = (BOOL(*)(objc_object *, SEL)) lookUpImpOrForwardTryCache((id)referent, @selector(allowsWeakReference), referent->getIsa()); if ((IMP)allowsWeakReference == _objc_msgForward) { return nil; } deallocating = ! (*allowsWeakReference)(referent, @selector(allowsWeakReference)); } // If it is destructed, If (Deallocating) {if (deallocatingOptions == CrashIfDeallocating) {_objc_fatal("Cannot form weak reference to instance (%p) of " "class %s. It is possible that this object was " "over-released, or is in the process of deallocation.", (void*)referent, object_getClassName((id)referent)); } else { return nil; }}} // Now remember it and where it is being stored /// Now remember it and where it was stored weak_entry_t *entry; Weak_entry_for_referent (weak_table, referent)) {// Append_referrer (entry, referrer); // Weak_entry_for_referent (weak_table, referent); Weak_entry_t new_entry(referent, referrer); weak_entry_t (referent, referrer); Weak_grow_maybe (weak_table); weak_grow_maybe(weak_table); Weak_entry_insert (weak_table, &new_entry); weak_entry_insert(weak_table, &new_entry); } // Do not set *referrer. objc_storeWeak() requires that the // value not change. return referent_id; }Copy the code

1, weak_grow_maybe

If the size is greater than or equal to 3/4 of the original table size, it needs to be expanded. If the original table does not exist, a new table size of 64 bytes is generated

// Grow the given zone's table of weak references if it is full. static void weak_grow_maybe(weak_table_t *weak_table) {  size_t old_size = TABLE_SIZE(weak_table); Weak_table ->num_entries >= old_size * 3/4) {weak_table->num_entries >= old_size * 3/4) {weak_table->num_entries >= old_size * 3/4) { Weak_resize (Weak_table, old_size? old_size*2 : 64); }}Copy the code

2, weak_entry_insert

Weak_entry_insert method does not judge whether the current object is directly inserted in the global weak reference table.

Here is the same idea as we write code, for better performance and reuse, business logic in the upper level of its own judgment, the lower level can only be manipulated.

/** * Add new_entry to the object's table of weak references. * Does not check whether the referent is already in the Table. */ /** * adds new_entry to the weak reference table of the object. * Do not check whether the referent is already in the table. */ static void weak_entry_insert(weak_table_t *weak_table, weak_entry_t *new_entry) { weak_entry_t *weak_entries = weak_table->weak_entries; ASSERT(weak_entries ! = nil); size_t begin = hash_pointer(new_entry->referent) & (weak_table->mask); size_t index = begin; size_t hash_displacement = 0; While (weak_entries[index].referent! = nil) { index = (index+1) & weak_table->mask; if (index == begin) bad_weak_table(weak_entries); hash_displacement++; Weak_entries [index] = *new_entry; weak_entries[index] = *new_entry; weak_entries[index] = *new_entry; Weak_table ->num_entries++; weak_table->num_entries++; Weak_table ->max_hash_displacement) {weak_table->max_hash_displacement = hash_displacement; }}Copy the code

3, append_referrer

/** * Add the given referrer to set of weak pointers in this entry. * Does not perform duplicate checking (b/c weak pointers are never * added to a set twice). * * @param entry The entry holding the set of weak pointers. * @param new_referrer The new weak pointer to be added. */ static void append_referrer(weak_entry_t *entry, Objc_object **new_referrer) {// If inline_referrers is used, insert if (! entry->out_of_line()) { // Try to insert inline. for (size_t i = 0; i < WEAK_INLINE_COUNT; i++) { if (entry->inline_referrers[i] == nil) { entry->inline_referrers[i] = new_referrer; return; }} // insert inline. Allocate out of line. // If inline_referrers is empty, Weak_referrer_t *referrers weak_Referrer_t * new_Referrers = (weak_referrer_t *) calloc(WEAK_INLINE_COUNT, sizeof(weak_referrer_t)); // This constructed table is invalid, But grow_refs_and_insert // will fix it and rehash it. i < WEAK_INLINE_COUNT; i++) { new_referrers[i] = entry->inline_referrers[i]; } entry->referrers = new_referrers; entry->num_refs = WEAK_INLINE_COUNT; // Referrers enable the corresponding out_of_lin() method call entry->out_of_line_ness = REFERRERS_OUT_OF_LINE; //mask = 3 entry->mask = WEAK_INLINE_COUNT-1; entry->max_hash_displacement = 0; } ASSERT(entry->out_of_line()); If (entry->num_refs >= TABLE_SIZE(entry) * 3/4) {return grow_refs_and_insert(entry, new_referrer); } size_t begin = w_hash_pointer(new_referrer) & (entry->mask); size_t index = begin; size_t hash_displacement = 0; While (entry->referrers[index]! = nil) { hash_displacement++; index = (index+1) & entry->mask; if (index == begin) bad_weak_table(entry); If (hash_displacement > entry->max_hash_displacement) {entry->max_hash_displacement = hash_displacement; Weak_referrer_t &ref = entry-> Referrers [index]; weak_referrer_t &ref = entry-> Referrers [index]; ref = new_referrer; Entry ->num_refs++; }Copy the code

6, callSetWeaklyReferenced

CallSetWeaklyReferenced The setWeaklyReferenced method is called to set the weak reference flag bit of an object to true.

// Call out to the _setWeaklyReferenced method on obj, if implemented. static void callSetWeaklyReferenced(id obj) { if (! obj) return; Class cls = obj->getIsa(); if (slowpath(cls->hasCustomRR() && ! object_isClass(obj))) { ASSERT(((objc_class *)cls)->isInitializing() || ((objc_class *)cls)->isInitialized()); void (*setWeaklyReferenced)(id, SEL) = (void(*)(id, SEL)) class_getMethodImplementation(cls, @selector(_setWeaklyReferenced)); if ((IMP)setWeaklyReferenced ! = _objc_msgForward) { (*setWeaklyReferenced)(obj, @selector(_setWeaklyReferenced)); }}}Copy the code

1, setWeaklyReferenced_nolock

Core code Newisa.Weakly_referenced = true; Sets the weak reference flag bit of the object to true.

inline void objc_object::setWeaklyReferenced_nolock() { isa_t newisa, oldisa = LoadExclusive(&isa.bits); do { newisa = oldisa; if (slowpath(! newisa.nonpointer)) { ClearExclusive(&isa.bits); sidetable_setWeaklyReferenced_nolock(); return; } if (newisa.weakly_referenced) { ClearExclusive(&isa.bits); return; } // Set newisa.weakly_referenced = true; } while (slowpath(! StoreExclusive(&isa.bits, &oldisa.bits, newisa.bits))); }Copy the code

7, objc_destroyWeak

The objc_destroyWeak process is the same as objc_initWeak except that the new object is passed in as nil and cleaned, not inserted.

void
objc_destroyWeak(id *location)
{
    (void)storeWeak<DoHaveOld, DontHaveNew, DontCrashIfDeallocating>
        (location, nil);
}
Copy the code

8. Clear weak objects in dealloc

Dealloc -> clearDeallocating -> sidetable_clearDeallocating() -> sidetable_clearDeallocating() -> weak_clear_no_lock()

Weak_clear_no_lock source code is as follows:

void weak_clear_no_lock(weak_table_t *weak_table, id referent_id) { objc_object *referent = (objc_object *)referent_id; weak_entry_t *entry = weak_entry_for_referent(weak_table, referent); if (entry == nil) { /// XXX shouldn't happen, but does with mismatched CF/objc //printf("XXX no entry for clear deallocating %p\n", referent); return; Weak_referrer_t *referrers; weak_referrer_t *referrers; size_t count; if (entry->out_of_line()) { referrers = entry->referrers; count = TABLE_SIZE(entry); } else { referrers = entry->inline_referrers; count = WEAK_INLINE_COUNT; } for (size_t I = 0; i < count; ++i) { objc_object **referrer = referrers[i]; if (referrer) { if (*referrer == referent) { *referrer = nil; } else if (*referrer) { _objc_inform("__weak variable at %p holds %p instead of %p. " "This is probably incorrect use of  " "objc_storeWeak() and objc_loadWeak(). " "Break on objc_weak_error to debug.\n", referrer, (void*)*referrer, (void*)referent); objc_weak_error(); Weak_entry_remove (weak_table, entry); weak_entry_remove(weak_table, entry); }Copy the code

Weak threads are unsafe

If we simplify the above code, we can see:

  • objc_initWeakInitialize theweakSomething of;
  • objc_destroyWeakThe destructionweakSome of the things.

objc_initWeak :

id objc_initWeak(id *location, id newObj)
{
    return storeWeak<DontHaveOld, DoHaveNew, DoCrashIfDeallocating>
        (location, (objc_object*)newObj);
}
Copy the code

objc_destroyWeak:

void objc_destroyWeak(id *location)
{
    (void)storeWeak<DoHaveOld, DontHaveNew, DontCrashIfDeallocating>
        (location, nil);
}
Copy the code

StoreWeak:

static id storeWeak(id *location, objc_object *newObj) { Class previouslyInitializedClass = nil; id oldObj; SideTable *oldTable; SideTable *newTable; retry: //... SideTable::lockTwo<haveOld, haveNew>(oldTable, newTable); / /... // unregister a registered weak reference. weak_unregister_no_lock(&oldTable->weak_table, oldObj, location); / /... // Insert new location newObj = (objc_object *) weak_register_no_lock(&newTable->weak_table, (id)newObj, location, crashIfDeallocating ? CrashIfDeallocating : ReturnNilIfDeallocating); / /... SideTable::unlockTwo<haveOld, haveNew>(oldTable, newTable); SideTable::unlockTwo<haveOld, haveNew>(oldTable, newTable); callSetWeaklyReferenced((id)newObj); return (id)newObj; }Copy the code

The above code can see that both objc_initWeak and objc_destroyWeak call storeWeak, When weak_unregister_NO_lock and weak_register_NO_lock are analyzed above, it is found that the operation of pointer address is in the two methods, but there is no lock in the method, so in the case of multi-threaded operation, it is only by locking the address to lock the sorting problem, Pointer operations are not protected, so the initialization and destruction of weak are made thread unsafe.

Here the analysis of the principle of weak is over. If there is any problem, welcome the boss to point out. We can learn together and make continuous progress.

PS: runnable and constantly annotated objC source address.