Android簡單的編寫一個txt閱讀器(沒有處理字元編碼),適用於新手學習

来源:http://www.cnblogs.com/hmxin/archive/2017/09/01/7461333.html
-Advertisement-
Play Games

Android簡單的編寫一個txt閱讀器(沒有處理字元編碼),適用於新手學習   本程式只是使用了一些基本的知識點編寫了一個比較簡單粗陋的txt文本閱讀器,效率不高,只適合新手練習。所以大神勿噴。   其實想到編寫這種程式源自本人之前喜歡看小說,而很多小說更新太慢,所以本人就只能找一個完本的.tx... ...


  本程式只是使用了一些基本的知識點編寫了一個比較簡單粗陋的txt文本閱讀器,效率不高,只適合新手練習。所以大神勿噴。

  其實想到編寫這種程式源自本人之前喜歡看小說,而很多小說更新太慢,所以本人就只能找一個完本的.txt小說下載下來,有沒有網路都可以看,當然現在不看那玩意了。

  廢話就不說了,程式中使用到的有4個類,5個xml文件,其中3個佈局文件、String.xml、AndroidManifest.xml。

  先看效果圖吧(雖然很醜):

     

  下麵就上代碼吧,本人都註釋好了,由於本人技術還不行,註釋不對的,望指正,謝謝>~<

  MainActivity.java:

 

package cn.hmxin.readbookproject;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;

import android.content.Intent;
import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.ActionBarActivity;
import android.util.Log;
import android.view.ContextMenu;
import android.view.ContextMenu.ContextMenuInfo;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ListView;
import android.widget.Toast;

public class MainActivity extends ActionBarActivity {

    private ListView bookList = null ;//顯示圖書(txt)列表組件
    private Button btnAdd = null ;//追加書按鈕組件
    private ArrayAdapter<String> adapter = null ;//適配器
    private List<String> data = null ; //用於存儲數據,載入到適配器中
    private List<String> pathData = null ;//記錄路徑
    private static final int REMOVE_BOOK = Menu.FIRST;//移除書上下文菜單標識
    //程式中要使用到的路徑
    public static final String MYPATH = Environment.getExternalStorageDirectory().getPath() + File.separator + "myBook";
    private int longClickPosition = 0 ;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        bookList = (ListView)findViewById(R.id.book_list);//綁定組件
        btnAdd = (Button)findViewById(R.id.btn_add);//綁定組件
        
        createOrRead();//創建或讀取文件方法,順帶顯示數據
        
        registerForContextMenu(bookList);//為txt列表組件註冊上下文菜單
        
