mysql set names和set_charset的區別

 首先介紹下三個MySQL的”環境變量”設置命令及其作用:

character_set_client:          MySQL服務器, 客戶端的編碼集
character_set_connection:傳輸給MySQL服務器的時候的編碼集
character_set_results:       期望MySQL返回的結果的編碼集
比如, 通過使用”SET NAMES utf8″, 就告訴服務器, 我用的是utf-8編碼, 我希望你也給我返回utf-8編碼的查詢結果. 

 一般情況下, 使用”SET NAMES”就足夠了, 也是可以保證正確的. 那麼爲什麼手冊又要說推薦使用mysqli_set_charset呢? 

首先, 我們看看mysqli_set_charset到底做了什麼(注意星號註釋處, mysql_set_charset類似):

 
  //php-5.2.11-SRC/ext/mysqli/mysqli_nonapi.c line 342 
  PHP_FUNCTION(mysqli_set_charset) 
  { 
  MY_MYSQL *mysql; 
  zval *mysql_link; 
  char *cs_name = NULL; 
  unsigned int len; 
  if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis() 
  , "Os", &mysql_link, mysqli_link_class_entry, &cs_name, &len) == FAILURE) { 
  return; 
  } 
  MYSQLI_FETCH_RESOURCE(mysql, MY_MYSQL*, &mysql_link, "mysqli_link" 
  , MYSQLI_STATUS_VALID); 
  if (mysql_set_character_set(mysql->mysql, cs_name)) { 
  //** 調用libmysql的對應函數 
  RETURN_FALSE; 
  } 
  RETURN_TRUE; 
   } 
    那mysql_set_character_set又做了什麼呢?

 
  //mysql-5.1.30-SRC/libmysql/client.c, line 3166: 
  int STDCALL mysql_set_character_set(MYSQL *mysql, const char *cs_name) 
  { 
  struct charset_info_st *cs; 
  const char *save_csdir= charsets_dir; 
  if (mysql->options.charset_dir) 
  charsets_dir= mysql->options.charset_dir; 
  if (strlen(cs_name) < MY_CS_NAME_SIZE && 
  (cs= get_charset_by_csname(cs_name, MY_CS_PRIMARY, MYF(0)))) 
  { 
  char buff[MY_CS_NAME_SIZE + 10]; 
  charsets_dir= save_csdir; 
  /* Skip execution of "SET NAMES" for pre-4.1 servers */ 
  if (mysql_get_server_version(mysql) < 40100) 
  return 0; 
  sprintf(buff, "SET NAMES %s", cs_name); 
  if (!mysql_real_query(mysql, buff, strlen(buff))) 
  { 
  mysql->charset= cs; 
  } 
  } 
  //以下省略 
 我們可以看到, mysqli_set_charset除了做了”SET NAMES”以外, 還多做了一步: 

 
  sprintf(buff, "SET NAMES %s", cs_name); 
  if (!mysql_real_query(mysql, buff, strlen(buff))) 
  { 
  mysql->charset= cs; 
  } 

       而對於mysql這個核心結構的成員charset又有什麼作用呢? 
  這就要說說mysql_real_escape_string()了, 這個函數和mysql_escape_string的區別就是, 它會考慮”當前”字符集. 那麼這個當前字符集從哪裏來呢? 
  對了, 你猜的沒錯, 就是mysql->charset. 
  mysql_real_string在判斷寬字符集的字符的時候, 就根據這個成員變量來分別採用不同的策略, 比如如果是utf-8, 那麼就會採用libmysql/ctype-utf8.c. 
  看個實例, 默認mysql連接字符集是latin-1, (經典的5c問題): 

 
  <?php 
  $db = mysql_connect('localhost:3737', 'root' ,'123456'); 
  mysql_select_db("test"); 
  $a = "\x91\x5c";//"憖"的gbk編碼, 低字節爲5c, 也就是ascii中的"\" 
  var_dump(addslashes($a)); 
  var_dump(mysql_real_escape_string($a, $db)); 
  mysql_query("set names gbk"); 
  var_dump(mysql_real_escape_string($a, $db)); 
  mysql_set_charset("gbk"); 
  var_dump(mysql_real_escape_string($a, $db)); 
  ?> 
      因爲, “憖”的gbk編碼低字節爲5c, 也就是ascii中的”\”, 而因爲除了mysql(i)_set_charset影響mysql->charset以外, 其他時刻mysql->charset都爲默認值,
      所以, 結果就是: 

 
  $ php -f 5c.php 
  string(3) "憖\" 
  string(3) "憖\" 
  string(3) "憖\" 
  string(2) "憖" 

 箇中細節有待進一步驗證。原文適當修改。文章摘至:深入理解mysql SET NAMES和mysql(i)_set_charset
 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章