Android客户端实现注册、登录详解(2)

上文中介绍了安卓客户端与服务器交互,实现注册功能,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":"nnn@aaa.com",
   "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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android登录注册功能 数据库SQLite验证

    本文实例为大家分享了Android登录注册功能的具体代码,供大家参考,具体内容如下 展示效果 代码区 MainActivity(登录方法) public class MainActivity extends AppCompatActivity { @BindView(R.id.editText) EditText editText; @BindView(R.id.editText2) EditText editText2; @BindView(R.id.button) Button button

  • Android手机注册登录时获取验证码之后倒计时功能(知识点总结)

    app注册界面经常会遇到一个场景:手机注册,点击获取验证码,验证码发送成功之后,开始倒计时 具体代码如下所示: private TimerTask timerTask; private Timer timer; private int time = 5000;//五秒 private int timess; /** * 开始倒计时 */ private void startTimer() { timess = time/1000; tvTime.setText(timess+"S");

  • Android注册登录实时自动获取短信验证码

    android应用的自动化测试必然会涉及到注册登录功能,而许多的注册登录或修改密码功能常常需要输入短信验证码,因此有必要能够自动获得下发的短信验证码. 主要就是实时获取短信信息. android上获取短信信息主要有BroadcastReceiver方式与数据库方式,要实时的话就BroadcastReceiver比较方便,分享一篇文章大家可以查看一下,点击. public class SMSReceiver extends BroadcastReceiver{ private String ver

  • Android开发之登录验证实例教程

    本文所述实例源自一个项目开发中的登录验证功能,具体的要求就是,在Android端输入用户名和密码,在服务器端验证MySQL数据库中是否有此用户,实现之前当然首要的是,如何使Android端的数据发送到服务器端,具体的实现方法如下: 服务器端:ManageServlet.java代码如下: public class ManageServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServle

  • Android实现登录注册功能封装

    我们都知道Android应用软件基本上都会用到登录注册功能,那么对一个一个好的登录注册模块进行封装就势在必行了.这里给大家介绍一下我的第一个项目中所用到的登录注册功能的,已经对其进行封装,希望能对大家有帮助,如果有什么错误或者改进的话希望各位可以指出. 我们都知道登录注册系列功能的实现有以下几步: 注册账号 登录账号 (第三方账号登录) 记住密码 自动登录 修改密码 大体的流程如下 对于需要获取用户登录状态的操作,先判断用户是否已经登录. 如果用户已经登录,则继续后面的操作,否则,跳转到登录页面

  • Android设计登录界面、找回密码、注册功能

    本文实例为大家分享了Android 登录.找回密码.注册功能的实现代码,供大家参考,具体内容如下 1.数据库的设计 我在数据库中添加了两张表,一张表用来存储用户信息,诸如用户名,密码,手机号等,可任意添加.另一张表用来存储上一个登录用户的账户信息,我是为了方便才另外创建了一张表去存储,而且这张表我设计了它只能存储一条信息,每次的存储都是对上一条记录的覆盖.事实上,我尝试过在存储用户信息的那张表内添加一个标识,用来标记上一次登录的是哪一个帐号,但是这样做的话,每次改变标识都需要遍历整张表,十分的麻

  • Android实现登录功能demo示例

    本文实例讲述了Android实现登录功能的方法.分享给大家供大家参考,具体如下: 安卓,在小编实习之前的那段岁月里面,小编都没有玩儿过,如果说玩儿过,那就是安卓手机了,咳咳,敲登录的时候有种特别久违的熟悉,这种熟悉的感觉就和当时敲机房收费系统一样,那叫一个艰难啊,不过小编相信,在小编的IT成长之路上,正是因为有了这些艰难险阻陪伴着小编一起成长,才让小编更加勇敢坚强,勇敢的面对一个又一个bug,坚强的敲完一行行代码,经过了几天的研究登录一条线的功能已经实现,现在小编就来简单的总结一下,还请小伙伴们

  • Android客户端实现注册、登录详解(1)

    我们在开发安卓App时难免要与服务器打交道,尤其是对于用户账号信息的注册与登录更是每个Android开发人员必须掌握的技能,本文将对客户端的注册/登录功能的实现进行分析,不到之处还请指出. 在这里我们仅讨论客户端如何请求服务器进行注册,而服务器在收到客户端请求后进行的一系列操作并不在本文所述范围内,有兴趣大家可以参考 请求服务器 客户端在进行用户信息的注册和登录时一般使用post请求(携带参数)去服务器.以volley框架请求服务器为例,真正与服务器进行交互的就是如下代码: StringRequ

  • Android开发之注册登录方法示例

    本文所述,继续上一篇关于Android端向服务器端发送数据的方法进一步完善注册登录的方法,由于版本问题出现一点瑕疵,今天经过调试已经解决,在这里给大家介绍一下. 在Android4.0以后版本的对于网络权限要求变得严格,致使上一篇所述的案例无法将数据发送到服务器端,当你一点击发送数据,Android控制台就会报错,错误当然是很让人头疼,基本上都是关于http的错误,所以可以肯定是Android虚拟机向服务器发送数据时出现了错误,经过一番检查与测试后才知道,4.0之后的版本,主线程中不允许调用网络

  • Android实现闪屏及注册和登录界面之间的切换效果

    在没给大家介绍正文之前先给大家说下实现思路: 先分别实现闪屏.注册界面.登录界面的活动,再用Intent将相关的活动连接起来,实现不同活动之间的跳转.此次试验代码较多,我只列出主要代码,详细的代码可用底部的下载链接下载. 一.实验效果图: 二.主要代码: (1)WelcomeActivity.Java(这部分代码实现的是第一页的欢迎页面) package com.example.flashscreendemo; import android.app.Activity; import androi

随机推荐