編輯:關於Android編程
很多時候我們的軟件需要對處理後的數據進行存儲或再次訪問。Android為數據存儲提供了多種方式,分別有如下幾種:
文件
SharedPreferences
SQLite數據庫
內容提供者(Content provider)
網絡
Rom:隨機存儲器--斷電不能保持
rom可以保存
存儲
首先給大家介紹使用文件如何對數據進行存儲,Activity提供了openFileOutput()方法可以用於把數據輸出到文件中,具體的實現過程與在J2SE環境中保存數據到文件中是一樣的。
public class FileActivity extends Activity {
@Override public void onCreate(Bundle savedInstanceState) {
...
FileOutputStream outStream = this.openFileOutput("itcast.txt", Context.MODE_PRIVATE);
outStream.write("趙雅智博客".getBytes());
outStream.close();
}
}
openFileOutput()方法
第一參數用於指定文件名稱,不能包含路徑分隔符“/” ,如果文件不存在,Android 會自動創建它。
創建的文件保存在/data/data/<package name>/files目錄,
如: /data/data/cn.itcast.action/files/itcast.txt ,通過點擊Eclipse菜單“Window”-“Show View”-“Other”,在對話窗口中展開android文件夾,選擇下面的File Explorer視圖,然後在File Explorer視圖中展開/data/data/<package name>/files目錄就可以看到該文件。
openFileOutput()方法的第二參數用於指定操作模式,有四種模式,分別為:
Context.MODE_PRIVATE = 0
Context.MODE_APPEND = 32768
Context.MODE_WORLD_READABLE = 1
Context.MODE_WORLD_WRITEABLE = 2
Context.MODE_PRIVATE:為默認操作模式,代表該文件是私有數據,只能被應用本身訪問,在該模式下,寫入的內容會覆蓋原文件的內容,如果想把新寫入的內容追加到原文件中。可以使用Context.MODE_APPEND
Context.MODE_APPEND:模式會檢查文件是否存在,存在就往文件追加內容,否則就創建新文件。
Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE用來控制其他應用是否有權限讀寫該文件。
MODE_WORLD_READABLE:
表示當前文件可以被其他應用讀取;MODE_WORLD_WRITEABLE:表示當前文件可以被其他應用寫入。
如果希望文件被其他應用讀和寫,可以傳入:
openFileOutput("itcast.txt", Context.MODE_WORLD_READABLE + Context.MODE_WORLD_WRITEABLE);
android有一套自己的安全模型,當應用程序(.apk)在安裝時系統就會分配給他一個userid,當該應用要去訪問其他資源比如文件的時候,就需要userid匹配。
默認情況下,任何應用創建的文件,sharedpreferences,數據庫都應該是私有的(位於/data/data/<package name>/files),其他程序無法訪問。除非在創建時指定了Context.MODE_WORLD_READABLE或者Context.MODE_WORLD_WRITEABLE ,只有這樣其他程序才能正確訪問。
讀取
如果要打開存放在/data/data/<package name>/files目錄應用私有的文件,可以使用Activity提供openFileInput()方法。
FileInputStream inStream = this.getContext().openFileInput("itcast.txt");
Log.i("FileTest", readInStream(inStream));
readInStream()的方法
public static String readInStream(FileInputStream inStream){
try {
ByteArrayOutputStream outStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int length = -1;
while((length = inStream.read(buffer)) != -1 ){
outStream.write(buffer, 0, length);
}
outStream.close();
inStream.close();
return outStream.toString();
} catch (IOException e) {
Log.i("FileTest", e.getMessage());
}
return null;
}
或者直接使用文件的絕對路徑:
File file = new File("/data/data/cn.itcast.action/files/itcast.txt");
FileInputStream inStream = new FileInputStream(file);
Log.i("FileTest", readInStream(inStream));
注意:上面文件路徑中的“cn.itcast.action”為應用所在包,當你在編寫代碼時應替換為你自己應用使用的包。
對於私有文件只能被創建該文件的應用訪問,如果希望文件能被其他應用讀和寫,可以在創建文件時,指定Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE權限。
Activity還提供了getCacheDir()和getFilesDir()方法:
getCacheDir()方法用於獲取/data/data/<package name>/cache目錄
getFilesDir()方法用於獲取/data/data/<package name>/files目錄
實例:login
AndroidManifest.xml
[html] view plaincopyprint?
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.lession02_login"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="17" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.lession02_login.LoginActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.lession02_login"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="17" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.lession02_login.LoginActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
布局文件:
[html] view plaincopyprint?
<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" >
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_name" />
<EditText
android:id="@+id/edit_name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_pass"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_pass" />
<EditText
android:id="@+id/edit_pass"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<Button
android:id="@+id/button_login"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_login" />
<CheckBox
android:id="@+id/check_remember"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="100dp"
android:text="@string/login_remember" />
</LinearLayout>
</LinearLayout>
<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" >
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_name" />
<EditText
android:id="@+id/edit_name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_pass"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_pass" />
<EditText
android:id="@+id/edit_pass"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<Button
android:id="@+id/button_login"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_login" />
<CheckBox
android:id="@+id/check_remember"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="100dp"
android:text="@string/login_remember" />
</LinearLayout>
</LinearLayout>
strings.xml
[html] view plaincopyprint?
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">lession02_login</string>
<string name="action_settings">Settings</string>
<string name="hello_world">Hello world!</string>
<string name="login_name">用戶名</string>
<string name="login_pass">密碼</string>
<string name="login_login">登陸</string>
<string name="login_remember">記住密碼</string>
</resources>
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">lession02_login</string>
<string name="action_settings">Settings</string>
<string name="hello_world">Hello world!</string>
<string name="login_name">用戶名</string>
<string name="login_pass">密碼</string>
<string name="login_login">登陸</string>
<string name="login_remember">記住密碼</string>
</resources>
字節輸出流對象:StreamTools.java
[java] view plaincopyprint?
package com.example.lession02_login.util;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
public class StreamTools {
public static String getValue(FileInputStream fis) throws Exception {
//字節的輸出流對象
ByteArrayOutputStream stream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int length = -1;
while ((length = fis.read(buffer)) != -1) {
stream.write(buffer, 0, length);
}
stream.flush();
stream.close();
String value = stream.toString();
return value;
}
}
package com.example.lession02_login.util;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
public class StreamTools {
public static String getValue(FileInputStream fis) throws Exception {
//字節的輸出流對象
ByteArrayOutputStream stream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int length = -1;
while ((length = fis.read(buffer)) != -1) {
stream.write(buffer, 0, length);
}
stream.flush();
stream.close();
String value = stream.toString();
return value;
}
}
LoginService.java
[java] view plaincopyprint?
package com.example.lession02_login.service;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.HashMap;
import java.util.Map;
import com.example.lession02_login.util.StreamTools;
import android.content.Context;
public class LoginService {
// 上下文對象
public Context context;
//通過上下文對象傳過來
public LoginService(Context context) {
super();
this.context = context;
}
/**
* 往手機內存上存儲用戶名與密碼的操作
* @param name
* @param pass
* @param fileName
* @return
*/
public boolean saveToRom(String name, String pass, String fileName) {
// 上下文對象的api
try {
// 通過openFileOutput()方法獲取一個文件的輸出流對象
//MODE_PRIVATE私有模式,只能被應用本身訪問,
/* FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_PRIVATE);*/
//MODE_APPEND追加
FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_APPEND);
// 拼接用戶名密碼
String result = name + ":" + pass;
// 寫入
fos.write(result.getBytes());
fos.flush();
fos.close();
} catch (Exception e) {
e.printStackTrace();
return false;
}
return true;
}
/**
* 讀取操作
* @param name
* @param pass
* @param fileName
* @return
*/
public Map<String, String> readFile(String fileName) {
Map<String, String> map = null;// new HashMap<String, String>();
try {
FileInputStream fis = context.openFileInput(fileName);
String value = StreamTools.getValue(fis);
String values[] = value.split(":");
if (values.length > 0) {
map = new HashMap<String, String>();
map.put("name", values[0]);
map.put("pass", values[1]);
}
} catch (Exception e) {
e.printStackTrace();
}
return map;
}
}
package com.example.lession02_login.service;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.HashMap;
import java.util.Map;
import com.example.lession02_login.util.StreamTools;
import android.content.Context;
public class LoginService {
// 上下文對象
public Context context;
//通過上下文對象傳過來
public LoginService(Context context) {
super();
this.context = context;
}
/**
* 往手機內存上存儲用戶名與密碼的操作
* @param name
* @param pass
* @param fileName
* @return
*/
public boolean saveToRom(String name, String pass, String fileName) {
// 上下文對象的api
try {
// 通過openFileOutput()方法獲取一個文件的輸出流對象
//MODE_PRIVATE私有模式,只能被應用本身訪問,
/* FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_PRIVATE);*/
//MODE_APPEND追加
FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_APPEND);
// 拼接用戶名密碼
String result = name + ":" + pass;
// 寫入
fos.write(result.getBytes());
fos.flush();
fos.close();
} catch (Exception e) {
e.printStackTrace();
return false;
}
return true;
}
/**
* 讀取操作
* @param name
* @param pass
* @param fileName
* @return
*/
public Map<String, String> readFile(String fileName) {
Map<String, String> map = null;// new HashMap<String, String>();
try {
FileInputStream fis = context.openFileInput(fileName);
String value = StreamTools.getValue(fis);
String values[] = value.split(":");
if (values.length > 0) {
map = new HashMap<String, String>();
map.put("name", values[0]);
map.put("pass", values[1]);
}
} catch (Exception e) {
e.printStackTrace();
}
return map;
}
}
LoginActivity.java
[java] view plaincopyprint?
package com.example.lession02_login;
import java.util.Map;
import com.example.lession02_login.service.LoginService;
import android.os.Bundle;
import android.app.Activity;
import android.support.v4.widget.SimpleCursorAdapter.ViewBinder;
import android.text.TextUtils;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.Checkable;
import android.widget.EditText;
import android.widget.Toast;
public class LoginActivity extends Activity {
// 聲明獲取的用戶名和密碼
private EditText edit_name, edit_pass;
// 聲明登陸按對象
private Button btn_login;
// 聲明復選框組件對象
private Checkable box_remember;
// 聲明業務對象
private LoginService loginService;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 設置顯示視圖
setContentView(R.layout.activity_login);
// 實例化業務對像
loginService = new LoginService(this);
// 根據id名獲取相應組件對象
edit_name = (EditText) findViewById(R.id.edit_name);
edit_pass = (EditText) findViewById(R.id.edit_pass);
btn_login = (Button) findViewById(R.id.button_login);
box_remember = (Checkable) findViewById(R.id.check_remember);
// 給按鈕注冊事件
btn_login.setOnClickListener(new MyOnclickListener());
//回顯數據
/*//私有模式
Map<String,String> map = loginService.readFile("private.txt");*/
Map<String,String> map = loginService.readFile("append.txt");
if(map!=null){
edit_name.setText(map.get("name"));
edit_pass.setText(map.get("pass"));
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.login, menu);
return true;
}
// 內部類 有關點擊的處理對象
class MyOnclickListener implements View.OnClickListener {
@Override
public void onClick(View v) {
int id = v.getId();
switch (id) {
case R.id.button_login:
// 獲取用戶名密碼
String name = edit_name.getText().toString();
String pass = edit_pass.getText().toString();
// 判斷用戶名密碼是否為空
if (TextUtils.isEmpty(name)) {
Toast.makeText(LoginActivity.this, "用戶名不可 為空",
Toast.LENGTH_LONG).show();
return;
} else if (TextUtils.isEmpty(pass)) {
Toast.makeText(LoginActivity.this, " 密碼不可 為空",
Toast.LENGTH_LONG).show();
return;
} else {
if (box_remember.isChecked()) {
// 進行保存
// 調用業務對象的業務方法
LoginActivity.this.loginService.saveToRom(name, pass,
"private.txt");
Toast.makeText(LoginActivity.this, "保存用戶名和密碼",
Toast.LENGTH_LONG).show();
} else {
Toast.makeText(LoginActivity.this, "不保存用戶名和密碼",
Toast.LENGTH_LONG).show();
}
}
break;
default:
break;
}
}
}
}
package com.example.lession02_login;
import java.util.Map;
import com.example.lession02_login.service.LoginService;
import android.os.Bundle;
import android.app.Activity;
import android.support.v4.widget.SimpleCursorAdapter.ViewBinder;
import android.text.TextUtils;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.Checkable;
import android.widget.EditText;
import android.widget.Toast;
public class LoginActivity extends Activity {
// 聲明獲取的用戶名和密碼
private EditText edit_name, edit_pass;
// 聲明登陸按對象
private Button btn_login;
// 聲明復選框組件對象
private Checkable box_remember;
// 聲明業務對象
private LoginService loginService;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 設置顯示視圖
setContentView(R.layout.activity_login);
// 實例化業務對像
loginService = new LoginService(this);
// 根據id名獲取相應組件對象
edit_name = (EditText) findViewById(R.id.edit_name);
edit_pass = (EditText) findViewById(R.id.edit_pass);
btn_login = (Button) findViewById(R.id.button_login);
box_remember = (Checkable) findViewById(R.id.check_remember);
// 給按鈕注冊事件
btn_login.setOnClickListener(new MyOnclickListener());
//回顯數據
/*//私有模式
Map<String,String> map = loginService.readFile("private.txt");*/
Map<String,String> map = loginService.readFile("append.txt");
if(map!=null){
edit_name.setText(map.get("name"));
edit_pass.setText(map.get("pass"));
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.login, menu);
return true;
}
// 內部類 有關點擊的處理對象
class MyOnclickListener implements View.OnClickListener {
@Override
public void onClick(View v) {
int id = v.getId();
switch (id) {
case R.id.button_login:
// 獲取用戶名密碼
String name = edit_name.getText().toString();
String pass = edit_pass.getText().toString();
// 判斷用戶名密碼是否為空
if (TextUtils.isEmpty(name)) {
Toast.makeText(LoginActivity.this, "用戶名不可 為空",
Toast.LENGTH_LONG).show();
return;
} else if (TextUtils.isEmpty(pass)) {
Toast.makeText(LoginActivity.this, " 密碼不可 為空",
Toast.LENGTH_LONG).show();
return;
} else {
if (box_remember.isChecked()) {
// 進行保存
// 調用業務對象的業務方法
LoginActivity.this.loginService.saveToRom(name, pass,
"private.txt");
Toast.makeText(LoginActivity.this, "保存用戶名和密碼",
Toast.LENGTH_LONG).show();
} else {
Toast.makeText(LoginActivity.this, "不保存用戶名和密碼",
Toast.LENGTH_LONG).show();
}
}
break;
default:
break;
}
}
}
}
D:文件夾
-文件
R:可讀
W:可寫
X:可執行
三三一組
前三:當前用戶,中三:組,後三全局
在android中不支持gif格式的圖片,但是由於我希望在我的程序中剛剛加載的時候有一個小人在跑步表示正在加載。而這個小人跑就是一個gif圖片。也就是希望程序一啟動時就加
前言雅虎天氣的界面上滑的時候背景圖片會跟著移動,最重要的是背景圖片會根據手指上下移動的距離來進行不同程度的模糊,感覺甚為驚奇,畢竟大家都知道,在Android平台上進行模
Android的PopupWindow是個很有用的widget,利用它可以實現懸浮窗體的效果,比如實現一個懸浮的菜單,最常見的應用就是在視頻播放界面裡,做一個工具欄,用來
概述Healthd是android4.4之後提出來的一種中介模型,該模型向下監聽來自底層的電池事件,向上傳遞電池數據信息給Framework層的BatteryServic