android APP实现更新 PHP后台服务器

简介: android APP实现更新 PHP后台服务器 最近自己在做一款APP,需要实现APP版本更新功能,从网上找了许多资料,只找到了关于移动端的实现。经过我的研究,终于实现了比较完整的android APP版本更新功能,在此分享给广大朋友,但是我的ios端还没实现,但是传输是基于的http协议,实现原理应该是大同小异的。

android APP实现更新 PHP后台服务器

最近自己在做一款APP,需要实现APP版本更新功能,从网上找了许多资料,只找到了关于移动端的实现。经过我的研究,终于实现了比较完整的android APP版本更新功能,在此分享给广大朋友,但是我的ios端还没实现,但是传输是基于的http协议,实现原理应该是大同小异的。接下来进入正文。

PHP后台服务器实现

后台的实现是基于的thinkPHP框架。对于PHP开发框架thinkPHP框架的朋友,只要去官网下载手册,看那么几页,我想你应该就懂了。

接下来直接给出PHP代码实现。

对于这个文件,大家需要修改的便是命名空间。

namespace API\Controller;
use Think\Controller;

至于为什么,大家需要了解下PHP的基础知识,和thinkPHP框架的开发流程就可以了,很快的。

<?php
namespace API\Controller;
use Think\Controller;
class ServerController extends Controller {
    public function index(){
        //$this->show('<style type="text/css">*{ padding: 0; margin: 0; } div{ padding: 4px 48px;} body{ background: #fff; font-family: "寰蒋闆呴粦"; color: #333;font-size:24px} h1{ font-size: 100px; font-weight: normal; margin-bottom: 12px; } p{ line-height: 1.8em; font-size: 36px }</style><div style="padding: 24px 48px;"> <h1>:)</h1><p>娆㈣繋浣跨敤 <b>ThinkPHP</b>锛?/p><br/>[ 鎮ㄧ幇鍦ㄨ闂殑鏄疕ome妯″潡鐨処ndex鎺у埗鍣?]</div><script type="text/javascript" src="http://tajs.qq.com/stats?sId=9347272" charset="UTF-8"></script>','utf-8');
        echo "hello";
    }

    function updateInfo(){
        $result=array(
            'version'=>'1.0',
            'description'=>'目前已经更新了卡屏功能',
            'url'=>'http://localhost/index.php/API/Server/download'
        );
        echo json_encode($result);
    }

    function download(){
        $file = "/tmp/test.apk";

        $filename = basename($file);//返回路径中的文件名部分。
        /*
         * header() 函数向客户端发送原始的 HTTP 报头。
         * 必须在任何实际的输出被发送之前调用 header() 函数
         * */
        header("Content-type: application/octet-stream");

//处理中文文件名
        $ua = $_SERVER["HTTP_USER_AGENT"];
        $encoded_filename = rawurlencode($filename);
        if (preg_match("/MSIE/", $ua)) {
            header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
        } else if (preg_match("/Firefox/", $ua)) {
            header("Content-Disposition: attachment; filename*=\"utf8''" . $filename . '"');
        } else {
            header('Content-Disposition: attachment; filename="' . $filename . '"');
        }

        header("Content-Length: ". filesize($file));

        /*
         * readfile() 函数输出一个文件。
         * 该函数读入一个文件并写入到输出缓冲。
         * 若成功,则返回从文件中读入的字节数。若失败,则返回 false。
         */
        ob_clean();
        flush();
        readfile($file);
    }
}



 

 
 

这里需要解释的是

function updateInfo()
这个方法实现的是获得最近的版本信息。然后和移动端进行匹配
URL地址是:http://localhost/index.php/API/Server/updateinfo,大家可以根据自己的服务器或者云服务器的IP地址进行修改。
 
