0

SOAP Web サービスからの応答を待つ必要があります。返されるデータを操作し、クラスから呼び出し元のクラスに返す必要があるため、NSURLConnection を介して呼び出しています。

これが私のコードです:

#import <Foundation/Foundation.h>


@interface UsersBLL : NSObject {

 NSMutableData *webData;
 NSMutableString *soapResults;
 NSXMLParser *xmlParser;
 BOOL *recordResults;
 NSNumber *EmailCount;
}

@property(nonatomic, retain) NSMutableData *webData;
@property(nonatomic, retain) NSMutableString *soapResults;
@property(nonatomic, retain) NSXMLParser *xmlParser;



-(int)checkEmailAddress:(NSString*)emailAddress;
@end

#import "UsersBLL.h"


@implementation UsersBLL
@synthesize webData;
@synthesize soapResults;
@synthesize xmlParser;

-(id)init {
 self = [super init];
 return self;
}

-(int)checkEmailAddress:(NSString*)emailAddress {
 // Build the SOAP envelope
 NSString *soapMessage = [NSString stringWithFormat:
        @"<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"
        "<soap:Envelope xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">\n"
        "<soap:Body>\n"
        "<CheckEmailAddress xmlns=\"http://tempuri.org/\">\n"
        "<EmailAddress>%@</EmailAddress>\n"
        "</CheckEmailAddress>\n"
        "</soap:Body>\n"
        "</soap:Envelope>\n", emailAddress];

 NSLog(soapMessage);

 NSURL *url = [NSURL URLWithString:@"http://photoswapper.mick-walker.co.uk/UsersService.asmx?op=CheckEmailAddress"];
 NSMutableURLRequest *theRequest = [NSMutableURLRequest requestWithURL:url];
 NSString *msgLength = [NSString stringWithFormat:@"%d", [soapMessage length]];

 [theRequest addValue: @"text/xml; charset=utf-8" forHTTPHeaderField:@"Content-Type"];
 [theRequest addValue: @"http://tempuri.org/CheckEmailAddress" forHTTPHeaderField:@"SOAPAction"];
 [theRequest addValue: msgLength forHTTPHeaderField:@"Content-Length"];
 [theRequest setHTTPMethod:@"POST"];
 [theRequest setHTTPBody: [soapMessage dataUsingEncoding:NSUTF8StringEncoding]];

 NSURLConnection *theConnection = [[NSURLConnection alloc] initWithRequest:theRequest delegate:self];

 if( theConnection )
 {
  webData = [[NSMutableData data] retain];
 }
 else
 {
  NSLog(@"theConnection is NULL");
 }
 NSLog(@"%@", EmailCount);
}

-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
 [webData setLength: 0];
}
-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
 [webData appendData:data];
}
-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
 NSLog(@"ERROR with theConenction");
 [connection release];
 [webData release];
}
-(void)connectionDidFinishLoading:(NSURLConnection *)connection
{
 NSLog(@"DONE. Received Bytes: %d", [webData length]);
 NSString *theXML = [[NSString alloc] initWithBytes: [webData mutableBytes] length:[webData length] encoding:NSUTF8StringEncoding];
 NSLog(theXML);
 [theXML release];

 if( xmlParser )
 {
  [xmlParser release];
 }

 xmlParser = [[NSXMLParser alloc] initWithData: webData];
 [xmlParser setDelegate: self];
 [xmlParser setShouldResolveExternalEntities: YES];
 [xmlParser parse];

 [connection release];
 [webData release];
}

-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *) namespaceURI qualifiedName:(NSString *)qName
   attributes: (NSDictionary *)attributeDict
{
 if( [elementName isEqualToString:@"CheckEmailAddressResult"])
 {
  if(!soapResults)
  {
   soapResults = [[NSMutableString alloc] init];
  }
  recordResults = TRUE;
 }
}
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
{
 if( recordResults )
 {
  [soapResults appendString: string];
 }
}
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
{
 if( [elementName isEqualToString:@"CheckEmailAddressResult"])
 {
  recordResults = FALSE;
  NSNumberFormatter *formatter = [[NSNumberFormatter alloc]init];
  EmailCount = [formatter numberFromString:soapResults];
            [formatter release];
  [soapResults release];
  soapResults = nil;
 }
}

@end

CheckEmailAddress は、整数値を返すものとして宣言されています (上記のサンプルでは何も返さないことがわかっています)。

私が理想的に望むのは、CheckEmailAddress メソッドを使用して、Web サービスから取得した値を返すことです。ただし、リクエストが完了するまで NSURLConnection の呼び出しは待機しないため、実行できません。

誰かが回避策の潜在的なアイデアを教えていただければ幸いです。

4

4 に答える 4

2

最も簡単な解決策は、を使用すること[NSURLConnection sendSynchronousRequest:returningResponse:error:]です。

採用したアプローチほど多くの制御はできませんが、通常はほとんどのアプリケーションで十分です。

于 2009-12-13T21:55:58.793 に答える
1

呼び出し元のスレッドをブロックできるように、非同期 NSURLConnection をラップするソリューションを投稿しました。標準よりも詳細な制御が必要な場合は[NSURLConnection sendSynchronousRequest:returningResponse:error:]、StackOverflow の次のリンクを確認してください。

セマフォで実装された NSURLConnection ブロッキング ラッパー

于 2012-12-05T22:16:19.573 に答える
0

それはすべて、必要な非同期性のレベルによって異なります。

  • リクエスト全体でブロックされたままでも問題ない場合は、使用することをお勧めします

      +[NSURLConnection sendSynchronousRequest:returningResponse:error:]
    

    ただし、Wade が提案したように、NSURLRequest にタイムアウトを追加するように注意してください。そうしないと、接続がブロックされ、アプリケーションがハングする可能性があります。

  • そうでない場合は、単に NSNotificationCenter を使用できます。ただし、特に複数のリクエストを処理している場合は、データの競合状態に注意する必要があります

于 2010-03-02T10:38:40.697 に答える
0

次の 2 つの選択肢があります。

  • 使用する+[NSURLConnection sendSynchronousRequest:returningResponse:error:]

  • カスタムランループモードで接続をスケジュールし、データが到着するか、接続をキャンセルする必要があるまで、そのモードでループを実行します

于 2009-12-14T11:14:25.357 に答える