智能文章系统实战-Android文章客户端(8)

发布于:2018-6-21 16:53 作者:admin 浏览:3031 

1. 列表模板XML

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    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="com.demo.article.MainActivity">

    <ListView
        android:id="@+id/listview"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>
</RelativeLayout>

 

 

2.列表项模板

<?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">
    <TextView
        android:id="@+id/title"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
    />
</LinearLayout>

 

3.文章类程序

public class Item {
    private int id;
	private String title;
    private String url;

    public Item(int id,String title, String url) {
        this.id = id;
        this.title = title;
		this.url = url;
    }

    public int getId() {
        return id;
    }

    public String getTitle() {
        return title;
    }
	
	public String getUrl() {
        return url;
    }
}

 

 

4.数据适配器

public class ItemAdapter extends ArrayAdapter<Item> {
    private int layoutId;

    public ItemAdapter(Context context, int layoutId, List<Item> list) {
        super(context, layoutId, list);
        this.layoutId = layoutId;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        
        View view;
        ViewHolder viewHolder;
        Item item = getItem(position);
        if (convertView == null) {
            view = LayoutInflater.from(getContext()).inflate(layoutId, parent, false);
            viewHolder = new ViewHolder();
            viewHolder.textView = (TextView) view.findViewById(R.id.title);
            view.setTag(viewHolder);
        } else {
            view = convertView;
            viewHolder = (ViewHolder) view.getTag();
        }
        viewHolder.textView.setText(item.getTitle());

        return view;
    }

    class ViewHolder {
        TextView textView;
    }
}

 

 

3.列表程序

public class MainActivity extends Activity {    
	private List<Item> list = new ArrayList<>();
	private ListView listView 
	private ItemAdapter<Item> itemAdapter;
    private ProgressDialog dialog;	
	
	@Override    
	protected void onCreate(Bundle savedInstanceState) {    
		super.onCreate(savedInstanceState);    
		setContentView(R.layout.activity_main);
		
		/*
		//初始化数据
		initList();
		
		//用自定义的数据适配器和自定义的布局显示列表
		ItemAdapter itemAdapter = new ItemAdapter(MainActivity.this, R.layout.item, list);    
		ListView listView = (ListView) findViewById(R.id.listview);    
		listView.setAdapter(itemAdapter);
		*/
		
		
		//进度条
		dialog = new ProgressDialog(MainActivity.this);
		
		//列表显示控件listView
		listView = (ListView) findViewById(R.id.listview); 
		
		//请求服务器获取数据,解析数据,加载列表
        new dataAsyncTask().execute("http://news.demo.com/app.php");
		
		

		//短暂点击跳到详情页
		listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
			
			@Override
			public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
				
				//详情页
				Item item = list.get(position);
				Intent intent = new Intent(MainActivity.this, ContentActivity.class);
				intent.putExtra("title",item.getTitle());
                intent.putExtra("url",item.getUrl());
                startActivity(intent);
				
			}
		});
		
		//长按Toast显示标题
		listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
			@Override
			public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) {
				Item item = list.get(position);
				Toast.makeText(MainActivity.this, "long click " + item.getTitle(), Toast.LENGTH_SHORT).show();

				return true;
			}
		});
	}

	//仅仅用来做测试,实际是对接的服务器接口的数据
    private void initList() {
        for (int i = 0; i < 100; i++) {
            //public Item(int id,String title, String url) 
			Item item = new Item(i,"T" + i, "#" + i);
            list.add(item);
        }
    }
	
	
	//解析JSON数据
    private List<Item> parseJSON(jsonStr)
        JSONObject obj = new JSONObject(jsonStr); //最外层的JSONObject对象
		String errCode = obj.getString("errCode");//通过errCode字段获取其所包含的字符串
		String errMsg =  obj.getString("errMsg"); //通过errMsg字段获取其所包含的字符串
		
        JSONArray array = obj.getJSONArray("data");
        for(int i = 0 ; i<array.length();i++){
            JSONObject jsonObj = array.getJSONObject(i); //索引值,获取数组中包含的值
			
			//System.out.println(jsonObj.getString("title"));
			
			//把即系的数据加载到列表里
			Item item = new Item(jsonObj.getInt("id"),jsonObj.getString("title"),jsonObj.getString("url"));
            list.add(item);

        }
		return list;
    }
	
	
	//请求服务器数据
    public String httpData(String path)
    {
        String result = "";
        HttpClient httpClient = new DefaultHttpClient();
        try
        {
            HttpPost httpPost = new HttpPost(path);
            HttpResponse httpResponse = httpClient.execute(httpPost);
            if(httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK)
            {
                HttpEntity httpEntity = httpResponse.getEntity();
                if(httpEntity != null)
                {
                    result = EntityUtils.toString(httpEntity, "utf-8");
                }
            }
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        finally
        {
            httpClient.getConnectionManager().shutdown();
        }
        
        return result;
    }
	
	
	
	public class dataAsyncTask extends AsyncTask<String, Void, List<String>>
    {
        @Override
        protected void onPreExecute()
        {
			//预加载
            dialog.setTitle("提示信息");
            dialog.setMessage("loading......");
            dialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);
            dialog.setCancelable(false);
			dialog.show();
        }
		
		
        @Override
        protected List<Item> doInBackground(String... params)
        {
			//后台解析数据
            List<Item> tempList = new ArrayList<Item>();
            String jsonStr = httpData(params[0]);
            //解析服务器端的json数据
            tempList = parseJSON(jsonStr);
			return tempList;
        }
		
        @Override
        protected void onPostExecute(List<Item> list)
        {
			//显示列表数据
			itemAdapter = new ItemAdapter(MainActivity.this, R.layout.item, list);    
			listView.setAdapter(itemAdapter);			
            dialog.dismiss();
        }
    }
	
	
	
	
		
}   

 

 

4.详细页模板

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/color_White">
    <WebView
        android:id="@+id/web_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
    />
		
</android.support.design.widget.CoordinatorLayout>

 

 

5.详细页程序

public class ContentActivity extends Activity {
    private WebView webView;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_content);

        

        webView = (WebView)findViewById(R.id.web_view);
        webView.getSettings().setJavaScriptEnabled(true);
        webView.setWebViewClient(new WebViewClient());

        String url = getIntent().getStringExtra("url");
        //String title = getIntent().getStringExtra("title");
        webView.loadUrl(url);

    }
}

 

 6.  临时总结

暂时没有实现 下拉刷新,上拉加载更多数据,目前仅仅加载第1页的数据。后期优化。

adapter.setData(list);//刷新列表数据

adapter.notifyDataSetChanged();

标签: Android

0

Android二维码识别 开源项目ZXing的编译

发布于:2015-3-4 12:31 作者:admin 浏览:1371 分类:Android

[该文章已设置加密,请点击标题输入密码访问]

标签: Android zxing jar

0

android项目红色感叹号

发布于:2015-3-4 11:30 作者:admin 浏览:1857 分类:Android

标签: Android 错误

0

Android数据库SQLite的使用示例

发布于:2014-7-22 9:55 作者:admin 浏览:2195 分类:Android
Android 提供了三种数据存储方式,第一种是文件存储;第二种是SharedPreferences存储;第三种就是数据库SQLiteDatabase存储。文件存储我就不用多说了,而SharedPreferences可以存取简单的数据(int,double,float.etc),它经常用于数据缓存,因为它读取存储简单。详细可以参见本系列。Android高手进阶教程(七)之----Android 中Preferences的使用!
今天我们将讲一下SQLiteDatabase的使用。而掌握SqliteDatabase,将会我们接下来掌握ContentProvider打下良好的基石。为了让大家更好的掌握,我们手把手完成该节的Demo。

第一步:新建一个Android工程,命名为SQLiteDatabaseDemo.
a1.jpg 

