Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android客戶端實現注冊、登錄詳解(2)

Android客戶端實現注冊、登錄詳解(2)

編輯:關於Android編程

上文中介紹了安卓客戶端與服務器交互,實現注冊功能,Android客戶端實現注冊/登錄詳解(一)

本文將繼續介紹App與服務器的交互實現登錄和自動登錄的功能,上文說到請求服務器進行注冊主要是通過POST請求攜帶參數實現,起作用的主要代碼:

 StringRequest request=new StringRequest(Method.POST, url, new Listener<String>() {

  //請求成功
  @Override
  public void onResponse(String s) {
   //執行請求成功的回調
   callback.onSuccess()
  }

 }, new ErrorListener() {

  //請求錯誤
  @Override
  public void onErrorResponse(VolleyError volleyError) {
   //執行請求失敗的回調
   callback.onFailure()
  }
 }){

  //攜帶參數(Map集合)
  @Override
  protected Map<String, String> getParams() throws AuthFailureError {
   return parames;
  }
 };

 //將請求添加到請求隊列中
 Volley.newRequestQueue(context).add(request);

其實登錄實現的原理也是一樣的,同樣是通過POST請求,而在本demo中則是把請求服務器的方法封裝在一起了,所以登錄的實現也是調用了RequestManager網絡請求處理類中的post方法

/**
 * post 請求數據
 *
 * @param app_url  公共的接口前綴 http://www.itlanbao.com/api/app/
 * @param tag_url  接口名稱,eg:users/user_register_Handler.ashx(注冊接口)
 * @param parameter 請求參數封裝對象
 * @param clazz  返回數據封裝對象,如果傳null,則直接返回String
 * @param callback 接口回調監聽
 */
public static <T> void post(final String app_url, final String tag_url, final HashMap<String, String> parameter, Class<T> clazz,
       final HttpResponeCallBack callback) {
 //發送post請求服務器
 post(app_url, tag_url, parameter, clazz, callback, Priority.NORMAL);
}

demo演示

實現代碼

1.服務器的數據格式

1.url:  http://www.itlanbao.com/api/app/users/user_login_handler.ashx

2.參數說明:
    email                  必須有         郵箱
    password            必須有         密碼
    accesstoken       必須有         md5(email+password+"雙方平台約定公鑰")     

3.請求方式:POST

4.返回值格式
    成功

 {
  "ret":0,
  "errcode":0,
  "msg":"登錄用戶接口調用成功",
  "data":{
   "userid":"16489",
   "email":"[email protected]",
   "nickname":"duss",
   "userhead":"http://img.itlanbao.com/avatar.png"
  }
 } 

    失敗

 {
  "ret":1,
  "errcode":1,
  "msg":"賬號或密碼錯誤"
 }

2.登錄界面(LoginActivity),點擊登錄按鈕

 //點擊登錄按鈕
 loginBtn.setOnClickListener(new Button.OnClickListener() {

  @Override
  public void onClick(View v) {
   // TODO Auto-generated method stub
   String account = loginAccount.getText().toString();//賬號
   String password = loginPassword.getText().toString();//密碼
   if (!TextUtils.isEmpty(account) && !TextUtils.isEmpty(password)
     && Utils.isEmail(account)) {
    RequestApiData.getInstance().getLoginData(account, password, UserBaseInfo.class, LoginActivity.this);
   } else {
    Toast.makeText(LoginActivity.this, "賬號或者密碼有誤", Toast.LENGTH_SHORT).show();
   }
  }
 });

核心代碼為:

 //傳入賬號名,密碼,解析數據的bean對象和回調(這裡傳入的是自身,所以LoginActivity也同樣實現了回調接口HttpResponeCallBack)
 RequestApiData.getInstance().getLoginData(account, password, UserBaseInfo.class, LoginActivity.this);

3.網絡接口類(RequestApiData)

 //創建接口對象
 public static RequestApiData getInstance() {
  if (instance == null) {
   instance = new RequestApiData();
  }
  return instance;
 }

 /**
  * 4.8登錄用戶接口
  * @param email 郵箱
  * @param password 密碼
  * @param clazz 數據返回的解析對象
  * @param callback 回調
  * 特別要注意參數位置不能變要根據文檔來
  * 請求方式:POST
  */
 public void getLoginData(String email ,String password,
   Class<UserBaseInfo> clazz,
   HttpResponeCallBack callback) {
   mCallBack = callback;
   //這是每一個接口的唯一標示
   String tagUrl = UrlConstance.KEY_LOGIN_INFO;//登錄接口
   HashMap<String, String> parameter = new HashMap<String, String>();
   parameter.put("email", email);
   parameter.put("password", password);

   //拼接參數信息,郵箱,密碼,公鑰,並用md5進行加密
   StringBuilder builder = new StringBuilder();
   builder.append(email);
   builder.append(password);
   builder.append(UrlConstance.PUBLIC_KEY);

   parameter.put(UrlConstance.ACCESSTOKEN_KEY,MD5Util.getMD5Str(builder.toString()));

   //請求數據接口
   RequestManager.post(UrlConstance.APP_URL,tagUrl, parameter, clazz, callback);

 }

