Android Studio - База данных SQLite (функция getWritableDatabase) не работает после копирования проекта

Мой проект работал до того, как я скопировал каталог со старого компьютера на новый.

Вот мой logcat:

06-15 22:28:03.169 3205-3205/com.example.moric.pronostic W/System: ClassLoader referenced unknown path: /data/app/com.example.moric.pronostic-2/lib/x86
06-15 22:28:03.674 3205-3205/com.example.moric.pronostic W/System: ClassLoader referenced unknown path: /data/app/com.example.moric.pronostic-2/lib/x86
06-15 22:28:03.830 3205-3205/com.example.moric.pronostic W/art: Before Android 4.1, method android.graphics.PorterDuffColorFilter android.support.graphics.drawable.VectorDrawableCompat.updateTintFilter(android.graphics.PorterDuffColorFilter, android.content.res.ColorStateList, android.graphics.PorterDuff$Mode) would have incorrectly overridden the package-private method in android.graphics.drawable.Drawable
06-15 22:28:03.860 3205-3205/com.example.moric.pronostic E/SQLiteLog: (1) unknown column "betUser" in foreign key definition
06-15 22:28:03.872 3205-3205/com.example.moric.pronostic D/AndroidRuntime: Shutting down VM
--------- beginning of crash
06-15 22:28:03.872 3205-3205/com.example.moric.pronostic E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.example.moric.pronostic, PID: 3205
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.moric.pronostic/com.example.moric.pronostic.MainActivity}: android.database.sqlite.SQLiteException: unknown column "betUser" in foreign key definition (code 1): , while compiling: CREATE TABLE betsTable (betId INTEGER PRIMARY KEY AUTOINCREMENT, betScore TEXT, FOREIGN KEY(betUser) REFERENCES userTable(userId), FOREIGN KEY(betMatch) REFERENCES matchTable(id));
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2416)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476)
    at android.app.ActivityThread.-wrap11(ActivityThread.java)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344)
    at android.os.Handler.dispatchMessage(Handler.java:102)
    at android.os.Looper.loop(Looper.java:148)
    at android.app.ActivityThread.main(ActivityThread.java:5417)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
Caused by: android.database.sqlite.SQLiteException: unknown column "betUser" in foreign key definition (code 1): , while compiling: CREATE TABLE betsTable (betId INTEGER PRIMARY KEY AUTOINCREMENT, betScore TEXT, FOREIGN KEY(betUser) REFERENCES userTable(userId), FOREIGN KEY(betMatch) REFERENCES matchTable(id));
    at android.database.sqlite.SQLiteConnection.nativePrepareStatement(Native Method)
    at android.database.sqlite.SQLiteConnection.acquirePreparedStatement(SQLiteConnection.java:887)
    at android.database.sqlite.SQLiteConnection.prepare(SQLiteConnection.java:498)
    at android.database.sqlite.SQLiteSession.prepare(SQLiteSession.java:588)
    at android.database.sqlite.SQLiteProgram.<init>(SQLiteProgram.java:58)
    at android.database.sqlite.SQLiteStatement.<init>(SQLiteStatement.java:31)
    at android.database.sqlite.SQLiteDatabase.executeSql(SQLiteDatabase.java:1674)
    at android.database.sqlite.SQLiteDatabase.execSQL(SQLiteDatabase.java:1605)
    at com.example.moric.pronostic.DataBaseHandler.onCreate(DataBaseHandler.java:17)
    at android.database.sqlite.SQLiteOpenHelper.getDatabaseLocked(SQLiteOpenHelper.java:251)
    at android.database.sqlite.SQLiteOpenHelper.getWritableDatabase(SQLiteOpenHelper.java:163)
    at com.example.moric.pronostic.MatchDataBase.openDb(MatchDataBase.java:33)
    at com.example.moric.pronostic.MainActivity.onCreate(MainActivity.java:29)
    at android.app.Activity.performCreate(Activity.java:6237)
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1107)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2369)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476) 
    at android.app.ActivityThread.-wrap11(ActivityThread.java) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344) 
    at android.os.Handler.dispatchMessage(Handler.java:102) 
    at android.os.Looper.loop(Looper.java:148) 
    at android.app.ActivityThread.main(ActivityThread.java:5417) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