        btnAdd.setOnClickListener(new OnClick());//添加圖書按鈕的單擊監聽處理
        bookList.setOnItemClickListener(new OnItemClick());//圖書列表的子項單擊監聽處理
        bookList.setOnItemLongClickListener(new OnItemLongClick());//圖書列表的子項單擊監聽處理
        
    }
    
    //文件的創建和讀取
    public void createOrRead(){
        File dirFile = new File(MYPATH);//聲明並實例化一個File類,路徑為MYPATH(路徑詳細看上面)
        if(!dirFile.exists()){//如果文件夾不存在
            dirFile.mkdirs();//創建文件夾
        }
        File file = new File(dirFile,"bookPath.txt");//聲明並實例化一個File類,父路徑為dirFile,文件名為bookPath.txt
        if(!file.exists()){//如果文件不存在
            try {
                file.createNewFile();//創建文件
            } catch (IOException e) {
                e.printStackTrace();
            }
        }else{//如果文件已經存在
            try {
                FileReader fr = new FileReader(file);//聲明並實例化FileReader類,路徑為file
                BufferedReader br = new BufferedReader(fr);//聲明並實例化BufferedFile類,緩衝讀取數據
                data = new ArrayList<String>();//實例化List類,用於添加數據,載入在適配器中
                pathData = new ArrayList<String>();//記錄路徑
                String line = "" ;//用於存儲讀取到的數據
                
                while((line = br.readLine()) != null){//如果還有下一行數據
                    int sub = line.lastIndexOf(File.separator);//截取文件名使用
                    String strName = line.substring(sub+1, line.length());//截取到文件名
                    data.add(strName);//添加文件名
                    pathData.add(line);//添加整個文件路徑
                }
                br.close();//關閉BufferedReader類
                fr.close();//關閉FileReader類
                //載入數據在適配器
                adapter = new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, data);
                bookList.setAdapter(adapter);//設置適配器
            } catch (FileNotFoundException e) {//拋出異常
                e.printStackTrace();
            } catch (IOException e) {//拋出異常
                e.printStackTrace();
            }
        }
    }
    
    //用來監聽圖書列表的選項單擊事件
    private class OnItemClick implements AdapterView.OnItemClickListener{
        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
            Intent intent = new Intent(MainActivity.this, ReadBookActivity.class);//用於跳轉Activity使用,此處跳轉到讀取圖書界面
            intent.putExtra("txtFilePath", pathData.get(position));//傳送要打開的txt文件路徑
            startActivity(intent);//跳轉界面
        }
        
    }
    //用來監聽圖書列表的選項長按事件
    private class OnItemLongClick implements AdapterView.OnItemLongClickListener{
        @Override
        public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) {
            longClickPosition = position;//記錄長按了列表哪個子項
            return false;
        }
    }
    
    //用來監聽添加圖書按鈕的單擊事件
    private class OnClick implements View.OnClickListener{
        @Override
        public void onClick(View v) {
            Intent intent = new Intent(MainActivity.this, FileBrowserActivity.class);//跳轉到文件瀏覽選擇界面
            startActivity(intent);//跳轉界面
        }
        
    }
    
    //創建上下文菜單
    @Override
    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) {
        menu.add(0, REMOVE_BOOK, 0, "移除讀書");//添加菜單子項
        super.onCreateContextMenu(menu, v, menuInfo);
    }
    
    @Override
    public boolean onContextItemSelected(MenuItem item) {
        if(item.getItemId() == REMOVE_BOOK){//移除圖書處理
            removeBook();
        }
        return super.onContextItemSelected(item);
    }
    
    //移除圖書處理
    public void removeBook(){
        try {
            data.remove(longClickPosition);//移除該項
            pathData.remove(longClickPosition);//移除該項
            FileWriter fw = new FileWriter(new File(MYPATH + File.separator + "bookPath.txt"));//設置讀取文件
            BufferedWriter bw = new BufferedWriter(fw);//用BufferedWriter類寫入數據
            for(String str : pathData){//遍歷pathData對象
                bw.write(str);//寫入數據
                bw.newLine();//新的一行
            }
            bw.close();
            //載入數據在適配器
            adapter = new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, data);
            bookList.setAdapter(adapter);//設置適配器
        } catch (IOException e) {//拋出異常
            e.printStackTrace();
        }
    }

}

 

activity_mai.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="cn.hmxin.readbookproject.MainActivity" >

    <LinearLayout 
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">
        
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="18dip"
        android:textStyle="bold"
        android:layout_weight="1"
        android:text="@string/list_title" />
    
    <Button 
        android:id="@+id/btn_add"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/add"/>
    </LinearLayout>
    
    <ListView 
        android:id="@+id/book_list"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        
    </ListView>

</LinearLayout>

SubFile.java

package cn.hmxin.readbookproject;
import java.io.File;

public class SubFile {
    
    private File file =null ;//封裝一個File類,用於接收傳進構造方法中的File類
    
    public SubFile(File file) {//構造方法
        this.file = file ;//本類中的file賦值為傳進來的file
    }
    
    public File getFile() {//getter方法,返回File類
        return this.file;//返回本類中的file對象
    }
    
    @Override
    public String toString() {
        String str = null; //聲明一個字元串用來返回值
        if(file.isDirectory()) {//是文件夾
            str = "[文件夾]" + file.getName();
        }else {//是文件
            if(file.getName().lastIndexOf(".txt")>-1) {//這裡是過濾掉別的文件,做文件瀏覽器,可以去掉
                str = "[文件]" + file.getName();//賦值"[文件]"+文件名字
            }
        }
        return str;//返回該字元串
    }
}

 