第二步:创建一个新的类BooksDB.java这个类要继承于android.database.sqlite.SQLiteOpenHelper抽象类,我们要实现其中两个方法:onCreate(),onUpdate.具体代码如下:
package com.android.tutor; 
import android.content.ContentValues; 
import android.content.Context; 
import android.database.Cursor; 
import android.database.sqlite.SQLiteDatabase; 
import android.database.sqlite.SQLiteOpenHelper; 
public class BooksDB extends SQLiteOpenHelper { 
private final static String DATABASE_NAME = "BOOKS.db"; 
private final static int DATABASE_VERSION = 1; 
private final static String TABLE_NAME = "books_table"; 
public final static String BOOK_ID = "book_id"; 
public final static String BOOK_NAME = "book_name"; 
public final static String BOOK_AUTHOR = "book_author"; 

public BooksDB(Context context) { 
// TODO Auto-generated constructor stub 
super(context, DATABASE_NAME, null, DATABASE_VERSION); 
} 
//创建table 
@Override 
public void onCreate(SQLiteDatabase db) { 
String sql = "CREATE TABLE " + TABLE_NAME + " (" + BOOK_ID 
+ " INTEGER primary key autoincrement, " + BOOK_NAME + " text, "+ BOOK_AUTHOR +" text);"; 
db.execSQL(sql); 
} 
@Override 
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { 
String sql = "DROP TABLE IF EXISTS " + TABLE_NAME; 
db.execSQL(sql); 
onCreate(db); 
} 

public Cursor select() { 
SQLiteDatabase db = this.getReadableDatabase(); 
Cursor cursor = db 
.query(TABLE_NAME, null, null, null, null, null, null); 
return cursor; 
} 
//增加操作 
public long insert(String bookname,String author) 
{ 
SQLiteDatabase db = this.getWritableDatabase(); 
/* ContentValues */ 
ContentValues cv = new ContentValues(); 
cv.put(BOOK_NAME, bookname); 
cv.put(BOOK_AUTHOR, author); 
long row = db.insert(TABLE_NAME, null, cv); 
return row; 
} 
//删除操作 
public void delete(int id) 
{ 
SQLiteDatabase db = this.getWritableDatabase(); 
String where = BOOK_ID + " = ?"; 
String[] whereValue ={ Integer.toString(id) }; 
db.delete(TABLE_NAME, where, whereValue); 
} 
//修改操作 
public void update(int id, String bookname,String author) 
{ 
SQLiteDatabase db = this.getWritableDatabase(); 
String where = BOOK_ID + " = ?"; 
String[] whereValue = { Integer.toString(id) }; 

ContentValues cv = new ContentValues(); 
cv.put(BOOK_NAME, bookname); 
cv.put(BOOK_AUTHOR, author); 
db.update(TABLE_NAME, cv, where, whereValue); 
} 
}
第三步:修改main.xml布局如下,由两个EditText和一个ListView组成,代码如下:
<?xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 

android:orientation="vertical" 
android:layout_width="fill_parent" 
android:layout_height="fill_parent" 
> 
<EditText 
android:id="@+id/bookname" 
android:layout_width="fill_parent" 
android:layout_height="wrap_content" 

> 
</EditText> 
<EditText 
android:id="@+id/author" 
android:layout_width="fill_parent" 
android:layout_height="wrap_content" 
> 
</EditText> 
<ListView 
android:id="@+id/bookslist" 

android:layout_width="fill_parent" 
android:layout_height="wrap_content" 
> 
</ListView> 
</LinearLayout> 
第四步:修改SQLiteDatabaseDemo.java代码如下:
package com.android.tutor; 
import android.app.Activity; 
import android.content.Context; 
import android.database.Cursor; 
import android.os.Bundle; 
import android.view.Menu; 
import android.view.MenuItem; 
import android.view.View; 
import android.view.ViewGroup; 
import android.widget.AdapterView; 
import android.widget.BaseAdapter; 
import android.widget.EditText; 
import android.widget.ListView; 
import android.widget.TextView; 
import android.widget.Toast; 
public class SQLiteDatabaseDemo extends Activity implements AdapterView.OnItemClickListener { 
private BooksDB mBooksDB; 
private Cursor mCursor; 
private EditText BookName; 
private EditText BookAuthor; 
private ListView BooksList; 

private int BOOK_ID = 0; 
protected final static int MENU_ADD = Menu.FIRST; 
protected final static int MENU_DELETE = Menu.FIRST + 1; 
protected final static int MENU_UPDATE = Menu.FIRST + 2; 

public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
setUpViews(); 
} 

public void setUpViews(){ 
mBooksDB = new BooksDB(this); 
mCursor = mBooksDB.select(); 

BookName = (EditText)findViewById(R.id.bookname); 
BookAuthor = (EditText)findViewById(R.id.author); 
BooksList = (ListView)findViewById(R.id.bookslist); 

BooksList.setAdapter(new BooksListAdapter(this, mCursor)); 
BooksList.setOnItemClickListener(this); 
} 

@Override 
public boolean onCreateOptionsMenu(Menu menu) { 
super.onCreateOptionsMenu(menu); 

menu.add(Menu.NONE, MENU_ADD, 0, "ADD"); 
menu.add(Menu.NONE, MENU_DELETE, 0, "DELETE"); 
menu.add(Menu.NONE, MENU_DELETE, 0, "UPDATE"); 
return true; 
} 

public boolean onOptionsItemSelected(MenuItem item) 
{ 
super.onOptionsItemSelected(item); 
switch (item.getItemId()) 
{ 
case MENU_ADD: 
add(); 
break; 
case MENU_DELETE: 
delete(); 
break; 
case MENU_UPDATE: 
update(); 
break; 
} 
return true; 
} 

public void add(){ 
String bookname = BookName.getText().toString(); 
String author = BookAuthor.getText().toString(); 
//书名和作者都不能为空,或者退出 
if (bookname.equals("") || author.equals("")){ 
return; 
} 
mBooksDB.insert(bookname, author); 
mCursor.requery(); 
BooksList.invalidateViews(); 
BookName.setText(""); 
BookAuthor.setText(""); 
Toast.makeText(this, "Add Successed!", Toast.LENGTH_SHORT).show(); 
} 

public void delete(){ 
if (BOOK_ID == 0) { 
return; 
} 
mBooksDB.delete(BOOK_ID); 
mCursor.requery(); 
BooksList.invalidateViews(); 
BookName.setText(""); 
BookAuthor.setText(""); 
Toast.makeText(this, "Delete Successed!", Toast.LENGTH_SHORT).show(); 
} 

public void update(){ 
String bookname = BookName.getText().toString(); 
String author = BookAuthor.getText().toString(); 
//书名和作者都不能为空,或者退出 
if (bookname.equals("") || author.equals("")){ 
return; 
} 
mBooksDB.update(BOOK_ID, bookname, author); 
mCursor.requery(); 
BooksList.invalidateViews(); 
BookName.setText(""); 
BookAuthor.setText(""); 
Toast.makeText(this, "Update Successed!", Toast.LENGTH_SHORT).show(); 
} 

@Override 
public void onItemClick(AdapterView<?> parent, View view, int position, long id) { 

mCursor.moveToPosition(position); 
BOOK_ID = mCursor.getInt(0); 
BookName.setText(mCursor.getString(1)); 
BookAuthor.setText(mCursor.getString(2)); 

} 

public class BooksListAdapter extends BaseAdapter{ 
private Context mContext; 
private Cursor mCursor; 
public BooksListAdapter(Context context,Cursor cursor) { 

mContext = context; 
mCursor = cursor; 
} 
@Override 
public int getCount() { 
return mCursor.getCount(); 
} 
@Override 
public Object getItem(int position) { 
return null; 
} 
@Override 
public long getItemId(int position) { 
return 0; 
} 
@Override 
public View getView(int position, View convertView, ViewGroup parent) { 
TextView mTextView = new TextView(mContext); 
mCursor.moveToPosition(position); 
mTextView.setText(mCursor.getString(1) + "___" + mCursor.getString(2)); 
return mTextView; 
} 

} 
} 
第五步:运行程序效果如下:
a2.jpg 
a3.jpg 
a5.jpg 
a6.jpg 
a7.jpg
a8.jpg 

第六步:查看我们所建的数据库。有两种方法:第一种用命令查看:adb shell ls data/data/com.android.tutor/databases。
另一种方法是用DDMS查看,在data/data下面对应的应用程序的包名 下会有如下数据库,如图所示:
a9.jpg 

标签: Android sqlite

0

ANDROID开发之SQLite详解

发布于:2014-7-22 9:47 作者:admin 浏览:1632 分类:Android

SQLite简介

Google为Andriod的较大的数据处理提供了SQLite,他在数据存储、管理、维护等各方面都相当出色,功能也非常的强大。SQLite具备下列特点:

 

1.轻量级

使用 SQLite 只需要带一个动态库,就可以享受它的全部功能,而且那个动态库的尺寸想当小。

2.独立性

SQLite 数据库的核心引擎不需要依赖第三方软件,也不需要所谓的“安装”。

3.隔离性

SQLite 数据库中所有的信息(比如表、视图、触发器等)都包含在一个文件夹内,方便管理和维护。

4.跨平台

SQLite 目前支持大部分操作系统,不至电脑操作系统更在众多的手机系统也是能够运行,比如:Android。

5.多语言接口

SQLite 数据库支持多语言编程接口。

6.安全性

SQLite 数据库通过数据库级上的独占性和共享锁来实现独立事务处理。这意味着多个进程可以在同一时间从同一数据库读取数据,但只能有一个可以写入数据。

Android中的SQLite使用

首先创建数据库类

public class DatabaseHelperextends SQLiteOpenHelper {
 
    private static final String DB_NAME ="mydata.db";//数据库名称
    private static final int version =1;//数据库版本
     
    public DatabaseHelper(Context context) {
        super(context, DB_NAME,null, version);
        // TODO Auto-generated constructor stub
    }
 
    @Override
    public void onCreate(SQLiteDatabase db) {
        String sql ="create table user(username varchar(20) not null , password varchar(60) not null );";         
        db.execSQL(sql);
    }
 