Вот мой MainActivity, он работает без openDb() а также closeDb() функции, я проверял это. Это нормально, что база данных не используется в моем коде, это тест.

import android.app.Activity;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.Toast;

import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.Vector;

public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    //Création de la base de données
    MatchDataBase matchDataBase = new MatchDataBase(this);
    Vector<Match> matchList = new Vector<Match>();
        matchList.addElement(new Match(1, "France", "Roumanie", "10/06/2016 - 15h00", new Score(2, 1), "Poule A"));
        matchList.addElement(new Match(2, "France", "Irlande", "12/06/2016 - 11h00", new Score(1, 0), "Quarts"));
        matchList.addElement(new Match(3, "France", "Allemagne", "15/06/2016 - 15h30", "Demi-finale"));
        matchList.addElement(new Match(4, "France", "Angleterre", "18/06/2016 - 18h00", "Finale"));
    matchDataBase.openDb();
    matchDataBase.closeDb();

}
}

Вот мой DataBaseHandler (класс, который расширяет SQLiteOpenHelper):

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;

public class DataBaseHandler extends SQLiteOpenHelper {

public DataBaseHandler(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) {
    super(context, name, factory, version);
}

@Override
public void onCreate(SQLiteDatabase db) {
    db.execSQL(TABLE_MATCH_CREATE);
    db.execSQL(TABLE_USER_CREATE);
    db.execSQL(TABLE_BETS_CREATE);
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
    db.execSQL(TABLE_MATCH_DROP);
    db.execSQL(TABLE_USER_DROP);
    db.execSQL(TABLE_BETS_DROP);
    onCreate(db);
}

//Caractéristiques de la table relative aux Matches, indépendants des utilisateurs
public static final String TABLE_MATCH = "matchTable";
public static final String MATCH_KEY = "id";
public static final String DATE = "date";
public static final String INFO = "info";
public static final String SCORE = "score";
public static final String TEAMDOM = "teamDom";
public static final String TEAMEXT = "teamExt";
public static final String TABLE_MATCH_CREATE = "CREATE TABLE " + TABLE_MATCH + " ("
                + MATCH_KEY + " INTEGER PRIMARY KEY AUTOINCREMENT, "
                + DATE + " TEXT, "
                + INFO + " TEXT, "
                + SCORE + " TEXT, "
                + TEAMDOM + " TEXT, "
                + TEAMEXT + " TEXT);";
public static final String TABLE_MATCH_DROP = "DROP TABLE IF EXISTS " + TABLE_MATCH + ";";

//Caractéristiques de la table relative aux Utilisateurs
public static final String TABLE_USER= "userTable";
public static final String USER_KEY = "userId";
public static final String USERNAME = "userName";
public static final String TABLE_USER_CREATE = "CREATE TABLE " + TABLE_USER + " ("
        + USER_KEY + " INTEGER PRIMARY KEY AUTOINCREMENT, "
        + USERNAME + " TEXT); ";
public static final String TABLE_USER_DROP = "DROP TABLE IF EXISTS " + TABLE_USER + ";";

//Caractéristiques de la table relative aux Pronostics
public static final String TABLE_BETS= "betsTable";
public static final String BET_KEY = "betId";
public static final String BET_SCORE = "betScore";
public static final String BET_USER = "betUser";
public static final String BET_MATCH = "betMatch";
public static final String TABLE_BETS_CREATE = "CREATE TABLE " + TABLE_BETS + " ("
        + BET_KEY + " INTEGER PRIMARY KEY AUTOINCREMENT, "
        + BET_SCORE + " TEXT, "
        + BET_USER + "INTEGER, "
        + BET_MATCH + "INTEGER, "
        + "FOREIGN KEY(" + BET_USER + ") REFERENCES " + TABLE_USER + "(" + USER_KEY + "), "
        + "FOREIGN KEY(" + BET_MATCH + ") REFERENCES " + TABLE_MATCH + "(" + MATCH_KEY + ")); ";
public static final String TABLE_BETS_DROP = "DROP TABLE IF EXISTS " + TABLE_BETS + ";";

}

И вот мой класс DataBase:

import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.support.annotation.Nullable;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.Vector;

public class MatchDataBase {

    private final static int VERSION = 1;
    private final static String NOM = "matches.db";
    private static SimpleDateFormat dateFormat = new SimpleDateFormat("dd/MM/yyyy' - 'HH'h'mm");

    private SQLiteDatabase mDb;
    private DataBaseHandler mHandler;
    public SQLiteDatabase getDb() {
        return mDb;
    }
    //Méthodes de création, d'ouverture et de fermeture de la base de données.
    public MatchDataBase(Context pContext) {
        this.mHandler = new DataBaseHandler(pContext, NOM, null, VERSION);
    }
    public void openDb() {
        //Pour ouvrir l'accès à la base de données.
        //getWritableDatabase ferme également la dernière base.
        this.mDb = this.mHandler.getWritableDatabase();
    }
    public void closeDb() {
        //Pour fermer l'accès à la base de données.
        this.mDb.close();
    }

    //Caractéristiques de la table des matches.
    private static final String TABLE_MATCH = "matchTable";
    private static final String MATCH_KEY = "id";
    private static final int NUM_COL_MATCH_KEY = 0;
    private static final String TEAMDOM = "teamDom";
    private static final int NUM_COL_TEAMDOM = 1;
    private static final String TEAMEXT = "teamExt";
    private static final int NUM_COL_TEAMEXT = 2;
    private static final String DATE = "date";
    private static final int NUM_COL_DATE = 3;
    private static final String SCORE = "score";
    private static final int NUM_COL_SCORE = 4;
    private static final String INFO = "info";
    private static final int NUM_COL_INFO = 5;

    //Fonctions de la table des matches
    //Cette fonction permet d'ajouter un match à la base de données.
    public void addMatch(Match match) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.MATCH_KEY, match.getId());
        value.put(MatchDataBase.DATE, match.getDate());
        value.put(MatchDataBase.INFO, match.getInfo());
        value.put(MatchDataBase.SCORE, match.getScore().toString());
        value.put(MatchDataBase.TEAMDOM, match.getTeamDom());
        value.put(MatchDataBase.TEAMEXT, match.getTeamExt());
        //L'on veut que le match déterminé remplace le précédent, donc on supprime celui qui existait déjà.
        this.deleteMatch(match.getId());
        mDb.insert(MatchDataBase.TABLE_MATCH, null, value);
    }
    //Cette fonction permet de supprimer un match de la base de données.
    public void deleteMatch(long id) {
        mDb.delete(TABLE_MATCH, MATCH_KEY + " = ?", new String[]{String.valueOf(id)});
    }
    //Cette fonction permet de modifier les informations d'un match, connaissant son ID.
    public void updateMatch(Match match) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.DATE, match.getDate());
        value.put(MatchDataBase.INFO, match.getInfo());
        value.put(MatchDataBase.SCORE, match.getScore().toString());
        value.put(MatchDataBase.TEAMDOM, match.getTeamDom());
        value.put(MatchDataBase.TEAMEXT, match.getTeamExt());
        mDb.update(TABLE_MATCH, value, MATCH_KEY + " = ?", new String[]{String.valueOf(match.getId())});
    }
    @Nullable //Cette fonction permet de récupérer la première ligne d'un cursor sous la forme d'un objet java de la classe Match.
    private Match cursorToMatch(Cursor c) {
        if (c.getCount() == 0) {
            return null;
        }
        //On se place au premier élément
        c.moveToFirst();

        Match returnedMatch = new Match();
        returnedMatch.setId(c.getLong(NUM_COL_MATCH_KEY));
        returnedMatch.setTeamDom(c.getString(NUM_COL_TEAMDOM));
        returnedMatch.setTeamExt(c.getString(NUM_COL_TEAMEXT));
        returnedMatch.setDate(c.getString(NUM_COL_DATE));
        String writtenScore = c.getString(NUM_COL_SCORE);
        returnedMatch.setScore(Score.convertStringToScore(writtenScore));
        returnedMatch.setInfo(c.getString(NUM_COL_INFO));

        c.close();
        return returnedMatch;
    }
    //Cette fonction permet de récupérer un match à partir de son ID.
    public Match getMatchWithId(long id) {
        Cursor c = mDb.query(TABLE_MATCH, new String[]{MATCH_KEY, TEAMDOM, TEAMEXT, DATE, SCORE, INFO},
                MATCH_KEY + "= ?", new String[]{String.valueOf(id)}, null, null, null);

        return cursorToMatch(c);
    }
    //Cette fonction permet de récupérer tous les matches qui auront lieu après une date donnée, dans une table de type ArrayList déjà créée.
    public void getMatchesAfterDate(String currentDateString, ArrayList<Match> listToComplete) {

        Date currentDate = new Date();
        Cursor c = mDb.query(TABLE_MATCH, new String[]{MATCH_KEY, TEAMDOM, TEAMEXT, DATE, SCORE, INFO},
                null, null, null, null, null);
        try {
            currentDate = dateFormat.parse(currentDateString);
        } catch (ParseException e) {
            e.printStackTrace();
        }
        if (c.getCount() == 0) {
        }
        for(c.moveToFirst(); !c.isAfterLast(); c.moveToNext()) {
            try {
                Date dateMatch = dateFormat.parse(c.getString(NUM_COL_DATE));

                if (dateMatch.after(currentDate)){
                    Match thisMatch = new Match();
                    thisMatch.setId(c.getLong(NUM_COL_MATCH_KEY));
                    thisMatch.setTeamDom(c.getString(NUM_COL_TEAMDOM));
                    thisMatch.setTeamExt(c.getString(NUM_COL_TEAMEXT));
                    thisMatch.setDate(c.getString(NUM_COL_DATE));
                    String writtenScore = c.getString(NUM_COL_SCORE);
                    thisMatch.setScore(Score.convertStringToScore(writtenScore));
                    thisMatch.setInfo(c.getString(NUM_COL_INFO));

                    listToComplete.add(thisMatch);
                }
            } catch (ParseException e) {
                e.printStackTrace();
            }
        }
        c.close();
    }

    //Caractéristiques de la table des utilisateurs.
    public static final String TABLE_USER= "userTable";
    public static final String USER_KEY = "userId";
    private static final int NUM_COL_USER_KEY = 0;
    public static final String USERNAME = "userName";
    private static final int NUM_COL_USERNAME = 1;

    //Fonctions de la table des utilisateurs.
    //Cette fonction permet d'ajouter un utilisateur à la base de données.
    public void addUser(User user) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.USER_KEY, user.getUserId());
        value.put(MatchDataBase.USERNAME, user.getUserName());
        this.deleteUser(user.getUserId());
        mDb.insert(MatchDataBase.TABLE_USER, null, value);
    }
    //Cette fonction permet de supprimer un match de la base de données.
    public void deleteUser(long id) {
        mDb.delete(TABLE_USER, USER_KEY + " = ?", new String[]{String.valueOf(id)});
    }
    //Cette fonction permet de modifier les informations d'un utilisateur, connaissant son ID.
    public void updateUser(User user) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.USERNAME, user.getUserName());
        mDb.update(TABLE_USER, value, USER_KEY + " = ?", new String[]{String.valueOf(user.getUserId())});
    }
    @Nullable //Cette fonction permet de récupérer la première ligne d'un cursor sous la forme d'un objet java de la classe User.
    private User cursorToUser(Cursor c) {
        if (c.getCount() == 0) {
            return null;
        }
        //On se place au premier élément
        c.moveToFirst();

        User returnedUser = new User();
        returnedUser.setUserId(c.getLong(NUM_COL_USER_KEY));
        returnedUser.setUserName(c.getString(NUM_COL_USERNAME));

        c.close();
        return returnedUser;
    }
    //Cette fonction permet de récupérer un utilisateur à partir de son ID.
    public User getUserWithId(long id) {
        Cursor c = mDb.query(TABLE_USER, new String[]{USER_KEY, USERNAME},
                USER_KEY + "= ?", new String[]{String.valueOf(id)}, null, null, null);
        return cursorToUser(c);
    }

    //Caractéristiques de la table des pronostics.
    public static final String TABLE_BETS= "betsTable";
    public static final String BET_KEY = "betId";
    private static final int NUM_COL_BET_KEY = 0;
    public static final String BET_SCORE = "betScore";
    private static final int NUM_COL_BET_SCORE = 1;
    public static final String BET_USER = "betUser";
    private static final int NUM_COL_BET_USER = 2;
    public static final String BET_MATCH = "betMatch";
    private static final int NUM_COL_BET_MATCH = 3;

    //Fonctions de la table des pronostics.
    //Cette fonction permet d'ajouter un utilisateur à la base de données.
    public void addBet(Bet bet) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.BET_KEY, bet.getBetId());
        value.put(MatchDataBase.BET_SCORE, bet.getBetScore().toString());
        value.put(MatchDataBase.BET_USER, bet.getBetUser());
        value.put(MatchDataBase.BET_MATCH, bet.getBetMatch());
        this.deleteUser(bet.getBetId());
        mDb.insert(MatchDataBase.TABLE_USER, null, value);
    }
    //Cette fonction permet de supprimer un match de la base de données.
    public void deleteBet(long id) {
        mDb.delete(TABLE_BETS, BET_KEY + " = ?", new String[]{String.valueOf(id)});
    }
    //Cette fonction permet de modifier les informations d'un utilisateur, connaissant son ID.
    public void updateBet(Bet bet) {
        ContentValues value = new ContentValues();
        value.put(MatchDataBase.BET_SCORE, bet.getBetScore().toString());
        value.put(MatchDataBase.BET_USER, bet.getBetUser());
        value.put(MatchDataBase.BET_MATCH, bet.getBetMatch());
        mDb.update(TABLE_BETS, value, BET_KEY + " = ?", new String[]{String.valueOf(bet.getBetId())});
    }
    @Nullable //Cette fonction permet de récupérer la première ligne d'un cursor sous la forme d'un objet java de la classe User.
    private Bet cursorToBet(Cursor c) {
        if (c.getCount() == 0) {
            return null;
        }
        //On se place au premier élément
        c.moveToFirst();

        Bet returnedBet = new Bet();
        returnedBet.setBetId(c.getLong(NUM_COL_BET_KEY));
        String writtenScore = c.getString(NUM_COL_BET_SCORE);
        returnedBet.setBetScore(Score.convertStringToScore(writtenScore));
        returnedBet.setBetUser(c.getLong(NUM_COL_BET_USER));
        returnedBet.setBetMatch(c.getLong(NUM_COL_BET_MATCH));

        c.close();
        return returnedBet;
    }
    //Cette fonction permet de récupérer un utilisateur à partir de son ID.
    public Bet geBetWithId(long id) {
        Cursor c = mDb.query(TABLE_BETS, new String[]{BET_KEY, BET_SCORE, BET_USER, BET_MATCH},
                BET_KEY + "= ?", new String[]{String.valueOf(id)}, null, null, null);
        return cursorToBet(c);
    }

}

Вы понимаете проблему? Это довольно странно, потому что (я повторяю) я получил тот же код на другом компьютере, и не было никаких проблем...

Спасибо,

Moric

1 ответ

Я не знаю, какие изменения вы сделали или нет, но ваш запрос на создание таблицы выглядит так:

Ваш запрос

    CREATE TABLE betsTable (betId INTEGER PRIMARY KEY AUTOINCREMENT, betScore TEXT, FOREIGN KEY(betUser) REFERENCES userTable(userId), FOREIGN KEY(betMatch) REFERENCES matchTable(id));

Там нет столбца с именем "betUser", поэтому сначала вам нужно создать столбец, а затем вы можете назначить ему внешний ключ,

Проверьте ниже новый запрос:

Правильный запрос для этого

  CREATE TABLE betsTable (betId INTEGER PRIMARY KEY AUTOINCREMENT, betScore TEXT, betUser TEXT, betMatch TEXT, FOREIGN KEY(betUser)  REFERENCES userTable(userId), FOREIGN KEY(betMatch) REFERENCES matchTable(id))
Другие вопросы по тегам