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/
目录
打赏
0
0
0
0
101
分享
相关文章
【Azure App Service】为部署在App Service上的PHP应用开启JIT编译器
【Azure App Service】为部署在App Service上的PHP应用开启JIT编译器
圈子社交app前端+后端源码,uniapp社交兴趣圈子开发,框架php圈子小程序安装搭建
本文介绍了圈子社交APP的源码获取、分析与定制,PHP实现的圈子框架设计及代码编写,以及圈子小程序的安装搭建。涵盖环境配置、数据库设计、前后端开发与接口对接等内容,确保平台的安全性、性能和功能完整性。通过详细指导,帮助开发者快速搭建稳定可靠的圈子社交平台。
75 17
使用php开发圈子系统特点,如何获取圈子系统源码,社交圈子运营以及圈子系统的功能特点,圈子系统,允许二开,免费源码,APP 小程序 H5
开发一个圈子系统(也称为社交网络或社群系统)可以是一个复杂但非常有趣的项目。以下是一些关键特点和步骤,帮助你理解如何开发、获取源码以及运营一个圈子系统。
86 3
PHP与Ajax在Web开发中的交互技术。PHP作为服务器端脚本语言,处理数据和业务逻辑
本文深入探讨了PHP与Ajax在Web开发中的交互技术。PHP作为服务器端脚本语言,处理数据和业务逻辑;Ajax则通过异步请求实现页面无刷新更新。文中详细介绍了两者的工作原理、数据传输格式选择、具体实现方法及实际应用案例,如实时数据更新、表单验证与提交、动态加载内容等。同时,针对跨域问题、数据安全与性能优化提出了建议。总结指出,PHP与Ajax的结合能显著提升Web应用的效率和用户体验。
60 3
|
2月前
|
优化 PHP-FPM 参数配置:实现服务器性能提升
优化PHP-FPM的参数配置可以显著提高服务器的性能和稳定性。通过合理设置 `pm.max_children`、`pm.start_servers`、`pm.min_spare_servers`、`pm.max_spare_servers`和 `pm.max_requests`等参数,并结合监控和调优措施,可以有效应对高并发和负载波动,确保Web应用程序的高效运行。希望本文提供的优化建议和配置示例能够帮助您实现服务器性能的提升。
87 3
|
2月前
|
PHP作为一门流行的服务器端脚本语言,深入理解PHP的命名空间
【10月更文挑战第22天】PHP作为一门流行的服务器端脚本语言,自1995年诞生以来,已经发展了二十多年。在这二十多年的时间里,PHP经历了多次重大版本的更新,不断增加新特性和改进。其中,命名空间(Namespace)是PHP 5.3.0引入的一个重要特性,它为PHP的代码组织和重用提供了一种新的方式。本文将从三个部分深入理解PHP的命名空间:一是命名空间的基本概念和作用;二是PHP命名空间的使用方法;三是通过实例讲解命名空间的应用。
31 4
|
3月前
|
PHP作为广受青睐的服务器端脚本语言,在Web开发中占据重要地位。理解其垃圾回收机制有助于开发高效稳定的PHP应用。
【10月更文挑战第1天】PHP作为广受青睐的服务器端脚本语言,在Web开发中占据重要地位。其垃圾回收机制包括引用计数与循环垃圾回收,对提升应用性能和稳定性至关重要。本文通过具体案例分析,详细探讨PHP垃圾回收机制的工作原理,特别是如何解决循环引用问题。在PHP 8中,垃圾回收机制得到进一步优化,提高了效率和准确性。理解这些机制有助于开发高效稳定的PHP应用。
57 3
PHP中的设计模式:如何提高代码的可维护性与扩展性在软件开发领域,PHP 是一种广泛使用的服务器端脚本语言。随着项目规模的扩大和复杂性的增加,保持代码的可维护性和可扩展性变得越来越重要。本文将探讨 PHP 中的设计模式,并通过实例展示如何应用这些模式来提高代码质量。
设计模式是经过验证的解决软件设计问题的方法。它们不是具体的代码,而是一种编码和设计经验的总结。在PHP开发中,合理地使用设计模式可以显著提高代码的可维护性、复用性和扩展性。本文将介绍几种常见的设计模式,包括单例模式、工厂模式和观察者模式,并通过具体的例子展示如何在PHP项目中应用这些模式。
PHP:一种强大的服务器端脚本语言
【8月更文挑战第23天】PHP是一种被广泛使用的开源服务器端脚本语言,它的强大和灵活性使其成为开发Web应用程序的首选。本文将从PHP的基本概念、主要特性以及应用场景等方面进行深入探讨,以期帮助读者更好地理解和掌握PHP。
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等