FileBrowserActivity.java

package cn.hmxin.readbookproject;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.List;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Environment;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Toast;

public class FileBrowserActivity extends Activity {
    
    private ListView fileList = null ;//顯示文件的列表
    private ArrayAdapter adapter = null ;//適配器    
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.file_browser_layout);
        setTitle("選擇您的文件名");//更改標題
        fileList = (ListView)findViewById(R.id.file_list);//綁定組件
        openFile(adapter, fileList);//打開文件和配置適配器,顯示在列表上
        
        fileList.setOnItemClickListener(new OnItemClick());//文件列表的子項單擊監聽處理
        
    }
    
    /**
     * 此方法用於裝載數據
     * @param adapter ArrayAdapter適配器
     * @param list 要顯示的ListView
     */
    public void openFile(ArrayAdapter adapter, ListView list){
        List<SubFile> ndata = new ArrayList<SubFile>();//用於存放文件名,載入到適配器中
        String strPath = getIntent().getStringExtra("filename");//獲取上個界面傳來的值
        if(strPath == null){//沒有有數據
            strPath = Environment.getExternalStorageDirectory().getPath();//根目錄
        }
        File pathFile = new File(strPath);//要顯示的目錄
        if(pathFile != null){//有這個目錄
            File[] files = pathFile.listFiles();//獲取目錄下的所有文件夾與文件
            for(File file : files){//全部遍歷
                if(new SubFile(file).toString()!=null){//如果不是篩選掉的文件
                    ndata.add(new SubFile(file));//添加到mdata對象中
                }
            }
            //配置適配器
            adapter = new ArrayAdapter(FileBrowserActivity.this, android.R.layout.simple_list_item_1, ndata);
            list.setAdapter(adapter);//設置列表的適配去
        }else{//空文件處理
            Toast.makeText(FileBrowserActivity.this , "查找文件為空!", Toast.LENGTH_SHORT).show();
        }
    }
    
    //文件列表子項的單擊監聽處理
    private class OnItemClick implements AdapterView.OnItemClickListener{
        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
            SubFile subFile = (SubFile) parent.getAdapter().getItem(position);//獲取子項存儲的SubFile類,該類可以獲取到子項所在路徑
            String filename = subFile.getFile().getPath();//filename賦值為subFile對象中的路徑
            Intent intent = null;//聲明Intent類,用於跳轉界面
            if(subFile.getFile().isDirectory()){//如果路徑為文件夾
                intent = new Intent(FileBrowserActivity.this, FileBrowserActivity.class);//還是跳轉到改Activity
                intent.putExtra("filename", filename);//傳入路徑
            }else{//文件的話
                intent = new Intent(FileBrowserActivity.this, MainActivity.class);//跳轉到主界面
                writeData(filename);//調用writeData方法,該方法用於寫入數據
            }
            startActivity(intent);//跳轉界面
            finish();//清除界面
        }
        
    }
    
    /**
     * 該方法用於寫入數據
     * @param str 要寫入的值
     */
    
    public void writeData(String str){
            File file = new File(MainActivity.MYPATH+File.separator+"bookPath.txt");//要操作的文件
            try {
                BufferedWriter bw = new BufferedWriter(new FileWriter(file,true));//用BufferedWriter類追加寫入數據
                bw.append(str);//追加數據
                bw.newLine();//新的一行
                bw.close();//關閉BufferedWrite類
            } catch (IOException e) {//拋出異常
                e.printStackTrace();
            }
    }
    
}

file_browser_layout.xml

<?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:layout_margin="10dp"
    android:orientation="vertical" >
    
     <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="18dip"
        android:textStyle="bold"
        android:text="@string/file_select_title" />
    
    <ListView 
        android:id="@+id/file_list"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
    </ListView>

</LinearLayout>

ReadBookActivity.java

package cn.hmxin.readbookproject;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;

import android.app.Activity;
import android.os.Bundle;
import android.text.method.ScrollingMovementMethod;
import android.util.Log;
import android.widget.TextView;
import android.widget.Toast;

public class ReadBookActivity extends Activity {
    
