Skip to main content

NSThread with Asynchronous Request

NSThread:

It is class used in Cocoa framework to create a thread.You have to specify a target and selector to create a new thread in objective c. So the selector or method is main entry point of your thread. 


What happen once the thread execute all the code in the method (Thread main routine)?

Thread is terminated after execute all the code in the entry(its main routine) routine. So how we can run an asynchronous request on NSThread ?

What is Async request?

It is non blocking request which permit the application to do other task side by side as request is in continue. So it enhance parallelism in the application.


Now question is how we can prepare a asynchronous request in iOS:

- (void) startAsyncRequestWithUrl:(NSString*)urlString{
    assert(urlString != nil && urlString.length > 0);
    NSURL* url = [[NSURL alloc] initWithString:urlString];
    NSURLRequest* urlRequest = [[NSURLRequest alloc] initWithURL:url cachePolicy:NSURLRequestReloadIgnoringCacheData timeoutInterval:60];
    theConnection = [[NSURLConnection alloc] initWithRequest:urlRequest delegate:self];
    [theConnection start];

}


So above code has an example of asynchronous http request in iOS. As we know we need to implement delegate method  for the Url connection. So our main motive is to understand what happen  when we set the above method as main routine or entry point for NSThred then what will happen ?

So for the all the delegates method and complete HTTP connection file you can download from following link:

HttpConnection Link:

Let come back to our question is delegate call for the connection ? If yes then on main thread or the new thread?

So Just try this code ?

 HttpConnection* conn = [[HttpConnection alloc] initWithDelegate:self];
    NSString* urlString = @"https://imagizer.imageshack.us/v2/235x352q90/716/nkm8.jpg";
    [NSThread detachNewThreadSelector:@selector(startAsyncRequestWithUrl:) toTarget:conn withObject:urlString];

Did you get call in delegates ?

It will not come. Delegate will never be called and the obvious reason is thread terminate (which init the request).

So  now try to call by following method:
[conn startAsyncRequestWithUrl:urlString];

instead of that

 [NSThread detachNewThreadSelector:@selector(startAsyncRequestWithUrl:) toTarget:conn withObject:urlString]; 

Delegate will be called :)

Is there any tactic so that it can work on NSThread as well?

Yes 

RunLoop:

A run loop is event processing loop and it associated with each thread. So your main thread also have run loop. If you add any source of event to run loop then it will execute on the respective thread to which that run loop begin.

So our solution also use the same approach.  The basic idea is we need to busy the run loop of thread so that our thread will not terminated.

We need to busy the run loop associated with NSThread so that our thread should not terminate.
So we need to block the run loop for indefinite time until we did not receive the callback
We can do that by using following method:

- (connectionStatus) startAsyncRequestWithUrl:(NSString*)urlString{
    self.status = connectionSetupStage;
    assert(urlString != nil && urlString.length > 0);
    NSURL* url = [[NSURL alloc] initWithString:urlString];
    NSURLRequest* urlRequest = [[NSURLRequest alloc] initWithURL:url cachePolicy:NSURLRequestReloadIgnoringCacheData timeoutInterval:CONNECTIONTIMEOUT];
    theConnection = [[NSURLConnection alloc] initWithRequest:urlRequest delegate:self];
    
    while(self.status != connectionSucess && self.status != connectionFail){
        BOOL isBlockForInput = [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        NSLog(@"Blocked:%d",isBlockForInput);
    }
    return self.status;

}

So we are calling the following run loop method in a loop

[[NSRunLoop currentRunLooprunMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]]

By calling above method run loop will be trying to process the input source if any input source not available then it will return immediately. So If it return after delay it means it processed some input and print 1 in that case for above example. 

Can we optimize the code here ?

I think if there is no input source then CPU cycle will waste to processing the loop until it does not have any source.

So we can add sleep method here for the thread like given below:
 

 while(self.status != connectionSucess && self.status != connectionFail){
        BOOL isBlockForInput = [[NSRunLoop currentRunLooprunMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
       sleep(1);
        NSLog(@"Blocked:%d",isBlockForInput);
    }








time depend upon the request if request take time you can increase the sleep time.

You can find more detail of this method here :
Detail:

So done with the today topic.
Please mail me if you need more information on this.




Comments

Popular posts from this blog

iOS8 UIWebView Remove or Modify Keyboard Toolbar

Remove Toolbar: One of my client requirements is to remove the default toolbar for the keyboard. I found numerous examples for iOS7 to remove toolbar from the keyboard window. But it did not work for iOS8. So i came to following solution after some Research on Google: Step1:   First we need to add observer for keyboard notification: [[ NSNotificationCenter defaultCenter ]  addObserver : self selector : @selector ( removeKeyboardTopBar :) name : UIKeyboardWillShowNotification object : nil ]; Step 2: We need to implement the method for observer: - ( void )removeKeyboardTopBar {     } Step 3: Find the keyboard window from all the application  windows:   - ( void )removeKeyboardTopBar {     UIWindow *keyboardWindow = nil ;     UIView * toolBarContainer = nil ;     NSArray * windows = [[ UIApplication sharedApplication ] windows ];     for ( U...

Implement orientation modes in iPhone Hybrid Applications

Let suppose you are working on a hybrid application which runs only in single (portrait) mode. One day a requirement come that PDF and Doc Viewer (HTML Page) should support both (landscape, portrait) mode. Your Application loads all the HTML content from the local html files and you need to implement the above functionality only for one HTML file. Let break the above task in the modules: Step 1: Application should detect when the PDF and Doc viewer is open in application. I setup location.href tag in html to " docvieweron:// " and " docvieweroff:// " when page is open and closed respectively. In this way I am getting a delegate callback in web view: WebViewDelegate: - ( BOOL ) webView:( UIWebView *)webView shouldStartLoadWithRequest:( NSURLRequest *)request   navigationType: ( UIWebViewNavigationType )navigationType {          NSString * urlString = [[request URL ] absoluteString ];...

Blocks are Objective-C Objects !!!

Blocks Blocks are executable code just like a function.  It can be written inside a function or we can store the reference in a variable and call it later.  But wait a minute these all functionality we can achieve by function pointer and functions.  Then what is so special about blocks? Let first go through how we can define blocks. int (^sum)( int , int ) = ^( int number1, int number2){ return number1+number2; }; The above block can be used to find the sum of two numbers.  If we recall the function pointer in c then it is very much similar to above: int sumFunction ( int number1, int number2){ return number1+number2; } int (*sum) ( int , int ) = sumFunction; //Function Pointer *sum Then Why we need Blocks if we already have Function pointer?  1) B locks can capture variables from the enclosing scope by simply referring to them within the block. How ? Let see the next Section 2)  Blocks...