I'm trying to rewrite an old website.
It is in Persian and uses Persian/Arabic characters.
CREATE DATABASE `db` DEFAULT CHARACTER SET utf8 COLLATE utf8_persian_ci; USE `db`;
Almost all my tables/columns have COLLATE set to utf8_persian_ci
I'm using codeigniter for my new script, and I've
'char_set' => 'utf8', 'dbcollat' => 'utf8_persian_ci',
is in the database settings, so no problem.
So this is the weird part
The old scripts used some kind of database engine called TUBADBENGINE
or TUBA DB ENGINE
... nothing special.
When I entered some data (in Farsi) into the database using an old script, when I looked at the database, the characters were stored as Ø1مران
.
The old script gets/displays the data fine, but the new script displays them using the same weird font/charset as the database
So when I type rather
the data stored in the database looks like Ø1مراÙ
and when I get it in a new script I get See Ø1مراÙ
but in the old script I see ?????
CREATE TABLE IF NOT EXISTS `tnewsgroups` ( `ID` int(11) NOT NULL AUTO_INCREMENT, `fName` varchar(200) COLLATE utf8_persian_ci DEFAULT NULL, PRIMARY KEY (`ID`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_persian_ci AUTO_INCREMENT=11 ; -- -- Dumping data for table `tnewsgroups` -- INSERT INTO `tnewsgroups` (`ID`, `fName`) VALUES (1, 'عمران'), (2, 'معماری'), (3, 'برق'), (4, 'مکانیک'), (5, 'test'), (6, 'test2');
On the other hand, when I enter ???
directly into the database
Of course, I stored the same in the database rather
The new script displays very well
but in the old script I get ? ? ?
Can anyone understand this?
This is a large engine
https://github.com/maxxxir/mz-codeigniter-crud/blob/master/tuba.php
Usage example of old script:
define("database_type" , "MYSQL"); define("database_ip" , "localhost"); define("database_un" , "root"); define("database_pw" , ""); define("database_name" , "nezam2"); define("database_connectionstring" , ""); $db = new TUBADBENGINE(database_type , database_ip , database_un , database_pw , database_name , database_connectionstring); $db->Select("SELECT * FROM tnews limit 3"); if ($db->Lasterror() != "") { echo "<B><Font color=red>ÎØÇ ! áØÝÇ ãÌÏøÏÇ ÊáÇÔ ˜äíÏ"; exit(); } for ($i = 0 ; $i < $db->Count() ; $i++) { $row = $db->Next(); var_dump($row); }
deceze's answer is very good, but I can add some information that might help with handling a large number of records without having to test them manually.
If the conversion
CONVERT(BINARY CONVERT(field_name USING latin1) USING utf8)
fails,NULL
will be printed instead of thefield_name
content.So I use this to find those records:
Or this:
UPDATE
with this clause only affects records that have been converted successfully:In short, because this question has been discussed a thousand times before:
"汉字"
, encoded in UTF-8. The byte isE6 BC A2 E5 AD 97
.database connection set to
latin1.E6 BC A2 E5 AD 97
and thought they representedlatin1
characters.æ¡ ¡ ¿ Li>- The same process in reverse causes PHP to receive the same bytes and then treat them as UTF-8. The round trip works just fine for PHP, although the database doesn't handle characters the way it should.
So the problem here is that the database connection is not set up correctly when the data is entered into the database. You have to convert the data in the database to the correct characters. Try this:
Maybe
utf8
is not what you need, please try it. If it works, change it to aUPDATE
statement to permanently update the data.