Web Service使用的主要協(xié)議是HTTP協(xié)議,即 超文本傳輸協(xié)議 。
10年積累的網(wǎng)站制作、網(wǎng)站建設(shè)經(jīng)驗(yàn),可以快速應(yīng)對客戶對網(wǎng)站的新想法和需求。提供各種問題對應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識你,你也不認(rèn)識我。但先網(wǎng)站設(shè)計后付款的網(wǎng)站建設(shè)流程,更有青田免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
HTTP/1.1協(xié)議共定義了8種請求方法(OPTIONS、HEAD、GET、POST、PUT、DELETE、TRACE、CONNECT)作為Web服務(wù)器。
GET方法 ,是向指定的資源發(fā)送請求,請求的參數(shù)“顯式”地在URL的后面。有點(diǎn)像明信片,把內(nèi)容“顯式”寫在外面,因此安全性比較差。一般使用于讀取數(shù)據(jù)、例如從服務(wù)器讀取靜態(tài)圖片、或查詢數(shù)據(jù)等。
POST方法 ,是向指定資源提交數(shù)據(jù),請求服務(wù)器進(jìn)行處理,數(shù)據(jù)包含在 請求體 中。參數(shù)和地址分開,放在body里面。有點(diǎn)像把信內(nèi)容放在信封中,接觸的人看不到,安全性比較高。一般用于例如提交表單、上傳文件等(請求的的動態(tài)資源,與查詢類似,每個方法調(diào)用都要傳遞很多參數(shù),因此需要使用NSMutableURLRequest創(chuàng)建請求。 )
iOS SDK中為HTTP請求提供了同步和異步請求這兩種不同的API,
同步請求,可以從因特網(wǎng)請求數(shù)據(jù),一旦發(fā)送同步請求,程序?qū)⑼V褂脩艚换ィ敝练?wù)器返回數(shù)據(jù)完成,才可以進(jìn)行下一步操作,意味著線程阻塞;
異步請求,不會阻塞主線程,而會建立一個新的線程來操作,用戶發(fā)出異步請求后,依然可以對UI進(jìn)行操作,程序可以繼續(xù)運(yùn)行;
它們的主要區(qū)別在于連接方式的不同。
下面通過請求一個登陸接口介紹有關(guān)于網(wǎng)絡(luò)請求中的不同情況。
【本次開發(fā)環(huán)境: Xcode:7.2 ? ??iOS Simulator:iphone6 ? By:啊左】
一、GET方法
1.同步get方法:
//1.創(chuàng)建一個web路徑?
NSString*webPath=[NSString?stringWithFormat:@";pass=%@btn=login",yourname,yourpass]; ? ? ?
webPath=[webPath stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];?//url不允許為中文等特殊字符,需要進(jìn)行字符串的轉(zhuǎn)碼為URL字符串,例如空格轉(zhuǎn)換后為“%20”;? ? ?
NSURL *url=[NSURL URLWithString:webPath];? ? ??
//2.根據(jù)WEB路徑創(chuàng)建一個請求? ? ?
NSURLRequest? *request=[NSURLRequest requestWithURL:url];? ? ? ? ?
NSURLResponse *respone;//獲取連接的響應(yīng)信息,可以為nil? ? ?
NSError *error;? ? ? ? //獲取連接的錯誤時的信息,可以為nil? ??
//3.得到服務(wù)器數(shù)據(jù)? ??
NSData*data=[NSURLConnection sendSynchronousRequest: request returningResponse: respone error: error];? ??
if(data==nil)? ? {? ? ? ? NSLog(@"登陸失敗:%@,請重試",error);? ? ? ? return;? ? }?
/*? ? ? ? 4.對服務(wù)器獲取的數(shù)據(jù)data進(jìn)行相應(yīng)的處理;? */
2.異步get方法:
異步請求與同步請求的不同在于使用NSURLConnectionDataDelegate委托協(xié)議,指定代理.
@interface ViewController : UIViewController // 遵循協(xié)議
@property (weak,nonatomic) NSMutableData *receiveData;? //創(chuàng)建一個可變data,用于異步接收服務(wù)器的數(shù)據(jù)
@end
創(chuàng)建網(wǎng)絡(luò)請求:
//1.創(chuàng)建一個web路徑
NSString*webPath=[NSString stringWithFormat: @";pass=%@btn=login",yourname,yourpass];
webPath=[webPath stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
NSURL? *url=[NSURL URLWithString:webPath];
//2.根據(jù)WEB路徑創(chuàng)建一個請求
NSURLRequest *request=[NSURLRequest requestWithURL:url];
//3.指定代理 以異步的方式接收數(shù)據(jù)NSURLConnectionDataDelegate
NSURLConnection? *con=[NSURLConnection connectionWithRequest:request delegate:self];
if(con==nil)
{
NSLog(@"創(chuàng)建連接失敗.");
return;
}
else//成功 準(zhǔn)備接數(shù)據(jù)
{
if(self.receiveData==nil)
{
self.receiveData=[[NSMutableData alloc] init];
}
}
異步的代理行為:
-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response{? ??
NSLog(@"已經(jīng)響應(yīng)成功.");
//清空 為當(dāng)前連接做準(zhǔn)備? ??
self.receiveData.length=0;
}
-(void) connection:(NSURLConnection *)connection didReceiveData:(NSData *)data{? ? NSLog(@"已經(jīng)接收到了數(shù)據(jù).");? ??
//追加接收到的數(shù)據(jù)? ??
[self.receiveData appendData:data];
}
-(void)connectionDidFinishLoading:(NSURLConnection *)connection{? ??
NSLog(@"接收數(shù)據(jù)已經(jīng)完成.");?
/*? ? ? 對服務(wù)器獲取的數(shù)據(jù)receiveData進(jìn)行相應(yīng)的處理;? */
}
-(void) connection:(NSURLConnection *)connection didFailWithError:(NSError *)error{? ? NSLog(@"連接失敗.");
}
二、POST方法
1.同步post方法:
//1.創(chuàng)建一個web路徑? ??
NSString? *webPath=@""; ??
webPath = [webPath stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];? ? NSURL *url=[NSURL URLWithString:webPath];
//2.建立一個帶協(xié)議緩存類型的請求 (使用NSMutableURLRequest,是post方法的關(guān)鍵)? ? NSMutableURLRequest? *request=[NSMutableURLRequest requestWithURL:url cachePolicy:(NSURLRequestUseProtocolCachePolicy) timeoutInterval:10];? ? ? ? //3.設(shè)置表單提交的方法(默認(rèn)為get)? ? [request setHTTPMethod: @"post"];
//4.設(shè)置要提交的參數(shù)? ? NSString? *args=[NSString stringWithFormat:@"uname=%@upas=%@btn=login",uname,upas];? ? [request setHTTPBody: [args dataUsingEncoding:NSUTF8StringEncoding]];
NSData *recvData=[NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil];? ? if(recvData!=nil)? ? {
/*
對服務(wù)器獲取的數(shù)據(jù)recvData進(jìn)行相應(yīng)的處理
*/
}
else
{
NSLog(@"連接失敗,請重試!");
}
2.post方法的異步與同步的區(qū)別在于使用NSURLConnectionDataDelegate委托協(xié)議,指定代理.
這一點(diǎn)與get方法一致,所以就不進(jìn)行長篇幅的演示了。
以上就是關(guān)于部分網(wǎng)絡(luò)同步異步請求,get、post請求方法的演示,由于UI控件還有其他的處理沒有附上,具體的讀者可以進(jìn)行相應(yīng)細(xì)節(jié)的調(diào)整,進(jìn)行完整的網(wǎng)絡(luò)請求項(xiàng)目開發(fā)。
由于iOS開始,引入了新的網(wǎng)絡(luò)接口NSURLSession,而在iOS9中NSURLConnection被宣布棄用,因此關(guān)于NSURLSession發(fā)送GET和POST請求的資料部分,有興趣的可以參考:
iOS開發(fā) GET、POST請求方法(NSURLSession篇)
? by:啊左~
HTTP(Hypertext Transfer Protocol)超文本傳輸協(xié)議是用來在Internet上傳送超文本的傳送協(xié)議,它可以使瀏覽器更加高效,使網(wǎng)絡(luò)傳輸減少。但HTTP協(xié)議采用明文傳輸信息,存在信息竊聽、信息篡改和信息劫持的風(fēng)險。
HTTPS(Secure Hypertext Transfer Protocol) 安全超文本傳輸協(xié)議是一個安全的通信通道,它基于HTTP開發(fā),用于在客戶計算機(jī)和服務(wù)器之間交換信息。HTTPS使用安全套接字層(SSL)進(jìn)行信息交換,簡單來說HTTPS是HTTP的安全版,是使用TLS/SSL加密的HTTP協(xié)議
由于xcode不支持http請求所以我們需要通過修改info.plist文件來讓xcode支持http請求,我們需要在info.plist文件中增加如下鍵值對
只要完成以上三步就可以實(shí)現(xiàn)GET的請求了
下面只列舉出GET方式的協(xié)議代理的請求方式,POST的請求方式只需要另外設(shè)置一下NSMutableURLRequest請求的方式,再設(shè)置一個參數(shù)就可以了,具體的協(xié)議代理的方法,和GET請求時使用的是一樣的 在使用協(xié)議的方法請求數(shù)據(jù)的時候,我們需要先遵循 NSURLSessionTaskDelegate,NSURLSessionDataDelegate 這兩個協(xié)議
至此,只要完成了以上六個步驟,并且實(shí)現(xiàn)了三個協(xié)議方法,就可以成功的通過協(xié)議代理的方式請求回來數(shù)據(jù)了。
一、網(wǎng)絡(luò)各個協(xié)議:TCP/IP、SOCKET、HTTP等
網(wǎng)絡(luò)七層由下往上分別為物理層、數(shù)據(jù)鏈路層、網(wǎng)絡(luò)層、傳輸層、會話層、表示層和應(yīng)用層。
其中物理層、數(shù)據(jù)鏈路層和網(wǎng)絡(luò)層通常被稱作媒體層,是網(wǎng)絡(luò)工程師所研究的對象;
傳輸層、會話層、表示層和應(yīng)用層則被稱作主機(jī)層,是用戶所面向和關(guān)心的內(nèi)容。
http協(xié)議對應(yīng)于應(yīng)用層
tcp協(xié)議對應(yīng)于傳輸層
ip協(xié)議對應(yīng)于網(wǎng)絡(luò)層
三者本質(zhì)上沒有可比性。 何況HTTP協(xié)議是基于TCP連接的。
TCP/IP是傳輸層協(xié)議,主要解決數(shù)據(jù)如何在網(wǎng)絡(luò)中傳輸;而HTTP是應(yīng)用層協(xié)議,主要解決如何包裝數(shù)據(jù)。
我 們在傳輸數(shù)據(jù)時,可以只使用傳輸層(TCP/IP),但是那樣的話,由于沒有應(yīng)用層,便無法識別數(shù)據(jù)內(nèi)容,如果想要使傳輸?shù)臄?shù)據(jù)有意義,則必須使用應(yīng)用層 協(xié)議,應(yīng)用層協(xié)議很多,有HTTP、FTP、TELNET等等,也可以自己定義應(yīng)用層協(xié)議。WEB使用HTTP作傳輸層協(xié)議,以封裝HTTP文本信息,然 后使用TCP/IP做傳輸層協(xié)議將它發(fā)送到網(wǎng)絡(luò)上。Socket是對TCP/IP協(xié)議的封裝,Socket本身并不是協(xié)議,而是一個調(diào)用接口(API),通過Socket,我們才能使用TCP/IP協(xié)議。
二、Http和Socket連接區(qū)別
相信不少初學(xué)手機(jī)聯(lián)網(wǎng)開發(fā)的朋友都想知道Http與Socket連接究竟有什么區(qū)別,希望通過自己的淺顯理解能對初學(xué)者有所幫助。
2.1、TCP連接
要想明白Socket連接,先要明白TCP連接。手機(jī)能夠使用聯(lián)網(wǎng)功能是因?yàn)槭謾C(jī)底層實(shí)現(xiàn)了TCP/IP協(xié)議,可以使手機(jī)終端通過無線網(wǎng)絡(luò)建立TCP連接。TCP協(xié)議可以對上層網(wǎng)絡(luò)提供接口,使上層網(wǎng)絡(luò)數(shù)據(jù)的傳輸建立在“無差別”的網(wǎng)絡(luò)之上。
建立起一個TCP連接需要經(jīng)過“三次握手”:
第一次握手:客戶端發(fā)送syn包(syn=j)到服務(wù)器,并進(jìn)入SYN_SEND狀態(tài),等待服務(wù)器確認(rèn);
第二次握手:服務(wù)器收到syn包,必須確認(rèn)客戶的SYN(ack=j+1),同時自己也發(fā)送一個SYN包(syn=k),即SYN+ACK包,此時服務(wù)器進(jìn)入SYN_RECV狀態(tài);
第三次握手:客戶端收到服務(wù)器的SYN+ACK包,向服務(wù)器發(fā)送確認(rèn)包ACK(ack=k+1),此包發(fā)送完畢,客戶端和服務(wù)器進(jìn)入ESTABLISHED狀態(tài),完成三次握手。
握
手過程中傳送的包里不包含數(shù)據(jù),三次握手完畢后,客戶端與服務(wù)器才正式開始傳送數(shù)據(jù)。理想狀態(tài)下,TCP連接一旦建立,在通信雙方中的任何一方主動關(guān)閉連
接之前,TCP
連接都將被一直保持下去。斷開連接時服務(wù)器和客戶端均可以主動發(fā)起斷開TCP連接的請求,斷開過程需要經(jīng)過“四次握手”(過程就不細(xì)寫了,就是服務(wù)器和客
戶端交互,最終確定斷開)
2.2、HTTP連接
HTTP協(xié)議即超文本傳送協(xié)議(HypertextTransfer Protocol ),是Web聯(lián)網(wǎng)的基礎(chǔ),也是手機(jī)聯(lián)網(wǎng)常用的協(xié)議之一,HTTP協(xié)議是建立在TCP協(xié)議之上的一種應(yīng)用。
HTTP連接最顯著的特點(diǎn)是客戶端發(fā)送的每次請求都需要服務(wù)器回送響應(yīng),在請求結(jié)束后,會主動釋放連接。從建立連接到關(guān)閉連接的過程稱為“一次連接”。
1)在HTTP 1.0中,客戶端的每次請求都要求建立一次單獨(dú)的連接,在處理完本次請求后,就自動釋放連接。
2)在HTTP 1.1中則可以在一次連接中處理多個請求,并且多個請求可以重疊進(jìn)行,不需要等待一個請求結(jié)束后再發(fā)送下一個請求。
由
于HTTP在每次請求結(jié)束后都會主動釋放連接,因此HTTP連接是一種“短連接”,要保持客戶端程序的在線狀態(tài),需要不斷地向服務(wù)器發(fā)起連接請求。通常的
做法是即時不需要獲得任何數(shù)據(jù),客戶端也保持每隔一段固定的時間向服務(wù)器發(fā)送一次“保持連接”的請求,服務(wù)器在收到該請求后對客戶端進(jìn)行回復(fù),表明知道客
戶端“在線”。若服務(wù)器長時間無法收到客戶端的請求,則認(rèn)為客戶端“下線”,若客戶端長時間無法收到服務(wù)器的回復(fù),則認(rèn)為網(wǎng)絡(luò)已經(jīng)斷開。
三、SOCKET原理
3.1、套接字(socket)概念
套接字(socket)是通信的基石,是支持TCP/IP協(xié)議的網(wǎng)絡(luò)通信的基本操作單元。它是網(wǎng)絡(luò)通信過程中端點(diǎn)的抽象表示,包含進(jìn)行網(wǎng)絡(luò)通信必須的五種信息:連接使用的協(xié)議,本地主機(jī)的IP地址,本地進(jìn)程的協(xié)議端口,遠(yuǎn)地主機(jī)的IP地址,遠(yuǎn)地進(jìn)程的協(xié)議端口。
應(yīng)
用層通過傳輸層進(jìn)行數(shù)據(jù)通信時,TCP會遇到同時為多個應(yīng)用程序進(jìn)程提供并發(fā)服務(wù)的問題。多個TCP連接或多個應(yīng)用程序進(jìn)程可能需要通過同一個
TCP協(xié)議端口傳輸數(shù)據(jù)。為了區(qū)別不同的應(yīng)用程序進(jìn)程和連接,許多計算機(jī)操作系統(tǒng)為應(yīng)用程序與TCP/IP協(xié)議交互提供了套接字(Socket)接口。應(yīng)
用層可以和傳輸層通過Socket接口,區(qū)分來自不同應(yīng)用程序進(jìn)程或網(wǎng)絡(luò)連接的通信,實(shí)現(xiàn)數(shù)據(jù)傳輸?shù)牟l(fā)服務(wù)。
3.2 、建立socket連接
建立Socket連接至少需要一對套接字,其中一個運(yùn)行于客戶端,稱為ClientSocket,另一個運(yùn)行于服務(wù)器端,稱為ServerSocket。
套接字之間的連接過程分為三個步驟:服務(wù)器監(jiān)聽,客戶端請求,連接確認(rèn)。
服務(wù)器監(jiān)聽:服務(wù)器端套接字并不定位具體的客戶端套接字,而是處于等待連接的狀態(tài),實(shí)時監(jiān)控網(wǎng)絡(luò)狀態(tài),等待客戶端的連接請求。
客戶端請求:指客戶端的套接字提出連接請求,要連接的目標(biāo)是服務(wù)器端的套接字。為此,客戶端的套接字必須首先描述它要連接的服務(wù)器的套接字,指出服務(wù)器端套接字的地址和端口號,然后就向服務(wù)器端套接字提出連接請求。
連
接確認(rèn):當(dāng)服務(wù)器端套接字監(jiān)聽到或者說接收到客戶端套接字的連接請求時,就響應(yīng)客戶端套接字的請求,建立一個新的線程,把服務(wù)器端套接字的描述發(fā)給客戶
端,一旦客戶端確認(rèn)了此描述,雙方就正式建立連接。而服務(wù)器端套接字繼續(xù)處于監(jiān)聽狀態(tài),繼續(xù)接收其他客戶端套接字的連接請求。
3.3、SOCKET連接與TCP連接
創(chuàng)建Socket連接時,可以指定使用的傳輸層協(xié)議,Socket可以支持不同的傳輸層協(xié)議(TCP或UDP),當(dāng)使用TCP協(xié)議進(jìn)行連接時,該Socket連接就是一個TCP連接。
3.4、Socket連接與HTTP連接
由
于通常情況下Socket連接就是TCP連接,因此Socket連接一旦建立,通信雙方即可開始相互發(fā)送數(shù)據(jù)內(nèi)容,直到雙方連接斷開。但在實(shí)際網(wǎng)絡(luò)應(yīng)用
中,客戶端到服務(wù)器之間的通信往往需要穿越多個中間節(jié)點(diǎn),例如路由器、網(wǎng)關(guān)、防火墻等,大部分防火墻默認(rèn)會關(guān)閉長時間處于非活躍狀態(tài)的連接而導(dǎo)致
Socket 連接斷連,因此需要通過輪詢告訴網(wǎng)絡(luò),該連接處于活躍狀態(tài)。
而HTTP連接使用的是“請求—響應(yīng)”的方式,不僅在請求時需要先建立連接,而且需要客戶端向服務(wù)器發(fā)出請求后,服務(wù)器端才能回復(fù)數(shù)據(jù)。
很
多情況下,需要服務(wù)器端主動向客戶端推送數(shù)據(jù),保持客戶端與服務(wù)器數(shù)據(jù)的實(shí)時與同步。此時若雙方建立的是Socket連接,服務(wù)器就可以直接將數(shù)據(jù)傳送給
客戶端;若雙方建立的是HTTP連接,則服務(wù)器需要等到客戶端發(fā)送一次請求后才能將數(shù)據(jù)傳回給客戶端,因此,客戶端定時向服務(wù)器端發(fā)送連接請求,不僅可以
保持在線,同時也是在“詢問”服務(wù)器是否有新的數(shù)據(jù),如果有就將數(shù)據(jù)傳給客戶端。
這里我們使用Socket實(shí)現(xiàn)一個聊天室的功能,關(guān)于服務(wù)器這里的就不介紹了
@interfaceViewController (){
NSInputStream *_inputStream;//對應(yīng)輸入流
NSOutputStream *_outputStream;//對應(yīng)輸出流
}
@property (weak, nonatomic) IBOutlet NSLayoutConstraint *inputViewConstraint;
@property (weak, nonatomic) IBOutlet UITableView *tableView;
@property (nonatomic, strong) NSMutableArray *chatMsgs;//聊天消息數(shù)組
@end
懶加載這個消息數(shù)組
//從主運(yùn)行循環(huán)移除
//1.建立連接
//定義C語言輸入輸出流
//把C語言的輸入輸出流轉(zhuǎn)化成OC對象
//設(shè)置代理
//把輸入輸入流添加到主運(yùn)行循環(huán)
//不添加主運(yùn)行循環(huán) 代理有可能不工作
//打開輸入輸出流
//登錄
//發(fā)送用戶名和密碼
//在這里做的時候,只發(fā)用戶名,密碼就不用發(fā)送
//如果要登錄,發(fā)送的數(shù)據(jù)格式為 "iam:zhangsan";
//如果要發(fā)送聊天消息,數(shù)據(jù)格式為 "msg:did you have dinner";
//登錄的指令11NSString *loginStr =@"iam:zhangsan";
//把Str轉(zhuǎn)成NSData
//建立一個緩沖區(qū) 可以放1024個字節(jié)
//返回實(shí)際裝的字節(jié)數(shù)
//把字節(jié)數(shù)組轉(zhuǎn)化成字符串
//從服務(wù)器接收到的數(shù)據(jù)
//聊天信息
//刷新表格
//發(fā)送數(shù)據(jù)
//發(fā)送完數(shù)據(jù),清空textField
//數(shù)據(jù)多,應(yīng)該往上滾動
}
//監(jiān)聽鍵盤
//獲取窗口的高度
//鍵盤結(jié)束的Frm
//獲取鍵盤結(jié)束的y值