`
kevinffk
  • 浏览: 33372 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

【原创】Active Android的一些见解与研究

阅读更多
这几天一直在找一个针对android数据库sqlite的一个orm框架,起初找了一个google上面的项目:android active record,无奈,demo可以运行,但自己写了一个应用,却怎么也跑不起来,原因是创建DB的sql语句有错误。于是debug了一番,然后发现错误不仅仅是这些。要我改原作者的代码是不可能的,更何况他的代码还没有完善与健壮。尔后几天一直在找有没有相应的框架可替代。找了个ormlite和sqlitegen,都不竟如人意,并不是说它们的框架设计得不合理,而是就简约和敏捷开发而言。orm的缺点就暴露了出来,令外,手机软件一般追求的是代码简洁,而J2EE的这一套显然在这里无法适用。终于,被我找到了一个商业版的active android。思想跟我找到那个google项目一样,都是基于RoR的activeRecord模式设计的。我下了一个试用版(正式版可要付$19.99 T_T)。再反编译一下,天,居然没有代码混淆。于是,看了下代码的实现。研究了大概1天半的时间。它的实例也跑过了,再把它应用于自己的小例子中,终于,跨过重重的困难。勉强OK了。但这里有个问题摆在眼前,试用版的,一些功能会有限制,果然,在它代码里面,我看到了它仅限于运行在模拟器。而且生成的DB名字不可变更。DB命名的那段逻辑在它自己实现的application上面call。且这些代码封装在它的jar包里面。这样就意味着application的命名要跟jar包里面的一致,否则无法应用。看来,要改它的代码是必然的,不然这些限制将不利于自己的应用(在这里,希望一些版权人士不要喷我,小弟只是抱着学习的态度去研究这框架,并未用于商业的开发)。于是,经过昨天的熬夜奋战,终于解决了这一问题,改了它的实现接口,在应用端方法只需写一个类继承那个jar包的application就可以了。另外,运行于真机器也没问题。好了,不多说了,上一个我的小例子ActiveAndriodDemo,方便大家理解。
1.AndroidManifest.xml上面的改动。
<application android:icon="@drawable/icon" android:label="@string/app_name" android:name="com.kevin.PersonApp">
...</application>

2.创建一个跟上面命名一样的PersonApp类,并继承activeAndroid.jar下面的applicatiion类。
package com.kevin;

import com.activeandroid.Application;

public class PersonApp extends Application {

	public PersonApp() {
                  //naming DB name and DB version		
                  super("PersonApp.db", 1);
	}

}


3.domain类(Person.java)
package com.kevin.entity;

import java.util.List;
import android.content.Context;
import com.activeandroid.ActiveRecordBase;
import com.activeandroid.annotation.Column;
import com.activeandroid.annotation.Table;

@Table(name = "Person")             //naming table name
public class Person extends ActiveRecordBase<Person>{

	public Person(Context context) {
		super(context);
	}
	
         //naming column name, just ignore length value setting
         @Column(name = "Name")
	public String name;
	@Column(name = "age")
	public String age;
	
	public static List<Person> getAll(Context context) {
		return Person.query(context, Person.class, null, null, null);
	}

}

4.创建一个适配器PeopleAdapter.java
package com.kevin.adapter;

import java.util.ArrayList;
import java.util.List;
import com.kevin.R;
import com.kevin.entity.Person;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

public class PeopleAdapter extends BaseAdapter {

	private LayoutInflater layoutInflater;
	private List<Person> people = new ArrayList<Person>();
	
	public PeopleAdapter(Context context, List<Person> personList) {
		layoutInflater = LayoutInflater.from(context);
		people.addAll(personList);
	}

	@Override
	public int getCount() {
		return people.size();
	}

	@Override
	public Object getItem(int position) {
		return people.get(position);
	}

	@Override
	public long getItemId(int position) {
		return people.get(position).getId();
	}

	@Override
	public View getView(int position, View convertView, ViewGroup parent) {
		ViewHolder holder = new ViewHolder();
		Person person = (Person) getItem(position);
		if (convertView == null) {
			convertView = layoutInflater.inflate(R.layout.list_item, null);
			holder.id = (TextView) convertView.findViewById(R.id.p_id);
			holder.name = (TextView) convertView.findViewById(R.id.p_name);
			holder.age = (TextView) convertView.findViewById(R.id.p_age);
			convertView.setTag(holder);
		} else {
			holder = (ViewHolder) convertView.getTag();
		}
		holder.id.setText(person.getId() + "");
		holder.name.setText(person.name);
		holder.age.setText(person.age);
		return convertView;
	}
	
	class ViewHolder{
		TextView id;
		TextView name;
		TextView age;
	}
}

5.Main.java主activity
package com.kevin;

import java.util.List;
import com.kevin.adapter.PeopleAdapter;
import com.kevin.entity.Person;
import android.app.AlertDialog;
import android.app.Dialog;
import android.app.ListActivity;
import android.content.DialogInterface;
import android.os.Bundle;
import android.view.ContextMenu;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.ContextMenu.ContextMenuInfo;
import android.view.MenuItem;
import android.view.View;
import android.widget.EditText;
import android.widget.AdapterView.AdapterContextMenuInfo;

public class Main extends ListActivity {
	
	private PeopleAdapter peopleAdapter;
	private static final int EDIT_DIALOG = 1;
	private static final int ADD_MENU = 0;
	private EditText editName;
	private EditText editAge;
	private Person person;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        loadList();
        registerForContextMenu(getListView());
    }

	@Override
	public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) {
		super.onCreateContextMenu(menu, v, menuInfo);
		MenuInflater inflater = getMenuInflater();
		inflater.inflate(R.menu.item_menu, menu);
	}

	@Override
	public boolean onContextItemSelected(MenuItem item) {
		AdapterContextMenuInfo info = (AdapterContextMenuInfo) item.getMenuInfo();
		switch (item.getItemId()) {
		case R.id.edit: {
			editPerson(info.id);
			return true;
		}
		case R.id.delete: {
			deletePerson(info.id);
			return true;
		}
		default:
			return super.onContextItemSelected(item);
		}
	}

	@Override
	public boolean onCreateOptionsMenu(Menu menu) {
		menu.add(0, ADD_MENU, 0, "Add Person");
		return true;
	}

	@Override
	public boolean onMenuItemSelected(int featureId, MenuItem item) {
		switch (item.getItemId()) {
		case ADD_MENU:
			addPerson();
			return true;
		default:
			return super.onMenuItemSelected(featureId, item);
		}
	}

	@Override
	protected Dialog onCreateDialog(int id) {
		switch (id) {
		case EDIT_DIALOG: {
			LayoutInflater inflater = LayoutInflater.from(this);
			View textEntryView = inflater.inflate(R.layout.alert_dialog, null);
			editName = (EditText) textEntryView.findViewById(R.id.name);
			editAge = (EditText) textEntryView.findViewById(R.id.age);
			AlertDialog.Builder builder = new AlertDialog.Builder(Main.this);
			builder.setView(textEntryView);
			String title = "Add Person";
			if (person != null) {
				title = "Edit Person";
				editName.setText(person.name);
				editAge.setText(person.age);
			}
			builder.setTitle(title);
			
			builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
				
				@Override
				public void onClick(DialogInterface dialog, int which) {
					String name = editName.getText().toString();
					String age = editAge.getText().toString();
					if(person == null) {
						person = new Person(Main.this);
					}
					person.name = name;
					person.age = age;
					person.save();
					loadList();
					removeDialog(EDIT_DIALOG);
				}
			});
			builder.setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() {
				@Override
				public void onClick(DialogInterface dialog, int which) {
					removeDialog(EDIT_DIALOG);
				}
			});
			return builder.create();
		}
		default:
			return super.onCreateDialog(id);
		}

	}
	
	private void loadList() {
		List<Person> personList = Person.getAll(this);
		peopleAdapter = new PeopleAdapter(this, personList);
		setListAdapter(peopleAdapter);
	}
	
	private void editPerson(long id) {
		person = Person.load(this, Person.class, id);
		showDialog(EDIT_DIALOG);
	}
	
	private void deletePerson(long id) {
		Person.delete(this, Person.class, id);
		loadList();
	}
	
	private void addPerson() {
		person = null;
		showDialog(EDIT_DIALOG);
	}
}

由于例子较长,部分代码(渲染界面和menu)忽略。
在这里总结一下ActiveAndroid。
优点:  方便,不用写sql,领域驱动模式可以使CRUD封装到domain上面,还可以扩展条件查询,支持简单的外键功能。
缺点:  每个表都默认有Id字段,并且是自增长,不可以自定义主键。不适应于小项目,性能较差。annotation的信息较小,生成的DDL十分的笼统。
修改:可以应用于真机器上面。可自定义数据库名或版本名,并且可自定义application name。

还有个小技巧可以教大家:
android 项目引入lib
1.在project上面添加"libs"
2.将library copy到libs下面
3.将library右键添加入项目
这样,lib会随着project一起安装。
   
  • 大小: 15.4 KB
  • 大小: 7.2 KB
  • 大小: 8 KB
分享到:
评论
1 楼 whawei 2014-04-23  
ActiveAndriodDemo 有工程源码吗?可否共享?

相关推荐

Global site tag (gtag.js) - Google Analytics