4.網絡請求處理類(RequestManager)中請求數據,和注冊執行了同樣的方法,只是這裡的傳入的tag_url為登錄的接口

 /**
  * post 請求數據
  *
  * @param app_url  公共的接口前綴 http://www.itlanbao.com/api/app/
  * @param tag_url  接口名稱,eg:users/user_login_handler.ashx(登錄接口)
  * @param parameter 請求參數封裝對象
  * @param clazz  返回數據封裝對象,如果傳null,則直接返回String
  * @param callback 接口回調監聽
  */
 public static <T> void post(final String app_url, final String tag_url, final HashMap<String, String> parameter, Class<T> clazz,
        final HttpResponeCallBack callback) {
  //發送post請求服務器
  post(app_url, tag_url, parameter, clazz, callback, Priority.NORMAL);
 }

 /**
  * post 請求數據
  *
  * @param app_url 路徑
  * @param url  接口名稱
  * @param parameter 請求參數封裝對象
  * @param clazz  返回數據封裝對象,如果傳null,則直接返回String
  * @param callback 接口回調監聽
  * @param priority 指定接口請求線程優先級
  */
 public static <T> void post(final String app_url, final String url, final HashMap<String, String> parameter, final Class<T> clazz,
        final HttpResponeCallBack callback, Priority priority) {
  if (callback != null) {
   callback.onResponeStart(url);//回調請求開始
  }

  initRequestQueue();

  //將公共的接口前綴和接口名稱拼接
  //eg:拼接成登錄的接口 http://www.itlanbao.com/api/app/users/user_login_handler.ashx
  StringBuilder builder = new StringBuilder(app_url);
  builder.append(url);

  {// 檢查當前網絡是否可用
   final NetworkUtils networkUtils = new NetworkUtils(ItLanbaoLibApplication.getInstance());

   if (!networkUtils.isNetworkConnected() && android.os.Build.VERSION.SDK_INT > 10) {
    if (callback != null) {
     callback.onFailure(url, null, 0, "網絡出錯");//回調請求失敗
     return;
    }
   }
  }

  /**
   * 使用Volley框架真正去請求服務器
   * Method.POST:請求方式為post
   * builder.toString():請求的鏈接
   * Listener<String>:監聽
   */
  StringRequest request = new StringRequest(Method.POST, builder.toString(),
    new Listener<String>() {

     @Override
     public void onResponse(String response) {
      // TODO Auto-generated method stub
//       這個位置先公共解析處理共同異常
      try {
       if (response != null && callback != null) {
        Gson gson = new Gson();
        //回調請求成功,同時url和解析的對象
        callback.onSuccess(url, gson.fromJson(response, clazz));

       }

      } catch (Exception e) {
       // TODO: handle exception
       if (callback != null) {
        //回調請求失敗--解析異常
        callback.onFailure(url, e, 0, "解析異常");
        return;
       }
      }


     }
    }, new ErrorListener() {
   //請求出錯的監聽
   @Override
   public void onErrorResponse(VolleyError error) {
    if (callback != null) {
     if (error != null) {
      callback.onFailure(url, error.getCause(), 0,
        error.getMessage());
     } else {
      callback.onFailure(url, null, 0, "");
     }
    }
   }
  }) {
   //post請求的參數信息
   protected Map<String, String> getParams() {
    return getPostApiParmes(parameter);
   }
  };

  //添加請求到請求隊列中
  addRequest(request, url);
 }


 /*
  * post參數
  * 
  * ts:時間戳 sign: 接口簽名 parms = 按文檔參數拼接 parm[0]+ … + parm[n-1] sign =
  * md5(parms+"雙方平台約定公鑰")
  */
 private static ApiParams getPostApiParmes(final HashMap<String, String> parameter) {
  ApiParams api = new ApiParams();
  for (Entry<String, String> entry : parameter.entrySet()) {
   api.with(entry.getKey(), entry.getValue());
  }
  return api;
 }

5.同樣回到LoginActivity中執行回調,失敗則提示,成功則將登錄信息保存到SP中和Application中

@Override
public void onResponeStart(String apiName) {
 // TODO Auto-generated method stub

 if (UrlConstance.KEY_LOGIN_INFO.equals(apiName)) {
  Toast.makeText(LoginActivity.this, "正在加載數據中", Toast.LENGTH_SHORT).show();
 }
}

@Override
public void onLoading(String apiName, long count, long current) {
 // TODO Auto-generated method stub
}

