當點選登陸之後,怎麼把server端返回的資料,寫到指定的控制元件上尼?,在android怎麼實現尼?以下我們通過詳細的程式碼進行分析和實現,希望能對你,在學習android知識上有所幫助。
以下通過程式碼說明:
package www.csdn.net.lesson03;
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLEncoder;
import android.app.Activity;
import android.os.Bundle;
import android.text.TextUtils;
import android.view.View;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
public class LoginActivity extends Activity {
// 宣告控制元件物件
private EditText et_name, et_pass;
// 宣告顯示返回資料庫的控制元件物件
private TextView tv_result;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 設定顯示的檢視
setContentView(R.layout.activity_login);
// 通過 findViewById(id)方法獲取username的控制元件物件
et_name = (EditText) findViewById(R.id.et_name);
// 通過 findViewById(id)方法獲取使用者password的控制元件物件
et_pass = (EditText) findViewById(R.id.et_pass);
// 通過 findViewById(id)方法獲取顯示返回資料的控制元件物件
tv_result = (TextView) findViewById(R.id.tv_result);
}
/**
* 通過android:onClick="login"指定的方法 , 要求這種方法中接受你點選控制元件物件的引數v
*
* @param v
*/
public void login(View v) {
// 獲取點選控制元件的id
int id = v.getId();
// 依據id進行推斷進行怎麼樣的處理
switch (id) {
// 登陸事件的處理
case R.id.btn_login:
// 獲取username
final String userName = et_name.getText().toString();
// 獲取使用者password
final String userPass = et_pass.getText().toString();
if (TextUtils.isEmpty(userName) || TextUtils.isEmpty(userPass)) {
Toast.makeText(this, "username或者password不能為空", Toast.LENGTH_LONG).show();
} else {
System.out
.println("----------------------傳送請求到server----------------------");
// 訪問網路 (須要一個網路的許可權) <uses-permission
// android:name="android.permission.INTERNET"/>
// 訪問網路(耗時的操作) 避免堵塞主執行緒(UI) 須要開啟新的子執行緒來處理
new Thread() {
public void run() {
// 呼叫loginByGet方法
loginByGet(userName, userPass);
};
}.start();
}
break;
default:
break;
}
}
/**
* 通過GET方式傳送的請求
*
* @param userName
* @param userPass
*/
public void loginByGet(String userName, String userPass) {
try {
// 設定請求的地址 通過URLEncoder.encode(String s, String enc)
// 使用指定的編碼機制將字串轉換為 application/x-www-form-urlencoded 格式
String spec = "http://172.16.237.200:8080/video/login.do?username="
+ URLEncoder.encode(userName, "UTF-8") + "&userpass="
+ URLEncoder.encode(userPass, "UTF-8");
// 依據地址建立URL物件(網路訪問的url)
URL url = new URL(spec);
// url.openConnection()開啟網路連結
HttpURLConnection urlConnection = (HttpURLConnection) url
.openConnection();
urlConnection.setRequestMethod("GET");// 設定請求的方式
urlConnection.setReadTimeout(5000);// 設定超時的時間
urlConnection.setConnectTimeout(5000);// 設定連結超時的時間
// 設定請求的頭
urlConnection
.setRequestProperty("User-Agent",
"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:27.0) Gecko/20100101 Firefox/27.0");
// 獲取響應的狀態碼 404 200 505 302
if (urlConnection.getResponseCode() == 200) {
// 獲取響應的輸入流物件
InputStream is = urlConnection.getInputStream();
// 建立位元組輸出流物件
ByteArrayOutputStream os = new ByteArrayOutputStream();
// 定義讀取的長度
int len = 0;
// 定義緩衝區
byte buffer[] = new byte[1024];
// 依照緩衝區的大小,迴圈讀取
while ((len = is.read(buffer)) != -1) {
// 依據讀取的長度寫入到os物件中
os.write(buffer, 0, len);
}
// 釋放資源
is.close();
os.close();
// 返回字串
String result = new String(os.toByteArray());
System.out.println("***************" + result+ "******************");
// 在這裡把返回的資料寫在控制元件上 會出現什麼情況尼
tv_result.setText(result);
} else {
System.out.println("------------------連結失敗-----------------");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
通過加入紅色的程式碼進行實現,會發現出現例如以下bug:
05-24 06:38:37.987: W/System.err(1170): android.view.ViewRootImpl$CalledFromWrongThreadException:
Only the original thread that created a view hierarchy can touch its views.
05-24 06:38:37.997: W/System.err(1170):
at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6094)
05-24 06:38:37.997: W/System.err(1170):
at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:824)
05-24 06:38:38.007: W/System.err(1170):
at android.view.View.requestLayout(View.java:16431)
05-24 06:38:38.007: W/System.err(1170):
at android.view.View.requestLayout(View.java:16431)
05-24 06:38:38.007: W/System.err(1170):
at android.view.View.requestLayout(View.java:16431)
05-24 06:38:38.007: W/System.err(1170):
at android.view.View.requestLayout(View.java:16431)
05-24 06:38:38.007: W/System.err(1170):
at android.widget.RelativeLayout.requestLayout(RelativeLayout.java:352)
05-24 06:38:38.007: W/System.err(1170):
at android.view.View.requestLayout(View.java:16431)
05-24 06:38:38.007: W/System.err(1170):
at android.widget.TextView.checkForRelayout(TextView.java:6600)
05-24 06:38:38.007: W/System.err(1170):
at android.widget.TextView.setText(TextView.java:3813)
05-24 06:38:38.007: W/System.err(1170):
at android.widget.TextView.setText(TextView.java:3671)
05-24 06:38:38.017: W/System.err(1170):
at android.widget.TextView.setText(TextView.java:3646)
05-24 06:38:38.017: W/System.err(1170):
at www.csdn.net.lesson03.LoginActivity.loginByGet(LoginActivity.java:134)
05-24 06:38:38.027: W/System.err(1170):
at www.csdn.net.lesson03.LoginActivity$1.run(LoginActivity.java:67)
錯誤的原因是:
僅僅有原來的執行緒建立的檢視層次能夠觸控它的控制元件.
所以僅僅有在主執行緒中,才可以設定返回資料控制元件的內容。
那麼怎麼實現效果,我們能夠使用Activity中給我們提供的runOnUiThread(action);的方法實現,把我們要進行設定返回資料控制元件的內容放在此方法中就可以。詳細的部分程式碼例如以下:
// 返回字串
final String result = new String(os.toByteArray());
System.out.println("***************" + result
+ "******************");
LoginActivity.this.runOnUiThread(new Runnable() { //才用此方法進行改動主執行緒中的UI控制元件內容
@Override
public void run() {
// 在這裡把返回的資料寫在控制元件上 會出現什麼情況尼
tv_result.setText(result);
}
});
測試效果例如以下:
以上程式碼僅供參考學習,如有問題請留言