    @Override
    public void onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion) {
        // TODO Auto-generated method stub
 
    }
 
}

SQLiteOpenHelper类介绍

SQLiteOpenHelper是SQLiteDatabase的一个帮助类,用来管理数据库的创建和版本的更新。一般是建立一个类继承它,并实现它的onCreate和onUpgrade方法。

方法名 方法描述
SQLiteOpenHelper(Context context,String name,SQLiteDatabase.CursorFactory factory,int version) 构造方法,一般是传递一个要创建的数据库名称那么参数
onCreate(SQLiteDatabase db) 创建数据库时调用
onUpgrade(SQLiteDatabase db,int oldVersion , int newVersion) 版本更新时调用
getReadableDatabase() 创建或打开一个只读数据库
getWritableDatabase() 创建或打开一个读写数据库

下面来介绍调用的方法

创建数据库

这里特别的地方是通过调用了SQLiteOpenHelper类的getReadableDatabase()方法来实现创建一个数据库的

1
2
3
DatabaseHelper database =new DatabaseHelper(this);//这段代码放到Activity类中才用this
SQLiteDatabase db =null;
db = database.getReadalbeDatabase();

SQLiteDatabase类为我们提供了很多种方法,而较常用的方法如下

(返回值)方法名 方法描述
(int) delete(String table,String whereClause,String[] whereArgs) 删除数据行的便捷方法
(long) insert(String table,String nullColumnHack,ContentValues values) 添加数据行的便捷方法
(int) update(String table, ContentValues values, String whereClause, String[] whereArgs) 更新数据行的便捷方法
(void) execSQL(String sql) 执行一个SQL语句,可以是一个select或其他的sql语句
(void) close() 关闭数据库
(Cursor) query(String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit) 查询指定的数据表返回一个带游标的数据集
(Cursor) rawQuery(String sql, String[] selectionArgs) 运行一个预置的SQL语句,返回带游标的数据集(与上面的语句最大的区别就是防止SQL注入)

数据的添删改查分别可以通过2种途径来实现

数据的添加

1.使用insert方法

1
2
3
ContentValues cv =new ContentValues();//实例化一个ContentValues用来装载待插入的数据cv.put("username","Jack Johnson");//添加用户名
cv.put("password","iLovePopMusic");//添加密码
db.insert("user",null,cv);//执行插入操作

2.使用execSQL方式来实现

1
2
String sql = "insert into user(username,password) values ('Jack Johnson','iLovePopMuisc');//插入操作的SQL语句
db.execSQL(sql);//执行SQL语句

数据的删除

同样有2种方式可以实现

1
2
3
String whereClause ="username=?";//删除的条件
String[] whereArgs = {"Jack Johnson"};//删除的条件参数
db.delete("user",whereClause,whereArgs);//执行删除

使用execSQL方式的实现

1
2
String sql ="delete from user where username='Jack Johnson'";//删除操作的SQL语句
db.execSQL(sql);//执行删除操作

数据修改

同上,仍是2种方式

1
2
3
4
5
ContentValues cv =new ContentValues();//实例化ContentValues
cv.put("password","iHatePopMusic");//添加要更改的字段及内容
String whereClause ="username=?";//修改条件
String[] whereArgs = {"Jack Johnson"};//修改条件的参数
db.update("user",cv,whereClause,whereArgs);//执行修改

使用execSQL方式的实现

1
2
String sql ="update [user] set password = 'iHatePopMusic' where username='Jack Johnson'";//修改的SQL语句
db.execSQL(sql);//执行修改

数据查询

数据查询相对前面几种方法就复杂一些了,因为查询会带有很多条件

通过query实现查询的

public Cursor query(String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit)

各参数说明:

  • table:表名称
  • colums:列名称数组
  • selection:条件子句,相当于where
  • selectionArgs:条件语句的参数数组
  • groupBy:分组
  • having:分组条件
  • orderBy:排序类
  • limit:分页查询的限制
  • Cursor:返回值,相当于结果集ResultSet

针对游标(Cursor)也提供了不少方法

方法名称 方法描述
getCount() 总记录条数
isFirst() 判断是否第一条记录
isLast() 判断是否最后一条记录
moveToFirst() 移动到第一条记录
moveToLast() 移动到最后一条记录
move(int offset) 移动到指定的记录
moveToNext() 移动到吓一条记录
moveToPrevious() 移动到上一条记录
getColumnIndex(String columnName) 获得指定列索引的int类型值

实现代码

1
2
3
4
5
6
7
8
Cursor c = db.query("user",null,null,null,null,null,null);//查询并获得游标
if(c.moveToFirst()){//判断游标是否为空
    for(int i=0;i<c.getCount();i++){
        c.move(i);//移动到指定记录
        String username = c.getString(c.getColumnIndex("username");
        String password = c.getString(c.getColumnIndex("password"));
    }
}

通过rawQuery实现的带参数查询

1
2
3
4
Cursor c = db.rawQuery("select * from user where username=?",new Stirng[]{"Jack Johnson"});
if(cursor.moveToFirst()) {
    String password = c.getString(c.getColumnIndex("password"));
}

标签: Android sqlite

0

Android中SQLite应用详解

发布于:2014-7-22 9:35 作者:admin 浏览:1682 分类:Android

现在的主流移动设备像Android、iPhone等都使用SQLite作为复杂数据的存储引擎,在我们为移动设备开发应用程序时,也许就要使用到SQLite来存储我们大量的数据,所以我们就需要掌握移动设备上的SQLite开发技巧。对于Android平台来说,系统内置了丰富的API来供开发人员操作SQLite,我们可以轻松的完成对数据的存取。

下面就向大家介绍一下SQLite常用的操作方法,为了方便,我将代码写在了Activity的onCreate中:



	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		
		//打开或创建test.db数据库
		SQLiteDatabase db = openOrCreateDatabase("test.db", Context.MODE_PRIVATE, null);
		db.execSQL("DROP TABLE IF EXISTS person");
		//创建person表
		db.execSQL("CREATE TABLE person (_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age SMALLINT)");
		Person person = new Person();
		person.name = "john";
		person.age = 30;
		//插入数据
		db.execSQL("INSERT INTO person VALUES (NULL, ?, ?)", new Object[]{person.name, person.age});
		
		person.name = "david";
		person.age = 33;
		//ContentValues以键值对的形式存放数据
		ContentValues cv = new ContentValues();
		cv.put("name", person.name);
		cv.put("age", person.age);
		//插入ContentValues中的数据
		db.insert("person", null, cv);
		
		cv = new ContentValues();
		cv.put("age", 35);
		//更新数据
		db.update("person", cv, "name = ?", new String[]{"john"});
		
		Cursor c = db.rawQuery("SELECT * FROM person WHERE age >= ?", new String[]{"33"});
		while (c.moveToNext()) {
			int _id = c.getInt(c.getColumnIndex("_id"));
			String name = c.getString(c.getColumnIndex("name"));
			int age = c.getInt(c.getColumnIndex("age"));
			Log.i("db", "_id=>" + _id + ", name=>" + name + ", age=>" + age);
		}
		c.close();
		
		//删除数据
		db.delete("person", "age < ?", new String[]{"35"});
		
		//关闭当前数据库
		db.close();
		
		//删除test.db数据库
//		deleteDatabase("test.db");
	}



在执行完上面的代码后,系统就会在/data/data/[PACKAGE_NAME]/databases目录下生成一个“test.db”的数据库文件,如图:



上面的代码中基本上囊括了大部分的数据库操作;对于添加、更新和删除来说,我们都可以使用

  1. db.executeSQL(String sql);  
  2. db.executeSQL(String sql, Object[] bindArgs);//sql语句中使用占位符,然后第二个参数是实际的参数集  
除了统一的形式之外,他们还有各自的操作方法:
  1. db.insert(String table, String nullColumnHack, ContentValues values);  
  2. db.update(String table, Contentvalues values, String whereClause, String whereArgs);  
  3. db.delete(String table, String whereClause, String whereArgs);  
以上三个方法的第一个参数都是表示要操作的表名;insert中的第二个参数表示如果插入的数据每一列都为空的话,需要指定此行中某一列的名称,系统将此列设置为NULL,不至于出现错误;insert中的第三个参数是ContentValues类型的变量,是键值对组成的Map,key代表列名,value代表该列要插入的值;update的第二个参数也很类似,只不过它是更新该字段key为最新的value值,第三个参数whereClause表示WHERE表达式,比如“age > ? and age < ?”等,最后的whereArgs参数是占位符的实际参数值;delete方法的参数也是一样。

下面来说说查询操作。查询操作相对于上面的几种操作要复杂些,因为我们经常要面对着各种各样的查询条件,所以系统也考虑到这种复杂性,为我们提供了较为丰富的查询形式:

  1. db.rawQuery(String sql, String[] selectionArgs);  
  2. db.query(String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy);  
  3. db.query(String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);  
  4. db.query(String distinct, String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);  
