热门标签 | HotTags
当前位置:  开发笔记 > 前端 > 正文

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

这篇文章主要为大家详细介绍了Android客户端实现注册、登录代码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

我们在开发安卓App时难免要与服务器打交道,尤其是对于用户账号信息的注册与登录更是每个Android开发人员必须掌握的技能,本文将对客户端的注册/登录功能的实现进行分析,不到之处还请指出。

在这里我们仅讨论客户端如何请求服务器进行注册,而服务器在收到客户端请求后进行的一系列操作并不在本文所述范围内,有兴趣大家可以参考

请求服务器

客户端在进行用户信息的注册和登录时一般使用post请求(携带参数)去服务器。以volley框架请求服务器为例,真正与服务器进行交互的就是如下代码:

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

 //请求成功
 @Override
 public void onResponse(String s) {
 //执行请求成功的回调
 callback.onSuccess()
 }

 }, new ErrorListener() {

 //请求错误
 @Override
 public void onErrorResponse(VolleyError volleyError) {
 //执行请求失败的回调
 callback.onFailure()
 }
 }){

 //携带参数(Map集合)
 @Override
 protected Map getParams() throws AuthFailureError {
 return parames;
 }
 };

 //将请求添加到请求队列中
 Volley.newRequestQueue(context).add(request);

 当然,我们在请求服务器成功或失败时应该设置相应的回调方法,让我们能够进行一些操作。
 •callback.onSuccess() //请求成功的回调
 1.保存用户注册信息(SP中和Application中)
 2.跳转到主页面 

•callback.onFailure() //请求失败的回调

 1.提示错误信息 

下面通过一个具体的demo来介绍
(声明:此demo为阿福老师IT蓝豹App的代码截取) 

注:我们在与服务器进行交互时,必须按照规定的接口和规则进行请求,这里使用的是IT蓝豹App中的服务器,服务器的注册接口数据格式如下

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

2.参数说明
    nickname       必须有    昵称
    email          必须有    邮箱
    password       必须有    密码
    accesstoken    必须有    签名md5(nickname+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":"接口调用失败"
 }

demo演示

主要实现代码(demo会在文章最后给出)

1.注册页面中(RegisterActivity),点击注册按钮

 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)) {//验证邮箱格式是否符合

 //调用RequestApiData中的getRegistData()方法进行注册,传入用户输入的昵称,邮箱、密码,以及解析数据的bean对象和callback对象(回调到自身)
 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();
 }
 }
 });

注意:这个注册的方法中,我们传入的最后一个参数是回调的对象,这里我们传入的是RegisterActivity自身,所以它需要实现

HttpResponeCallBack接口
 RequestApiData.getInstance().getRegistData(nick, emailStr, passwordStr,
 AnalyticalRegistInfo.class, RegisterActivity.this);