function download(){
这个方法实现的是下载功能,也就是核心方法。移动端需要访问到这个IP地址才能进行文件的下载。
URL地址是:http://localhost/index.php/API/Server/download,大家可以根据自己的服务器或者云服务器的IP地址进行修改。
 
关于需要下载的文件路径,我是存放在ubuntu系统的/tmp目录下的
 
android移动端实现
 
移动端我直接给出了实现放的类和方法以及活动,就不解释了。
只给出基本的原理
1、从这个URL地址http://localhost/index.php/API/Server/updateinfo获取最新的版本信息,通过http协议和json协议解析出基本的版本信息,然后存储在UpdateInfo类中,然后和APP的当前版本进行比较,如果有最新版则下载。
2、关于下载,android端的核心实现代码是
new Thread() {
         public void run() {        
            HttpClient client = new DefaultHttpClient();
            HttpGet get = new HttpGet(url);
            HttpResponse response;
            try {
               response = client.execute(get);
               HttpEntity entity = response.getEntity();
               int length = (int) entity.getContentLength();
                                        progressDialog.setMax(length);
               InputStream is = entity.getContent();
               FileOutputStream fileOutputStream = null;
               if (is != null) {
                  File file = new File(
                        Environment.getExternalStorageDirectory(),
                        "Test.apk");
                  fileOutputStream = new FileOutputStream(file);
                        byte[] buf = new byte[10];   
                  int ch = -1;
                  int process = 0;
                  while ((ch = is.read(buf)) != -1) {       
                     fileOutputStream.write(buf, 0, ch);
                     process += ch;
                     progressDialog.setProgress(process);
                  }
               }
               fileOutputStream.flush();
               if (fileOutputStream != null) {
                  fileOutputStream.close();
               }
               down();
            } catch (ClientProtocolException e) {
               e.printStackTrace();
            } catch (IOException e) {
               e.printStackTrace();
            }
         }

      }.start();

而PHP的实现代码是download中的

ob_clean();
flush();
readfile($file);

至于底层的原理是,readfile方法封装了,http文件传输的基本实现,大家可以百度下,以上两段代码配合,一个客户端,一个服务器,就能够实现基于http文件的传输.

 
最后给出android端的代码
 
 
import android.annotation.SuppressLint;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.ProgressDialog;
import android.content.DialogInterface;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;

import com.example.administrator.dongzhiwuapp.R;

public class SetActivity extends Activity {

   private UpdateInfo info;
   private ProgressDialog progressDialog;//下载进度条窗口
   UpdateInfoService updateInfoService;

   @Override
   protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.update);
        Button button=(Button)findViewById(R.id.button1);
       button.setOnClickListener(new OnClickListener() {
         @Override
         public void onClick(View v) {
            checkUpdate();
         }
      });
   }



   private void checkUpdate(){
      new Thread() {
         public void run() {
            try {
               updateInfoService = new UpdateInfoService(SetActivity.this);
               info = updateInfoService.getUpDateInfo();
               handler1.sendEmptyMessage(0);
            } catch (Exception e) {
               e.printStackTrace();
            }
         };
      }.start();
   }
   
   @SuppressLint("HandlerLeak")
   private Handler handler1 = new Handler() {
      public void handleMessage(Message msg) {
         if (updateInfoService.isNeedUpdate()) {
            showUpdateDialog();
         }
      }
   };

   private void showUpdateDialog() {
      AlertDialog.Builder builder = new AlertDialog.Builder(this);
      builder.setIcon(android.R.drawable.ic_dialog_info);
      builder.setTitle("请升级APP版本至" + info.getVersion());
      builder.setMessage(info.getDescription());
      builder.setCancelable(false);
      builder.setPositiveButton("确定", new DialogInterface.OnClickListener() {
         @Override
         public void onClick(DialogInterface dialog, int which) {
            if (Environment.getExternalStorageState().equals(
                  Environment.MEDIA_MOUNTED)) {
               downFile(info.getUrl());
            } else {
               Toast.makeText(SetActivity.this, "SD卡不可用,请插入SD卡",
                     Toast.LENGTH_SHORT).show();
            }
         }
      });
      builder.setNegativeButton("取消", new DialogInterface.OnClickListener() {
         @Override
         public void onClick(DialogInterface dialog, int which) {
         }
      });
      builder.create().show();
   }


   //进入下载
   void downFile(final String url) { 
      progressDialog = new ProgressDialog(SetActivity.this);
      progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
      progressDialog.setTitle("正在下载");
      progressDialog.setMessage("请稍后...");
      progressDialog.setProgress(0);
      progressDialog.show();
      updateInfoService.downLoadFile(url, progressDialog,handler1);
   }
   
}
 
 
 
import android.annotation.SuppressLint;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.ProgressDialog;
import android.content.DialogInterface;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;

import com.example.administrator.dongzhiwuapp.R;

public class SetActivity extends Activity {

   private UpdateInfo info;
   private ProgressDialog progressDialog;//下载进度条窗口
   UpdateInfoService updateInfoService;