上面几种都是常用的查询方法,第一种最为简单,将所有的SQL语句都组织到一个字符串中,使用占位符代替实际参数,selectionArgs就是占位符实际参数集;下面的几种参数都很类似,columns表示要查询的列所有名称集,selection表示WHERE之后的条件语句,可以使用占位符,groupBy指定分组的列名,having指定分组条件,配合groupBy使用,orderBy指定排序的列名,limit指定分页参数,distinct可以指定“true”或“false”表示要不要过滤重复值。需要注意的是,selection、groupBy、having、orderBy、limit这几个参数中不包括“WHERE”、“GROUP BY”、“HAVING”、“ORDER BY”、“LIMIT”等SQL关键字。
最后,他们同时返回一个Cursor对象,代表数据集的游标,有点类似于JavaSE中的ResultSet。

下面是Cursor对象的常用方法

  1. c.move(int offset); //以当前位置为参考,移动到指定行  
  2. c.moveToFirst();    //移动到第一行  
  3. c.moveToLast();     //移动到最后一行  
  4. c.moveToPosition(int position); //移动到指定行  
  5. c.moveToPrevious(); //移动到前一行  
  6. c.moveToNext();     //移动到下一行  
  7. c.isFirst();        //是否指向第一条  
  8. c.isLast();     //是否指向最后一条  
  9. c.isBeforeFirst();  //是否指向第一条之前  
  10. c.isAfterLast();    //是否指向最后一条之后  
  11. c.isNull(int columnIndex);  //指定列是否为空(列基数为0)  
  12. c.isClosed();       //游标是否已关闭  
  13. c.getCount();       //总数据项数  
  14. c.getPosition();    //返回当前游标所指向的行数  
  15. c.getColumnIndex(String columnName);//返回某列名对应的列索引值  
  16. c.getString(int columnIndex);   //返回当前行指定列的值  

在上面的代码示例中,已经用到了这几个常用方法中的一些,关于更多的信息,大家可以参考官方文档中的说明。

最后当我们完成了对数据库的操作后,记得调用SQLiteDatabase的close()方法释放数据库连接,否则容易出现SQLiteException。

上面就是SQLite的基本应用,但在实际开发中,为了能够更好的管理和维护数据库,我们会封装一个继承自SQLiteOpenHelper类的数据库操作类,然后以这个类为基础,再封装我们的业务逻辑方法。

下面,我们就以一个实例来讲解具体的用法,我们新建一个名为db的项目,结构如下:


其中DBHelper继承了SQLiteOpenHelper,作为维护和管理数据库的基类,DBManager是建立在DBHelper之上,封装了常用的业务方法,Person是我们的person表对应的JavaBean,MainActivity就是我们显示的界面。

下面我们先来看一下DBHelper:

  1. package com.scott.db;  
  2.   
  3. import android.content.Context;  
  4. import android.database.sqlite.SQLiteDatabase;  
  5. import android.database.sqlite.SQLiteOpenHelper;  
  6.   
  7. public class DBHelper extends SQLiteOpenHelper {  
  8.   
  9.     private static final String DATABASE_NAME = "test.db";  
  10.     private static final int DATABASE_VERSION = 1;  
  11.       
  12.     public DBHelper(Context context) {  
  13.         //CursorFactory设置为null,使用默认值  
  14.         super(context, DATABASE_NAME, null, DATABASE_VERSION);  
  15.     }  
  16.   
  17.     //数据库第一次被创建时onCreate会被调用  
  18.     @Override  
  19.     public void onCreate(SQLiteDatabase db) {  
  20.         db.execSQL("CREATE TABLE IF NOT EXISTS person" +  
  21.                 "(_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age INTEGER, info TEXT)");  
  22.     }  
  23.   
  24.     //如果DATABASE_VERSION值被改为2,系统发现现有数据库版本不同,即会调用onUpgrade  
  25.     @Override  
  26.     public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {  
  27.         db.execSQL("ALTER TABLE person ADD COLUMN other STRING");  
  28.     }  
  29. }  
正如上面所述,数据库第一次创建时onCreate方法会被调用,我们可以执行创建表的语句,当系统发现版本变化之后,会调用onUpgrade方法,我们可以执行修改表结构等语句。

为了方便我们面向对象的使用数据,我们建一个Person类,对应person表中的字段,如下:

  1. package com.scott.db;  
  2.   
  3. public class Person {  
  4.     public int _id;  
  5.     public String name;  
  6.     public int age;  
  7.     public String info;  
  8.       
  9.     public Person() {  
  10.     }  
  11.       
  12.     public Person(String name, int age, String info) {  
  13.         this.name = name;  
  14.         this.age = age;  
  15.         this.info = info;  
  16.     }  
  17. }  
然后,我们需要一个DBManager,来封装我们所有的业务方法,代码如下:
  1. package com.scott.db;  
  2.   
  3. import java.util.ArrayList;  
  4. import java.util.List;  
  5.   
  6. import android.content.ContentValues;  
  7. import android.content.Context;  
  8. import android.database.Cursor;  
  9. import android.database.sqlite.SQLiteDatabase;  
  10.   
  11. public class DBManager {  
  12.     private DBHelper helper;  
  13.     private SQLiteDatabase db;  
  14.       
  15.     public DBManager(Context context) {  
  16.         helper = new DBHelper(context);  
  17.         //因为getWritableDatabase内部调用了mContext.openOrCreateDatabase(mName, 0, mFactory);  
  18.         //所以要确保context已初始化,我们可以把实例化DBManager的步骤放在Activity的onCreate里  
  19.         db = helper.getWritableDatabase();  
  20.     }  
  21.       
  22.     /** 
  23.      * add persons 
  24.      * @param persons 
  25.      */  
  26.     public void add(List<Person> persons) {  
  27.         db.beginTransaction();  //开始事务  
  28.         try {  
  29.             for (Person person : persons) {  
  30.                 db.execSQL("INSERT INTO person VALUES(null, ?, ?, ?)"new Object[]{person.name, person.age, person.info});  
  31.             }  
  32.             db.setTransactionSuccessful();  //设置事务成功完成  
  33.         } finally {  
  34.             db.endTransaction();    //结束事务  
  35.         }  
  36.     }  
  37.       
  38.     /** 
  39.      * update person's age 
  40.      * @param person 
  41.      */  
  42.     public void updateAge(Person person) {  
  43.         ContentValues cv = new ContentValues();  
  44.         cv.put("age", person.age);  
  45.         db.update("person", cv, "name = ?"new String[]{person.name});  
  46.     }  
  47.       
  48.     /** 
  49.      * delete old person 
  50.      * @param person 
  51.      */  
  52.     public void deleteOldPerson(Person person) {  
  53.         db.delete("person""age >= ?"new String[]{String.valueOf(person.age)});  
  54.     }  
  55.       
  56.     /** 
  57.      * query all persons, return list 
  58.      * @return List<Person> 
  59.      */  
  60.     public List<Person> query() {  
  61.         ArrayList<Person> persons = new ArrayList<Person>();  
  62.         Cursor c = queryTheCursor();  
  63.         while (c.moveToNext()) {  
  64.             Person person = new Person();  
  65.             person._id = c.getInt(c.getColumnIndex("_id"));  
  66.             person.name = c.getString(c.getColumnIndex("name"));  
  67.             person.age = c.getInt(c.getColumnIndex("age"));  
  68.             person.info = c.getString(c.getColumnIndex("info"));  
  69.             persons.add(person);  
  70.         }  
  71.         c.close();  
  72.         return persons;  
  73.     }  
  74.       
  75.     /** 
  76.      * query all persons, return cursor 
  77.      * @return  Cursor 
  78.      */  
  79.     public Cursor queryTheCursor() {  
  80.         Cursor c = db.rawQuery("SELECT * FROM person"null);  
  81.         return c;  
  82.     }  
  83.       
  84.     /** 
  85.      * close database 
  86.      */  
  87.     public void closeDB() {  
  88.         db.close();  
  89.     }  
  90. }  
我们在DBManager构造方法中实例化DBHelper并获取一个SQLiteDatabase对象,作为整个应用的数据库实例;在添加多个Person信息时,我们采用了事务处理,确保数据完整性;最后我们提供了一个closeDB方法,释放数据库资源,这一个步骤在我们整个应用关闭时执行,这个环节容易被忘记,所以朋友们要注意。

我们获取数据库实例时使用了getWritableDatabase()方法,也许朋友们会有疑问,在getWritableDatabase()和getReadableDatabase()中,你为什么选择前者作为整个应用的数据库实例呢?在这里我想和大家着重分析一下这一点。

