最近在学习android,领导想做一个测试利用局域网进行电视apk的自动升级程序,最新代码是放在本地局域网电脑上面的,在此写下代码调试过程中遇到的问题及解决方法;一开始用想用FTPClient和HttpURLConnection来读取本地文件(并且电脑有登录密码)没能成功(本人小菜),后来在网上看到看到利用基于SMB协议共享文件也可以对局域网文件进行读写,所以尝试利用此方法;不多说了,上代码先;
public class FtpClientUtil {
/* 下载中 */
private static final int DOWNLOAD = 1;
/* 下载结束 */
private static final int DOWNLOAD_FINISH = 2;
private static final int DOWNLOAD_DLG = 3;
// 保存解析的XML信息
private Context mContext;
static HashMap mHashMap;
/* 下载保存路径 */
private String mSavePath;
/* 记录进度条数量 */
private int progress;
/* 是否取消更新 */
private boolean cancelUpdate = false;
/* 更新进度条 */
private ProgressBar mProgress;
private Dialog mDownloadDialog;
public static final String vurl = "smb://liuyun:123456@192.168.1.8/testfile/version.xml";
public static final String appurl = "smb://liuyun:123456@192.168.1.8/testfile/SocketClient.apk";
public FtpClientUtil(Context context)
{
this.mContext = context;
}
private Handler mHandler = new Handler()
{
public void handleMessage(Message msg)
{
switch (msg.what)
{
// 正在下载
case DOWNLOAD:
// 设置进度条位置
mProgress.setProgress(progress);
break;
case DOWNLOAD_FINISH:
// 安装文件
installApk();
break;
case DOWNLOAD_DLG:
//显示版本更新及下载提示
Log.d("mHandler","---downloadApk");
showNoticeDialog();
break;
default:
break;
}
};
};
public void CheckUpdate(){
MutliThread m = new MutliThread();
Thread t1 = new Thread(m);
t1.start();
}
private class MutliThread implements Runnable{
public void run(){
if(isUpdate()){
mHandler.sendEmptyMessage(DOWNLOAD_DLG);
}
else
{
Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_LONG).show();
}
}
}
private boolean isUpdate() {
try {
//建立连接
SmbFile smbFile = new SmbFile(vurl);
// 建立smb文件输入流
InputStream in = new BufferedInputStream(new SmbFileInputStream(smbFile));
return parserUpdateVesion(in);//检测是否有新的版本
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
private boolean parserUpdateVesion(InputStream is)
{
// 获取当前软件版本
int versionCode = getVersionCode(mContext);
// InputStream inStream = ParseXmlService.class.getClassLoader().getResourceAsStream("version.xml");
// 解析XML文件,使用DOM方式进行解析
ParseXmlService service = new ParseXmlService();
try
{
mHashMap = service.parseXml(is);
} catch (Exception e)
{
e.printStackTrace();
}
if (null != mHashMap)
{
int serviceCode = Integer.valueOf(mHashMap.get("version"));
// 版本判断
Log.d("parserUpdateVesion","versionCode="+String.valueOf(serviceCode));
if (serviceCode > versionCode)
{
//showNoticeDialog();
try
{
is.close();
}
catch (IOException e)
{
e.printStackTrace();
}
return true;
}
}
return false;
}
private int getVersionCode(Context context)
{
int versionCode = 0;
try
{
// 获取软件版本号,对应AndroidManifest.xml下android:versionCode
versionCode = context.getPackageManager().getPackageInfo("com.example.update", 0).versionCode;
} catch (NameNotFoundException e)
{
e.printStackTrace();
}
return versionCode;
}
/**
* 显示软件更新对话框
*/
private void showNoticeDialog()
{
// 构造对话框
AlertDialog.Builder builder = new Builder(mContext);
builder.setTitle(R.string.soft_update_title);
builder.setMessage(R.string.soft_update_info);
// 更新
builder.setPositiveButton(R.string.soft_update_updatebtn, new OnClickListener()
{
@Override
public void onClick(DialogInterface dialog, int which)
{
dialog.dismiss();
// 显示下载对话框
showDownloadDialog();
}
});
// 稍后更新
builder.setNegativeButton(R.string.soft_update_later, new OnClickListener()
{
@Override
public void onClick(DialogInterface dialog, int which)
{
dialog.dismiss();
}
});
Dialog noticeDialog = builder.create();
noticeDialog.show();
}
/**
* 显示软件下载对话框
*/
private void showDownloadDialog()
{
// 构造软件下载对话框
AlertDialog.Builder builder = new Builder(mContext);
builder.setTitle(R.string.soft_updating);
// 给下载对话框增加进度条
final LayoutInflater inflater = LayoutInflater.from(mContext);
View v = inflater.inflate(R.layout.softupdate_progress, null);
mProgress = (ProgressBar) v.findViewById(R.id.update_progress);
builder.setView(v);
// 取消更新
builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener()
{
@Override
public void onClick(DialogInterface dialog, int which)
{
dialog.dismiss();
// 设置取消状态
cancelUpdate = true;
}
});
mDownloadDialog = builder.create();
mDownloadDialog.show();
// 下载文件
downloadApk();
}
/**
* 下载apk文件
*/
private void downloadApk()
{
// 启动新线程下载软件
new downloadApkThread().start();
}
/**
* 下载文件线程
*/
private class downloadApkThread extends Thread
{
@Override
public void run()
{
try
{
// 判断SD卡是否存在,并且是否具有读写权限
if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED))
{
// 获得存储卡的路径
String sdpath = Environment.getExternalStorageDirectory() + "/";
mSavePath = sdpath + "download";
Log.d("installApk",mSavePath);
//创建连接
SmbFile smbFile = new SmbFile(appurl);
int length = smbFile.getContentLength();// 得到文件的大小
InputStream in = new BufferedInputStream(new SmbFileInputStream(smbFile));
File file = new File(mSavePath);
// 判断文件目录是否存在
if (!file.exists())
{
file.mkdir();
}
File apkFile = new File(mSavePath, "SocketClient");
FileOutputStream fos = new FileOutputStream(apkFile);
int count = 0;
// 缓存
byte buf[] = new byte[1024];
// 写入到文件中
do
{
int numread = in.read(buf);
count += numread;
// 计算进度条位置
progress = (int) (((float) count / length) * 100);
// 更新进度
mHandler.sendEmptyMessage(DOWNLOAD);
if (numread <= 0)
{
// 下载完成
mHandler.sendEmptyMessage(DOWNLOAD_FINISH);
break;
}
// 写入文件
fos.write(buf, 0, numread);
} while (!cancelUpdate);// 点击取消就停止下载.
fos.close();
in.close();
}
} catch (MalformedURLException e)
{
e.printStackTrace();
} catch (IOException e)
{
e.printStackTrace();
}
// 取消下载对话框显示
mDownloadDialog.dismiss();
}
};
/**
* 安装APK文件
*/
private void installApk()
{
File apkfile = new File(mSavePath, mHashMap.get("name"));
if (!apkfile.exists())
{
return;
}
Log.d("installApk","---start");
// 通过Intent安装APK文件
Intent i = new Intent(Intent.ACTION_VIEW);
i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive");
mContext.startActivity(i);
}
}代码注释也挺详细,在此不再多说;
说说在此过程中遇到的一个问题,本来是想在
这个地方直接调用showNoticeDialog();但是运行时却有个异常,导致dialog没有显示出来:java.lang.RuntimeException:Can't create handler inside thread that has not called Looper.prepare();
后来查资料才得知在android子线程中不能直接控制UI组件;可以通过mHandler.sendMessage来实现;
经过调试成从本地局域网电脑获取共享的文件,并且升级成功;