   @Override
   protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.update);
        Button button=(Button)findViewById(R.id.button1);
       button.setOnClickListener(new OnClickListener() {
         @Override
         public void onClick(View v) {
            checkUpdate();
         }
      });
   }



   private void checkUpdate(){
      new Thread() {
         public void run() {
            try {
               updateInfoService = new UpdateInfoService(SetActivity.this);
               info = updateInfoService.getUpDateInfo();
               handler1.sendEmptyMessage(0);
            } catch (Exception e) {
               e.printStackTrace();
            }
         };
      }.start();
   }
   
   @SuppressLint("HandlerLeak")
   private Handler handler1 = new Handler() {
      public void handleMessage(Message msg) {
         if (updateInfoService.isNeedUpdate()) {
            showUpdateDialog();
         }
      }
   };

   private void showUpdateDialog() {
      AlertDialog.Builder builder = new AlertDialog.Builder(this);
      builder.setIcon(android.R.drawable.ic_dialog_info);
      builder.setTitle("请升级APP版本至" + info.getVersion());
      builder.setMessage(info.getDescription());
      builder.setCancelable(false);
      builder.setPositiveButton("确定", new DialogInterface.OnClickListener() {
         @Override
         public void onClick(DialogInterface dialog, int which) {
            if (Environment.getExternalStorageState().equals(
                  Environment.MEDIA_MOUNTED)) {
               downFile(info.getUrl());
            } else {
               Toast.makeText(SetActivity.this, "SD卡不可用,请插入SD卡",
                     Toast.LENGTH_SHORT).show();
            }
         }
      });
      builder.setNegativeButton("取消", new DialogInterface.OnClickListener() {
         @Override
         public void onClick(DialogInterface dialog, int which) {
         }
      });
      builder.create().show();
   }


   //进入下载
   void downFile(final String url) { 
      progressDialog = new ProgressDialog(SetActivity.this);
      progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
      progressDialog.setTitle("正在下载");
      progressDialog.setMessage("请稍后...");
      progressDialog.setProgress(0);
      progressDialog.show();
      updateInfoService.downLoadFile(url, progressDialog,handler1);
   }
   
}
/**
 * 获取服务器地址
 */

public class GetServerUrl{
   static String url="http://";
         
   public static String getUrl() {
      return url;
   }
}
/*
*
*  版本更新信息类
*  1、获取版本字符串信息
*  2、设置版本字符串信息
*  3、获取版本描述
*  4、设置版本描述
*  5、获取URL地址
*  6、设置URL地址
* */

public class UpdateInfo
{
        private String version;
        private String description;
        private String url;
        
        public String getVersion()
        {
                return version;
        }
        public void setVersion(String version)
        {
                this.version = version;
        }
        public String getDescription()
        {
                return description;
        }
        public void setDescription(String description)
        {
                this.description = description;
        }
        public String getUrl()
        {
                return url;
        }
        public void setUrl(String url)
        {
                this.url = url;
        }
        
}
 
 
import android.app.ProgressDialog;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.PackageManager.NameNotFoundException;
import android.net.Uri;
import android.os.Environment;
import android.os.Handler;
import android.util.Log;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
import org.json.JSONObject;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;

public class UpdateInfoService {
   ProgressDialog progressDialog;
   Handler handler;
   Context context;
   UpdateInfo updateInfo;
   
   public UpdateInfoService(Context context){
      this.context=context;
   }


   /*
   * 获取更新信息
   * 版本信息存储在服务器的update.txt文件中
   * 更新信息包括:
   * 1、版本信息
   * 2、最新版本具体描述
   * 3、下载地址
   *
   * */
   public UpdateInfo getUpDateInfo() throws Exception {
      HttpClient httpClient = new DefaultHttpClient();
      HttpGet httpGet = new HttpGet("http://localhost/index.php/API/Server/updateInfo");

      HttpResponse httpResponse = httpClient.execute(httpGet);
      UpdateInfo updateInfo = new UpdateInfo();
      if (httpResponse.getStatusLine().getStatusCode() == 200)//请求相应成功
      {
         HttpEntity entity = httpResponse.getEntity();
         String response = EntityUtils.toString(entity, "utf-8");
         JSONObject jsonObject=new JSONObject(response);

         Log.d("dsdsada", "getUpDateInfo: "+response);
         updateInfo.setVersion(jsonObject.getString("version"));
         updateInfo.setDescription(jsonObject.getString("description"));
         updateInfo.setUrl(jsonObject.getString("url"));
         this.updateInfo = updateInfo;
         return updateInfo;
      }
      updateInfo.setVersion("");
      updateInfo.setDescription("");
      updateInfo.setUrl("");
      this.updateInfo = updateInfo;

      return updateInfo;
   }

   /*
    *判断是否需要更新
    */
   public boolean isNeedUpdate(){
         String new_version = updateInfo.getVersion();
         String now_version="";
         try {
            PackageManager packageManager = context.getPackageManager();
            PackageInfo packageInfo = packageManager.getPackageInfo(
                  context.getPackageName(), 0);
            now_version= packageInfo.versionName;
         } catch (NameNotFoundException e) {
            e.printStackTrace();
         }
         if (new_version.equals(now_version)) {
            return false;
         } else {
            return true;
         }
   }

