Java Demo
本文介紹如何使用阿里云智能語(yǔ)音服務(wù)提供的Java SDK,包括SDK的安裝方法及SDK代碼示例。
前提條件
使用SDK前,請(qǐng)先閱讀接口說(shuō)明,詳情請(qǐng)參見接口說(shuō)明。
已開通智能語(yǔ)音交互并獲取AccessKey ID和AccessKey Secret,詳情請(qǐng)參見從這里開始。
SDK說(shuō)明
錄音文件識(shí)別的Java示例使用了阿里云Java SDK的CommonRequest提交錄音文件識(shí)別請(qǐng)求和識(shí)別結(jié)果查詢,采用的是RPC風(fēng)格的POP API調(diào)用。阿里云Java SDK CommonRequest的使用方法請(qǐng)參見使用CommonRequest進(jìn)行調(diào)用。
阿里云Java SDK不支持Android開發(fā)。
Java依賴
您只需依賴阿里云Java SDK的核心庫(kù)與阿里巴巴開源庫(kù)fastjson。阿里云Java SDK的核心庫(kù)版本支持3.5.0及以上(如果版本在4.0.0及以上,需要根據(jù)錯(cuò)誤提示補(bǔ)充對(duì)應(yīng)的第三方依賴)。
<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun-java-sdk-core</artifactId>
<version>3.7.1</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.83</version>
</dependency>
示例說(shuō)明
下載nls-sample-16k.wav。示例中使用的WAV錄音文件為PCM編碼格式16000 Hz采樣率,模型設(shè)置為通用模型。
調(diào)用接口前,需配置環(huán)境變量,通過環(huán)境變量讀取訪問憑證。智能語(yǔ)音交互的AccessKey ID、AccessKey Secret和AppKey的環(huán)境變量名:ALIYUN_AK_ID、ALIYUN_AK_SECRET、NLS_APP_KEY。
鑒權(quán)
通過傳入阿里云賬號(hào)的AccessKey ID和AccessKey Secret(獲取方式請(qǐng)參見開通服務(wù)),調(diào)用阿里云Java SDK得到client,示例如下:
final String accessKeyId = System.getenv().get("ALIYUN_AK_ID");
final String accessKeySecret = System.getenv().get("ALIYUN_AK_SECRET");
/**
* 地域ID
*/
final String regionId = "cn-shanghai";
final String endpointName = "cn-shanghai";
final String product = "SpeechFileTranscriberLite";
final String domain = "speechfiletranscriberlite.cn-shanghai.aliyuncs.com";
IAcsClient client;
// 設(shè)置endpoint
DefaultProfile.addEndpoint(endpointName, regionId, product, domain);
// 創(chuàng)建DefaultAcsClient實(shí)例并初始化
DefaultProfile profile = DefaultProfile.getProfile(regionId, accessKeyId, accessKeySecret);
client = new DefaultAcsClient(profile);
錄音文件識(shí)別請(qǐng)求調(diào)用接口
Java 示例采用輪詢的方式,提交錄音文件識(shí)別請(qǐng)求,獲取任務(wù)ID,供后續(xù)輪詢使用。
參數(shù)設(shè)置請(qǐng)參見接口說(shuō)明,只需設(shè)置JSON字符串中的參數(shù),其他方法的參數(shù)值保持不變。
/**
* 創(chuàng)建CommonRequest 設(shè)置請(qǐng)求參數(shù)
*/
CommonRequest postRequest = new CommonRequest();
postRequest.setDomain("speechfiletranscriberlite.cn-shanghai.aliyuncs.com"); // 設(shè)置域名,固定值。
postRequest.setVersion("2021-12-21"); // 設(shè)置API的版本號(hào),固定值。
postRequest.setAction("SubmitTask"); // 設(shè)置action,固定值。
postRequest.setProduct("SpeechFileTranscriberLite"); // 設(shè)置產(chǎn)品名稱,固定值。
// 設(shè)置錄音文件識(shí)別請(qǐng)求參數(shù),以JSON字符串的格式設(shè)置到請(qǐng)求Body中。
JSONObject taskObject = new JSONObject();
taskObject.put("appkey", "您的appkey"); // 項(xiàng)目的Appkey,獲取Appkey請(qǐng)前往控制臺(tái):https://nls-portal.console.aliyun.com/applist
taskObject.put("file_link", "您的錄音文件訪問鏈接"); // 設(shè)置錄音文件的鏈接
String task = taskObject.toJSONString();
postRequest.putBodyParameter("Task", task); // 設(shè)置以上JSON字符串為Body參數(shù)。
postRequest.setMethod(MethodType.POST); // 設(shè)置為POST方式請(qǐng)求。
// postRequest.setHttpContentType(FormatType.JSON); //當(dāng)aliyun-java-sdk-core 版本為4.6.0及以上時(shí),請(qǐng)取消該行注釋
/**
* 提交錄音文件識(shí)別請(qǐng)求
*/
String taskId = ""; // 獲取錄音文件識(shí)別請(qǐng)求任務(wù)的ID,以供識(shí)別結(jié)果查詢使用。
CommonResponse postResponse = client.getCommonResponse(postRequest);
if (postResponse.getHttpStatus() == 200) {
JSONObject result = JSONObject.parseObject(postResponse.getData());
String statusText = result.getString("StatusText");
if ("SUCCESS".equals(statusText)) {
System.out.println("錄音文件識(shí)別請(qǐng)求成功響應(yīng): " + result.toJSONString());
taskId = result.getString("TaskId");
}
else {
System.out.println("錄音文件識(shí)別請(qǐng)求失敗: " + result.toJSONString());
return;
}
}
else {
System.err.println("錄音文件識(shí)別請(qǐng)求失敗,Http錯(cuò)誤碼:" + postResponse.getHttpStatus());
System.err.println("錄音文件識(shí)別請(qǐng)求失敗響應(yīng):" + JSONObject.toJSONString(postResponse));
return;
}
錄音文件識(shí)別結(jié)果查詢
使用獲取的任務(wù)ID,查詢錄音文件識(shí)別的結(jié)果。
/**
* 創(chuàng)建CommonRequest,設(shè)置任務(wù)ID。
*/
CommonRequest getRequest = new CommonRequest();
getRequest.setDomain("speechfiletranscriberlite.cn-shanghai.aliyuncs.com"); // 設(shè)置域名,固定值。
getRequest.setVersion("2021-12-21"); // 設(shè)置API版本,固定值。
getRequest.setAction("GetTaskResult"); // 設(shè)置action,固定值。
getRequest.setProduct("SpeechFileTranscriberLite"); // 設(shè)置產(chǎn)品名稱,固定值。
getRequest.putQueryParameter("TaskId", taskId); // 設(shè)置任務(wù)ID為查詢參數(shù)。
getRequest.setMethod(MethodType.GET); // 設(shè)置為GET方式的請(qǐng)求。
/**
* 提交錄音文件識(shí)別結(jié)果查詢請(qǐng)求
* 以輪詢的方式進(jìn)行識(shí)別結(jié)果的查詢,直到服務(wù)端返回的狀態(tài)描述為“SUCCESS”、“SUCCESS_WITH_NO_VALID_FRAGMENT”,或者為錯(cuò)誤描述,則結(jié)束輪詢。
*/
String statusText = "";
while (true) {
CommonResponse getResponse = client.getCommonResponse(getRequest);
if (getResponse.getHttpStatus() != 200) {
System.err.println("識(shí)別結(jié)果查詢請(qǐng)求失敗,Http錯(cuò)誤碼: " + getResponse.getHttpStatus());
System.err.println("識(shí)別結(jié)果查詢請(qǐng)求失敗: " + getResponse.getData());
break;
}
JSONObject result = JSONObject.parseObject(getResponse.getData());
System.out.println("識(shí)別查詢結(jié)果:" + result.toJSONString());
statusText = result.getString("StatusText");
if ("RUNNING".equals(statusText) || "QUEUEING".equals(statusText)) {
// 繼續(xù)輪詢
Thread.sleep(3000);
}
else {
break;
}
}
if ("SUCCESS".equals(statusText) || "SUCCESS_WITH_NO_VALID_FRAGMENT".equals(statusText)) {
System.out.println("錄音文件識(shí)別成功!");
}
else {
System.err.println("錄音文件識(shí)別失敗!");
}
示例代碼
調(diào)用接口前,需配置環(huán)境變量,通過環(huán)境變量讀取訪問憑證。智能語(yǔ)音交互的AccessKey ID、AccessKey Secret和AppKey的環(huán)境變量名:ALIYUN_AK_ID、ALIYUN_AK_SECRET、NLS_APP_KEY。
import com.alibaba.fastjson.JSONObject;
import com.aliyuncs.CommonRequest;
import com.aliyuncs.CommonResponse;
import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.http.MethodType;
import com.aliyuncs.profile.DefaultProfile;
public class FileTransJavaDemo {
// 地域ID,常量,固定值。
public static final String REGIONID = "cn-shanghai";
public static final String ENDPOINTNAME = "cn-shanghai";
public static final String PRODUCT = "SpeechFileTranscriberLite";
public static final String DOMAIN = "speechfiletranscriberlite.cn-shanghai.aliyuncs.com";
public static final String API_VERSION = "2021-12-21";
public static final String POST_REQUEST_ACTION = "SubmitTask";
public static final String GET_REQUEST_ACTION = "GetTaskResult";
// 請(qǐng)求參數(shù)
public static final String KEY_APP_KEY = "appkey";
public static final String KEY_FILE_LINK = "file_link";
public static final String KEY_VERSION = "version";
public static final String KEY_ENABLE_WORDS = "enable_words";
// 響應(yīng)參數(shù)
public static final String KEY_TASK = "Task";
public static final String KEY_TASK_ID = "TaskId";
public static final String KEY_STATUS_TEXT = "StatusText";
public static final String KEY_RESULT = "Result";
// 狀態(tài)值
public static final String STATUS_SUCCESS = "SUCCESS";
private static final String STATUS_RUNNING = "RUNNING";
private static final String STATUS_QUEUEING = "QUEUEING";
// 阿里云鑒權(quán)client
IAcsClient client;
public FileTransJavaDemo(String accessKeyId, String accessKeySecret) {
// 設(shè)置endpoint
try {
DefaultProfile.addEndpoint(ENDPOINTNAME, REGIONID, PRODUCT, DOMAIN);
} catch (ClientException e) {
e.printStackTrace();
}
// 創(chuàng)建DefaultAcsClient實(shí)例并初始化
DefaultProfile profile = DefaultProfile.getProfile(REGIONID, accessKeyId, accessKeySecret);
this.client = new DefaultAcsClient(profile);
}
public String submitFileTransRequest(String appKey, String fileLink) {
/**
* 1. 創(chuàng)建CommonRequest,設(shè)置請(qǐng)求參數(shù)。
*/
CommonRequest postRequest = new CommonRequest();
// 設(shè)置域名
postRequest.setDomain(DOMAIN);
// 設(shè)置API的版本號(hào),格式為YYYY-MM-DD。
postRequest.setVersion(API_VERSION);
// 設(shè)置action
postRequest.setAction(POST_REQUEST_ACTION);
// 設(shè)置產(chǎn)品名稱
postRequest.setProduct(PRODUCT);
/**
* 2. 設(shè)置錄音文件識(shí)別請(qǐng)求參數(shù),以JSON字符串的格式設(shè)置到請(qǐng)求Body中。
*/
JSONObject taskObject = new JSONObject();
// 設(shè)置appkey
taskObject.put(KEY_APP_KEY, appKey);
// 設(shè)置音頻文件訪問鏈接
taskObject.put(KEY_FILE_LINK, fileLink);
// 設(shè)置是否輸出詞信息,默認(rèn)為false。
taskObject.put(KEY_ENABLE_WORDS, true);
String task = taskObject.toJSONString();
System.out.println(task);
// 設(shè)置以上JSON字符串為Body參數(shù)。
postRequest.putBodyParameter(KEY_TASK, task);
// 設(shè)置為POST方式的請(qǐng)求。
postRequest.setMethod(MethodType.POST);
// postRequest.setHttpContentType(FormatType.JSON); //當(dāng)aliyun-java-sdk-core 版本為4.6.0及以上時(shí),請(qǐng)取消該行注釋
/**
* 3. 提交錄音文件識(shí)別請(qǐng)求,獲取錄音文件識(shí)別請(qǐng)求任務(wù)的ID,以供識(shí)別結(jié)果查詢使用。
*/
String taskId = null;
try {
CommonResponse postResponse = client.getCommonResponse(postRequest);
System.err.println("提交錄音文件識(shí)別請(qǐng)求的響應(yīng):" + postResponse.getData());
if (postResponse.getHttpStatus() == 200) {
JSONObject result = JSONObject.parseObject(postResponse.getData());
String statusText = result.getString(KEY_STATUS_TEXT);
if (STATUS_SUCCESS.equals(statusText)) {
taskId = result.getString(KEY_TASK_ID);
}
}
} catch (ClientException e) {
e.printStackTrace();
}
return taskId;
}
public String getFileTransResult(String taskId) {
/**
* 1. 創(chuàng)建CommonRequest,設(shè)置任務(wù)ID。
*/
CommonRequest getRequest = new CommonRequest();
// 設(shè)置域名
getRequest.setDomain(DOMAIN);
// 設(shè)置API版本
getRequest.setVersion(API_VERSION);
// 設(shè)置action
getRequest.setAction(GET_REQUEST_ACTION);
// 設(shè)置產(chǎn)品名稱
getRequest.setProduct(PRODUCT);
// 設(shè)置任務(wù)ID為查詢參數(shù)
getRequest.putQueryParameter(KEY_TASK_ID, taskId);
// 設(shè)置為GET方式的請(qǐng)求
getRequest.setMethod(MethodType.GET);
/**
* 2. 提交錄音文件識(shí)別結(jié)果查詢請(qǐng)求
* 以輪詢的方式進(jìn)行識(shí)別結(jié)果的查詢,直到服務(wù)端返回的狀態(tài)描述為“SUCCESS”或錯(cuò)誤描述,則結(jié)束輪詢。
*/
String result = null;
while (true) {
try {
CommonResponse getResponse = client.getCommonResponse(getRequest);
System.err.println("識(shí)別查詢結(jié)果:" + getResponse.getData());
if (getResponse.getHttpStatus() != 200) {
break;
}
JSONObject rootObj = JSONObject.parseObject(getResponse.getData());
String statusText = rootObj.getString(KEY_STATUS_TEXT);
if (STATUS_RUNNING.equals(statusText) || STATUS_QUEUEING.equals(statusText)) {
// 繼續(xù)輪詢,注意設(shè)置輪詢時(shí)間間隔。
Thread.sleep(10000);
}
else {
// 狀態(tài)信息為成功,返回識(shí)別結(jié)果;狀態(tài)信息為異常,返回空。
if (STATUS_SUCCESS.equals(statusText)) {
result = rootObj.getString(KEY_RESULT);
// 狀態(tài)信息為成功,但沒有識(shí)別結(jié)果,則可能是由于文件里全是靜音、噪音等導(dǎo)致識(shí)別為空。
if(result == null) {
result = "";
}
}
break;
}
} catch (Exception e) {
e.printStackTrace();
}
}
return result;
}
public static void main(String args[]) throws Exception {
final String accessKeyId = System.getenv().get("ALIYUN_AK_ID");
final String accessKeySecret = System.getenv().get("ALIYUN_AK_SECRET");
final String appKey = System.getenv().get("NLS_APP_KEY");
String fileLink = "https://gw.alipayobjects.com/os/bmw-prod/0574ee2e-f494-45a5-820f-63aee583045a.wav";
FileTransJavaDemo demo = new FileTransJavaDemo(accessKeyId, accessKeySecret);
// 第一步:提交錄音文件識(shí)別請(qǐng)求,獲取任務(wù)ID用于后續(xù)的識(shí)別結(jié)果輪詢。
String taskId = demo.submitFileTransRequest(appKey, fileLink);
if (taskId != null) {
System.out.println("錄音文件識(shí)別請(qǐng)求成功,task_id: " + taskId);
}
else {
System.out.println("錄音文件識(shí)別請(qǐng)求失敗!");
return;
}
// 第二步:根據(jù)任務(wù)ID輪詢識(shí)別結(jié)果。
String result = demo.getFileTransResult(taskId);
if (result != null) {
System.out.println("錄音文件識(shí)別結(jié)果查詢成功:" + result);
}
else {
System.out.println("錄音文件識(shí)別結(jié)果查詢失敗!");
}
}
}
如果使用回調(diào)方式,請(qǐng)?jiān)O(shè)置如下enable_callback
、callback_url
參數(shù):
taskObject.put("enable_callback", true);
taskObject.put("callback_url", "回調(diào)地址");
回調(diào)服務(wù)示例:該服務(wù)用于回調(diào)方式獲取識(shí)別結(jié)果,假設(shè)設(shè)置回調(diào)地址為http://ip:port/filetrans/callback/result。
package com.example.filetrans;
import com.alibaba.fastjson.JSONObject;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
@RequestMapping("/filetrans/callback")
@RestController
public class FiletransCallBack {
// 以4開頭的狀態(tài)碼是客戶端錯(cuò)誤。
private static final Pattern PATTERN_CLIENT_ERR = Pattern.compile("4105[0-9]*");
// 以5開頭的狀態(tài)碼是服務(wù)端錯(cuò)誤。
private static final Pattern PATTERN_SERVER_ERR = Pattern.compile("5105[0-9]*");
// 必須是post方式
@RequestMapping(value = "result", method = RequestMethod.POST)
public void GetResult(@RequestBody String body) {
System.out.println("body: " + body);
try {
// 獲取JSON格式的文件識(shí)別結(jié)果。
String result = body;
JSONObject jsonResult = JSONObject.parseObject(result);
// 解析并輸出相關(guān)結(jié)果內(nèi)容。
System.out.println("獲取文件中轉(zhuǎn)寫回調(diào)結(jié)果:" + result);
System.out.println("TaskId: " + jsonResult.getString("TaskId"));
System.out.println("StatusCode: " + jsonResult.getString("StatusCode"));
System.out.println("StatusText: " + jsonResult.getString("StatusText"));
Matcher matcherClient = PATTERN_CLIENT_ERR.matcher(jsonResult.getString("StatusCode"));
Matcher matcherServer = PATTERN_SERVER_ERR.matcher(jsonResult.getString("StatusCode"));
// 以2開頭狀態(tài)碼為正常狀態(tài)碼,回調(diào)方式正常狀態(tài)只返回“21050000”。
if("21050000".equals(jsonResult.getString("StatusCode"))) {
System.out.println("RequestTime: " + jsonResult.getString("RequestTime"));
System.out.println("SolveTime: " + jsonResult.getString("SolveTime"));
System.out.println("BizDuration: " + jsonResult.getString("BizDuration"));
System.out.println("Result.Sentences.size: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").size());
for (int i = 0; i < jsonResult.getJSONObject("Result").getJSONArray("Sentences").size(); i++) {
System.out.println("Result.Sentences[" + i + "].BeginTime: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("BeginTime"));
System.out.println("Result.Sentences[" + i + "].EndTime: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("EndTime"));
System.out.println("Result.Sentences[" + i + "].SilenceDuration: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("SilenceDuration"));
System.out.println("Result.Sentences[" + i + "].Text: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("Text"));
System.out.println("Result.Sentences[" + i + "].ChannelId: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("ChannelId"));
System.out.println("Result.Sentences[" + i + "].SpeechRate: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("SpeechRate"));
System.out.println("Result.Sentences[" + i + "].EmotionValue: " +
jsonResult.getJSONObject("Result").getJSONArray("Sentences").getJSONObject(i).getString("EmotionValue"));
}
}
else if(matcherClient.matches()) {
System.out.println("狀態(tài)碼以4開頭表示客戶端錯(cuò)誤......");
}
else if(matcherServer.matches()) {
System.out.println("狀態(tài)碼以5開頭表示服務(wù)端錯(cuò)誤......");
}
else {
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
常見問題
如何結(jié)合SDK日志分析延遲問題?
以Java SDK日志為例。
一句話識(shí)別的延遲為一句話說(shuō)完開始,到收到最終識(shí)別結(jié)果為止,消耗的時(shí)間。
在日志中搜索關(guān)鍵字
StopRecognition
以及RecognitionCompleted
,分別找到語(yǔ)音發(fā)送完畢時(shí)的日志,以及一句話識(shí)別結(jié)束的日志。記錄的時(shí)間差即為SDK端記錄的一句話延時(shí),如下日志延遲為:984-844=140(ms)。14:24:44.844 DEBUG [ main] [c.a.n.c.transport.netty4.NettyConnection] thread:1,send:{"header":{"namespace":"SpeechRecognizer","name":"StopRecognition","message_id":"bccac69b505f4e2897d12940e5b38953","appkey":"FWpPCaVYDRp6J1rO","task_id":"8c5c28d9a40c4a229a5345c09bc9c968"}} 14:24:44.984 DEBUG [ntLoopGroup-2-1] [ c.a.n.c.p.asr.SpeechRecognizerListener] on message:{"header":{"namespace":"SpeechRecognizer","name":"RecognitionCompleted","status":20000000,"message_id":"2869e93427b9429190206123b7a3d397","task_id":"8c5c28d9a40c4a229a5345c09bc9c968","status_text":"Gateway:SUCCESS:Success."},"payload":{"result":"北京的天氣。","duration":2959}}
語(yǔ)音合成關(guān)注首包延遲,即從發(fā)送合成請(qǐng)求開始,到收到第一個(gè)語(yǔ)音包為止,消耗的時(shí)間。
日志中搜索關(guān)鍵字
send
,找到這條日志和緊隨其后的一條收到語(yǔ)音包的日志。記錄的時(shí)間差即為SDK端記錄的首包延時(shí)。如下日志延時(shí)為1035-813=222(ms)。14:32:13.813 DEBUG [ main] [c.a.n.c.transport.netty4.NettyConnection] thread:1,send:{"payload":{"volume":50,"voice":"Ruoxi","sample_rate":8000,"format":"wav","text":"國(guó)家是由領(lǐng)土、人民、文化和政府四個(gè)要素組成的,國(guó)家也是政治地理學(xué)名詞。從廣義的角度,國(guó)家是指擁有共同的語(yǔ)言、文化、血統(tǒng)、領(lǐng)土、政府或者歷史等的社會(huì)群體。從狹義的角度,國(guó)家是一定范圍內(nèi)的人群所形成的共同體形式。"},"context":{"sdk":{"name":"nls-sdk-java","version":"2.1.0"},"network":{"upgrade_cost":160,"connect_cost":212}},"header":{"namespace":"SpeechSynthesizer","name":"StartSynthesis","message_id":"6bf2a84444434c0299974d8242380d6c","appkey":"FWpPCaVYDRp6J1rO","task_id":"affa5c90986e4378907fbf49eddd283a"}} 14:32:14.035 INFO [ntLoopGroup-2-1] [ c.a.n.c.protocol.tts.SpeechSynthesizer] write array:6896
實(shí)時(shí)語(yǔ)音識(shí)別SDK日志類似一句話識(shí)別,可以從日志中計(jì)算語(yǔ)音末尾處延遲(關(guān)鍵字:
StopTranscription
和TranscriptionCompleted
)。RESTful形式訪問,客戶端自帶日志中沒有體現(xiàn)延遲。需要用戶自己編寫代碼,或者查看服務(wù)端日志。
Java SDK找不到com.alibaba的JAR包,如何安裝Alibaba Cloud SDK for Java?
請(qǐng)參見V1.0 Java SDK(不推薦)安裝Alibaba Cloud SDK for Java。
Java SDK通過傳入阿里云賬號(hào)的AccessKey ID和AccessKey Secret,調(diào)用阿里云Java SDK得到client提示錯(cuò)誤org.json.JSONArray.iterator()Ljava/util/Iterator如何解決?
請(qǐng)確認(rèn)依賴包是否完整,查找并添加如下兩個(gè)依賴JAR包。
<dependency>
<groupId>org.json</groupId>
<artifactId>json</artifactId>
<version>20170516</version>
</dependency>
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.8.2</version>
</dependency>