0

ListView に表示される sqlite エントリの選択があり、並列配列にそれらの主キーがあります。ユーザーがアイテムを選択すると、そのアイテムはエディター アクティビティに送信されます (ユーザーはそのエントリを編集できます)。私がプログラムした方法では、Editor アクティビティ クラスは、Cursor の moveToPosition 関数を使用して、行番号 (つまり位置) を使用してデータベースからエントリを取得します。

エントリの主キーを使用して行番号を調べるにはどうすればよいですか?

編集:
私は最初にデータベースからエントリの選択を取得し、次に戻ってそれらのエントリの主キーを取得する解決策を見つけました。エントリを 1 つの配列に格納し、主キーを別の配列に格納します。

コードを見た後、より明確な回答が得られることを願っています。これは機能しているようですが、よりエレガントなソリューションを探しています。

関連するコードは次のとおりです。

CalendarMenu.java

@Override
public void onCreate(Bundle savedInstance){
  incomeManager = new SQLManagerIncome(this);
incomeManager.open();

  //This method (see below) gets all of the SQLite entries between two dates
    incomeArray = incomeManager.getDatedEntriesString(
            currentDay.getTimeInMillis(), 
              nextDay.getTimeInMillis());

  //This method (see below) gets the primary keys of all the entries retrieved 
  //in the previous method

    incomeArrayKeys = incomeManager.getDatedKeys(
           currentDay.getTimeInMillis(),
            nextDay.getTimeInMillis());
    incomeManager.close();

    // Sending the SQL info to the ListViews
    incomeAdapter = new ArrayAdapter<String>(this,
            android.R.layout.simple_list_item_1, incomeArray);
    dayList.setAdapter(incomeAdapter);
}


    @Override
public void onItemClick(AdapterView<?> listView, View view, int pos, long id) {
    Bundle basket = new Bundle();
    // we need to send the position of an entry inside the entire database to the
    // Editor class. So we'll use the primary key of an entry to retrieve its position.

    switch (listView.getId()) {
    case R.id.dayList:
        int entryPos = 0;
        //incomeArrayKeys holds all of the primary keys. It is a parallel
        //array to the array that is displayed in the listview 
        //(using a standard ArrayAdapter)
            incomeManager.open();
            entryPos = incomeManager.findPosByKey(incomeArrayKeys[pos]);
            incomeManager.close();

        basket.putLong("index", entryPos);
                    Class dayClass;
            try {
                dayClass = Class.forName("com.shulim.maaser.EditIncome");
                Intent dayIntent = new Intent(CalendarMenu.this, dayClass);
                dayIntent.putExtras(basket);
                startActivityForResult(dayIntent, 0);
            } catch (ClassNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
        break;
}

SQLManagerIncome.java

public String[] getDatedEntriesString(Long firstDay, Long lastDay){
    String[] columns = new String[] { KEY_ROWID, KEY_DESCRIPTION,
            KEY_AMOUNT, KEY_DATE, KEY_OWED };
    String selection = KEY_DATE + " >=" + firstDay + " AND " + KEY_DATE
            + "<" + lastDay;
    Cursor c = ourDatabase.query(INCOME_TABLE, columns, selection, null, null, null, null);
    int iDescription = c.getColumnIndex(KEY_DESCRIPTION);
    int iAmount = c.getColumnIndex(KEY_AMOUNT);
    ArrayList<String> keys = new ArrayList<String>();

    for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
        Long dateMillis = c.getLong(3);
        final Calendar cal = Calendar.getInstance();
        cal.setTimeInMillis(dateMillis);
        int mYear = cal.get(Calendar.YEAR);
        int mMonth = cal.get(Calendar.MONTH)+1;
        int mDay = cal.get(Calendar.DAY_OF_MONTH);
        String date = mMonth + "-" + mDay + "-" + mYear;
        String result = "Desc: " + c.getString(iDescription) 
                + ". Amount: $"
                + c.getString(iAmount) + ". Date: "
                + date + "\n";
        keys.add(result);
    }
    String[] stringDateKeys = new String[keys.size()];
    stringDateKeys = keys.toArray(stringDateKeys);
    c.close();
    return stringDateKeys;
}

public Integer[] getDatedKeys(Long firstDay, Long lastDay) {
    String[] columns = new String[] { KEY_ROWID, KEY_DESCRIPTION,
             KEY_AMOUNT, KEY_DATE };
    String selection = KEY_DATE + " >=" + firstDay + " AND " + KEY_DATE
            + "<" + lastDay;
    Cursor c = ourDatabase.query(INCOME_TABLE, columns, selection, null,
            null, null, null);
    int iRow = c.getColumnIndex(KEY_ROWID);
    ArrayList<Integer> keys = new ArrayList<Integer>();

    for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
        int result = c.getInt(iRow);
        keys.add(result);
    }

        Integer[] intKeys = new Integer[keys.size()];
        intKeys = keys.toArray(intKeys);
        c.close();
        return intKeys;
}

public int findPosByKey(Integer integer) {
    String[] columns = new String[] { KEY_ROWID, KEY_DESCRIPTION,
            KEY_AMOUNT, KEY_DATE };
    Cursor c = ourDatabase.query(INCOME_TABLE, columns, null, null,
            null, null, null);
    int iRow = c.getColumnIndex(KEY_ROWID);
    int position = 0;
    for (c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
        if (c.getInt(iRow) == integer) {
            break;
        }
        position++;
    }
    c.close();
    return position;
}   
4

3 に答える 3

2

私がプログラムした方法では、Editor アクティビティ クラスは、Cursor の moveToPosition 関数を使用して、行番号 (つまり位置) を使用してデータベースからエントリを取得します。

カーソルの moveToPosition 関数を使用する必要はありません。代わりに、はるかに優れた解決策があります。既に使用可能な主キーを使用します。主キーを使用すると、必要な行のみを返す非常に単純で効率的な SQL クエリが可能になります。列名はわかりませんが、この行をロードする SQL クエリの例は次のようになります。

SELECT foo, bar, baz
FROM FooBar
WHERE pk = userSelectedPrimaryKey;

whereuserSelectedPrimaryKeyは、クエリ ビルダーを介してステートメントに挿入されます。

もちろん、これは大まかなスケッチですが、ポイントは、このクエリが正確に 1 つの行を返すことです。これは、ユーザーが選択した主キー値を持つ唯一の行です。これはより単純で、不要な行を反復する必要がないため、パフォーマンスが向上します。

于 2012-04-26T02:57:15.610 に答える
0

最初にデータベースからエントリの選択を取得し、次に戻ってそれらのエントリの主キーを取得するという解決策を見つけました。エントリを 1 つの配列に格納し、主キーを別の配列に格納します。質問に投稿したコード I を参照してください。

これは今のところうまくいくようですが、よりエレガントなソリューションを探しています。より最適化されたものを期待しています。

于 2012-04-29T08:22:12.203 に答える
0

この値を呼び出してに渡すことI have their primary keys in a parallel array.により、この配列から選択された項目の主キーを取得するオプションがあると述べました。Adam Mihalcin が回答で述べたように、この主キーを where 条件に渡すことで、編集したい行を簡単に取得できます。int pk= array[position]EditorActivityEditorActivitySQL query

于 2012-04-26T05:51:26.043 に答える