   /*
   *
   * 下载文件
   * */
   public void downLoadFile(final String url,final ProgressDialog pDialog,Handler h){
      progressDialog=pDialog;
      handler=h;
      new Thread() {
         public void run() {        
            HttpClient client = new DefaultHttpClient();
            HttpGet get = new HttpGet(url);
            HttpResponse response;
            try {
               response = client.execute(get);
               HttpEntity entity = response.getEntity();
               int length = (int) entity.getContentLength();
                                        progressDialog.setMax(length);
               InputStream is = entity.getContent();
               FileOutputStream fileOutputStream = null;
               if (is != null) {
                  File file = new File(
                        Environment.getExternalStorageDirectory(),
                        "Test.apk");
                  fileOutputStream = new FileOutputStream(file);
                        byte[] buf = new byte[10];   
                  int ch = -1;
                  int process = 0;
                  while ((ch = is.read(buf)) != -1) {       
                     fileOutputStream.write(buf, 0, ch);
                     process += ch;
                     progressDialog.setProgress(process);
                  }
               }
               fileOutputStream.flush();
               if (fileOutputStream != null) {
                  fileOutputStream.close();
               }
               down();
            } catch (ClientProtocolException e) {
               e.printStackTrace();
            } catch (IOException e) {
               e.printStackTrace();
            }
         }

      }.start();
   }


   //下载完毕,退出下载,进入安装APP步骤
   void down() {
      handler.post(new Runnable() {
         public void run() {
            progressDialog.cancel();
            update();
         }
      });
   }


   //安装最新版本APP
   void update() {
      Intent intent = new Intent(Intent.ACTION_VIEW);
      intent.setDataAndType(Uri.fromFile(new File(Environment
            .getExternalStorageDirectory(), "Test.apk")),
            "application/vnd.android.package-archive");
      context.startActivity(intent);
   }

   
}
 
github项目地址:https://github.com/huanghuaisong/android-php-appUpdate/
相关文章
|
1月前
|
Ubuntu 网络协议 Java
【Android平板编程】远程Ubuntu服务器code-server编程写代码
【Android平板编程】远程Ubuntu服务器code-server编程写代码
|
3月前
|
Android开发 开发者 iOS开发
APP开发后如何上架,上架Android应用市场前要准备什么
移动应用程序(APP)的开发已经成为现代企业和开发者的常见实践。然而,开发一个成功的APP只是第一步,将其上架到应用商店让用户下载和使用是实现其潜力的关键一步。
|
4天前
|
测试技术 Android开发
Android App获取不到pkgInfo信息问题原因
Android App获取不到pkgInfo信息问题原因
14 0
|
1月前
|
监控 安全 容灾
PHP服务器稳定性保障
确保PHP服务器稳定性,需关注以下几点:配置合适硬件及优化操作系统;使用最新稳定版PHP,及时更新安全补丁;编写高质量代码并优化性能;处理异常,记录日志以便监控;管理资源,使用性能监控工具;加强安全防护,如权限设置、防注入攻击;采用自动化部署和持续集成工具;定期备份数据,建立容灾机制。
19 0
|
1月前
|
前端开发 Android开发 iOS开发
应用研发平台EMAS使用 aliyun-react-native-push 库接入推送和辅助通道,推送都可以收到,但是在App切到后台或者杀掉进程之后就收不到推送了,是需要配置什么吗?
【2月更文挑战第31天】应用研发平台EMAS使用 aliyun-react-native-push 库接入推送和辅助通道,推送都可以收到,但是在App切到后台或者杀掉进程之后就收不到推送了,是需要配置什么吗?
32 2
|
1月前
|
设计模式 测试技术 数据库
基于Android的食堂点餐APP的设计与实现(论文+源码)_kaic
基于Android的食堂点餐APP的设计与实现(论文+源码)_kaic
|
1月前
|
Ubuntu 网络协议 Java
在Android平板上使用code-server公网远程Ubuntu服务器编程
在Android平板上使用code-server公网远程Ubuntu服务器编程
|
2月前
|
弹性计算 PHP
ECS续费问题之PHP运行环境到期如何解决
ECS续费是指对已创建的阿里云ECS实例执行的续费操作,以延长其服务期限;本合集旨在为用户提供ECS续费的操作步骤、策略选择和注意事项,确保业务的持续性和成本的优化。
ECS续费问题之PHP运行环境到期如何解决
|
2月前
|
安全 Java 数据挖掘
当 App 有了系统权限,真的可以为所欲为? Android Performance Systrace
当 App 有了系统权限,真的可以为所欲为? Android Performance Systrace 转载自: https://androidperformance.com/2023/05/14/bad-android-app-with-system-permissions/#/0-Dex-%E6%96%87%E4%BB%B6%E4%BF%A1%E6%81%AF
31 0