iOS混合程式設計之使用Swift+Objective-C呼叫WebService
最近由於專案的需要,要使用Swift語言開發個iOS專案,某個簡單的需求就是要呼叫遠端的WebService資料。問題出現了,網上有很多使用OC呼叫WebService的例子,但是幾乎沒有找到使用Swift呼叫WebService的例子,我苦苦搜尋了好幾個小時,就是沒有滿足我要求的,怎麼辦,專案要泡湯了?Swift和OC,我該怎麼選擇?
天無絕人之路,還好Swift和OC可以完美的實現混合程式設計,可以相互實現呼叫。然後又在網上找到一個使用OC訪問WebService的例子http://my.oschina.net/plumsoft/blog/75277,也非常感謝這位博主。好吧,我來寫一個基於Swift和OC混合程式設計呼叫WebService的例子吧。最下面可以下載我的原始碼哦。
(1)新建一個Swift專案,在Main.storyboard中設計介面如下,分別繫結號碼輸入框和查詢按鈕到ViewControlle.swift中。
。
注意以下程式碼是Swift:
import UIKit
class ViewController: UIViewController {
@IBOutlet weak var phoneNumber: UITextField!//輸入手機號碼的文字框;
override func viewDidLoad() {
super.viewDidLoad()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
//查詢按鈕的響應;
@IBAction func beginQuery(sender: AnyObject) {
}
}
(2)右鍵專案New Files.選擇新建一個Cocoa Touch Class,命名為OCWebService,然後會彈出一個是否建立一個Bridging-Header.h檔案,點選Yes即可。注意非常重要:需要在Bridging-Header.h橋接標頭檔案中import如這個OC的標頭檔案!!
。
。
(3)然後在OCWebService.h中宣告如下屬性和方法,OCWebService.h具體程式碼如下:注意這個程式碼是OC。
#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
//需要有兩個Delegate,一個是解析XML用的,一個是網路連線用的;
@interface OCWebService : NSObject<NSXMLParserDelegate, NSURLConnectionDelegate>
@property (strong, nonatomic) NSMutableData *webData;
@property (strong, nonatomic) NSMutableString *soapResults;
@property (strong, nonatomic) NSXMLParser *xmlParser;
@property (nonatomic) BOOL elementFound;
@property (strong, nonatomic) NSString *matchingElement;
@property (strong, nonatomic) NSURLConnection *conn;
-(void)query:(NSString*)phoneNumber;//在標頭檔案中宣告查詢方法;
@end
(4)OCWebService.m中具體的方法實現如下:注意這個程式碼是OC。
#import "OCWebService.h"
@implementation OCWebService
@synthesize webData;
@synthesize soapResults;
@synthesize xmlParser;
@synthesize elementFound;
@synthesize matchingElement;
@synthesize conn;
-(void)query:(NSString*)phoneNumber{
// 設定我們之後解析XML時用的關鍵字,與響應報文中Body標籤之間的getMobileCodeInfoResult標籤對應
matchingElement = @"getMobileCodeInfoResult";
// 建立SOAP訊息,內容格式就是網站上提示的請求報文的主體實體部分 這裡使用了SOAP1.2;
NSString *soapMsg = [NSString stringWithFormat:
@"<?xml version=\"1.0\" encoding=\"utf-8\"?>"
"<soap12:Envelope "
"xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" "
"xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" "
"xmlns:soap12=\"http://www.w3.org/2003/05/soap-envelope\">"
"<soap12:Body>"
"<getMobileCodeInfo xmlns=\"http://WebXml.com.cn/\">"
"<mobileCode>%@</mobileCode>"
"<userID>%@</userID>"
"</getMobileCodeInfo>"
"</soap12:Body>"
"</soap12:Envelope>", phoneNumber, @""];
// 將這個XML字串列印出來
NSLog(@"%@", soapMsg);
// 建立URL,內容是前面的請求報文報文中第二行主機地址加上第一行URL欄位
NSURL *url = [NSURL URLWithString: @"http://webservice.webxml.com.cn/WebServices/MobileCodeWS.asmx"];
// 根據上面的URL建立一個請求
NSMutableURLRequest *req = [NSMutableURLRequest requestWithURL:url];
NSString *msgLength = [NSString stringWithFormat:@"%lu", (unsigned long)[soapMsg length]];
// 新增請求的詳細資訊,與請求報文前半部分的各欄位對應
[req addValue:@"application/soap+xml; charset=utf-8" forHTTPHeaderField:@"Content-Type"];
[req addValue:msgLength forHTTPHeaderField:@"Content-Length"];
// 設定請求行方法為POST,與請求報文第一行對應
[req setHTTPMethod:@"POST"];
// 將SOAP訊息加到請求中
[req setHTTPBody: [soapMsg dataUsingEncoding:NSUTF8StringEncoding]];
// 建立連線
conn = [[NSURLConnection alloc] initWithRequest:req delegate:self];
if (conn) {
webData = [NSMutableData data];
}
}
// 剛開始接受響應時呼叫
-(void) connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *) response{
[webData setLength: 0];
}
// 每接收到一部分資料就追加到webData中
-(void) connection:(NSURLConnection *)connection didReceiveData:(NSData *) data {
[webData appendData:data];
}
// 出現錯誤時
-(void) connection:(NSURLConnection *)connection didFailWithError:(NSError *) error {
conn = nil;
webData = nil;
}
// 完成接收資料時呼叫
-(void) connectionDidFinishLoading:(NSURLConnection *) connection {
NSString *theXML = [[NSString alloc] initWithBytes:[webData mutableBytes]
length:[webData length]
encoding:NSUTF8StringEncoding];
// 列印出得到的XML
NSLog(@"%@", theXML);
// 使用NSXMLParser解析出我們想要的結果
xmlParser = [[NSXMLParser alloc] initWithData: webData];
[xmlParser setDelegate: self];
[xmlParser setShouldResolveExternalEntities: YES];
[xmlParser parse];
}
#pragma mark -
#pragma mark XML Parser Delegate Methods
// 開始解析一個元素名
-(void) parser:(NSXMLParser *) parser didStartElement:(NSString *) elementName namespaceURI:(NSString *) namespaceURI qualifiedName:(NSString *) qName attributes:(NSDictionary *) attributeDict {
if ([elementName isEqualToString:matchingElement]) {
if (!soapResults) {
soapResults = [[NSMutableString alloc] init];
}
elementFound = YES;
}
}
// 追加找到的元素值,一個元素值可能要分幾次追加
-(void)parser:(NSXMLParser *) parser foundCharacters:(NSString *)string {
if (elementFound) {
[soapResults appendString: string];
}
}
// 結束解析這個元素名
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {
if ([elementName isEqualToString:matchingElement]) {
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"手機號碼資訊"
message:[NSString stringWithFormat:@"%@", soapResults]
delegate:self
cancelButtonTitle:@"確定"
otherButtonTitles:nil];
[alert show];
elementFound = FALSE;
// 強制放棄解析
[xmlParser abortParsing];
}
}
// 解析整個檔案結束後
- (void)parserDidEndDocument:(NSXMLParser *)parser {
if (soapResults) {
soapResults = nil;
}
}
// 出錯時,例如強制結束解析
- (void) parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError {
if (soapResults) {
soapResults = nil;
}
}
@end
(5)最後再來到ViewController.swift的專案主檔案使用Swift呼叫上述用OC寫的方法:主要實現是在beginQuery()方法中。注意這個是Swift程式碼
//查詢按鈕的響應;
@IBAction func beginQuery(sender: AnyObject) {
var webservice = OCWebService()//主要在OCWebService這個OC寫的類中實現對WebService的呼叫;
webservice.query(phoneNumber.text)//呼叫query方法開始訪問WebService,傳遞的引數是電話號碼;
}
(6)執行程式,檢視效果。
。
。
該專案Demo下載地址:http://pan.baidu.com/s/1hqgwnJI 。以後我再也不糾結選擇Swift還是OC了。
github主頁:https://github.com/chenyufeng1991 。歡迎大家訪問!
相關文章
- 【混合程式設計】C/C++呼叫Fortran的DLL程式設計C++
- QML之C++混合程式設計C++程式設計
- 好程式設計師分享WebService的簡單使用程式設計師Web
- webservice介面呼叫Web
- webapi建立和呼叫WebServiceWebAPI
- 一種WebService的呼叫方式Web
- OC/Swift/C/C++混合使用的程式設計姿勢SwiftC++程式設計
- Groovy + Java 混合程式設計方案:GMavenJava程式設計Maven
- CUDA 8的混合精度程式設計程式設計
- C++ & Intel MKL 混合程式設計C++Intel程式設計
- 騰訊WebService Api 跨域呼叫WebAPI跨域
- 併發程式設計之:非同步呼叫獲取返回值程式設計非同步
- 函數語言程式設計之尾呼叫和尾遞迴函數程式設計遞迴
- iOS開發 面向切面程式設計之 Aspects 原始碼解析iOS程式設計原始碼
- Python 程式設計之Tkinter的使用01Python程式設計
- 非同步程式設計之使用yield from非同步程式設計
- 【IDL】 IDL與C#混合程式設計技術C#程式設計
- C++:與C混合程式設計 CMake undefined reference toC++程式設計Undefined
- Java併發程式設計之CyclicBarrier使用指南Java程式設計
- Flutter混合開發之FlutterFragment使用FlutterFragment
- iOS程式設計師如何成為程式設計高手,並以此創業iOS程式設計師創業
- 程式設計思想之冪等性 | 程式設計之道程式設計
- iOS進階課程-Newsstand程式設計iOS程式設計
- iOS 逆向程式設計(入門條件)iOS程式設計
- ios 面向協議程式設計資源iOS協議程式設計
- 【iOS印象】 併發程式設計:Operation QueuesiOS程式設計
- BIRT 怎麼呼叫 Webservice 作為資料來源Web
- 用WebService呼叫第三方天氣介面Web
- XCode 中 Swift / Objective-C / C / C++ 混合程式設計XCodeSwiftObjectC++程式設計
- 市場不缺IOS程式設計師,缺的是IOS大牛iOS程式設計師
- 譯:原生iOS應用程式和原生Android應用程式設計之間的差異iOSAndroid程式設計
- Flutter混合開發-iOSFlutteriOS
- [Go語言寫介面]三、使用介面設計器設計視窗,在程式碼中呼叫,背景編輯器的使用Go
- 防禦式程式設計之斷言assert的使用程式設計
- python之超程式設計Python程式設計
- Spring之切面程式設計Spring程式設計
- Linux之shell程式設計Linux程式設計
- 使用python程式設計實現資料清洗之呼叫百度API實現地域維度規範化Python程式設計API
- iOS音訊程式設計之實時語音通訊(對講機功能)iOS音訊程式設計