我们来看一下SQLiteOpenHelper中的getReadableDatabase()方法:

  1. public synchronized SQLiteDatabase getReadableDatabase() {  
  2.     if (mDatabase != null && mDatabase.isOpen()) {  
  3.         // 如果发现mDatabase不为空并且已经打开则直接返回  
  4.         return mDatabase;  
  5.     }  
  6.   
  7.     if (mIsInitializing) {  
  8.         // 如果正在初始化则抛出异常  
  9.         throw new IllegalStateException("getReadableDatabase called recursively");  
  10.     }  
  11.   
  12.     // 开始实例化数据库mDatabase  
  13.   
  14.     try {  
  15.         // 注意这里是调用了getWritableDatabase()方法  
  16.         return getWritableDatabase();  
  17.     } catch (SQLiteException e) {  
  18.         if (mName == null)  
  19.             throw e; // Can't open a temp database read-only!  
  20.         Log.e(TAG, "Couldn't open " + mName + " for writing (will try read-only):", e);  
  21.     }  
  22.   
  23.     // 如果无法以可读写模式打开数据库 则以只读方式打开  
  24.   
  25.     SQLiteDatabase db = null;  
  26.     try {  
  27.         mIsInitializing = true;  
  28.         String path = mContext.getDatabasePath(mName).getPath();// 获取数据库路径  
  29.         // 以只读方式打开数据库  
  30.         db = SQLiteDatabase.openDatabase(path, mFactory, SQLiteDatabase.OPEN_READONLY);  
  31.         if (db.getVersion() != mNewVersion) {  
  32.             throw new SQLiteException("Can't upgrade read-only database from version " + db.getVersion() + " to "  
  33.                     + mNewVersion + ": " + path);  
  34.         }  
  35.   
  36.         onOpen(db);  
  37.         Log.w(TAG, "Opened " + mName + " in read-only mode");  
  38.         mDatabase = db;// 为mDatabase指定新打开的数据库  
  39.         return mDatabase;// 返回打开的数据库  
  40.     } finally {  
  41.         mIsInitializing = false;  
  42.         if (db != null && db != mDatabase)  
  43.             db.close();  
  44.     }  
  45. }  
在getReadableDatabase()方法中,首先判断是否已存在数据库实例并且是打开状态,如果是,则直接返回该实例,否则试图获取一个可读写模式的数据库实例,如果遇到磁盘空间已满等情况获取失败的话,再以只读模式打开数据库,获取数据库实例并返回,然后为mDatabase赋值为最新打开的数据库实例。既然有可能调用到getWritableDatabase()方法,我们就要看一下了:
  1. public synchronized SQLiteDatabase getWritableDatabase() {  
  2.     if (mDatabase != null && mDatabase.isOpen() && !mDatabase.isReadOnly()) {  
  3.         // 如果mDatabase不为空已打开并且不是只读模式 则返回该实例  
  4.         return mDatabase;  
  5.     }  
  6.   
  7.     if (mIsInitializing) {  
  8.         throw new IllegalStateException("getWritableDatabase called recursively");  
  9.     }  
  10.   
  11.     // If we have a read-only database open, someone could be using it  
  12.     // (though they shouldn't), which would cause a lock to be held on  
  13.     // the file, and our attempts to open the database read-write would  
  14.     // fail waiting for the file lock. To prevent that, we acquire the  
  15.     // lock on the read-only database, which shuts out other users.  
  16.   
  17.     boolean success = false;  
  18.     SQLiteDatabase db = null;  
  19.     // 如果mDatabase不为空则加锁 阻止其他的操作  
  20.     if (mDatabase != null)  
  21.         mDatabase.lock();  
  22.     try {  
  23.         mIsInitializing = true;  
  24.         if (mName == null) {  
  25.             db = SQLiteDatabase.create(null);  
  26.         } else {  
  27.             // 打开或创建数据库  
  28.             db = mContext.openOrCreateDatabase(mName, 0, mFactory);  
  29.         }  
  30.         // 获取数据库版本(如果刚创建的数据库,版本为0)  
  31.         int version = db.getVersion();  
  32.         // 比较版本(我们代码中的版本mNewVersion为1)  
  33.         if (version != mNewVersion) {  
  34.             db.beginTransaction();// 开始事务  
  35.             try {  
  36.                 if (version == 0) {  
  37.                     // 执行我们的onCreate方法  
  38.                     onCreate(db);  
  39.                 } else {  
  40.                     // 如果我们应用升级了mNewVersion为2,而原版本为1则执行onUpgrade方法  
  41.                     onUpgrade(db, version, mNewVersion);  
  42.                 }  
  43.                 db.setVersion(mNewVersion);// 设置最新版本  
  44.                 db.setTransactionSuccessful();// 设置事务成功  
  45.             } finally {  
  46.                 db.endTransaction();// 结束事务  
  47.             }  
  48.         }  
  49.   
  50.         onOpen(db);  
  51.         success = true;  
  52.         return db;// 返回可读写模式的数据库实例  
  53.     } finally {  
  54.         mIsInitializing = false;  
  55.         if (success) {  
  56.             // 打开成功  
  57.             if (mDatabase != null) {  
  58.                 // 如果mDatabase有值则先关闭  
  59.                 try {  
  60.                     mDatabase.close();  
  61.                 } catch (Exception e) {  
  62.                 }  
  63.                 mDatabase.unlock();// 解锁  
  64.             }  
  65.             mDatabase = db;// 赋值给mDatabase  
  66.         } else {  
  67.             // 打开失败的情况:解锁、关闭  
  68.             if (mDatabase != null)  
  69.                 mDatabase.unlock();  
  70.             if (db != null)  
  71.                 db.close();  
  72.         }  
  73.     }  
  74. }  
大家可以看到,几个关键步骤是,首先判断mDatabase如果不为空已打开并不是只读模式则直接返回,否则如果mDatabase不为空则加锁,然后开始打开或创建数据库,比较版本,根据版本号来调用相应的方法,为数据库设置新版本号,最后释放旧的不为空的mDatabase并解锁,把新打开的数据库实例赋予mDatabase,并返回最新实例。

看完上面的过程之后,大家或许就清楚了许多,如果不是在遇到磁盘空间已满等情况,getReadableDatabase()一般都会返回和getWritableDatabase()一样的数据库实例,所以我们在DBManager构造方法中使用getWritableDatabase()获取整个应用所使用的数据库实例是可行的。当然如果你真的担心这种情况会发生,那么你可以先用getWritableDatabase()获取数据实例,如果遇到异常,再试图用getReadableDatabase()获取实例,当然这个时候你获取的实例只能读不能写了。

最后,让我们看一下如何使用这些数据操作方法来显示数据,下面是MainActivity.java的布局文件和代码:

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:orientation="vertical"  
  4.     android:layout_width="fill_parent"  
  5.     android:layout_height="fill_parent">  
  6.     <Button  
  7.         android:layout_width="fill_parent"  
  8.         android:layout_height="wrap_content"  
  9.         android:text="add"  
  10.         android:onClick="add"/>  
  11.     <Button  
  12.         android:layout_width="fill_parent"  
  13.         android:layout_height="wrap_content"  
  14.         android:text="update"  
  15.         android:onClick="update"/>  
  16.     <Button  
  17.         android:layout_width="fill_parent"  
  18.         android:layout_height="wrap_content"  
  19.         android:text="delete"  
  20.         android:onClick="delete"/>  
  21.     <Button  
  22.         android:layout_width="fill_parent"  
  23.         android:layout_height="wrap_content"  
  24.         android:text="query"  
  25.         android:onClick="query"/>  
  26.     <Button  
  27.         android:layout_width="fill_parent"  
  28.         android:layout_height="wrap_content"  
  29.         android:text="queryTheCursor"  
  30.         android:onClick="queryTheCursor"/>  
  31.     <ListView  
  32.         android:id="@+id/listView"  
  33.         android:layout_width="fill_parent"  
  34.         android:layout_height="wrap_content"/>  
  35. </LinearLayout>  

  1. package com.scott.db;  
  2.   
  3. import java.util.ArrayList;  
  4. import java.util.HashMap;  
  5. import java.util.List;  
  6. import java.util.Map;  
  7.   
  8. import android.app.Activity;  
  9. import android.database.Cursor;  
  10. import android.database.CursorWrapper;  
  11. import android.os.Bundle;  
  12. import android.view.View;  
  13. import android.widget.ListView;  
  14. import android.widget.SimpleAdapter;  
  15. import android.widget.SimpleCursorAdapter;  
  16.   
  17.   
  18. public class MainActivity extends Activity {  
  19.      
  20.     private DBManager mgr;  
  21.     private ListView listView;  
  22.       
  23.     @Override  
  24.     public void onCreate(Bundle savedInstanceState) {  
  25.         super.onCreate(savedInstanceState);  
  26.         setContentView(R.layout.main);  
  27.         listView = (ListView) findViewById(R.id.listView);  
  28.         //初始化DBManager  
  29.         mgr = new DBManager(this);  
  30.     }  
  31.       
  32.     @Override  
  33.     protected void onDestroy() {  
  34.         super.onDestroy();  
  35.         //应用的最后一个Activity关闭时应释放DB  
  36.         mgr.closeDB();  
  37.     }  
  38.       
  39.     public void add(View view) {  
  40.         ArrayList<Person> persons = new ArrayList<Person>();  
  41.           
  42.         Person person1 = new Person("Ella"22"lively girl");  
  43.         Person person2 = new Person("Jenny"22"beautiful girl");  
  44.         Person person3 = new Person("Jessica"23"sexy girl");  
  45.         Person person4 = new Person("Kelly"23"hot baby");  
  46.         Person person5 = new Person("Jane"25"a pretty woman");  
  47.           
  48.         persons.add(person1);  
  49.         persons.add(person2);  
  50.         persons.add(person3);  
  51.         persons.add(person4);  
  52.         persons.add(person5);  
  53.           
  54.         mgr.add(persons);  
  55.     }  
  56.       
  57.     public void update(View view) {  
  58.         Person person = new Person();  
  59.         person.name = "Jane";  
  60.         person.age = 30;  
  61.         mgr.updateAge(person);  
  62.     }  
  63.       
  64.     public void delete(View view) {  
  65.         Person person = new Person();  
  66.         person.age = 30;  
  67.         mgr.deleteOldPerson(person);  
  68.     }  
  69.       
  70.     public void query(View view) {  
  71.         List<Person> persons = mgr.query();  
  72.         ArrayList<Map<String, String>> list = new ArrayList<Map<String, String>>();  
  73.         for (Person person : persons) {  
  74.             HashMap<String, String> map = new HashMap<String, String>();  
  75.             map.put("name", person.name);  
  76.             map.put("info", person.age + " years old, " + person.info);  
  77.             list.add(map);  
  78.         }  
  79.         SimpleAdapter adapter = new SimpleAdapter(this, list, android.R.layout.simple_list_item_2,  
  80.                     new String[]{"name""info"}, new int[]{android.R.id.text1, android.R.id.text2});  
  81.         listView.setAdapter(adapter);  
  82.     }  
  83.       
  84.     public void queryTheCursor(View view) {  
  85.         Cursor c = mgr.queryTheCursor();  
  86.         startManagingCursor(c); //托付给activity根据自己的生命周期去管理Cursor的生命周期  
  87.         CursorWrapper cursorWrapper = new CursorWrapper(c) {  
  88.             @Override  
  89.             public String getString(int columnIndex) {  
  90.                 //将简介前加上年龄  
  91.                 if (getColumnName(columnIndex).equals("info")) {  
  92.                     int age = getInt(getColumnIndex("age"));  
  93.                     return age + " years old, " + super.getString(columnIndex);  
  94.                 }  
  95.                 return super.getString(columnIndex);  
  96.             }  
  97.         };  
  98.         //确保查询结果中有"_id"列  
  99.         SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, android.R.layout.simple_list_item_2,   
  100.                 cursorWrapper, new String[]{"name""info"}, new int[]{android.R.id.text1, android.R.id.text2});  
  101.         ListView listView = (ListView) findViewById(R.id.listView);  
  102.         listView.setAdapter(adapter);  
  103.     }  
  104. }  