    private TextView bookContent = null ;//顯示文本的組件
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.read_book_layout);
        
        bookContent = (TextView)findViewById(R.id.book_content);//綁定文本組件
        bookContent.setMovementMethod(ScrollingMovementMethod.getInstance());//給文本組件加入滾動條
        
        String txtFilePath = getIntent().getStringExtra("txtFilePath");//獲取傳進來的文本路徑
        if(txtFilePath != null){//如果有傳進數據
            int index = txtFilePath.lastIndexOf(File.separator);//用來截取文本的名字的第一個參數
            String name = txtFilePath.substring(index+1, txtFilePath.length());//截取文本名字
            setTitle(name);//設置標題為文本名字
            try {
                FileInputStream fr = new FileInputStream(txtFilePath);//文件輸出流
                BufferedReader br = new BufferedReader(new InputStreamReader(fr, "utf-8"));//緩衝讀取文件數據
                String line = "" ;//記錄每一行數據
                String content = "" ;
                while((line = br.readLine()) != null){//如果還有下一行數據
                    content += line + "\n" ;
                }
                bookContent.setText(content);;//追加顯示數據
                br.close();//關閉文件輸出流
                fr.close();//關閉緩衝區
            } catch (IOException e) {//拋出異常
                Toast.makeText(ReadBookActivity.this, "沒有此文件!", Toast.LENGTH_SHORT).show();//提示異常
                finish();//直接關閉界面
            }
        }
    }
}

read_book_layout.xml

<?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" >
    <TextView 
        android:id="@+id/book_content"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:textSize="14dip"/>
</LinearLayout>

strings.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <string name="app_name">圖書閱讀程式(txt)</string>
    <string name="list_title">圖書列表:</string>
    <string name="add">添加圖書</string>
    <string name="file_select_title">請選擇.txt文件:</string>

</resources>

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.hmxin.readbookproject"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="21" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".ReadBookActivity">
            
        </activity>
        <activity android:name=".FileBrowserActivity">
            
        </activity>
    </application>

</manifest>

 

  最後想說的是,新手們可以著重看SubFile.java和FileBrowserActivity.java這兩個文件,這兩個文件實現的是一個文件瀏覽器,方法簡單至極,是本人在看教程中學到的。到最後你要是想做界面漂亮的瀏覽器,也可以借鑒借鑒。要源碼的話,可以聯繫下本人。

《黑幕下的人》

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 之前一直聽一些大牛談論什麼架構、框架,給我的直觀感覺就是一樣的,但是看瞭如下博文,我恍然大悟,原來是兩碼事。 人們對軟體架構存在非常多的誤解,其中一個最為普遍的誤解就是:將架構(Architecture)和框架(Framework)混為一談。其實很簡單,一句話:框架是軟體,架構不是軟體。框架落腳在“ ...
  • 當您選擇將應用程式構建成為一組微服務時,您需要決定應用程式客戶端將如何與微服務進行交互。單體應用程式只有一組端點(endpoint),通常使用複製(replicated)結合負載均衡來分配流量。然而,在微服務架構中,每個微服務都暴露一組通常比較細顆粒的端點。在本文中,我們將研究如何改進客戶端通信,並... ...
  • 1、下載 Creating a Download Task 2、上傳 Creating an Upload Task 3、批量上傳 Creating an Upload Task for a Multi-Part Request, with Progress 4、數據任務 Creating a Da ...
  • 有不清楚的地方歡迎各位朋友們留言 ...
  • 核心代碼 控制點的簡單計算 將繪製的曲線添加到UIScrollView上 ...
  • 蘋果電腦獲取Android Studio的發佈版SHA1和開發版SHA1 ...
  • 一、 安裝插件 cordova plugin add ionic-plugin-keyboard 二、 軟鍵盤顯示監聽 window.addEventListener('native.keyboardshow', function (e) { // todo 進行鍵盤可用時操作 //e.keyboa ...
  • 一、當Activity啟動後EditText直接獲取了焦點,此時軟鍵盤會自動彈出,這種體驗並不是很好,因此要做的Activity啟動不自動彈出軟鍵盤,只需要在Manifest中對應的Activity添加以下這句話就可以了: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...