php mssql扩展SQL查询中文字段名解决方法

 更新时间:2016年11月25日 16:36  点击:1329


一、问题:

      数据库是MS SQLServer2000,要把SQLServer2000里的一张表的数据导入MySQL5,其中SQLServer2000表的字段以简体中文命名(强烈建议不要以中文做为字段名)。其实操作就是对SQLServer查询记录,插入到MySQL里。选择的脚本语言是PHP,PHP打开MSSQL和MySQL扩展,对这两个数据库操作都是很容易的问题。

    问题就出现在SQLServer中表的字段名是中文,写好的查询语句在SQLServe里测试是通过有记录返回,用PHP的MSSQL扩展查询就是报错。

    上网查了资料,网上相关的信息不太多,很多网友都认为是PHP的MSSQL扩展不支持SQL语句中有中文。查了一下资料,PHP的MSSQL是支持SQL中有中文的语句。出现报错问题大多是因为编码的问题,编码数据库和编码与查询语句编码不统一,查询语句到SQLServer里中文部分就成了乱码,造成查询失败。

 二、解决方法:

      知道了原因,接下来分析解决,确认是编码不统一的问题。解决分以下几步:

           1、确认SQLServer 数据库的编码,我的数据编码是GBK。

           2、确认当前PHP脚本文件的编码,我的编码是UTF-8。

           3、转换SQL查询语句的的编码。

           补充:有的网友提到要把PHP的脚本文件编码转成和数据库编码一致,其实这一步大可不必,只要确认你的SQL语句和数据库的编码一致就可以,这里建议不必转换的原因是如果你的PHP脚本文件里包含其它PHP脚本,那也得对所有include或require的脚本文件编码转换,不然PHP脚本编码不统一很容易出错,若互相关联的文件很多,这也是一件很麻烦的问题而且把事情复杂化了。

 

三、方案:

写一个转换函数,在把SQL操作前把SQL语句编码转换。下面贴出我的范例代码:

//编码转换函数

 代码如下 复制代码

function utf8togb($s) {

return iconv('utf-8', 'gbk//IGNORE', $s);  // IGNORE 参数是遇到不成转换的字符时忽略

}

//建议把所有中文字段用英文别名替换,方便下面操作还有编码转换等问题

$sql="SELECT [id], [栏目] as typeid, [正题] as title, [作者] as author, convert(text, [正文]) as body FROM [文章表];";

$sql = utf8togb($sql);

php将可以正常的和apache结合,并且常见的诸如:PHP Warning: PHP Startup: Unable to load dynamic library……、Fatal error: Call to undefined function mysql_connect() 错误

不过今天在弄一个CMS时还是出现了一个不可饶恕的错误,无法连接mysql,写了一个测试mysql的php文件如下:

 代码如下 复制代码
<?php
$link=mysql_connect('localhost','root','root');
if(!$link) echo "失败!";
else echo "成功!";
mysql_close();
?>    

 

结果报错误:Fatal error: Call to undefined function mysql_connect(),明显没法找到mysql相应的库,百度了一大堆,又是什么复制mysql的lib库、又是复制到windows的system32目录,过于复杂,其实正确的配置应该是在apache的httpd.conf文件中加入下设置:

 代码如下 复制代码

LoadModule php5_module d:/tools/php-5.3.1/php5apache2_2.dll
AddType application/x-httpd-php .php
PHPIniDir d:/tools/php-5.3.1     

注意,最后一行是关键,告诉apache你的php配置文件在哪里,不然你就得把php.ini文件复制到windows的系统目录下了。一般我们通过这里配置就可以完成apache和php的结合。而在php.ini文件中只需要修改如下几个地方即可:

# 修改php的扩展库目录为你的实际路径

 代码如下 复制代码
extension_dir = "D:/tools/php-5.3.1/ext"

# 去掉前面的;

 代码如下 复制代码
extension=php_mysql.dll     

最后一个步骤就是在系统环境变量(注意是系统环境变量,而非用户环境变量)path中加入你的PHP安装路径,在设置成功后,通过phpinfo()应该可以在Apache Environment段的path中可以看到你的PHP安装目录,否则就是环境变量没有配置成功(注意,配置环境变量后重启下apache)。

完成这些步骤后就可以正确连接数据库了,不过前提是你的apache和php的整合是正常的,然后再考虑这个mysql配置,如果还不能够达到效果,你可以质问我,呵呵!


总结,windows平台下apache和php(包括mysql)整合的正确配置步骤如下:

1、安装apache,并保证apache可以单独正常运行

2、将php(如5.2.17或5.3.1)的zip包解压到一个目录,比如D:php-5.3.1