这里需要注意的是SimpleCursorAdapter的应用,当我们使用这个适配器时,我们必须先得到一个Cursor对象,这里面有几个问题:如何管理Cursor的生命周期,如果包装Cursor,Cursor结果集都需要注意什么。

如果手动去管理Cursor的话会非常的麻烦,还有一定的风险,处理不当的话运行期间就会出现异常,幸好Activity为我们提供了startManagingCursor(Cursor cursor)方法,它会根据Activity的生命周期去管理当前的Cursor对象,下面是该方法的说明:

  1. /** 
  2.      * This method allows the activity to take care of managing the given 
  3.      * {@link Cursor}'s lifecycle for you based on the activity's lifecycle. 
  4.      * That is, when the activity is stopped it will automatically call 
  5.      * {@link Cursor#deactivate} on the given Cursor, and when it is later restarted 
  6.      * it will call {@link Cursor#requery} for you.  When the activity is 
  7.      * destroyed, all managed Cursors will be closed automatically. 
  8.      *  
  9.      * @param c The Cursor to be managed. 
  10.      *  
  11.      * @see #managedQuery(android.net.Uri , String[], String, String[], String) 
  12.      * @see #stopManagingCursor 
  13.      */  
文中提到,startManagingCursor方法会根据Activity的生命周期去管理当前的Cursor对象的生命周期,就是说当Activity停止时他会自动调用Cursor的deactivate方法,禁用游标,当Activity重新回到屏幕时它会调用Cursor的requery方法再次查询,当Activity摧毁时,被管理的Cursor都会自动关闭释放。

如何包装Cursor:我们会使用到CursorWrapper对象去包装我们的Cursor对象,实现我们需要的数据转换工作,这个CursorWrapper实际上是实现了Cursor接口。我们查询获取到的Cursor其实是Cursor的引用,而系统实际返回给我们的必然是Cursor接口的一个实现类的对象实例,我们用CursorWrapper包装这个实例,然后再使用SimpleCursorAdapter将结果显示到列表上。

Cursor结果集需要注意些什么:一个最需要注意的是,在我们的结果集中必须要包含一个“_id”的列,否则SimpleCursorAdapter就会翻脸不认人,为什么一定要这样呢?因为这源于SQLite的规范,主键以“_id”为标准。解决办法有三:第一,建表时根据规范去做;第二,查询时用别名,例如:SELECT id AS _id FROM person;第三,在CursorWrapper里做文章:

[java] view plaincopy
  1. CursorWrapper cursorWrapper = new CursorWrapper(c) {  
  2.     @Override  
  3.     public int getColumnIndexOrThrow(String columnName) throws IllegalArgumentException {  
  4.         if (columnName.equals("_id")) {  
  5.             return super.getColumnIndex("id");  
  6.         }  
  7.         return super.getColumnIndexOrThrow(columnName);  
  8.     }  
  9. };  
如果试图从CursorWrapper里获取“_id”对应的列索引,我们就返回查询结果里“id”对应的列索引即可。





标签: Android sqlite

0

android 参数含义

发布于:2014-7-21 13:38 作者:admin 浏览:1811 
android:layout_width 设置组件的宽度
android:layout_height 设置组件的高度
android:id 给组件定义一个id值,供后期使用
android:background 设置组件的背景颜色或背景图片
android:text 设置组件的显示文字
android:textColor 设置组件的显示文字的颜色
android:layout_below 组件在参考组件的下面
android:alignTop 同指定组件的顶平行
android:maxLength="6" 限制输入字数
android:digits='012356789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'限制输入数字和大写小写字母
1. 开发更简单,执行速度高效。 2. 输入法默认会根据情况变动,比如说设置为numeric后输入法会自动仅显示数字,不会出现Qwerty中的字母。
下面我们通过EditText的layout xml文件中的相关属性来实现:
1. 密码框属性 android:password='true' 这条可以让EditText显示的内容自动为 星号,输入时内容会在1秒内变成*字样。
2. 纯数字 android:numeric='true' 这条可以让输入法自动变为数字输入键盘,同时仅允许0-9的数字输入
3. 仅允许 android:capitalize='cwj1987' 这样仅允许接受输入cwj1987,一般用于密码验证
下面是一些扩展的风格属性
android:editable='false' 设置EditText不可编辑
android:singleLine='true' 强制输入的内容在单行
android:ellipsize='end' 自动隐藏尾部溢出数据,一般用于文字内容过长一行无法全部显示时。

android:autoLink
设置是否当文本为URL链接/email/电话号码/map时,文本显示为可点击的链接。可选值(none/web/email/phone/map/all)

android:autoText
如果设置,将自动执行输入值的拼写纠正。此处无效果,在显示输入法并输入的时候起作用。

android:bufferType
指定getText()方式取得的文本类别。选项editable 类似于StringBuilder可追加字符,
也就是说getText后可调用append方法设置文本内容。spannable 则可在给定的字符区域使用样式,参见这里1、这里2。

android:capitalize
设置英文字母大写类型。此处无效果,需要弹出输入法才能看得到,参见EditText此属性说明。

android:cursorVisible
设定光标为显示/隐藏,默认显示。

android:digits
设置允许输入哪些字符。如“1234567890.+-*/%\n()”

android:drawableBottom
在text的下方输出一个drawable,如图片。如果指定一个颜色的话会把text的背景设为该颜色,并且同时和background使用时覆盖后者。

android:drawableLeft
在text的左边输出一个drawable,如图片。

android:drawablePadding
设置text与drawable(图片)的间隔,与drawableLeft、drawableRight、drawableTop、drawableBottom一起使用,可设置为负数,单独使用没有效果。

android:drawableRight
在text的右边输出一个drawable,如图片。

android:drawableTop
在text的正上方输出一个drawable,如图片。

android:editable
设置是否可编辑。这里无效果,参见EditView。

android:editorExtras
设置文本的额外的输入数据。在EditView再讨论。

android:ellipsize
设置当文字过长时,该控件该如何显示。有如下值设置:”start”—–省略号显示在开头;”end”——省略号显示在结尾;”middle”—-省略号显示在中间;”marquee” ——以跑马灯的方式显示(动画横向移动)

android:freezesText
设置保存文本的内容以及光标的位置。参见:这里。

android:gravity
设置文本位置,如设置成“center”,文本将居中显示。