2.请求服务器的回调接口(HttpResponeCallBack)

 public interface HttpResponeCallBack {
 public void onResponeStart(String apiName);

 /**
 * 此回调只有调用download方法下载数据时才生效
 * 
 * @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.网络接口类(RequestApiData)

 public class RequestApiData {
 private static RequestApiData instance = null;
 private HttpResponeCallBack mCallBack = null;

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

 /**
 * 4.6注册用户接口
 * @param nickname 昵称
 * @param email 邮箱
 * @param password 密码
 * @param clazz 数据返回的解析对象
 * @param callback 回调
 * 特别要注意参数位置不能变要根据文档来
 * 请求方式:POST
 */
 public void getRegistData(String nickname,String email
 ,String password, Class clazz,
 HttpResponeCallBack callback) {
 mCallBack = callback;
 //这是每一个接口的唯一标示
 String tagUrl = UrlConstance.KEY_REGIST_INFO;//注册接口
 //将注册的信息保存在map中(须和服务器端一致)
 HashMap parameter = new HashMap();
 parameter.put("nickname", nickname);
 parameter.put("email",email);
 parameter.put("password",password);

 //拼接参数信息,昵称,邮箱,密码,公钥,并用md5进行加密
 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()));

 //调用RequestManager的post方法,请求服务器
 RequestManager.post(UrlConstance.APP_URL,tagUrl, parameter, clazz, callback);

 }
}

 4.网络请求处理的类(RequestManager)

 public class RequestManager {
 private static RequestQueue mRequestQueue;
 private static ImageLoader mImageLoader;


 private synchronized static void initRequestQueue() {
 if (mRequestQueue == null) {
 //创建一个请求队列(使用Volley框架)
 mRequestQueue = Volley.newRequestQueue(ItLanbaoLibApplication.getInstance());
 }
 }


 /**
 * 添加请求到请求队列中
 * @param request
 * @param tag
 */
 private static void addRequest(Request<&#63;> request, Object tag) {
 if (tag != null) {
 request.setTag(tag);
 }
 mRequestQueue.add(request);
 }

 /**
 * 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  void post(final String app_url, final String tag_url, final HashMap parameter, Class 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  void post(final String app_url, final String url, final HashMap parameter, final Class clazz,
 final HttpResponeCallBack callback, Priority priority) {
 if (callback != null) {
 callback.onResponeStart(url);//回调请求开始
 }

 //初始化请求队列
 initRequestQueue();

 //将公共的接口前缀和接口名称拼接
 //eg:拼接成注册的接口 http://www.itlanbao.com/api/app/users/user_register_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:监听
 */
 StringRequest request = new StringRequest(Method.POST, builder.toString(),
 new Listener() {

 @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 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 parameter) {
 ApiParams api = new ApiParams();
 for (Entry entry : parameter.entrySet()) {
 api.with(entry.getKey(), entry.getValue());
 }
 return api;
 }

}

 5.在请求服务器成功/失败后会执行回调方法,而我们传入的callback对象是自身(RegisterActivity),所以现在我们回到注册页面

 @Override
 public void onResponeStart(String apiName) {
 // TODO Auto-generated method stub
 Toast.makeText(RegisterActivity.this, "正在请求数据...", 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
 //注册接口
 if (UrlConstance.KEY_REGIST_INFO.equals(apiName)) {
 if (object != null && object instanceof AnalyticalRegistInfo) {
 AnalyticalRegistInfo info = (AnalyticalRegistInfo) object;
 String successCode = info.getRet();
 //请求成功
 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, "注册成功...", Toast.LENGTH_SHORT).show();

 RegisterActivity.this.finish();

 } else {
 Toast.makeText(RegisterActivity.this, "注册失败", 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下载地址

至此,安卓客户端的注册功能就实现了,下一篇中将会介绍登录和自动登录的实现,尽请关注。


推荐阅读
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 本文介绍了高校天文共享平台的开发过程中的思考和规划。该平台旨在为高校学生提供天象预报、科普知识、观测活动、图片分享等功能。文章分析了项目的技术栈选择、网站前端布局、业务流程、数据库结构等方面,并总结了项目存在的问题,如前后端未分离、代码混乱等。作者表示希望通过记录和规划,能够理清思路,进一步完善该平台。 ... [详细]
  • 使用nodejs爬取b站番剧数据,计算最佳追番推荐
    本文介绍了如何使用nodejs爬取b站番剧数据,并通过计算得出最佳追番推荐。通过调用相关接口获取番剧数据和评分数据,以及使用相应的算法进行计算。该方法可以帮助用户找到适合自己的番剧进行观看。 ... [详细]
  • YOLOv7基于自己的数据集从零构建模型完整训练、推理计算超详细教程
    本文介绍了关于人工智能、神经网络和深度学习的知识点,并提供了YOLOv7基于自己的数据集从零构建模型完整训练、推理计算的详细教程。文章还提到了郑州最低生活保障的话题。对于从事目标检测任务的人来说,YOLO是一个熟悉的模型。文章还提到了yolov4和yolov6的相关内容,以及选择模型的优化思路。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文讲述了作者通过点火测试男友的性格和承受能力,以考验婚姻问题。作者故意不安慰男友并再次点火,观察他的反应。这个行为是善意的玩人,旨在了解男友的性格和避免婚姻问题。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
author-avatar
民海育来仁湖
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有