在開發過程當中,常常須要接入第三方支付。下面對支付進行一個歸納。java
支付寶SDK下載地址:https://doc.open.alipay.com/doc2/detail?treeId=54&articleId=103419&docType=1xcode
在支付寶集成的過程當中,會遇到一些報錯,下面就碰到的報錯,和你們一塊兒解決。安全
集成參考官方文檔:app
導入代碼文檔:https://doc.open.alipay.com/doc2/detail.htm?spm=0.0.0.0.RiFaVa&treeId=59&articleId=103676&docType=1dom
Demo運行注意:https://doc.open.alipay.com/doc2/detail.htm?spm=0.0.0.0.Ravnv1&treeId=59&articleId=103677&docType=1ide
1.新建一個aliPaySDK文件夾,提取SDK包中如下文件到此文件夾裏,把文件拉到xcode裏測試
2.在xcode中,點擊項目名,選擇Build Phases -->Link Binary With Libraries 添加依賴庫。ui
注意:若是漏了部分沒有添加的,有時會報其它各類錯。編碼
3.編譯項目,會出現如下錯誤。spa
1)Cannot find
interface
declaration
for
'NSObject'
, superclass of
'Base64'
那麼須要打開報錯了的文件,增長頭文件。
#import <Foundation/Foundation.h
2)"Unknown type name ‘NSString‘ "或者"Unknown type name ‘NSData‘ "等不識別常見類的問題。
這是由於缺乏Foundation類庫和UIKit類庫,支付寶Demo中之因此沒有出現此錯誤,是由於在.pch文件中導入過這些類庫
解決辦法:只須要在出現錯誤的文件中導入這些類庫便可,能夠在.pch中添加,也能夠直接在openssl_wrapper.h中添加
#import <UIKit/UIKit.h> #import <Foundation/Foundation.h>
3)‘openssl/asn1.h‘ file not found
這個處理方式,參考上面連接「Demo的運行注意」第4點。
點擊項目名稱,點擊「Build Settings」選項卡,在搜索框中,以關鍵字「search」搜索,對「Header Search Paths」增長頭文件路徑:$(SRCROOT)/項目名稱。若是頭文件信息已增長,可沒必要再增長。
$(SRCROOT)/項目名稱 必定要注意這個路徑。
4)若是你的app基於9.0編譯,那麼爲了適配iOS9.0中的App Transport Security(ATS)對http的限制,這裏須要對支付寶的請求地址alipay.com、alipayobjects.com作例外,在app對應的info.list中添加以下配置(文中以XML格式描述)。
<key>NSAppTransportSecurity</key> <dict> <key>NSExceptionDomains</key> <dict> <key>alipay.com</key> <dict> <key>NSIncludesSubdomains</key> <true/> <key>NSTemporaryExceptionAllowsInsecureHTTPLoads</key> <true/> <key>NSTemporaryExceptionMinimumTLSVersion</key> <string>TLSv1.0</string> <key>NSTemporaryExceptionRequiresForwardSecrecy</key> <false/> </dict> <key>alipayobjects.com</key> <dict> <key>NSIncludesSubdomains</key> <true/> <key>NSTemporaryExceptionAllowsInsecureHTTPLoads</key> <true/> <key>NSTemporaryExceptionMinimumTLSVersion</key> <string>TLSv1.0</string> <key>NSTemporaryExceptionRequiresForwardSecrecy</key> <false/> </dict> </dict> </dict>
5)配置支付寶返回
注意:定義的返回types,不能帶下畫線,建議字母或者字母+數字
//應用註冊scheme,在AlixPayDemo-Info.plist定義URL types NSString *appScheme = @"alisdkPaydemo";
基本上已經完成了,下面只須要拷貝demo裏的測試代碼,便可以進行支付寶支付了。
#pragma mark ==============產生隨機訂單號============== - (NSString *)generateTradeNO { static int kNumber = 15; NSString *sourceStr = @"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"; NSMutableString *resultStr = [[NSMutableString alloc] init]; srand((unsigned)time(0)); for (int i = 0; i < kNumber; i++) { unsigned index = rand() % [sourceStr length]; NSString *oneStr = [sourceStr substringWithRange:NSMakeRange(index, 1)]; [resultStr appendString:oneStr]; } return resultStr; } - (void)payForAlipay { /* *點擊獲取prodcut實例並初始化訂單信息 */ //Product *product = [self.productList objectAtIndex:indexPath.row]; NSString *productName = @"支付寶測試商品"; //商品標題 NSString *productDescription = @"支付寶測試商品描述"; //商品描述 NSString *amount = @"0.01"; //商品價格 NSString *notifyURL = @"http://www.xxx.com"; //回調URL /* *商戶的惟一的parnter和seller。 *簽約後,支付寶會爲每一個商戶分配一個惟一的 parnter 和 seller。 */ /*============================================================================*/ /*=======================須要填寫商戶app申請的===================================*/ /*============================================================================*/ NSString *partner = @"2088811956033819"; // 這是支付寶文檔裏的測試帳號 NSString *seller = @"xinyuxingbangxinxi@163.com"; // 這是支付寶文檔裏的測試帳號 NSString *privateKey = @"MIICdwIBADANBgkqhkiG9w0BAQEFAASCAmEwggJdAgEAAoGBAJ3U/F+CuYUTwIlefRyWcwNvecIx756fbYFKp3Ymfy3MNdzccPc7Kk62RKCpVTgshHGVx0UUpRGzSV1y6M17teOTY/1KVSxhB6wvlOkTxF9SYiXWyr6ioHyl1etxjnQOtL0zqA3ID1vjOzMZZXM0+QE8dnc3rXoOSE7xOvkIb+RJAgMBAAECgYAdE0Rer+1PN6FLbQ2tO4X6hwmuHZbf6My6ea8508OwAyOVCUMCOHMFxwwDcM5TJ9hKOGZaMoBqL1X/khCS8gxCkwVEsIqr0/A4b2wBcJqtYXYx9onhUDjpfc/DjJ/DJx0VDDuEpeM5++djBTDxEjzDmEgK27trfPwm7cNbJjxPJQJBANb6bBpmUnml22bUu4jMeVAQZekg+ho3tMr8aa/np0CK8Jdq9je/HBhPXkVMGDhXlX4hAOYGI6wF2vrmz7ExRdMCQQC78v+lCXRtmsMzJQzE6tZAVG8ErFYpfm+23Ebn+36w8E+VNT+8wquoCD8tXsBssvBwdT6ZRqmEeEV77mdZ18/zAkBvcl1OhlMlW1VVht09uvr9BbM/W2gs5UolnRtRJN+w9xZo+PtxxPJUq/isJhm8Q7NtMsDbfr1JdbOjNLrhGjEfAkEArFeroeskjuit+7UKm3r3ka+ayX851vywdc5RWqGbz6XcY+abFnyvqPo+7FyJOGNw5L4t86D/CpC6rmSy8ohZjwJBALiHGihuWHU8Xw9Qz5l6nWzN2w/vQa9Brm3XOkVLfoirPdOx0oo6OU7wwdgBMz+86+QLMAwx+ZooIHpGiZ0mWR8="; /*============================================================================*/ /*============================================================================*/ /*============================================================================*/ //partner和seller獲取失敗,提示 if ([partner length] == 0 || [seller length] == 0 || [privateKey length] == 0) { UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:@"缺乏partner或者seller或者私鑰。" delegate:self cancelButtonTitle:@"肯定" otherButtonTitles:nil]; [alert show]; return; } /* *生成訂單信息及簽名 */ //將商品信息賦予AlixPayOrder的成員變量 Order *order = [[Order alloc] init]; order.partner = partner; order.seller = seller; order.tradeNO = [self generateTradeNO]; //訂單ID(由商家自行制定) order.productName = productName; //商品標題 order.productDescription = productDescription; //商品描述 order.amount = [NSString stringWithFormat:@"%.2f",[amount doubleValue]]; //商品價格 order.notifyURL = notifyURL; //回調URL order.service = @"mobile.securitypay.pay"; order.paymentType = @"1"; order.inputCharset = @"utf-8"; order.itBPay = @"30m"; order.showUrl = @"m.alipay.com"; //應用註冊scheme,在AlixPayDemo-Info.plist定義URL types NSString *appScheme = @"alisdkPaydemo"; //將商品信息拼接成字符串 NSString *orderSpec = [order description]; NSLog(@"orderSpec = %@",orderSpec); //獲取私鑰並將商戶信息簽名,外部商戶能夠根據狀況存放私鑰和簽名,只須要遵循RSA簽名規範,並將簽名字符串base64編碼和UrlEncode id<DataSigner> signer = CreateRSADataSigner(privateKey); NSString *signedString = [signer signString:orderSpec]; //將簽名成功字符串格式化爲訂單字符串,請嚴格按照該格式 NSString *orderString = nil; if (signedString != nil) { orderString = [NSString stringWithFormat:@"%@&sign=\"%@\"&sign_type=\"%@\"", orderSpec, signedString, @"RSA"]; NSLog(@"%@",orderString); [[AlipaySDK defaultService] payOrder:orderString fromScheme:appScheme callback:^(NSDictionary *resultDic) { NSLog(@"reslut = %@",resultDic); // 這裏填寫須要回調的頁面,好比,跳轉到支付成功頁面。 // BSPayResultViewController *payState = [[BSPayResultViewController alloc] init]; // payState.title =[resultDic[@"resultStatus"] isEqualToString:@"9000"]?@"支付成功": @"支付失敗"; // payState.orderId = self.paymentModel.order_id; // payState.amount = self.paymentModel.order_amount; // payState.payType = @"支付寶支付"; // payState.failure = resultDic[@"memo"]; // [self.navigationController pushViewController:payState animated:NO]; }]; } }
固然,這個是在客戶端進行簽名的,實際上,客戶端不須要知道RAS的私密,簽名經過請求服務端獲取。官方是推薦服務端簽名的,爲了安全考慮。