android:hint
Text为空时显示的文字提示信息,可通过textColorHint设置提示信息的颜色。此属性在EditView中使用,但是这里也可以用。

android:imeOptions
附加功能,设置右下角IME动作与编辑框相关的动作,如actionDone右下角将显示一个“完成”,而不设置默认是一个回车符号。这个在EditText中再详细说明,此处无用。

android:imeActionId
设置IME动作ID。在EditText再做说明,可以先看这篇帖子:这里。

android:imeActionLabel
设置IME动作标签。在EditText再做说明。

android:includeFontPadding
设置文本是否包含顶部和底部额外空白,默认为true。

android:inputMethod
为文本指定输入法,需要完全限定名(完整的包名)。例如:com.google.android.inputmethod.pinyin,但是这里报错找不到。

android:inputType
设置文本的类型,用于帮助输入法显示合适的键盘类型。在EditText中再详细说明,这里无效果。

android:linksClickable
设置链接是否点击连接,即使设置了autoLink。

android:marqueeRepeatLimit
在ellipsize指定marquee的情况下,设置重复滚动的次数,当设置为marquee_forever时表示无限次。

android:ems
设置TextView的宽度为N个字符的宽度。这里测试为一个汉字字符宽度,如图:

android:maxEms
设置TextView的宽度为最长为N个字符的宽度。与ems同时使用时覆盖ems选项。

android:minEms
设置TextView的宽度为最短为N个字符的宽度。与ems同时使用时覆盖ems选项。

android:maxLength
限制显示的文本长度,超出部分不显示。

android:lines
设置文本的行数,设置两行就显示两行,即使第二行没有数据。

android:maxLines
设置文本的最大显示行数,与width或者layout_width结合使用,超出部分自动换行,超出行数将不显示。

android:minLines
设置文本的最小行数,与lines类似。

android:lineSpacingExtra
设置行间距。

android:lineSpacingMultiplier
设置行间距的倍数。如”1.2”

android:numeric
如果被设置,该TextView有一个数字输入法。此处无用,设置后唯一效果是TextView有点击效果,此属性在EditText将详细说明。

android:password
以小点”.”显示文本

android:phoneNumber
设置为电话号码的输入方式。

android:privateImeOptions
设置输入法选项,此处无用,在EditText将进一步讨论。

android:scrollHorizontally
设置文本超出TextView的宽度的情况下,是否出现横拉条。

android:selectAllOnFocus
如果文本是可选择的,让他获取焦点而不是将光标移动为文本的开始位置或者末尾位置。EditText中设置后无效果。

android:shadowColor
指定文本阴影的颜色,需要与shadowRadius一起使用。效果:

android:shadowDx
设置阴影横向坐标开始位置。

android:shadowDy
设置阴影纵向坐标开始位置。

android:shadowRadius
设置阴影的半径。设置为0.1就变成字体的颜色了,一般设置为3.0的效果比较好。

android:singleLine
设置单行显示。如果和layout_width一起使用,当文本不能全部显示时,后面用“…”来表示。如android:text="test_ singleLine " android:singleLine="true" android:layout_width="20dp"将只显示“t…”。如果不设置singleLine或者设置为false,文本将自动换行

android:text
设置显示文本.

android:textAppearance
设置文字外观。如“?android:attr/textAppearanceLargeInverse
”这里引用的是系统自带的一个外观,?表示系统是否有这种外观,否则使用默认的外观。可设置的值如下:textAppearanceButton/textAppearanceInverse/textAppearanceLarge/textAppearanceLargeInverse/textAppearanceMedium/textAppearanceMediumInverse/textAppearanceSmall/textAppearanceSmallInverse

android:textColor
设置文本颜色

android:textColorHighlight
被选中文字的底色,默认为蓝色

android:textColorHint
设置提示信息文字的颜色,默认为灰色。与hint一起使用。

android:textColorLink
文字链接的颜色.

android:textScaleX
设置文字之间间隔,默认为1.0f。分别设置0.5f/1.0f/1.5f/2.0f效果如下:


android:textSize
设置文字大小,推荐度量单位”sp”,如”15sp”

android:textStyle
设置字形[bold(粗体) 0, italic(斜体) 1, bolditalic(又粗又斜) 2] 可以设置一个或多个,用“|”隔开

android:typeface
设置文本字体,必须是以下常量值之一:normal 0, sans 1, serif 2, monospace(等宽字体) 3]

android:height
设置文本区域的高度,支持度量单位:px(像素)/dp/sp/in/mm(毫米)

android:maxHeight
设置文本区域的最大高度

android:minHeight
设置文本区域的最小高度

android:width
设置文本区域的宽度,支持度量单位:px(像素)/dp/sp/in/mm(毫米),与layout_width的区别看这里。

android:maxWidth
设置文本区域的最大宽度

android:minWidth
设置文本区域的最小宽度

标签: Android

0

Android Notification通知详解

发布于:2014-7-18 9:38 作者:admin 浏览:1659 分类:Android
根据activity的生命周期,在activity不显示时,会执行onStop函数(比如按下home键),所以你在onStop函数(按退出键除外)里面把notification放在通知栏里,再此显示时,把notification从通知栏里去掉。或者,只要程序在运行就一直显示通知栏图标。

        

下面对Notification类中的一些常量,字段,方法简单介绍一下:
常量:
DEFAULT_ALL    使用所有默认值,比如声音,震动,闪屏等等
DEFAULT_LIGHTS 使用默认闪光提示
DEFAULT_SOUNDS 使用默认提示声音
DEFAULT_VIBRATE 使用默认手机震动 
【说明】:加入手机震动,一定要在manifest.xml中加入权限:
&lt;uses-permission android:name="android.permission.VIBRATE" /&gt;
以上的效果常量可以叠加,即通过
notification.defaults =DEFAULT_SOUND|DEFAULT_VIBRATE;  
notification.defaults |= DEFAULT_SOUND (最好在真机上测试,震动效果模拟器上没有)

            

//设置flag位
FLAG_AUTO_CANCEL  该通知能被状态栏的清除按钮给清除掉
FLAG_NO_CLEAR     该通知能被状态栏的清除按钮给清除掉
FLAG_ONGOING_EVENT 通知放置在正在运行
FLAG_INSISTENT 是否一直进行,比如音乐一直播放,知道用户响应

          

常用字段:
contentIntent  设置PendingIntent对象,点击时发送该Intent
defaults 添加默认效果
flags 设置flag位,例如FLAG_NO_CLEAR等
icon 设置图标
sound 设置声音
tickerText 显示在状态栏中的文字
when 发送此通知的时间戳

                

NotificationManager常用方法介绍:
public void cancelAll() 移除所有通知(只是针对当前Context下的Notification)
public  void cancel(int id) 移除标记为id的通知 (只是针对当前Context下的所有Notification)
public  void notify(String tag ,int id, Notification notification) 将通知加入状态栏,标签为tag,标记为id
public  void notify(int id, Notification notification) 将通知加入状态栏,标记为id

             


package com.ljq.activity;
 
import android.app.Activity;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.graphics.Color;
import android.os.Bundle;
 
public class MainActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        clearNotification();
    }
     
    @Override
    protected void onStop() {
        showNotification();
        super.onStop();
    }
     
    @Override
    protected void onStart() {
        clearNotification();
        super.onStart();
    }
     
    /**
     * 在状态栏显示通知
     */
    private void showNotification(){
        // 创建一个NotificationManager的引用   
        NotificationManager notificationManager = (NotificationManager)    
            this.getSystemService(android.content.Context.NOTIFICATION_SERVICE);   
         
        // 定义Notification的各种属性   
        Notification notification =new Notification(R.drawable.icon,   
                "督导系统", System.currentTimeMillis()); 
        //FLAG_AUTO_CANCEL   该通知能被状态栏的清除按钮给清除掉
        //FLAG_NO_CLEAR      该通知不能被状态栏的清除按钮给清除掉
        //FLAG_ONGOING_EVENT 通知放置在正在运行
        //FLAG_INSISTENT     是否一直进行,比如音乐一直播放,知道用户响应
        notification.flags |= Notification.FLAG_ONGOING_EVENT; // 将此通知放到通知栏的"Ongoing"即"正在运行"组中   
        notification.flags |= Notification.FLAG_NO_CLEAR; // 表明在点击了通知栏中的"清除通知"后,此通知不清除,经常与FLAG_ONGOING_EVENT一起使用   
        notification.flags |= Notification.FLAG_SHOW_LIGHTS;   
        //DEFAULT_ALL     使用所有默认值,比如声音,震动,闪屏等等
        //DEFAULT_LIGHTS  使用默认闪光提示
        //DEFAULT_SOUNDS  使用默认提示声音
        //DEFAULT_VIBRATE 使用默认手机震动,需加上&lt;uses-permission android:name="android.permission.VIBRATE" /&gt;权限
        notification.defaults = Notification.DEFAULT_LIGHTS; 
        //叠加效果常量
        //notification.defaults=Notification.DEFAULT_LIGHTS|Notification.DEFAULT_SOUND;
        notification.ledARGB = Color.BLUE;   
        notification.ledOnMS =5000; //闪光时间,毫秒
         
        // 设置通知的事件消息   
        CharSequence contentTitle ="督导系统标题"; // 通知栏标题   
        CharSequence contentText ="督导系统内容"; // 通知栏内容   
        Intent notificationIntent =new Intent(MainActivity.this, MainActivity.class); // 点击该通知后要跳转的Activity   
        PendingIntent contentItent = PendingIntent.getActivity(this, 0, notificationIntent, 0);   
        notification.setLatestEventInfo(this, contentTitle, contentText, contentItent);   
         
        // 把Notification传递给NotificationManager   
        notificationManager.notify(0, notification);   
    }