@Override
public void onSuccess(String apiName, Object object) {
 // TODO Auto-generated method stub
 if (UrlConstance.KEY_LOGIN_INFO.equals(apiName)) {
  //郵箱登錄返回數據
  if (object != null && object instanceof UserBaseInfo) {
   UserBaseInfo info = (UserBaseInfo) object;
   if (info.getRet().equals(Constant.KEY_SUCCESS)) {

    //登錄成功,保存登錄信息
    ItLanBaoApplication.getInstance().setBaseUser(info);//保存到Application中

    //保存到SP中
    UserPreference.save(KeyConstance.IS_USER_ID, String.valueOf(info.getUserid()));

    UserPreference.save(KeyConstance.IS_USER_ACCOUNT, info.getEmail());
    UserPreference.save(KeyConstance.IS_USER_PASSWORD, loginPassword.getText().toString());


    Intent intent = new Intent();
    intent.setClass(LoginActivity.this, MainActivity.class);
    startActivity(intent);
    overridePendingTransition(android.R.anim.slide_in_left,
      android.R.anim.slide_out_right);
    finish();

   } else {
    Log.e("TAG", "info="+info.toString());
    if (info.getErrcode().equals(Constant.KEY_NO_REGIST)) {
     Toast.makeText(LoginActivity.this, "登錄失敗", Toast.LENGTH_SHORT).show();
    } else {
     Toast.makeText(LoginActivity.this, info.getMsg(), Toast.LENGTH_SHORT).show();
     Log.e("TAG", "info.getMsg()="+info.getMsg());
    }

   }
  }
 }

}

@Override
public void onFailure(String apiName, Throwable t, int errorNo,
      String strMsg) {
 // TODO Auto-generated method stub
 Toast.makeText(LoginActivity.this, "Failure", Toast.LENGTH_SHORT).show(); 
}

6.自動登陸的實現,其實就是我們在歡迎頁面進行一個判斷:讀取SP中的信息,如有登錄的信息,則取出,攜帶此信息請求服務器(同登錄的請求),若成功,則直接跳轉到主頁面;如果登錄不成功或者SP中沒有保存的登錄信息,則跳轉到登錄界面,代碼如下(WelcomeActivity中)

   String userAccount = UserPreference.read(KeyConstance.IS_USER_ACCOUNT, null);//軟件還沒有保持賬號
   String userPassword = UserPreference.read(KeyConstance.IS_USER_PASSWORD, null);
   String userid = UserPreference.read(KeyConstance.IS_USER_ID, null);

   if (TextUtils.isEmpty(userAccount)) {//沒有保存的登錄信息跳轉到登錄界面
    //空的,表示沒有注冊,或者清除數據
    Intent intent = new Intent();
    intent.setClass(WelcomeActiviy.this, LoginActivity.class);
    startActivity(intent);
    overridePendingTransition(android.R.anim.slide_in_left, android.R.anim.slide_out_right);
    finish();
   } else {
    //用保存的信息直接登錄
    RequestApiData.getInstance().getLoginData(userAccount, userPassword,
      UserBaseInfo.class, WelcomeActiviy.this);

   }
WelcomeActivity也同樣實現了HttpResponeCallBack接口,所以傳入的callback對象也是自身,我們在回調方法中判斷是否登錄成功

@Override
public void onResponeStart(String apiName) {

}

@Override
public void onLoading(String apiName, long count, long current) {

}

@Override
public void onSuccess(String apiName, Object object) {
 //當前接口是否是獲取用戶的基本信息的接口
 if (UrlConstance.KEY_USER_BASE_INFO.equals(apiName)) {
  if (object != null && object instanceof UserBaseInfo) {
   UserBaseInfo info = (UserBaseInfo) object;
   ItLanBaoApplication.getInstance().setBaseUser(info);//把數據放入到Application裡面,全局
   UserPreference.save(KeyConstance.IS_USER_ID, String.valueOf(info.getUserid()));

   Intent intent = new Intent();
   intent.setClass(WelcomeActiviy.this, MainActivity.class);
   startActivity(intent);
   overridePendingTransition(android.R.anim.slide_in_left,
     android.R.anim.slide_out_right);
   finish();

  } else {
   Toast.makeText(WelcomeActiviy.this, "加載失敗", Toast.LENGTH_SHORT).show();
  }
 } else if (UrlConstance.KEY_LOGIN_INFO.equals(apiName)) {//當前接口是登錄的接口
  //登錄返回數據
  if (object != null && object instanceof UserBaseInfo) {
   UserBaseInfo info = (UserBaseInfo) object;
   if (Constant.KEY_SUCCESS.equals(info.getRet())) {

    ItLanBaoApplication.getInstance().setBaseUser(info);//將用戶信息保存在Application中
    UserPreference.save(KeyConstance.IS_USER_ID, String.valueOf(info.getUserid()));

    Intent intent = new Intent();
    intent.setClass(WelcomeActiviy.this, MainActivity.class);
    startActivity(intent);
    overridePendingTransition(android.R.anim.slide_in_left,
      android.R.anim.slide_out_right);
    finish();

   } else {
    Toast.makeText(WelcomeActiviy.this, info.getMsg(), Toast.LENGTH_SHORT).show();
   }
  }
 }
}

@Override
public void onFailure(String apiName, Throwable t, int errorNo, String strMsg) {
 Toast.makeText(WelcomeActiviy.this, "Failure", Toast.LENGTH_SHORT).show();
}


demo下載地址http://xiazai.jb51.net/201611/yuanma/Androidlogindemo(jb51.net).rar

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved