今天寫了個輸入網址從網絡下載一個apk安裝包的小例子。把它記錄下來之後方便本身使用。java
運行效果圖:android
注意事項:安卓6.0系統以上要動態申請一些危險的權限,就是要主動彈出對話框問用戶願不肯意調取通訊錄名單,得到存儲權限,建立文件權限等等,安卓6.0之前是安裝時在Manifest寫入的權限默認就擁有了,不須要詢問客戶。今天我寫的程序是沒有主動詢問客戶的代碼(動態申請權限),因此在manifest中對uses-sdkVersion要注意,最大的版本要小於23,這樣6.0以上手機就知道幫助你用兼容模式運行程序,否則是運行不出來的。數組
Manifest代碼:網絡
<uses-sdk android:minSdkVersion="8" android:targetSdkVersion="22" /> <uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
XML代碼:ide
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:gravity="center" > <TextView android:id="@+id/tv_address" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="文件存放位置" /> <TextView android:id="@+id/tv_size" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="SD卡剩餘空間" /> <Button android:id="@+id/b_click" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="點擊下載文件" /> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <!-- android:indeterminate="false"flase纔是精確顯示,開始寫true不顯示 !--> <ProgressBar android:id="@+id/pb_scheduel" style="?android:attr/progressBarStyleHorizontal" android:layout_width="200dp" android:layout_height="50dp" android:layout_centerHorizontal="true" android:indeterminate="false" android:progress="0" android:max="100"/> <TextView android:id="@+id/tv_scheduel" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginLeft="5dp" android:layout_toRightOf="@+id/pb_scheduel" android:textColor="#000000" android:layout_centerHorizontal="true" android:text="0%" /> </RelativeLayout> </LinearLayout>
JAVA代碼:ui
public class DownLoadFile extends FragmentActivity { private Handler handler; private TextView tv_address, tv_size, tv_scheduel; private Button b_click; private ProgressBar pb_scheduel; private URL url; private Message msg; double totalLength = 0.0; String downloadFileDirectory; InputStream stream; OutputStream output; @Override protected void onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.download_file); tv_address = (TextView) findViewById(R.id.tv_address); tv_size = (TextView) findViewById(R.id.tv_size); b_click = (Button) findViewById(R.id.b_click); pb_scheduel = (ProgressBar) findViewById(R.id.pb_scheduel); tv_scheduel = (TextView) findViewById(R.id.tv_scheduel); pb_scheduel.setProgress(0); try { // url=new // URL("https://clfile.imooc.com/class/assist/119/4858629/Android%206.0%20%E5%8A%A8%E6%80%81%E6%9D%83%E9%99%90%E7%94%B3%E8%AF%B7.pdf"); url = new URL("https://www.imooc.com/mobile/mukewang.apk"); } catch (MalformedURLException e) { // TODO Auto-generated catch block e.printStackTrace(); } handler = new myHandler(); b_click.setOnClickListener(new OnClickListener() { @Override public void onClick(View arg0) { // TODO Auto-generated method stub Thread thread = new myThread(); thread.start(); } }); } class myHandler extends Handler { int Progress; @Override public void handleMessage(Message msg) { // TODO Auto-generated method stub super.handleMessage(msg); if (msg.what == 555) { Progress = msg.arg1;//handler建立在主線程中,如今的代碼運行在主線程中 pb_scheduel.setProgress(Progress);// 設置進度條進度 tv_scheduel.setText(Progress + "%");// 顯示 } else if (msg.what == 777) { tv_size.setText("SD卡剩餘空間:" + msg.getData().getString("sd"));// bundle的方式取數據 tv_address.setText("SD卡路徑:" + msg.getData().getString("a")); } } } class myThread extends Thread { @SuppressWarnings("unused") @Override public void run() { // TODO Auto-generated method stub super.run(); // 判斷存儲卡是否存在是否可讀可寫 if (Environment.MEDIA_MOUNTED.equals(Environment .getExternalStorageState())) { downloadFileDirectory = Environment .getExternalStorageDirectory() + File.separator + "downSB" + File.separator + "mukewang.apk";// 下載的文件路徑 File file = new File(downloadFileDirectory); if (!file.exists())// 判斷文件目錄是否存在 file.getParentFile().mkdirs();// File.getParentFile()獲取父目錄不包括文件名,不存在則建立 try { file.createNewFile();// 建立what.apk文件 } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } URLConnection connect = null; try { if (url != null) connect = url.openConnection();// 返回一個connection else { System.out.println("你提供那個的網址不能下載。"); return; } totalLength = connect.getContentLength();// 下載的文件總長度 msg = handler.obtainMessage(777); Bundle bundle = new Bundle();// bundle方式傳遞數據 bundle.putString("a", Environment .getExternalStorageDirectory().getPath()); bundle.putString("sd", SDStorage .GetSDAvailableSize(getApplicationContext())); msg.setData(bundle);// bundle和msg的聯繫 handler.sendMessage(msg); if (totalLength < SDStorage .GetSDAvailableSize2(getApplicationContext())) {// 錯誤點看錯了,long和double能夠比較 int length = 0; stream = connect.getInputStream();// 返回輸入流 byte[] b = new byte[1024]; output = new FileOutputStream(file);// 建立輸出到文件的輸出流 int downloadSize = 0;// 已經下載的文件長度 while ((length = stream.read(b)) != -1) {// 每次讀取到數組b中返回實際讀取的字節數 output.write(b,0,length);// 從b數組中拿到往文件中寫入 downloadSize += length; msg = handler.obtainMessage(555);// 555標誌消息 msg.arg1 = (int) (downloadSize * 100 / totalLength);// 求出如今的進度 handler.sendMessage(msg);// 發送到主線程更新進度條 } stream.close(); output.close(); } else { // Toast.makeText(DownLoadFile.this, "手機存儲空間不夠", // 1).show();//子線程中不顯示toast,不應寫 } } catch (IOException e1) { // TODO Auto-generated catch block e1.printStackTrace(); } } else { // Toast.makeText(DownLoadFile.this, "您的存儲設備不容許使用", // Toast.LENGTH_LONG).show(); } } } }
SD卡剩餘空間的計算代碼:this
public class SDStorage { /** * @獲取手機SD卡里剩餘可用空間大小 * 存儲區域被劃分爲多個塊,有些塊不能使用 */ @SuppressLint("NewApi") public static String GetSDAvailableSize(Context context){ long blockSize=0;//單個存儲分塊大小 long availableBlocks;//可用的存儲分塊數 File file=Environment.getExternalStorageDirectory();//獲取外部存儲設備的抽象形式,SD卡 StatFs statFs=new StatFs(file.getPath());//file.getPath()獲取文件的路徑,statFs獲取存儲設備的使用狀況的抽象 if(Build.VERSION.SDK_INT>Build.VERSION_CODES.JELLY_BEAN_MR2 ){//安卓API的版本號>18,進build文件看每一個版本的常量名 blockSize=statFs.getBlockSizeLong();//獲取存儲塊的大小 availableBlocks=statFs.getAvailableBlocksLong();//獲取能使用的存儲快數 }else{ blockSize=statFs.getBlockSize(); availableBlocks=statFs.getAvailableBlocks(); } return Formatter.formatFileSize(context, availableBlocks*blockSize);//long字節轉化爲字符串GB字節 } @SuppressLint("NewApi") public static long GetSDAvailableSize2(Context context){ long blockSize=0;//單個存儲分塊大小 long availableBlocks;//可用的存儲分塊數 File file=Environment.getExternalStorageDirectory();//獲取外部存儲設備的抽象形式,SD卡 StatFs statFs=new StatFs(file.getPath());//file.getPath()獲取文件的路徑,statFs獲取存儲設備的使用狀況的抽象 if(Build.VERSION.SDK_INT>Build.VERSION_CODES.JELLY_BEAN_MR2 ){//安卓API的版本號>18,進build文件看每一個版本的常量名 blockSize=statFs.getBlockSizeLong();//獲取存儲塊的大小 availableBlocks=statFs.getAvailableBlocksLong();//獲取能使用的存儲快數 }else{ blockSize=statFs.getBlockSize(); availableBlocks=statFs.getAvailableBlocks(); } return blockSize*availableBlocks; } }