
Libffiを使用してObjective-Cブロックをフックします。ブロック用の強力なAOPツールです。 BlockHookは、ブロックを呼び出した後、/代わりに/代わりにコードを実行できます。 BlockHookは、ブロックが取引をしたときに通知することさえできます。ブロックフックを使用して、ブロックのライフサイクル全体をトレースできます!
メソッドを通過するブロックをフックしたいですか?ブロックトラッカーを試してみてください!
BlockHookには、iOS、TVOS、およびMacOSをサポートするLibffiが必要です。 BlockHookSample iOS 、 BlockHookSample tvOS 、またはBlockHookSample macOSターゲットを実行できます。
4つのモードを使用してブロックをフックすることができます(前/代わり/後/死)。このメソッドは、より多くの制御のためにBHTokenインスタンスを返します。 BHToken remove 、 retValueプロパティにカスタムリターン値を設定したりできます。 invokeOriginalBlockメソッドを呼び出すと、ブロックの元の実装が呼び出されます。
- (BHToken *)block_hookWithMode:(BlockHookMode)mode
usingBlock:( id )blockブロックフックは使いやすいです。そのAPIは、側面によって模範を示します。これがブロックフックの完全な使用セットです。
これは、すべてのモードでフックブロックの例です。ブロック戻り値を8から15に変更できます。次に、フックを取り外して、成功しているかどうかを確認できます。最後に、Block Deallocのときにコールバックを取得します。
NSObject *z = NSObject .new;
int (^block)( int x, int y) = ^ int ( int x, int y) {
int result = x + y;
NSLog ( @" %d + %d = %d , z is a NSObject: %@ " , x, y, result, z);
return result;
};
BHToken *token = [block block_hookWithMode: BlockHookModeDead|BlockHookModeBefore|BlockHookModeInstead|BlockHookModeAfter usingBlock: ^(BHInvocation *invocation, int x, int y) {
int ret = 0 ;
[invocation getReturnValue: &ret];
switch (invocation. mode ) {
case BlockHookModeBefore:
// BHInvocation has to be the first arg.
NSLog ( @" hook before block! invocation: %@ " , invocation);
break ;
case BlockHookModeInstead:
[invocation invokeOriginalBlock ];
NSLog ( @" let me see original result: %d " , ret);
// change the block imp and result
ret = x * y;
[invocation setReturnValue: &ret];
NSLog ( @" hook instead: '+' -> '*' " );
break ;
case BlockHookModeAfter:
// print args and result
NSLog ( @" hook after block! %d * %d = %d " , x, y, ret);
break ;
case BlockHookModeDead:
// BHInvocation is the only arg.
NSLog ( @" block dead! token: %@ " , invocation. token );
break ;
default :
break ;
}
}];
NSLog ( @" hooked block " );
int ret = block( 3 , 5 );
NSLog ( @" hooked result: %d " , ret);
// remove token.
[token remove ];
NSLog ( @" remove tokens, original block " );
ret = block( 3 , 5 );
NSLog ( @" original result: %d " , ret);これがログです:
hooked block
hook before block! invocation:<BHInvocation: 0x60b00003c370>
3 + 5 = 8, z is a NSObject: <NSObject: 0x6020000279d0>
let me see original result: 0
hook instead: '+' -> '*'
hook after block! 3 * 5 = 15
hooked result:15
block dead! token:<BHToken: 0x60d000004bd0>
remove tokens, original block
3 + 5 = 8, z is a NSObject: <NSObject: 0x6020000279d0>
original result:8
他のコンポーネントにルーティングする前に、最初にユーザーログインが必要な場合があります。ルーターのコードにハッキングせずにブロックを傍受するには、ブロックインターセプターを使用できます。
NSObject *testArg = [ NSObject new ];
NSObject *testArg1 = [ NSObject new ];
NSObject *(^testblock)( NSObject *) = ^( NSObject *a) {
return [ NSObject new ];
};
[testblock block_interceptor: ^(BHInvocation *invocation, IntercepterCompletion _Nonnull completion) {
dispatch_after ( dispatch_time (DISPATCH_TIME_NOW, ( int64_t )( 0.5 * NSEC_PER_SEC)), dispatch_get_main_queue (), ^{
NSObject * __unsafe_unretained arg;
[invocation getArgument: &arg atIndex: 1 ];
NSLog ( @" Original argument: %@ " , arg);
[invocation setArgument: ( void *)&testArg1 atIndex: 1 ];
completion ();
});
}];
testblock (testArg);Cocoapodsは、Cocoaプロジェクトの依存マネージャーです。次のコマンドでインストールできます。
$ gem install cocoapods cocoapodsを使用してブロックフックをXcodeプロジェクトに統合するには、 Podfileで指定します。
source 'https://github.com/CocoaPods/Specs.git'
platform :ios, '8.0'
use_frameworks!
target 'MyApp' do
pod 'BlockHook'
end
「myApp」をプロジェクトの名前に置き換える必要があります。
次に、次のコマンドを実行します。
$ pod installCarthageは、依存関係を構築し、バイナリフレームワークを提供する分散型依存関係マネージャーです。
次のコマンドを使用して、HomeBrewでCarthageをインストールできます。
$ brew update
$ brew install carthage Carthageを使用してBlockhookをXcodeプロジェクトに統合するには、 Cartfileで指定します。
github "yulingtianxia/BlockHook"
carthage updateを実行してフレームワークを構築し、構築されたBlockHook.frameworkをXcodeプロジェクトにドラッグします。
libffiをインポートした後、2つのファイルBlockHook.h/mプロジェクトに追加するだけです。
Yulingtianxia、[email protected]
BlockHookはMITライセンスの下で利用できます。詳細については、ライセンスファイルを参照してください。
MablockClosureとAscepsに感謝します!