?
    //删除通知    
    private void clearNotification(){
        // 启动后删除之前我们定义的通知   
        NotificationManager notificationManager = (NotificationManager) this 
                .getSystemService(NOTIFICATION_SERVICE);   
        notificationManager.cancel(0);  
 
    }
}

Android之HelloWorld

发布于:2013-12-3 15:42 作者:admin 浏览:1639 分类:Android

在开始Android开发之旅启动之前,首先要搭建环境,然后创建一个简单的HelloWorld。本文的主题如下:

  • 1、环境搭建     
    • 1.1、JDK安装
    • 1.2、Eclipse安装
    • 1.3、Android SDK安装
    • 1.4、ADT安装
    • 1.5、创建AVD
  • 2、HelloWorld

1、环境搭建

1.1、JDK安装

如果你还没有JDK的话,可以去这里下载,接下来的工作就是安装提示一步一步走。设置环境变量步骤如下:

  1. 我的电脑->属性->高级->环境变量->系统变量中添加以下环境变量:
  2. JAVA_HOME值为: D:\Program Files\Java\jdk1.6.0_18(你安装JDK的目录
  3. CLASSPATH值为:.;%JAVA_HOME%\lib\tools.jar;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\bin;
  4. Path:  在开始追加 %JAVA_HOME%\bin;
  5. NOTE:前面四步设置环境变量对搭建Android开发环境不是必须的,可以跳过。

安装完成之后,可以在检查JDK是否安装成功。打开cmd窗口,输入java –version 查看JDK的版本信息。出现类似下面的画面表示安装成功了:

image

图1、验证JDK安装是否成功

1.2、Eclipse安装

如果你还么有Eclipse的话,可以去这里下载,下载如下图所示的Eclipse IDE for Java Developers(92M)的win 32bit版:

image 图2、Eclipse下载

解压之后即可使用。

1.3、Android SDK安装

在Android Developers下载android-sdk_r05-windows.zip,下载完成后解压到任意路径。

  • 运行SDK Setup.exe,点击Available Packages。如果没有出现可安装的包,请点击Settings,选中Misc中的"Force https://..."这项,再点击Available Packages 。
  • 选择希望安装的SDK及其文档或者其它包,点击Installation Selected、Accept All、Install Accepted,开始下载安装所选包
  • 在用户变量中新建PATH值为:Android SDK中的tools绝对路径(本机为D:\AndroidDevelop\android-sdk-windows\tools)。

image图2、设置Android SDK的环境变量

“确定”后,重新启动计算机。重启计算机以后,进入cmd命令窗口,检查SDK是不是安装成功。    
运行 android –h 如果有类似以下的输出,表明安装成功:

image图3、验证Android SDK是否安装成功

1.4、ADT安装

  • 打开 Eclipse IDE,进入菜单中的 "Help" -> "Install New Software"
  • 点击Add...按钮,弹出对话框要求输入Name和Location:Name自己随便取,Location输入http://dl-ssl.google.com/android/eclipse。如下图所示:

image

  • 确定返回后,在work with后的下拉列表中选择我们刚才添加的ADT,我们会看到下面出有Developer Tools,展开它会有Android DDMS和Android Development Tool,勾选他们。 如下图所示:

image

  • 然后就是按提示一步一步next。

完成之后:

  • 选择Window > Preferences...
  • 在左边的面板选择Android,然后在右侧点击Browse...并选中SDK路径,本机为:      
    D:\AndroidDevelop\android-sdk-windows
  • 点击Apply、OK。配置完成。

1.5、创建AVD

为使Android应用程序可以在模拟器上运行,必须创建AVD。

  • 1、在Eclipse中。选择Windows > Android SDK and AVD Manager
  • 2、点击左侧面板的Virtual Devices,再右侧点击New
  • 3、填入Name,选择Target的API,SD Card大小任意,Skin随便选,Hardware目前保持默认值      
  • 4、点击Create AVD即可完成创建AVD

注意:如果你点击左侧面板的Virtual Devices,再右侧点击New ,而target下拉列表没有可选项时,这时候你:

image

    • 然后点击Install Selected按钮,接下来就是按提示做就行了

要做这两步,原因是在1.3、Android SDK安装中没有安装一些必要的可用包(Available Packages)。

2、HelloWorld

  • 通过File -> New -> Project 菜单,建立新项目"Android Project"
  • 然后填写必要的参数,如下图所示:(注意这里我勾选的是Google APIs,你可以选你喜欢的,但你要创建相应的AVD)

image

相关参数的说明:

  1. Project Name: 包含这个项目的文件夹的名称。
  2. Package Name: 包名,遵循JAVA规范,用包名来区分不同的类是很重要的,我用的是helloworld.test。
  3. Activity Name: 这是项目的主类名,这个类将会是Android的Activity类的子类。一个Activity类是一个简单的启动程序和控制程序的类。它可以根据需要创建界面,但不是必须的。
  4. Application Name: 一个易读的标题在你的应用程序上。
  5. 在"选择栏"的 "Use default location" 选项,允许你选择一个已存在的项目。
  • 点击Finish后,点击Eclipse的Run菜单选择Run Configurations…
  • 选择“Android Application”,点击在左上角(按钮像一张纸上有个“+”号)或者双击“Android Application”, 有个新的选项“New_configuration”(可以改为我们喜欢的名字)。
  • 在右侧Android面板中点击Browse…,选择HelloWorld
  • 在Target面板的Automatic中勾选相应的AVD,如果没有可用的AVD的话,你需要点击右下角的Manager…,然后新建相应的AVD。如下图所示:

image

  • 然后点Run按钮即可,运行成功的话会有Android的模拟器界面,如下图所示:

image

 

标签: Android

0

Eclipse中搭建Android开发环境

发布于:2013-12-3 15:18 作者:admin 浏览:1727 分类:Android

1、下载Eclipse3.7,登录http://www.eclipse.org/downloads/,下载Eclipse Classic 3.7:

2、安装ADT插件:下载好Eclipse后解压,运行Eclipse,第一次运行会提示设置workspace;在菜单栏选择Help—Install New Software,在Work with栏输入http://dl-ssl.google.com/android/eclipse/并回车,稍后显示如下图:

选择Select All按钮,单击Next——Next,选中“I Accept ”,再单击Finish,开始安装插件:

安装期间会提示“Warning(此处略去若干字)”,选择OK。安装完成后选择Restart,重启Eclispe。

此时选择Window,会出现如下选项:

并且Preference窗口中会出现Android选项,说明ADT插件安装成功。

3、安装Android SDK:登录http://developer.android.com/sdk/index.html,下载android-sdk_r12-windows.zip:

得到的是一个压缩包,解压到D盘根目录下。这里请注意,解压之后的SDK Manager.exe文件的绝对地址不要带有空格,否则后面启动模拟器的时候会有如下报错:

invalid command-line parameter: Files\android-sdk-windows\tools/emulator-arm.exe.
Hint: use '@foo' to launch a virtual device named 'foo'.
please use -help for more information

这里文件的绝地地址是D:\android-sdk-windows\SDK Manager.exe。

4、运行D:\android-sdk-windows\SDK Manager.exe,会自动检查更新。我们下载的android-sdk_r12-windows.zip。

在上图中,你可以直接选择Install或者选择Accept All之后再选择Install。

5、安装完毕后,关闭SDK Manager.exe。运行Eclipse,选择Window——Preference——Android,在SDK Location选择D:\android-sdk-windows,单击Apply后在Target Name中选择Android 2.3.3,单击OK,以关闭Preference对话框。

6、创建Android模拟器:在Eclipse中选择Window——Android SDK and AVD Manager,在对话框的左边选中Virtual decices,单击右部的New按钮(你可能需要把对话框拉宽才能看见这个按钮),仿照下图填写参数:

单击Create AVD按钮,完成模拟器的创建。

7、在Android SDK and AVD Manager窗口选中Android2.3.3,单击Start,弹出对话框以设定参数。为了使模拟器大小适中,可以仿照下图填写参数

设定参数后,单击Launch,启动模拟器:

上图显示的模拟器我进行了一些设置,如时区选择、语言选择。

至此,环境搭建完毕,可以进行开发了。

 

 

安装文档下载

标签: Android

0