Android客戶端實(shí)現(xiàn)注冊(cè)、登錄詳解(1)
我們?cè)陂_發(fā)安卓App時(shí)難免要與服務(wù)器打交道,尤其是對(duì)于用戶賬號(hào)信息的注冊(cè)與登錄更是每個(gè)Android開發(fā)人員必須掌握的技能,本文將對(duì)客戶端的注冊(cè)/登錄功能的實(shí)現(xiàn)進(jìn)行分析,不到之處還請(qǐng)指出。

在這里我們僅討論客戶端如何請(qǐng)求服務(wù)器進(jìn)行注冊(cè),而服務(wù)器在收到客戶端請(qǐng)求后進(jìn)行的一系列操作并不在本文所述范圍內(nèi),有興趣大家可以參考
請(qǐng)求服務(wù)器
客戶端在進(jìn)行用戶信息的注冊(cè)和登錄時(shí)一般使用post請(qǐng)求(攜帶參數(shù))去服務(wù)器。以volley框架請(qǐng)求服務(wù)器為例,真正與服務(wù)器進(jìn)行交互的就是如下代碼:
StringRequest request=new StringRequest(Method.POST, url, new Listener<String>() {
//請(qǐng)求成功
@Override
public void onResponse(String s) {
//執(zhí)行請(qǐng)求成功的回調(diào)
callback.onSuccess()
}
}, new ErrorListener() {
//請(qǐng)求錯(cuò)誤
@Override
public void onErrorResponse(VolleyError volleyError) {
//執(zhí)行請(qǐng)求失敗的回調(diào)
callback.onFailure()
}
}){
//攜帶參數(shù)(Map集合)
@Override
protected Map<String, String> getParams() throws AuthFailureError {
return parames;
}
};
//將請(qǐng)求添加到請(qǐng)求隊(duì)列中
Volley.newRequestQueue(context).add(request);
當(dāng)然,我們?cè)谡?qǐng)求服務(wù)器成功或失敗時(shí)應(yīng)該設(shè)置相應(yīng)的回調(diào)方法,讓我們能夠進(jìn)行一些操作。
•callback.onSuccess() //請(qǐng)求成功的回調(diào)
1.保存用戶注冊(cè)信息(SP中和Application中)
2.跳轉(zhuǎn)到主頁(yè)面
•callback.onFailure() //請(qǐng)求失敗的回調(diào)
1.提示錯(cuò)誤信息
下面通過(guò)一個(gè)具體的demo來(lái)介紹
(聲明:此demo為阿福老師IT藍(lán)豹App的代碼截取)
注:我們?cè)谂c服務(wù)器進(jìn)行交互時(shí),必須按照規(guī)定的接口和規(guī)則進(jìn)行請(qǐng)求,這里使用的是IT藍(lán)豹App中的服務(wù)器,服務(wù)器的注冊(cè)接口數(shù)據(jù)格式如下
1.url:http://www.itlanbao.com/api/app/users/user_register_Handler.ashx
2.參數(shù)說(shuō)明
nickname 必須有 昵稱
email 必須有 郵箱
password 必須有 密碼
accesstoken 必須有 簽名md5(nickname+email+password+"雙方平臺(tái)約定公鑰")
3.請(qǐng)求方式:POST
4.返回值格式:
成功
{
"ret":0,
"errcode":0,
"msg":"接口調(diào)用成功",
"data":{
"userid":"16489",
"email":"nnn@aaa.com",
"nickname":"duss",
"userhead":"http://img.itlanbao.com/avatar.png"
}
}
失敗
{
"ret":1,
"errcode":1,
"msg":"接口調(diào)用失敗"
}
demo演示

