Android简单的编写一个txt阅读器(没有处理字符编码),适用于新手学习


  对于下方评论的朋友说声道歉,由于只是2017随手写的博客,最后没时间使用博客,所以不知道有朋友关注。抱歉。

  提示:这个是2017年写的,读取文件流效率很低(可能导致程序奔溃),建议使用比较小(最好几十kb)的txt文件进行测试。

 

  使用工具: eclipse + Android adt

 

  SDK依赖:4.0.3(也就是android-15)

 

  包依赖: android-support-v7-appcompat (所以也需要导入这个包)

 

 

  本程序只是使用了一些基本的知识点编写了一个比较简单粗陋的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这两个文件,这两个文件实现的是一个文件浏览器,方法简单至极,是本人在看教程中学到的。到最后你要是想做界面漂亮的浏览器,也可以借鉴借鉴。

  源码链接:https://files.cnblogs.com/files/hmxin/ReadBookProject.zip

《黑幕下的人》

 


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM