Diskuze: SQL Character set
V předchozím kvízu, Online test znalostí SQL a databází, jsme si ověřili nabyté zkušenosti z kurzu.
Člen
Zobrazeno 8 zpráv z 8.
//= Settings::TRACKING_CODE_B ?> //= Settings::TRACKING_CODE ?>
V předchozím kvízu, Online test znalostí SQL a databází, jsme si ověřili nabyté zkušenosti z kurzu.
Ahoj, to tvé kódování (CHARACTER SET) je nějaké divoké. Jelikož chceš nastavit porovnání (COLLATE) na nějaký typ UTF-8, chtělo by to i odpovídající kódování.
Nevím, jak to máš vymyšlené, ale lehce to zavání chybným návrhem. Pokud v tabulce ještě nejsou data, doporučil bych změnu kódování a následnou změnu porovnávání (což požaduješ).
Nevím, zda jsem byl pochopen. Databáze je naplněna. Při spuštění dotazu z phpmyadmin konzole to vrací korektní výsledky. Pouze když to pustím ze svého PHP skriptu, tak dojde k té chybě. Standardně používám porovnávání utf8_bin kvůli přesnému vyhledávání češtiny. (tak je to nastaveno na tabulce). Nicméně pro určitý typ výsledku potřebuji porovnávání utf8mb4.
Návrh mohu mít špatně, zase tak moc tomu nerozumím.
Nicméně se domnívám, že dotaz zní - jak mu nastavit ten character set jinak, než binary? Případně neskrývá se za onou hláškou něco jiného?
Tak když to zadám natvrdo před hledaný text, tak to funguje:
SELECT `cesky`, `madarsky`, `vyslovnost`
FROM `madarska_slovicka`, `ceska_slovicka`, `vazebni_tabulka_slovicka`
WHERE (`ID_madarsky` = `ID_madarsky_vazba` AND `ID_cesky_vazba` = `ID_cesky` AND `cesky` LIKE _utf8mb4"rec" COLLATE utf8mb4_general_ci)
ORDER BY `cesky`
LIMIT 30
Tedy to krásně najde jak "rec", tak i "řeč".
Jenže já tam potřebuji dosazovat proměnnou. A když tam dám místo řetězce tu proměnnou, tak to hlásí chybu syntaxe
$slovicka = Db::queryAll('
SELECT `cesky`, `madarsky`, `vyslovnost`
FROM `madarska_slovicka`, `ceska_slovicka`, `vazebni_tabulka_slovicka`
WHERE (`ID_madarsky` = `ID_madarsky_vazba` AND `ID_cesky_vazba` = `ID_cesky` AND `cesky` LIKE _utf8mb4? COLLATE utf8mb4_general_ci)
ORDER BY `cesky`
LIMIT 30
', $heslo1);
Poradíte prosím, jak to s tím otazníkem zadat správně?
Tady nastává problém s těmi daty. Když se změní kódování, může se z textů stát "rozsypaný čaj," což není přípustné. Spusť si v PHPMyAdmin tento dotaz:
SELECT CCSA.character_set_name FROM information_schema.`TABLES` T,
information_schema.`COLLATION_CHARACTER_SET_APPLICABILITY` CCSA
WHERE CCSA.collation_name = T.table_collation
AND T.table_schema = "nazev-databaze"
AND T.table_name = "nazev-tabulky";
Myslím si, že ti to vrátí právě "binary." Je to kvůli tomu, že jsi kódování nenastavil již při vytváření tabulky/databáze a výchozím je právě "binary."
Co se týče řešení, můžeš zkusit zálohovat danou tabulku a pokusit se převést jednotlivé sloupce na požadované kódování "utf8." Nějaké čtení je například tady. Ale každopádně bez zálohy nic nezkoušet.
Díky.
Výsledek je character_set_name: utf8
Ale nemyslím si, že by se to rozsypalo, když to funguje i v PHP. Jde jen o to, jak do toho dostat ty otazníky místo proměnné místo přímého textu.
Když napíšu přímo
LIKE _utf8mb4"rec" COLLATE utf8mb4_general_ci
, tak to funguje. Já potřebuji ten řetězec "řeč" nahradit proměnnou.
To se dělá otazníkem a následně uvedenou proměnnou. Jenže když
napíšu:
LIKE _utf8mb4"?" COLLATE utf8mb4_general_ci nebo
LIKE _utf8mb4(?) COLLATE utf8mb4_general_ci
tak to nefunguje a hodí chybu syntaxe.
Když to napíšu do proměnné $retezec = "_utf8mb4$promenna" nebo
$retezec = "_utf8mb4'$promenna'" nebo
$retezec = '_utf8mb4"'.$promenna.'"'
tak to vyhledá celý řetězec a to _utf8mb4 to ignoruje ..
Takže jak jinak to zapsat?
Mně šlo o to, aby sis opravil tu tabulku a na data se mohl dotazovat běžným způsobem.
Pokud trváš na svém řešení, můžeš využít CONCAT(), který spojuje části dotazu.
-- Nemusí fungovat (viz dále)
... CONCAT(_utf8mb4, ?) ...
Toto tvrzení nemám podložené, píši již z telefonu a nemohu to vyzkoušet.
Díky, ale cesta to nebyla správná. Správné řešení je popsané na https://stackoverflow.com/…ed-statement
Zobrazeno 8 zpráv z 8.