3、在apache的httpd.conf文件的最后加入如下段:

 代码如下 复制代码
# php5 supportLoadModule php5_module D:/php-5.3.1/php5apache2_2.dll
 代码如下 复制代码
AddType application/x-httpd-php .php
 代码如下 复制代码
PHPIniDir D:/php-5.3.14

、在操作系统的环境变量(必须是系统变量而非用户变量)的path中加入D:/php-5.3.1路径

5、将php安装目录下的php.ini-dist重命名为php.ini,并修改如下内容:

# 设置php的扩展文件目录

 代码如下 复制代码
extension_dir = "D:/php-5.3.17/ext"

 # 开启mysql扩展,去掉前面的;注释

 代码如下 复制代码
extension=php_mysql.dll
php的mysqli扩展被封装到一个类中,是一种面向对象的技术,执行速度更快,与传统的过程化方法相比更方便也更高效

默认情况下mysqli在php是未开启的我们需在
要在PHP中使用mysqli扩展,需要在配置文件php.ini中添加如下的设置:

 代码如下 复制代码
extension=php_mysqli.dll

如果配置文件中已有上述设置,确保extension前面没有“;”,否则将其去掉。下面开始介绍如何使用mysqli扩展来存取数据库,即可了.

 代码如下 复制代码
<?php
$db_host="localhost";                                           //连接的服务器地址
$db_user="root";                                                  //连接数据库的用户名
$db_psw="root";                                                  //连接数据库的密码
$db_name="sunyang";                                           //连接的数据库名称
$mysqli=new mysqli();
$mysqli->connect($db_host,$db_user,$db_psw,$db_name);
?>

关闭与MySQL服务器的连接通过mysqli对象调用close()方法即可,例如:
$mysqli->close();

 代码如下 复制代码

<?php
$connection = mysqli_connect("localhost","root","root","sunyang");
if ( $connection ) {
         echo "数据库连接成功";
}else {
         echo "数据库连接失败";
}
?>

数据查询

 代码如下 复制代码

<?php
$mysqli=new mysqli("localhost","root","root","sunyang");     //实例化mysqli
$query="select * from employee";
$result=$mysqli->query($query);
if ($result) {
         if($result->num_rows>0){                                               //判断结果集中行的数目是否大于0
                  while($row =$result->fetch_array() ){                        //循环输出结果集中的记录
                           echo ($row[0])."<br>";
                           echo ($row[1])."<br>";
                           echo ($row[2])."<br>";
                           echo ($row[3])."<br>";
                           echo "<hr>";
                  }
         }
}else {
         echo "查询失败";
}
$result->free();
$mysqli->close();
?>

其它的像数据保存等待


mysqli类中的成员方法

__construct():构造方法,用于创建一个mysqli对象,也可以建立一个连接。

autocommit():开启或关闭数据库修改自动提交。

change_user():改变数据库连接所指定的用户。

character_set_name():返回数据库连接默认字符集。

close():关闭先前打开的连接。

commit():提交当前的事物。

connect():打开一个新的连接到mysql数据库服务器。

debug():执行调试操作。

dump_debug_info():转储调试信息。

get_client_info():返回客户端版本。

get_host_info():返回一个字符串代表的连接使用类型,如:Localhost via UNIX socket

get_server_info():返回mysql服务器版本。

get_server_version():返回整数形式的mysql服务器版本。

init():初始化一个mysqli并返回一个资源。

info():检索有关最近执行的查询。

kill():杀死一个mysql线程。

multi_query():执行多个查询语句。

more_results():从多查询语句中检索是否有任何更多的查询结果。

next_result():从当前执行的多查询中读取下一个结果。

options():设置选项。

ping():如果没有连接,ping一台服务器连接或重新连接。

prepare():准备一个sql语句的执行,返回mysqli_stmt对象。

query():与数据库交互都是通过查询进行的,该方法向数据库发送查询来执行,执行失败返回FALSE。

real_connect():试图打开一个连接到mysql数据库服务器。

escape_string():转义特殊字符的字符串。

rollback():回滚当前的事务。

select_db():为数据库查询选择一个默认的数据库。

set_charset():设置默认客户端字符集。


ssl_set():使用ssl用于建立安全连接。

stat():获取当前的系统状态。

stmt_init():初始化一个声明,返回一个mysql_stmt对象。

store_result():从最后查询中转让结果集。

thread_safe():是否考虑返回安全的线程。

 


mysql类中的成员属性

$affected_rows:前一个mysql操作中影响的行数。

$client_info:mysql客户端版本(字符串)。

$client_version:mysql客户端版本(整数)。

$errno:最近函数调用的错误代码。

$error:最近函数调用的错误信息字符串。

$field_count():查询获取的列数。

$host_info:连接类型使用(字符串)。

$info:最近执行的查询。

$insert_id:最后查询自动生成的编号。

$protocol_version:mysql协议使用的版本。

$sqlstate:包含SQLSTATE错误码的最后一个错误。

$thread_id:当前连接线程ID。

$warning_count:前一个sql语句执行过程中产生的警告数量。

php mysql_insert_id()返回数据库最新id实现方法 有需要同学可参考一下。
 代码如下 复制代码

mysql_insert_id()
mysql_insert_id() 函数返回上一步 INSERT 操作产生的 ID。如果上一查询没有产生 AUTO_INCREMENT 的 ID,则 mysql_insert_id() 返回 0。

语法:mysql_insert_id(connection)

参数connection,可选。规定 MySQL 连接。如果未规定,则使用上一个连接。

mysql_insert_id() 返回给定的 connection 中上一步 INSERT 查询中产生的 AUTO_INCREMENT 的 ID 号。如果没有指定 connection ,则使用上一个打开的连接。

如果需要保存该值以后使用,要确保在产生了值的查询之后立即调用 mysql_insert_id()。

 代码如下 复制代码

<?php
$con = mysql_connect("localhost", "hello", "321");
if (!$con)
{
 die('Could not connect: ' . mysql_error());
}

$db_selected = mysql_select_db("test_db",$con);

$sql = "INSERT INTO person VALUES ('Carter','Thomas','Beijing')";
$result = mysql_query($sql,$con);
echo "ID of last inserted record is: " . mysql_insert_id();

mysql_close($con);
?>

mysql_insert_id() 返回给定的 link_identifier 中上一步 INSERT 查询中产生的 AUTO_INCREMENT 的 ID 号。如果没有指定 link_identifier,则使用上一个打开的连接。如果上一查询没有产生 AUTO_INCREMENT 的值,则 mysql_insert_id() 返回 0。如果需要保存该值以后使用,要确保在产生了值的查询之后立即调用 mysql_insert_id()。

LAST_INSERT_ID()
MySQL也提供了这么一个同样功能的API,它总是保存着最新产生的AUTO_INCREMENT值,并且不会在查询语句之间被重置,也就是说,在执行INSERT操作后,执行SELECT、UPDATE、DELETE语句都不会影响该API的返回值。

可以用 SELECT LAST_INSERT_ID(); 来查询LAST_INSERT_ID() 的返回值。

使用单条INSERT语句插入多条记录,,LAST_INSERT_ID() 只返回插入的第一条记录产生的AUTO_INCREMENT值。

出现中文乱码一般情况是文档编码与数据库编码不一致导致的,下面我们来分析MYSQL中文乱码的原因吧。

其MYSQL的默认字符集竟然是瑞典文(latin1_swedish_ci),害的我手动一个表一个表地修改字符集为Unicode多语言(utf8_unicode_ci)。

只好查资料,发现了一个解决方法,就是在mysql_connect后面加一句SET NAMES UTF8,即可使得UTF8的数据库消除乱码,对于GBK的数据库则使用SET NAMES GBK,代码如下:

 代码如下 复制代码

$mysql_mylink = mysql_connect($mysql_host, $mysql_user, $mysql_pass);
mysql_query("SET NAMES 'GBK'");

[!--infotagslink--]

相关文章

  • 浅谈C# 字段和属性

    这篇文章主要介绍了C# 字段和属性的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下...2020-11-03
  • js URLdecode()与urlencode方法支持中文解码

    下面来介绍在js中来利用urlencode对中文编码与接受到数据后利用URLdecode()对编码进行解码,有需要学习的机友可参考参考。 代码如下 复制代码 ...2016-09-20
  • php 中file_get_contents超时问题的解决方法

    file_get_contents超时我知道最多的原因就是你机器访问远程机器过慢,导致php脚本超时了,但也有其它很多原因,下面我来总结file_get_contents超时问题的解决方法总结。...2016-11-25
  • Mybatis Plus select 实现只查询部分字段

    这篇文章主要介绍了Mybatis Plus select 实现只查询部分字段的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-09-01
  • HTTP 408错误是什么 HTTP 408错误解决方法

    相信很多站长都遇到过这样一个问题,访问页面时出现408错误,下面一聚教程网将为大家介绍408错误出现的原因以及408错误的解决办法。 HTTP 408错误出现原因: HTT...2017-01-22
  • mybatis-plus 返回部分字段的解决方式

    这篇文章主要介绍了mybatis-plus 返回部分字段的解决方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-10-02
  • 安卓手机wifi打不开修复教程,安卓手机wifi打不开解决方法

    手机wifi打不开?让小编来告诉你如何解决。还不知道的朋友快来看看。 手机wifi是现在生活中最常用的手机功能,但是遇到手机wifi打不开的情况该怎么办呢?如果手机wifi...2016-12-21
  • PHP 验证码不显示只有一个小红叉的解决方法

    最近想自学PHP ,做了个验证码,但不知道怎么搞的,总出现一个如下图的小红叉,但验证码就是显示不出来,原因如下 未修改之前,出现如下错误; (1)修改步骤如下,原因如下,原因是apache权限没开, (2)点击打开php.int., 搜索extension=ph...2013-10-04
  • postgresql数据添加两个字段联合唯一的操作

    这篇文章主要介绍了postgresql数据添加两个字段联合唯一的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-02-04
  • 连接MySql速度慢的解决方法(skip-name-resolve)

    最近在Linux服务器上安装MySql5后,本地使用客户端连MySql速度超慢,本地程序连接也超慢。 解决方法:在配置文件my.cnf的[mysqld]下加入skip-name-resolve。原因是默认安装的MySql开启了DNS的反向解析。如果禁用的话就不能...2015-10-21
  • 关于Mysql中文乱码问题该如何解决(乱码问题完美解决方案)

    最近两天做项目总是被乱码问题困扰着,这不刚把mysql中文乱码问题解决了,下面小编把我的解决方案分享给大家,供大家参考,也方便以后自己查阅。首先:用show variables like “%colla%”;show varables like “%char%”;这两条...2015-11-24
  • C#读取中文文件出现乱码的解决方法

    这篇文章主要介绍了C#读取中文文件出现乱码的解决方法,涉及C#中文编码的操作技巧,非常具有实用价值,需要的朋友可以参考下...2020-06-25
  • PHP 二维数组根据某个字段排序的具体实现

    本文记录的要实现的功能类似于 MySQL 中的 ORDER BY,上个项目中有遇到这样的一个需求。 要求:从两个不同的表中获取各自的4条数据,然后整合(array_merge)成一个数组,再根据数据的创建时间降序排序取前4条。 遇到这个...2014-06-07
  • Mysql在debian系统中不能插入中文的终极解决方案

    在debian环境下,彻底解决mysql无法插入和显示中文的问题Linux下Mysql插入中文显示乱码解决方案mysql -uroot -p 回车输入密码进入mysql查看状态如下:默认的是客户端和服务器都用了latin1,所以会乱码。解决方案:mysql>use...2013-10-04
  • 总结android studio注意事项及打不开等问题解决方法

    经过一段时间的使用,总结了android studio打不开等问题的6种解决方法及android studio注意事项,希望对大家有所帮助。 1 首次运行,建立好项目需要下载一些东西,如果...2016-09-20
  • Windows服务器MySQL中文乱码的解决方法

    我们自己鼓捣mysql时,总免不了会遇到这个问题:插入中文字符出现乱码,虽然这是运维先给配好的环境,但是在自己机子上玩的时候咧,总得知道个一二吧,不然以后如何优雅的吹牛B。...2015-03-15
  • linux mint 下mysql中文支持问题

    一.mysql默认不支持中文,它的server和db默认是latin1编码.所以我们要将其改变为utf-8编码,因为utf-8包含了地球上大部分语言的二进制编码 1.关闭mysql服务 sudo /etc/init.d/mysql stop 2.修改mysql配置文件 mysql配...2015-10-21
  • php怎么用拼音 简单的php中文转拼音的实现代码

    小编分享了一段简单的php中文转拼音的实现代码,代码简单易懂,适合初学php的同学参考学习。 代码如下 复制代码 <?phpfunction Pinyin($_String...2017-07-06
  • MySQL ERROR 2013 (HY000)错误解决方法

    当通过 TCP/IP 连接 MySQL 远程主机时,出现 ERROR 2013 (HY000): Lost connection to MySQL server at 'reading initial communication packet', system error: 104 。如果是在linux shell命令行中直接打 mysql 命令,...2015-03-15
  • IE6-IE9中tbody的innerHTML不能赋值的解决方法

    IE6-IE9中tbody的innerHTML不能赋值,重现代码如下 复制代码 代码如下: <!DOCTYPE html> <html> <head> <meta charset="utf-8"/> <title>IE6-IE9中tbody的innerHTML不能复制bug</title> </head> <body style="height:3...2014-06-07