主要實(shí)現(xiàn)代碼(demo會(huì)在文章最后給出)
1.注冊(cè)頁(yè)面中(RegisterActivity),點(diǎn)擊注冊(cè)按鈕
registBtn.setOnClickListener(new Button.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
//獲得用戶輸入的信息
String nick = loginNick.getText().toString();
String emailStr = email.getText().toString();
String passwordStr = password.getText().toString();
if (!TextUtils.isEmpty(nick) &&
!TextUtils.isEmpty(emailStr)
&& !TextUtils.isEmpty(passwordStr)) {
if (Utils.isEmail(emailStr)) {//驗(yàn)證郵箱格式是否符合
//調(diào)用RequestApiData中的getRegistData()方法進(jìn)行注冊(cè),傳入用戶輸入的昵稱,郵箱、密碼,以及解析數(shù)據(jù)的bean對(duì)象和callback對(duì)象(回調(diào)到自身)
RequestApiData.getInstance().getRegistData(nick, emailStr, passwordStr,
AnalyticalRegistInfo.class, RegisterActivity.this);
} else {
Toast.makeText(RegisterActivity.this, "輸入郵箱有誤", Toast.LENGTH_SHORT).show();
}
} else {
Toast.makeText(RegisterActivity.this, "輸入信息未完全", Toast.LENGTH_SHORT).show();
}
}
});
注意:這個(gè)注冊(cè)的方法中,我們傳入的最后一個(gè)參數(shù)是回調(diào)的對(duì)象,這里我們傳入的是RegisterActivity自身,所以它需要實(shí)現(xiàn)
HttpResponeCallBack接口 RequestApiData.getInstance().getRegistData(nick, emailStr, passwordStr, AnalyticalRegistInfo.class, RegisterActivity.this);
2.請(qǐng)求服務(wù)器的回調(diào)接口(HttpResponeCallBack)
public interface HttpResponeCallBack {
public void onResponeStart(String apiName);
/**
* 此回調(diào)只有調(diào)用download方法下載數(shù)據(jù)時(shí)才生效
*
* @param apiName
* @param count
* @param current
*/
public void onLoading(String apiName, long count, long current);
public void onSuccess(String apiName, Object object);
public void onFailure(String apiName, Throwable t, int errorNo, String strMsg);
}
3.網(wǎng)絡(luò)接口類(RequestApiData)
public class RequestApiData {
private static RequestApiData instance = null;
private HttpResponeCallBack mCallBack = null;
//創(chuàng)建接口對(duì)象
public static RequestApiData getInstance() {
if (instance == null) {
instance = new RequestApiData();
}
return instance;
}
/**
* 4.6注冊(cè)用戶接口
* @param nickname 昵稱
* @param email 郵箱
* @param password 密碼
* @param clazz 數(shù)據(jù)返回的解析對(duì)象
* @param callback 回調(diào)
* 特別要注意參數(shù)位置不能變要根據(jù)文檔來(lái)
* 請(qǐng)求方式:POST
*/
public void getRegistData(String nickname,String email
,String password, Class<AnalyticalRegistInfo> clazz,
HttpResponeCallBack callback) {
mCallBack = callback;
//這是每一個(gè)接口的唯一標(biāo)示
String tagUrl = UrlConstance.KEY_REGIST_INFO;//注冊(cè)接口
//將注冊(cè)的信息保存在map中(須和服務(wù)器端一致)
HashMap<String, String> parameter = new HashMap<String, String>();
parameter.put("nickname", nickname);
parameter.put("email",email);
parameter.put("password",password);
//拼接參數(shù)信息,昵稱,郵箱,密碼,公鑰,并用md5進(jìn)行加密
StringBuilder builder = new StringBuilder();
builder.append(nickname);
builder.append(email);
builder.append(password);
builder.append(UrlConstance.PUBLIC_KEY);
parameter.put(UrlConstance.ACCESSTOKEN_KEY,MD5Util.getMD5Str(builder.toString()));
//調(diào)用RequestManager的post方法,請(qǐng)求服務(wù)器
RequestManager.post(UrlConstance.APP_URL,tagUrl, parameter, clazz, callback);
}
}
4.網(wǎng)絡(luò)請(qǐng)求處理的類(RequestManager)
public class RequestManager {
private static RequestQueue mRequestQueue;
private static ImageLoader mImageLoader;
private synchronized static void initRequestQueue() {
if (mRequestQueue == null) {
//創(chuàng)建一個(gè)請(qǐng)求隊(duì)列(使用Volley框架)
mRequestQueue = Volley.newRequestQueue(ItLanbaoLibApplication.getInstance());
}
}
/**
* 添加請(qǐng)求到請(qǐng)求隊(duì)列中
* @param request
* @param tag
*/
private static void addRequest(Request<?> request, Object tag) {
if (tag != null) {
request.setTag(tag);
}
mRequestQueue.add(request);
}
/**
* post 請(qǐng)求數(shù)據(jù)
*
* @param app_url 公共的接口前綴 http://www.itlanbao.com/api/app/
* @param tag_url 接口名稱,eg:users/user_register_Handler.ashx(注冊(cè)接口)
* @param parameter 請(qǐng)求參數(shù)封裝對(duì)象
* @param clazz 返回?cái)?shù)據(jù)封裝對(duì)象,如果傳null,則直接返回String
* @param callback 接口回調(diào)監(jiān)聽(tīng)
*/
public static <T> void post(final String app_url, final String tag_url, final HashMap<String, String> parameter, Class<T> clazz,
final HttpResponeCallBack callback) {
//發(fā)送post請(qǐng)求服務(wù)器
post(app_url, tag_url, parameter, clazz, callback, Priority.NORMAL);
}
/**
* post 請(qǐng)求數(shù)據(jù)
*
* @param app_url 路徑
* @param url 接口名稱
* @param parameter 請(qǐng)求參數(shù)封裝對(duì)象
* @param clazz 返回?cái)?shù)據(jù)封裝對(duì)象,如果傳null,則直接返回String
* @param callback 接口回調(diào)監(jiān)聽(tīng)
* @param priority 指定接口請(qǐng)求線程優(yōu)先級(jí)
*/
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);//回調(diào)請(qǐng)求開始
}
//初始化請(qǐng)求隊(duì)列
initRequestQueue();
//將公共的接口前綴和接口名稱拼接
//eg:拼接成注冊(cè)的接口 http://www.itlanbao.com/api/app/users/user_register_Handler.ashx
StringBuilder builder = new StringBuilder(app_url);
builder.append(url);
{// 檢查當(dāng)前網(wǎng)絡(luò)是否可用
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, "網(wǎng)絡(luò)出錯(cuò)");//回調(diào)請(qǐng)求失敗
return;
}
}
}
/**
* 使用Volley框架真正去請(qǐng)求服務(wù)器
* Method.POST:請(qǐng)求方式為post
* builder.toString():請(qǐng)求的鏈接
* Listener<String>:監(jiān)聽(tīng)
*/
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();
//回調(diào)請(qǐng)求成功,傳入url和解析的對(duì)象
callback.onSuccess(url, gson.fromJson(response, clazz));
}
} catch (Exception e) {
// TODO: handle exception
if (callback != null) {
//回調(diào)請(qǐng)求失敗--解析異常
callback.onFailure(url, e, 0, "解析異常");
return;
}
}
}
}, new ErrorListener() {
//請(qǐng)求出錯(cuò)的監(jiān)聽(tīng)
@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請(qǐng)求的參數(shù)信息
protected Map<String, String> getParams() {
return getPostApiParmes(parameter);
}
};
//添加請(qǐng)求到請(qǐng)求隊(duì)列中
addRequest(request, url);
}
/*
* post參數(shù)
*
* ts:時(shí)間戳 sign: 接口簽名 parms = 按文檔參數(shù)拼接 parm[0]+ … + parm[n-1] sign =
* md5(parms+"雙方平臺(tái)約定公鑰")
*/
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.在請(qǐng)求服務(wù)器成功/失敗后會(huì)執(zhí)行回調(diào)方法,而我們傳入的callback對(duì)象是自身(RegisterActivity),所以現(xiàn)在我們回到注冊(cè)頁(yè)面
@Override
public void onResponeStart(String apiName) {
// TODO Auto-generated method stub
Toast.makeText(RegisterActivity.this, "正在請(qǐng)求數(shù)據(jù)...", Toast.LENGTH_SHORT).show();
}
@Override
public void onLoading(String apiName, long count, long current) {
Toast.makeText(RegisterActivity.this, "Loading...", Toast.LENGTH_SHORT).show();
}
@Override
public void onSuccess(String apiName, Object object) {
// TODO Auto-generated method stub
//注冊(cè)接口
if (UrlConstance.KEY_REGIST_INFO.equals(apiName)) {
if (object != null && object instanceof AnalyticalRegistInfo) {
AnalyticalRegistInfo info = (AnalyticalRegistInfo) object;
String successCode = info.getRet();
//請(qǐng)求成功
if (successCode.equals(Constant.KEY_SUCCESS)) {
UserBaseInfo baseUser = new UserBaseInfo();
baseUser.setEmail(info.getEmail());
baseUser.setNickname(info.getNickname());
baseUser.setUserhead(info.getUserhead());
baseUser.setUserid(String.valueOf(info.getUserid()));
ItLanBaoApplication.getInstance().setBaseUser(baseUser);
UserPreference.save(KeyConstance.IS_USER_ID, String.valueOf(info.getUserid()));
UserPreference.save(KeyConstance.IS_USER_ACCOUNT, info.getEmail());
UserPreference.save(KeyConstance.IS_USER_PASSWORD, password.getText().toString());
Intent intent = new Intent(RegisterActivity.this, MainActivity.class);
RegisterActivity.this.startActivity(intent);
Toast.makeText(RegisterActivity.this, "注冊(cè)成功...", Toast.LENGTH_SHORT).show();
RegisterActivity.this.finish();
} else {
Toast.makeText(RegisterActivity.this, "注冊(cè)失敗", Toast.LENGTH_SHORT).show();
}
}
}
}
@Override
public void onFailure(String apiName, Throwable t, int errorNo, String strMsg) {
Toast.makeText(RegisterActivity.this, "Failure", Toast.LENGTH_SHORT).show();
}
demo下載地址
至此,安卓客戶端的注冊(cè)功能就實(shí)現(xiàn)了,下一篇中將會(huì)介紹登錄和自動(dòng)登錄的實(shí)現(xiàn),盡請(qǐng)關(guān)注。
- Android用SharedPreferences實(shí)現(xiàn)登錄注冊(cè)注銷功能
- android實(shí)現(xiàn)注冊(cè)登錄程序
- Android實(shí)現(xiàn)登錄注冊(cè)頁(yè)面(下)
- Android實(shí)現(xiàn)登錄注冊(cè)功能
- Android基于Sqlite實(shí)現(xiàn)注冊(cè)和登錄功能
- Android實(shí)現(xiàn)注冊(cè)登錄界面的實(shí)例代碼
- Android設(shè)計(jì)登錄界面、找回密碼、注冊(cè)功能
- Android登錄注冊(cè)功能 數(shù)據(jù)庫(kù)SQLite驗(yàn)證
- Android Studio連接SQLite數(shù)據(jù)庫(kù)的登錄注冊(cè)實(shí)現(xiàn)
- Android使用http實(shí)現(xiàn)注冊(cè)登錄功能
相關(guān)文章
利用Android中的TextView實(shí)現(xiàn)逐字顯示動(dòng)畫
在安卓程序啟動(dòng)的時(shí)候,想逐字顯示一段話,每個(gè)字都有一個(gè)從透明到不透明的漸變動(dòng)畫。那如何顯示這個(gè)效果,下面一起來(lái)看看。2016-08-08
Android中自定義PopupWindow實(shí)現(xiàn)彈出框并帶有動(dòng)畫效果
這篇文章主要介紹了Android中自定義PopupWindow實(shí)現(xiàn)彈出框并帶有動(dòng)畫效果的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09
Android實(shí)現(xiàn)藍(lán)牙串口通訊
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)藍(lán)牙串口通訊,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08
一文詳解Jetpack?Android新一代導(dǎo)航管理Navigation
這篇文章主要為大家介紹了Jetpack?Android新一代導(dǎo)航管理Navigation詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03
android ScrollView實(shí)現(xiàn)下拉放大頭部圖片
這篇文章主要為大家詳細(xì)介紹了android ScrollView實(shí)現(xiàn)下拉放大頭部圖片,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
Android搭建本地Tomcat服務(wù)器及相關(guān)配置
這篇文章主要介紹了Android搭建本地Tomcat服務(wù)器及相關(guān)配置,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07
android實(shí)現(xiàn)通知欄下載更新app示例
這篇文章主要介紹了android實(shí)現(xiàn)通知欄下載更新app示例,需要的朋友可以參考下2014-03-03
Kotlin中ListView與RecyclerView的應(yīng)用講解
這篇文章主要介紹了Kotlin中ListView與RecyclerView的應(yīng)用講解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09

