天天躁日日躁狠狠躁AV麻豆-天天躁人人躁人人躁狂躁-天天澡夜夜澡人人澡-天天影视香色欲综合网-国产成人女人在线视频观看-国产成人女人视频在线观看

mysqli_set_charset和SET NAMES使用抉擇及優劣分析

最近公司組織了個php安全編程的培訓, 其中涉及到一部分關于Mysql的”SET NAMES”和mysql_set_charset (mysqli_set_charset)的內容:
說到, 盡量使用mysqli_set_charset(mysqli:set_charset)而不是”SET NAMES”, 當然, 這個內容在php手冊中也有敘及, 但是卻沒有解釋為什么.

最近有好幾個朋友問我這個問題, 到底為什么?
問的人多了, 我也就覺得可以寫篇blog, 專門介紹下這部分的內容了.
首先, 很多人都不知道”SET NAMES”到底是做了什么,
我之前的文章深入MySQL字符集設置中, 曾經介紹過character_set_client/character_set_connection/character_set_results這三個MySQL的”環境變量”, 這里再簡單介紹下,
這三個變量, 分別告訴MySQL服務器, 客戶端的編碼集, 在傳輸給MySQL服務器的時候的編碼集, 以及期望MySQL返回的結果的編碼集.
比如, 通過使用”SET NAMES utf8″, 就告訴服務器, 我用的是utf-8編碼, 我希望你也給我返回utf-8編碼的查詢結果.

一般情況下, 使用”SET NAMES”就足夠了, 也是可以保證正確的. 那么為什么手冊又要說推薦使用mysqli_set_charset(php>=5.0.5)呢?
首先, 我們看看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 STDCALLmysql_set_character_set(MYSQL*mysql, const char *cs_name)
{
structcharset_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";//" 主站蜘蛛池模板: 9797在线看片亚洲精品 | 国产99视频精品免费播放 | 成人在线不卡视频 | YELLOW日本动漫免费动漫 | 一区三区不卡高清影视 | 免费A级毛片无码无遮挡 | 窝窝影院午夜看片毛片 | 美女视频秀色福利视频 | 日本熟妇乱妇熟色A片蜜桃亚洲 | 国产成人精品视频免费大全 | 久久性生大片免费观看性 | c了瑜伽老师嗷嗷叫一节课视频 | 狠狠爱亚洲五月婷婷av | 国产精品99久久久久久WWW | 国产亚洲精品福利视频 | 一二三四在线观看高清电视剧 | 囯产免费久久久久久国产免费 | 亚洲精品AV无码重口另类 | 国产免费人成在线看视频 | 国产亚洲精品久久久久 | 草莓西瓜樱桃香蕉直播视频 | 俄罗斯雏妓的BBB孩交 | 伊在香蕉国产在线视频 | 午夜A级理论片左线播放 | 日本九九热在线观看官网 | 久久夜色撩人精品国产 | 老王午夜69精品影院 | 亚洲一区二区三区乱码在线欧洲 | 亚洲AV无码乱码国产麻豆P | 边摸边吃奶边做下面视频 | 老师你下面好紧夹死了 | 精品免费在线视频 | 国产传媒18精品A片在线观看 | 午夜AV内射一区二区三区红桃视 | 伊人久久国产 | 麻豆久久婷婷五月国产 | 玩弄放荡人妻一区二区三区 | 99国内偷揿国产精品人妻 | 亚洲日韩一区精品射精 | 成年人在线视频免费观看 | 久久99国产亚洲高清观着 |