Android studio開發(fā)小型對話機器人app(實例代碼)
前言
偶然在網(wǎng)上看到一個免費機器人接口,所以生此想法,接口地址:http://api.qingyunke.com/,Android開發(fā)比爬蟲要繁瑣得多,所以本文我將細(xì)說接口的調(diào)用方法,讀者可根據(jù)思路去網(wǎng)上找一些免費接口拿來玩,其他代碼一帶而過,詳細(xì)源碼見文末。
成品展示:

開發(fā)步驟
1)新建項目empty,必要可github托管。
2)先寫頁面在res->layout目錄下新建兩個layout xml file

activity_main作為主頁面,msg_item為附在其上的消息頁面。(具體代碼見文末)
這里再acitvity_main采用的不是ListView而是RecyclerView。
(3)寫接口調(diào)用首先封裝一個RobotManager類,用于接收用戶輸入之后對Url進行封裝。
public class RobotManager {
private static String url = "http://api.qingyunke.com/api.php?key=free&appid=0&msg=!!";
public static String getUrl(String question){
String real_url = url.replace("!!",question);//將url中的!!替換為用戶輸入的內(nèi)容
return real_url;
}
}
其次在java->第一個包下創(chuàng)建一個接口GetConnection,用來接收接口返回的返回。
public interface GetConnection {
void onFinish(String response); //返回正常
void onError(Exception e); //返回錯誤
}
然后開始寫MyConnection類,調(diào)用接口,給參,并通過GetConnection接口將網(wǎng)上接口返回的數(shù)據(jù)傳遞出去。
public class MyConnection {
public static void getResponse(final String url,final GetConnection getConnection){
new Thread(new Runnable() {
@Override
public void run() {
HttpURLConnection connection;
try{
URL Url = new URL(url);
connection = (HttpURLConnection) Url.openConnection();
connection.setRequestMethod("GET");//Get方法
connection.setConnectTimeout(2000);//延時時間
connection.setReadTimeout(3000);
connection.setDoInput(true);
InputStream in = connection.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
StringBuilder response = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
response.append(line);
}
Log.e("result",response.toString());
if (getConnection != null) {
getConnection.onFinish(response.toString());
}else {
Log.e("WeiNull","WeiNull");//測試時用的,無須關(guān)心。
}
}catch(IOException io) {
if (getConnection != null) {
getConnection.onError(io);
}
}
}
}).start();
}
}
(4)寫頁面邏輯頁面邏輯主要是將從接口得到的內(nèi)容渲染到頁面上。首先進行一系列初始化,綁定布局,然后監(jiān)聽按鈕點擊接收用戶輸入,調(diào)用MyConnection類傳入url,輸入內(nèi)容接收接口返回內(nèi)容,同時用handler動態(tài)更新頁面。
public class MainActivity extends AppCompatActivity {
private List<Msg> msgList = new ArrayList<Msg>();
private EditText inputText;
private Button send;
private RecyclerView msgRecyclerView;
private MsgAdapter adapter;
private Handler handler = new Handler(){
@Override
public void handleMessage(@NonNull Message msg) {
switch (msg.what){
case 1:{
Bundle data = msg.getData();
String result = data.getString("result");
Msg robotMsg = new Msg(result,Msg.TYPE_RECEIVED);
msgList.add(robotMsg);
adapter.notifyItemInserted(msgList.size() - 1); // 當(dāng)有新消息時,刷新ListView中的顯示
msgRecyclerView.scrollToPosition(msgList.size() - 1); // 將ListView定位到最后一行
}break;
case 2:{}break;
default:break;
}
}
};
private void getInter(String content){
MyConnection.getResponse(RobotManager.getUrl(content), new GetConnection() {
@Override
public void onFinish(String response) {
ContentBean contentBean = new ContentBean();
Log.e("getResult",response);
Message msg = new Message();
Bundle data = new Bundle();
Gson gson = new Gson();
contentBean = gson.fromJson(response,ContentBean.class); //用Gson將返回內(nèi)容序列化為ContentBean對象。
if(contentBean.getResult()==0){
data.putString("result",contentBean.getContent());
}else{
data.putString("result","我聽不懂你在說什么呀!");
}
msg.setData(data);
msg.what = 1;
handler.sendMessage(msg);
}
@Override
public void onError(Exception e) {
e.printStackTrace();
}
});
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initMsgs(); // 初始化消息數(shù)據(jù)
inputText = findViewById(R.id.input_text);
send = findViewById(R.id.send);
msgRecyclerView = findViewById(R.id.msg_recycler_view);
LinearLayoutManager layoutManager = new LinearLayoutManager(this);
msgRecyclerView.setLayoutManager(layoutManager);
adapter = new MsgAdapter(msgList);
msgRecyclerView.setAdapter(adapter);
send.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String content = inputText.getText().toString();
if (!"".equals(content)) {
Msg msg = new Msg(content, Msg.TYPE_SENT);
msgList.add(msg);
getInter(content);
Log.e("url",RobotManager.getUrl(content));
adapter.notifyItemInserted(msgList.size() - 1); // 當(dāng)有新消息時,刷新ListView中的顯示
msgRecyclerView.scrollToPosition(msgList.size() - 1); // 將ListView定位到最后一行
inputText.setText(""); // 清空輸入框中的內(nèi)容
}
}
});
}
private void initMsgs() {
Msg msg1 = new Msg("我是菲菲,快來和我聊天吧* ( ´͈ ᵕ `͈ )◞♡", Msg.TYPE_RECEIVED);
msgList.add(msg1);
}
}
由于接口返回的數(shù)據(jù)是Json格式,所以我們需要對Json進行解析,本文采用google的Gson庫對其進行解析,將接口內(nèi)容序列化為一個類ContentBean。解析過程見上面代碼。
public class ContentBean {
/**
* result : 0
* content : [04月08日] 邯鄲天氣:小雨,白天 17℃,夜晚 9℃,微風(fēng),<3級轉(zhuǎn)3-4級
*/
private int result;
private String content;
public int getResult() {
return result;
}
public void setResult(int result) {
this.result = result;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
}
(5)其余代碼Msg和MsgAdapter用于解析消息是發(fā)送方還是接收方
并展示在消息列表上Msg
public class Msg {
public static final int TYPE_RECEIVED = 0;
public static final int TYPE_SENT = 1;
private String content;
private int type;
public Msg(String content, int type) {
this.content = content;
this.type = type;
}
public String getContent() {
return content;
}
public int getType() {
return type;
}
}
MsgAdapter
public class MsgAdapter extends RecyclerView.Adapter<MsgAdapter.ViewHolder> {
private List<Msg> mMsgList;
static class ViewHolder extends RecyclerView.ViewHolder {
LinearLayout leftLayout;
LinearLayout rightLayout;
TextView leftMsg;
TextView rightMsg;
public ViewHolder(View view) {
super(view);
leftLayout = (LinearLayout) view.findViewById(R.id.left_layout);
rightLayout = (LinearLayout) view.findViewById(R.id.right_layout);
leftMsg = (TextView) view.findViewById(R.id.left_msg);
rightMsg = (TextView) view.findViewById(R.id.right_msg);
}
}
public MsgAdapter(List<Msg> msgList) {
mMsgList = msgList;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.msg_item, parent, false);
return new ViewHolder(view);
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
Msg msg = mMsgList.get(position);
if (msg.getType() == Msg.TYPE_RECEIVED) {
// 如果是收到的消息,則顯示左邊的消息布局,將右邊的消息布局隱藏
holder.leftLayout.setVisibility(View.VISIBLE);
holder.rightLayout.setVisibility(View.GONE);
holder.leftMsg.setText(msg.getContent());
} else if(msg.getType() == Msg.TYPE_SENT) {
// 如果是發(fā)出的消息,則顯示右邊的消息布局,將左邊的消息布局隱藏
holder.rightLayout.setVisibility(View.VISIBLE);
holder.leftLayout.setVisibility(View.GONE);
holder.rightMsg.setText(msg.getContent());
}
}
@Override
public int getItemCount() {
return mMsgList.size();
}
}
尾聲
以上代碼可復(fù)制使用,但其中有一些圖片,類庫什么的需要自行導(dǎo)入,完整代碼見geithub:點擊此處直達(dá)
該項目是我早期項目,只是為了記錄當(dāng)初的開發(fā)流程,所以并不存在什么設(shè)計模式,代碼結(jié)構(gòu)也比較亂,注釋也比較少,如果有任何問題,歡迎添加我的QQ詢問!
總結(jié)
到此這篇關(guān)于Android studio開發(fā)一個小型對話機器人app的文章就介紹到這了,更多相關(guān)android studio 對話機器人內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android源碼探究之BaseDexClassLoader的使用
今天解決一個插件化問題的時候,竟然發(fā)現(xiàn)SO沒有正常加載,很怪異,最終排查下來發(fā)現(xiàn)竟然是參數(shù)傳入錯誤導(dǎo)致的。這就扯到了本文的標(biāo)題上了,BaseDexClassLoader中的4個參數(shù)該如何傳入,傳入的又是什么呢2022-08-08
Android事件分發(fā)機制(下) View的事件處理
這篇文章主要介紹了Android事件分發(fā)機制下篇, View的事件處理的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-01-01
Android 數(shù)據(jù)存儲之 FileInputStream 工具類及FileInputStream類的使用
這篇文章主要介紹了Android 數(shù)據(jù)存儲之 FileInputStream 工具類及FileInputStream類的使用的相關(guān)資料,需要的朋友可以參考下2015-11-11
Android Studio3.0升級后使用注意事項及解決方法
這篇文章主要介紹了Android Studio3.0升級后使用注意事項及解決方法,需要的朋友參考下吧2017-12-12
Android提高之BLE開發(fā)Android手機搜索iBeacon基站
這篇文章主要介紹了BLE開發(fā)Android手機搜索iBeacon基站,需要的朋友可以參考下2014-08-08
Android點擊Button實現(xiàn)功能的幾種方法總結(jié)
當(dāng)Button有多個或者Button的使用次數(shù)很多時,我們需要采用綁定監(jiān)聽器的做法,其實,綁定監(jiān)聽器也有幾種方法,不過,我在這里就不一一列舉了,畢竟那些方法在實際的應(yīng)用中也不常見2013-10-10

