【发布时间】:2013-12-04 05:41:39
【问题描述】:
我想将图像存储在我的数据库中。我还想检查图像和标题是否已经在数据库中。如果是这样,它不会将它们添加到数据库中。这是我的课。
景点
public class Attractions extends ListActivity {
DataBaseHandler db = new DataBaseHandler(this);
ArrayList<Contact> imageArry = new ArrayList<Contact>();
List<Contact> contacts;
ContactImageAdapter adapter;
int ctr, loaded;
int [] landmarkImages={R.drawable.oblation,R.drawable.eastwood,R.drawable.ecopark,R.drawable.circle};
String []landmarkDetails = { "Oblation", "Eastwood", "Ecopark", "QC Circle"};
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_attractions);
ctr = db.checkContact(landmarkDetails[loaded]);
// get image from drawable
/**
* CRUD Operations
* */
// Inserting Contacts
Log.d("Insert: ", "Inserting ..");
for(loaded=0; loaded <landmarkDetails.length;loaded++){
Bitmap image = BitmapFactory.decodeResource(getResources(),
landmarkImages[loaded]);
// convert bitmap to byte
ByteArrayOutputStream stream = new ByteArrayOutputStream();
image.compress(Bitmap.CompressFormat.JPEG, 100, stream);
byte imageInByte[] = stream.toByteArray();
Log.d("Going to load images", "Image "+ loaded);
Log.d("Goind to load objects", "loading");
if(ctr == 0){
Log.d("Nothing Loaded", "Loading Now");
db.addContact(new Contact(landmarkDetails[loaded], imageInByte));}
Log.d(landmarkDetails[loaded], "Loaded!");
image.recycle();
}
loadFromDb();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.attractions, menu);
return true;
}
public void loadFromDb(){
// Reading all contacts from database
contacts = db.getAllContacts();
for (Contact cn : contacts) {
String log = "ID:" + cn.getID() + " Name: " + cn.getName()
+ " ,Image: " + cn.getImage();
// Writing Contacts to log
Log.d("Result: ", log);
//add contacts data in arrayList
imageArry.add(cn);
}
adapter = new ContactImageAdapter(this, R.layout.screen_list,
imageArry);
ListView dataList = (ListView) findViewById(android.R.id.list);
dataList.setAdapter(adapter);
}
public void onPause(){
super.onPause();
}
public void onResume(){
super.onResume();
}
}
它在模拟器上运行良好,但我尝试在我的 S4 上进行测试,然后在尝试了 3 次上这个课程后,它强制停止。我用usb调试试过了,logcat显示java.lang.outofmemoryerror。 logcat 指出了我的联系人图像适配器中的错误。
ContactImageAdapter
public class ContactImageAdapter extends ArrayAdapter<Contact>{
Context context;
int layoutResourceId;
// BcardImage data[] = null;
ArrayList<Contact> data=new ArrayList<Contact>();
public ContactImageAdapter(Context context, int layoutResourceId, ArrayList<Contact> data) {
super(context, layoutResourceId, data);
this.layoutResourceId = layoutResourceId;
this.context = context;
this.data = data;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View row = convertView;
ImageHolder holder = null;
if(row == null)
{
LayoutInflater inflater = ((Activity)context).getLayoutInflater();
row = inflater.inflate(layoutResourceId, parent, false);
holder = new ImageHolder();
holder.txtTitle = (TextView)row.findViewById(R.id.txtTitle);
holder.imgIcon = (ImageView)row.findViewById(R.id.imgIcon);
row.setTag(holder);
}
else
{
holder = (ImageHolder)row.getTag();
}
Contact picture = data.get(position);
holder.txtTitle.setText(picture._name);
//convert byte to bitmap take from contact class
byte[] outImage=picture._image;
ByteArrayInputStream imageStream = new ByteArrayInputStream(outImage);
Bitmap theImage = BitmapFactory.decodeStream(imageStream);
holder.imgIcon.setImageBitmap(theImage);
return row;
}
static class ImageHolder
{
ImageView imgIcon;
TextView txtTitle;
}
}
并指向这条线Bitmap theImage = BitmapFactory.decodeStream(imageStream);
我对管理图像和存储图像知之甚少(几乎没有)。我也启用了android:largeHeap,但多次尝试仍然强制关闭。我希望有人可以帮助我解决这个问题,或者至少向我展示一种将文本和图像存储到 sqlite db 的不同方式。非常感谢!
【问题讨论】:
-
一般建议将文件存储在App Directory(外部)中,并根据需要在数据库中维护文件路径插入图像和从数据库中检索图像是复杂的操作
-
图片有多大?也许您需要在存储之前调整它的大小并压缩它。作为一名业余摄影师,我知道您可以将 jpg 压缩到至少 80%,而对质量没有明显影响,在大多数情况下也可以压缩到 70%(假设您不在同一张图像上多次执行此操作)。您目前拥有 100% 的质量比。如果您调整大小并进行压缩,这可能会对所需的大小产生很大的影响,从而影响所需的内存。
-
检查所有图片有多少mb。如果它更大,你会得到oom error
-
向任何数据库中添加图像或大块,无论您使用的是 SQLite、Oracle 还是 MS SQL Server,都是一个糟糕的想法,它将导致数据库的所有表中的巨大性能损失。
标签: java android sqlite bitmap out-of-memory