<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>程序猿二三事儿</title>
	<atom:link href="http://www.waicai.org/?feed=rss2" rel="self" type="application/rss+xml" />
	<link>http://www.waicai.org</link>
	<description>歪才的程序猿手记</description>
	<lastBuildDate>Wed, 25 Dec 2024 08:50:38 +0000</lastBuildDate>
	<language>zh-CN</language>
		<sy:updatePeriod>hourly</sy:updatePeriod>
		<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.8.1</generator>
	<item>
		<title>查看Navicat已保存数据库连接的密码</title>
		<link>http://www.waicai.org/?p=105</link>
		<comments>http://www.waicai.org/?p=105#comments</comments>
		<pubDate>Wed, 11 Dec 2024 02:22:54 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[数据库]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=105</guid>
		<description><![CDATA[1.导出数据库连接 connections.ncx 文件，导出时要勾选导出密码 2.使用文本编辑工具打开导出的 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>1.导出数据库连接 connections.ncx 文件，导出时要勾选导出密码</p>
<p>2.使用文本编辑工具打开导出的connections.ncx 文件，找到Password字段中的加密数据</p>
<p>3.JAVA解密脚本</p>
<p>import javax.crypto.Cipher;<br />
import javax.crypto.spec.IvParameterSpec;<br />
import javax.crypto.spec.SecretKeySpec;<br />
import java.security.MessageDigest;<br />
import java.util.Arrays;</p>
<p>public class NavicatPassword {<br />
public static void main(String[] args) throws Exception {<br />
NavicatPassword navicatPassword = new NavicatPassword();</p>
<p>// 解密11版本及以前的密码<br />
//String decode = navicatPassword.decrypt(&#8220;15057D7BA390&#8243;, 11);</p>
<p>// 解密12版本及以后的密码<br />
//String decode = navicatPassword.decrypt(&#8220;15057D7BA390&#8243;, 12);<br />
String decode = navicatPassword.decrypt(&#8220;解密密码&#8221;, 12);<br />
System.out.println(decode);<br />
}</p>
<p>private static final String AES_KEY = &#8220;libcckeylibcckey&#8221;;<br />
private static final String AES_IV = &#8220;libcciv libcciv &#8220;;<br />
private static final String BLOW_KEY = &#8220;3DC5CA39&#8243;;<br />
private static final String BLOW_IV = &#8220;d9c7c3c8870d64bd&#8221;;</p>
<p>public static String encrypt(String plaintext, int version) throws Exception {<br />
switch (version) {<br />
case 11:<br />
return encryptEleven(plaintext);<br />
case 12:<br />
return encryptTwelve(plaintext);<br />
default:<br />
throw new IllegalArgumentException(&#8220;Unsupported version&#8221;);<br />
}<br />
}</p>
<p>public static String decrypt(String ciphertext, int version) throws Exception {<br />
switch (version) {<br />
case 11:<br />
return decryptEleven(ciphertext);<br />
case 12:<br />
return decryptTwelve(ciphertext);<br />
default:<br />
throw new IllegalArgumentException(&#8220;Unsupported version&#8221;);<br />
}<br />
}</p>
<p>private static String encryptEleven(String plaintext) throws Exception {<br />
byte[] iv = hexStringToByteArray(BLOW_IV);<br />
byte[] key = hashToBytes(BLOW_KEY);</p>
<p>int round = plaintext.length() / 8;<br />
int leftLength = plaintext.length() % 8;<br />
StringBuilder result = new StringBuilder();<br />
byte[] currentVector = iv.clone();</p>
<p>Cipher cipher = Cipher.getInstance(&#8220;Blowfish/ECB/NoPadding&#8221;);<br />
SecretKeySpec secretKeySpec = new SecretKeySpec(key, &#8220;Blowfish&#8221;);<br />
cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec);</p>
<p>for (int i = 0; i &lt; round; i++) {<br />
byte[] block = xorBytes(plaintext.substring(i * 8, (i + 1) * 8).getBytes(),currentVector);<br />
byte[] temp = cipher.doFinal(block);<br />
currentVector = xorBytes(currentVector, temp);<br />
result.append(bytesToHex(temp));<br />
}</p>
<p>if (leftLength &gt; 0) {<br />
currentVector = cipher.doFinal(currentVector);<br />
byte[] block = xorBytes(plaintext.substring(round * 8).getBytes(), currentVector);<br />
result.append(bytesToHex(block));<br />
}</p>
<p>return result.toString().toUpperCase();<br />
}</p>
<p>private static String encryptTwelve(String plaintext) throws Exception {<br />
byte[] iv = AES_IV.getBytes();<br />
byte[] key = AES_KEY.getBytes();</p>
<p>Cipher cipher = Cipher.getInstance(&#8220;AES/CBC/NoPadding&#8221;);<br />
SecretKeySpec secretKeySpec = new SecretKeySpec(key, &#8220;AES&#8221;);<br />
IvParameterSpec ivParameterSpec = new IvParameterSpec(iv);<br />
cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, ivParameterSpec);</p>
<p>byte[] result = cipher.doFinal(plaintext.getBytes());<br />
return bytesToHex(result).toUpperCase();<br />
}</p>
<p>private static String decryptEleven(String ciphertext) throws Exception {<br />
byte[] iv = hexStringToByteArray(BLOW_IV);<br />
byte[] key = hashToBytes(BLOW_KEY);<br />
byte[] encrypted = hexStringToByteArray(ciphertext.toLowerCase());</p>
<p>int round = encrypted.length / 8;<br />
int leftLength = encrypted.length % 8;<br />
StringBuilder result = new StringBuilder();<br />
byte[] currentVector = iv.clone();</p>
<p>Cipher cipher = Cipher.getInstance(&#8220;Blowfish/ECB/NoPadding&#8221;);<br />
SecretKeySpec secretKeySpec = new SecretKeySpec(key, &#8220;Blowfish&#8221;);<br />
cipher.init(Cipher.DECRYPT_MODE, secretKeySpec);</p>
<p>for (int i = 0; i &lt; round; i++) {<br />
byte[] block = Arrays.copyOfRange(encrypted, i * 8, (i + 1) * 8);<br />
byte[] temp = xorBytes(cipher.doFinal(block), currentVector);<br />
currentVector = xorBytes(currentVector, block);<br />
result.append(new String(temp));<br />
}</p>
<p>if (leftLength &gt; 0) {<br />
currentVector = cipher.doFinal(currentVector);<br />
byte[] block = Arrays.copyOfRange(encrypted, round * 8, round * 8 + leftLength);<br />
result.append(new String(xorBytes(block, currentVector)));<br />
}</p>
<p>return result.toString();<br />
}</p>
<p>private static String decryptTwelve(String ciphertext) throws Exception {<br />
byte[] iv = AES_IV.getBytes();<br />
byte[] key = AES_KEY.getBytes();<br />
byte[] encrypted = hexStringToByteArray(ciphertext.toLowerCase());</p>
<p>Cipher cipher = Cipher.getInstance(&#8220;AES/CBC/NoPadding&#8221;);<br />
SecretKeySpec secretKeySpec = new SecretKeySpec(key, &#8220;AES&#8221;);<br />
IvParameterSpec ivParameterSpec = new IvParameterSpec(iv);<br />
cipher.init(Cipher.DECRYPT_MODE, secretKeySpec, ivParameterSpec);</p>
<p>byte[] result = cipher.doFinal(encrypted);<br />
return new String(result);<br />
}</p>
<p>private static byte[] xorBytes(byte[] bytes1, byte[] bytes2) {<br />
byte[] result = new byte[bytes1.length];<br />
for (int i = 0; i &lt; bytes1.length; i++) {<br />
result[i] = (byte) (bytes1[i] ^ bytes2[i]);<br />
}<br />
return result;<br />
}</p>
<p>private static byte[] hexStringToByteArray(String s) {<br />
int len = s.length();<br />
byte[] data = new byte[len / 2];<br />
for (int i = 0; i &lt; len; i += 2) {<br />
data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) &lt;&lt; 4)<br />
+ Character.digit(s.charAt(i + 1), 16));<br />
}<br />
return data;<br />
}</p>
<p>private static byte[] hashToBytes(String s) throws Exception {<br />
return MessageDigest.getInstance(&#8220;SHA-1&#8243;).digest(s.getBytes());<br />
}</p>
<p>private static String bytesToHex(byte[] bytes) {<br />
StringBuilder result = new StringBuilder();<br />
for (byte b : bytes) {<br />
result.append(String.format(&#8220;%02X&#8221;, b));<br />
}<br />
return result.toString();<br />
}<br />
}<br />
4.PHP解密脚本</p>
<p>&lt;?php</p>
<p>namespace FatSmallTools;</p>
<p>class NavicatPassword<br />
{<br />
protected $version = 0;<br />
protected $aesKey = &#8216;libcckeylibcckey&#8217;;<br />
protected $aesIv = &#8216;libcciv libcciv &#8216;;<br />
protected $blowString = &#8217;3DC5CA39&#8242;;<br />
protected $blowKey = null;<br />
protected $blowIv = null;</p>
<p>public function __construct($version = 12)<br />
{<br />
$this-&gt;version = $version;<br />
$this-&gt;blowKey = sha1(&#8217;3DC5CA39&#8242;, true);<br />
$this-&gt;blowIv = hex2bin(&#8216;d9c7c3c8870d64bd&#8217;);<br />
}</p>
<p>public function encrypt($string)<br />
{<br />
$result = FALSE;<br />
switch ($this-&gt;version) {<br />
case 11:<br />
$result = $this-&gt;encryptEleven($string);<br />
break;<br />
case 12:<br />
$result = $this-&gt;encryptTwelve($string);<br />
break;<br />
default:<br />
break;<br />
}</p>
<p>return $result;<br />
}</p>
<p>protected function encryptEleven($string)<br />
{<br />
$round = intval(floor(strlen($string) / 8));<br />
$leftLength = strlen($string) % 8;<br />
$result = &#8221;;<br />
$currentVector = $this-&gt;blowIv;</p>
<p>for ($i = 0; $i &lt; $round; $i++) {<br />
$temp = $this-&gt;encryptBlock($this-&gt;xorBytes(substr($string, 8 * $i, 8), $currentVector));<br />
$currentVector = $this-&gt;xorBytes($currentVector, $temp);<br />
$result .= $temp;<br />
}</p>
<p>if ($leftLength) {<br />
$currentVector = $this-&gt;encryptBlock($currentVector);<br />
$result .= $this-&gt;xorBytes(substr($string, 8 * $i, $leftLength), $currentVector);<br />
}</p>
<p>return strtoupper(bin2hex($result));<br />
}</p>
<p>protected function encryptBlock($block)<br />
{<br />
return openssl_encrypt($block, &#8216;BF-ECB&#8217;, $this-&gt;blowKey, OPENSSL_RAW_DATA|OPENSSL_NO_PADDING);<br />
}</p>
<p>protected function decryptBlock($block)<br />
{<br />
return openssl_decrypt($block, &#8216;BF-ECB&#8217;, $this-&gt;blowKey, OPENSSL_RAW_DATA|OPENSSL_NO_PADDING);<br />
}</p>
<p>protected function xorBytes($str1, $str2)<br />
{<br />
$result = &#8221;;<br />
for ($i = 0; $i &lt; strlen($str1); $i++) {<br />
$result .= chr(ord($str1[$i]) ^ ord($str2[$i]));<br />
}</p>
<p>return $result;<br />
}</p>
<p>protected function encryptTwelve($string)<br />
{<br />
$result = openssl_encrypt($string, &#8216;AES-128-CBC&#8217;, $this-&gt;aesKey, OPENSSL_RAW_DATA, $this-&gt;aesIv);<br />
return strtoupper(bin2hex($result));<br />
}</p>
<p>public function decrypt($string)<br />
{<br />
$result = FALSE;<br />
switch ($this-&gt;version) {<br />
case 11:<br />
$result = $this-&gt;decryptEleven($string);<br />
break;<br />
case 12:<br />
$result = $this-&gt;decryptTwelve($string);<br />
break;<br />
default:<br />
break;<br />
}</p>
<p>return $result;<br />
}</p>
<p>protected function decryptEleven($upperString)<br />
{<br />
$string = hex2bin(strtolower($upperString));</p>
<p>$round = intval(floor(strlen($string) / 8));<br />
$leftLength = strlen($string) % 8;<br />
$result = &#8221;;<br />
$currentVector = $this-&gt;blowIv;</p>
<p>for ($i = 0; $i &lt; $round; $i++) {<br />
$encryptedBlock = substr($string, 8 * $i, 8);<br />
$temp = $this-&gt;xorBytes($this-&gt;decryptBlock($encryptedBlock), $currentVector);<br />
$currentVector = $this-&gt;xorBytes($currentVector, $encryptedBlock);<br />
$result .= $temp;<br />
}</p>
<p>if ($leftLength) {<br />
$currentVector = $this-&gt;encryptBlock($currentVector);<br />
$result .= $this-&gt;xorBytes(substr($string, 8 * $i, $leftLength), $currentVector);<br />
}</p>
<p>return $result;<br />
}</p>
<p>protected function decryptTwelve($upperString)<br />
{<br />
$string = hex2bin(strtolower($upperString));<br />
return openssl_decrypt($string, &#8216;AES-128-CBC&#8217;, $this-&gt;aesKey, OPENSSL_RAW_DATA, $this-&gt;aesIv);<br />
}<br />
}</p>
<p>use FatSmallTools\NavicatPassword;</p>
<p>//需要指定版本，11或12<br />
$navicatPassword = new NavicatPassword(12);<br />
//$navicatPassword = new NavicatPassword(11);</p>
<p>//解密<br />
//$decode = $navicatPassword-&gt;decrypt(&#8217;15057D7BA390&#8242;);<br />
$decode = $navicatPassword-&gt;decrypt(&#8216;解密密码&#8217;);<br />
echo $decode.&#8221;\n&#8221;;</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=105</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>驻足</title>
		<link>http://www.waicai.org/?p=101</link>
		<comments>http://www.waicai.org/?p=101#comments</comments>
		<pubDate>Thu, 07 Dec 2023 10:00:19 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[随笔]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=101</guid>
		<description><![CDATA[站在天桥上 吹着暖风 看着下边的车流 想着当年的我们 &#160; 还记得 当年的我们 每天都盼着体活课 可以 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>站在天桥上</p>
<p>吹着暖风</p>
<p>看着下边的车流</p>
<p>想着当年的我们</p>
<p>&nbsp;</p>
<p>还记得</p>
<p>当年的我们</p>
<p>每天都盼着体活课</p>
<p>可以在操场上疯跑</p>
<p>&nbsp;</p>
<p>还记得</p>
<p>当年的我们</p>
<p>一起站在天桥看夕阳</p>
<p>一起打闹着骑车回家</p>
<p>&nbsp;</p>
<p>还记得</p>
<p>当年的我们</p>
<p>天刚亮就在操场上打球</p>
<p>放学后在街边吃饱了才回家</p>
<p>&nbsp;</p>
<p>还记得</p>
<p>当年的我们</p>
<p>牵着你的手走遍校园的每一个角落</p>
<p>把你拥在怀里畅想着关于我们的未来</p>
<p>&nbsp;</p>
<p>当年的我们</p>
<p>现在都已经长大</p>
<p>一起2B的日子已经远去</p>
<p>各自NB的日子不知何时才能到来</p>
<p>&nbsp;</p>
<p>多年后</p>
<p>再一次驻足天桥</p>
<p>当年的你们是否还会记得我</p>
<p>我永远都不会忘记你们</p>
<p>&nbsp;</p>
<p>&#8211;写于2011-06-13 23:00:00</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=101</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>卡塔尔世界杯之诸神的黄昏</title>
		<link>http://www.waicai.org/?p=96</link>
		<comments>http://www.waicai.org/?p=96#comments</comments>
		<pubDate>Fri, 25 Nov 2022 08:36:22 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[随笔]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=96</guid>
		<description><![CDATA[时间流逝，一晃疫情已经过去了三年，卡塔尔世界杯也随之到来。 随着第一轮结束，英格兰和西班牙让我们看到了恐怖的攻 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>时间流逝，一晃疫情已经过去了三年，卡塔尔世界杯也随之到来。<br />
随着第一轮结束，英格兰和西班牙让我们看到了恐怖的攻击力，沙特和日本的胜利，给亚洲足球带来了荣光。<br />
这次世界杯我愿意称之为诸神的黄昏，足坛双子星，35岁的梅西和37岁的C罗，恐怕是最后一次世界杯之旅了，对于80年代生人，还有很多陪伴多年的球星，在这之后也会永远告别世界杯的舞台。33岁的贝尔，33岁的二娃，34岁的莱万，34岁的迪马利亚，34岁的布教授，35岁的卡瓦尼，35岁的苏牙，36岁的大吉鲁，36岁的诺伊尔，37岁的魔笛，38岁的蒂亚戈席尔瓦，39岁的阿尔维斯，39岁的武僧佩佩，还有没能来的新金球奖得主35岁的本泽马，很多叱咤足坛多年的人，都踏上了最后一次世界杯舞台。<br />
当然，这也是新老交替的一届世界杯。随着老人的离去，新星们也在冉冉升起。23岁的德里赫特，23岁的芒特，23岁的加克波，23岁的姆巴佩，23岁的哈弗茨，23岁的菲利克斯，22岁的福登，22岁的维尼修斯，22岁的安东尼，22岁的费兰托雷斯，21岁的萨卡，20岁的法蒂，20岁的佩德里，19岁的穆西亚拉，19岁的贝林厄姆，18岁的穆科科，18岁的加维，这些人有的已经成名，有的还在成长，早晚有一天，会闪耀球场。<br />
第二轮即将开始了，英格兰和西班牙会不会继续大胜，阿根廷和德国能否走出输球的阴霾，C罗能否带领葡萄牙继续前进，没有内马尔的巴西还能不能走得更远，让我们拭目以待吧。</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=96</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>PHP序列化和反序列化</title>
		<link>http://www.waicai.org/?p=85</link>
		<comments>http://www.waicai.org/?p=85#comments</comments>
		<pubDate>Wed, 24 Feb 2021 05:19:21 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[PHP]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=85</guid>
		<description><![CDATA[序列化 序列化格式 在PHP中，序列化用于存储或传递 PHP 的值的过程中，同时不丢失其类型和结构。 序列化函 [&#8230;]]]></description>
				<content:encoded><![CDATA[<h1><strong>序列化</strong></h1>
<h2>序列化格式</h2>
<p>在PHP中，序列化用于存储或传递 PHP 的值的过程中，同时不丢失其类型和结构。</p>
<p>序列化函数原型如下：</p>
<pre><code>string serialize ( mixed $value )</code></pre>
<p>例子：</p>
<pre><code>class TEST {
    public $TEST_data = 'test_data';
}
class C_TEST extends TEST{
    const SECOND = 60;

    public $data;
    private $pass;

    public function __construct($data, $pass)
    {
        $this-&gt;data = $data;
        $this-&gt;pass = $pass;
    }

    public function setPass($pass)
    {
        $this-&gt;pass = $pass;
    }
}
$test = new C_TEST('test', true);
$number = 100;
$str = 'test';
$bool = true;
$null = NULL;
$arr = array('a' =&gt; 1, 'b' =&gt; 2);
var_dump(serialize($number));
var_dump(serialize($str));
var_dump(serialize($bool));
var_dump(serialize($null));
var_dump(serialize($arr));
var_dump(serialize($test));</code></pre>
<p>输出结果为：</p>
<pre><code>string(6)"i:100;"
string(11)"s:4:"test";"
string(4)"b:1;"
string(2)"N;"
string(30)"a:2:{s:1:"a";i:1;s:1:"b";i:2;}"
string(87)"O:6:"C_TEST":3{s:4:"data";s:4:"test";s:12:"C_TESTpass";b:1;s:9:"TEST_data";s:9:"test_data";}"</code></pre>
<p>序列化对象时，不会保存常量的值。对于父类中的变量，则会保留。</p>
<p>序列化对于不同类型得到的字符串格式为：</p>
<ul>
<li>String : s:size:value;</li>
<li>Integer : i:value;</li>
<li>Boolean : b:value;(保存1或0)</li>
<li>Null : N;</li>
<li>Array : a:size:{key definition;value definition;(repeated per element)}</li>
<li>Object : O:strlen(object name):object name:object size:{s:strlen(property name):property name:property definition;(repeated per property)}</li>
</ul>
<h2>对象序列化自定义</h2>
<p>在序列化对象的时候，对于对象中的一些敏感属性，我们不需要保存时，要用到一个魔术方法，当调用serialize()函数序列化对象时，该函数会检查类中是否存在魔术方法__sleep()。如果存在，该方法会先被调用，然后才执行序列化操作。可以通过重载这个方法，从而自定义序列化行为。该方法原型如下：</p>
<pre><code>public array __sleep ( void )</code></pre>
<ul>
<li>该方法返回一个包含对象中所有应被序列化的变量名称的数组</li>
<li>该方法未返回任何内容，则 NULL 被序列化，并产生一个E_NOTICE级别的错误</li>
<li>__sleep()不能返回父类的私有成员的名字。这样做会产生一个E_NOTICE级别的错误。这时只能用Serializable接口来替代。</li>
<li>常用于保存那些大对象时的清理工作，避免保存过多冗余数据</li>
</ul>
<p>例子：</p>
<pre><code>class User{
    const SITE = 'uusama';

    public $username;
    public $nickname;
    private $password;

    public function __construct($username, $nickname, $password)
    {
        $this-&gt;username = $username;
        $this-&gt;nickname = $nickname;
        $this-&gt;password = $password;
    }

    // 重载序列化调用的方法
    public function __sleep()
    {
        // 返回需要序列化的变量名，过滤掉password变量
        return array('username', 'nickname');
    }
}
$user = new User('uusama', 'uu', '123456');
var_dump(serialize($user));</code></pre>
<p>返回结果如下，序列化的时候忽略了 password 字段的值。</p>
<pre><code>string(67) "O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}"</code></pre>
<h2>序列化对象存储</h2>
<p>通过上面的介绍，我们可以把一个复制的对象或者数据序列化成一个序列字符串，保存值的同事还保存了他们的结构。</p>
<p>我们可以把序列化之后的值保存起来，存在文件或者缓存里面。不推荐存在数据库里面，可读性查，而且不便于迁移维护，不便于查询。</p>
<pre><code>$user = new User('uusama', 'uu', '123456');
$ser = serialize($user);
// 保存在本地
file_put_contents('user.ser', $ser);</code></pre>
<h1>反序列化</h1>
<h2>反序列函数</h2>
<pre><code>mixed unserialize ( string $str )</code></pre>
<p>unserialize()反序列化函数用于将单一的已序列化的变量转换回 PHP 的值。</p>
<ul>
<li>如果传递的字符串不可解序列化，则返回 FALSE，并产生一个E_NOTICE</li>
<li>返回的是转换之后的值，可为integer&#8220;float、string、array或object</li>
<li>若被反序列化的变量是一个对象，在成功重新构造对象之后，PHP会自动地试图去调用__wakeup()成员函数（如果存在的话）</li>
</ul>
<p>例子：</p>
<pre><code>class User{
    const SITE = 'uusama';

    public $username;
    public $nickname;
    private $password;
    private $order;

    public function __construct($username, $nickname, $password)
    {
        $this-&gt;username = $username;
        $this-&gt;nickname = $nickname;
        $this-&gt;password = $password;
    }

    // 定义反序列化后调用的方法
    public function __wakeup()
    {
        $this-&gt;password = $this-&gt;username;
    }
}
$user_ser = 'O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}';
var_dump(unserialize($user_ser));</code></pre>
<p>输出结果为：</p>
<pre><code>object(User)#1 (4) {
  ["username"]=&gt;
  string(6) "uusama"
  ["nickname"]=&gt;
  string(2) "uu"
  ["password":"User":private]=&gt;
  string(6) "uusama"
  ["order":"User":private]=&gt;
  NULL
}</code></pre>
<ul>
<li>__wakeup()函数在对象被构建以后执行，所以$this-&gt;username的值不为空</li>
<li>反序列化时，会尽量将变量值进行匹配并复制给序列化后的对象</li>
</ul>
<h2>未定义类的处理</h2>
<p>在上面的例子中，我们在调用反序列化函数unserialize()之前，提前定义了User类</p>
<p>没有定义例：</p>
<div>
<pre><code>$user_ser = 'O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}';
var_dump(unserialize($user_ser));</code></pre>
</div>
<p>得到的结果：</p>
<div>
<pre><code>object(__PHP_Incomplete_Class)#1 (3) {
  ["__PHP_Incomplete_Class_Name"]=&gt;
  string(4) "User"
  ["username"]=&gt;
  string(6) "uusama"
  ["nickname"]=&gt;
  string(2) "uu"
}</code></pre>
</div>
<p>这个例子中，没有定义任何的User类，反序列化正常执行，并没有报错，对比之前定义了User类的结果，这儿反序列化得到的对象是__PHP_Incomplete_Class，并指定了未定义类的类名。</p>
<p>如果这个时候去使用这个反序列化后的不明对象，则会抛出E_NOTICE。</p>
<p>有两种解决方案。</p>
<ul>
<li>定义__autoload()等函数，指定发现未定义类时加载类的定义文件</li>
<li>可通过 php.ini、ini_set() 或 .htaccess 定义unserialize_callback_func。每次实例化一个未定义类时它都会被调用</li>
</ul>
<p>以上两种方案的实现如下：</p>
<div>
<pre><code>// unserialize_callback_func 从 PHP 4.2.0 起可用
ini_set('unserialize_callback_func', 'mycallback'); // 设置您的回调函数
function mycallback($classname)
{
   // 只需包含含有类定义的文件
   // $classname 指出需要的是哪一个类
}

// 建议使用下面的函数，代替__autoload()
spl_autoload_register(function ($class_name) {
    // 动态加载未定义类的定义文件
    require_once $class_name . '.php';
});</code></pre>
</div>
<h2>PHP预定义序列化接口Serializable</h2>
<p>上面在将序列化过程中遇到的：无法在__sleep()方法中返回父类对象的问题吗，方法就是实现序列化接口Serializable。</p>
<p>该接口的原型如下：</p>
<div>
<pre><code>Serializable {
    abstract public string serialize ( void )
    abstract public mixed unserialize ( string $serialized )
}</code></pre>
</div>
<p>需要注意的是，如果定义的类实现了Serializable接口，那么序列化和反序列化的时候，PHP就不会再去调用__sleep()方法和__wakeup()方法。</p>
<div>
<pre><code>class CB implements Serializable{
    public $CB_data = '';
    private $CB_password = 'ttt';
    public function setCBPassword($password)
    {
        $this-&gt;CB_password = $password;
    }

    public function serialize()
    {
        echo __METHOD__ . "\n";
        return serialize($this-&gt;CB_password);
    }

    public function unserialize($serialized)
    {
        echo __METHOD__ . "\n";
    }
}

class CC extends CB {
    const SECOND = 60;

    public $data;
    private $pass;
    public function __construct($data, $pass)
    {
        $this-&gt;data = $data;
        $this-&gt;pass = $pass;
    }
    public function __sleep()
    {
        // 输出调用了该方法名
        echo __METHOD__ . "\n";
    }

    public function __wakeup()
    {
        // 输出调用了该方法名
        echo __METHOD__ . "\n";
    }
}

$cc = new CC('uu', true);
$ser = serialize($cc);
var_dump($ser);
$un_cc = unserialize($ser);
var_dump($un_cc);
</code></pre>
</div>
<p>运行结果为：</p>
<div>
<pre><code>CB::serialize
string(24) "C:2:"CC":10:{s:3:"ttt";}"
CB::unserialize
object(CC)#2 (4) {
  ["data"]=&gt;
  NULL
  ["pass":"CC":private]=&gt;
  NULL
  ["CB_data"]=&gt;
  string(0) ""
  ["CB_password":"CB":private]=&gt;
  string(3) "ttt"
}</code></pre>
</div>
<p>可以完全定义serialize()方法，该方法返回的值就是序列化后大括号内的值，只要保证自定义序列化和反序列化的规则一致即可。</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=85</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>MySQL常用命令</title>
		<link>http://www.waicai.org/?p=79</link>
		<comments>http://www.waicai.org/?p=79#comments</comments>
		<pubDate>Thu, 15 Nov 2018 08:35:33 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[数据库]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=79</guid>
		<description><![CDATA[1.连接命令：mysql -h 数据库连接 -u 用户名 -p 2.显示数据库：show databases; [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>1.连接命令：mysql -h 数据库连接 -u 用户名 -p<br />
2.显示数据库：show databases;<br />
3.选择数据库：use 库名;<br />
4.显示所有表：show tables;<br />
5.显示表结构：describe 表名;<br />
6.创建数据库：create database 库名;<br />
7.增删改查：insert,delete,update,select<br />
8.删库删表：drop database 库名;drop table 表名;</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=79</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>CentOS 7 安装 LNMP</title>
		<link>http://www.waicai.org/?p=65</link>
		<comments>http://www.waicai.org/?p=65#comments</comments>
		<pubDate>Fri, 13 Jul 2018 05:42:52 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[PHP]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=65</guid>
		<description><![CDATA[记录在CentOS 7安装LNMP环境（PHP7 + MySQL5.7 + Nginx1.10）的过程。 一  [&#8230;]]]></description>
				<content:encoded><![CDATA[<p><span style="font-family: Bitter, Georgia, serif; font-size: 22px; line-height: 1.3;">记录在CentOS 7安装LNMP环境（PHP7 + MySQL5.7 + Nginx1.10）的过程。</span></p>
<p>一 、修改yum源</p>
<pre><code>rpm -Uvh https://dl.Fedoraproject.org/pub/epel/epel-release-latest-7.noarch.rpm

rpm -Uvh https://mirror.webtatic.com/yum/el7/webtatic-release.rpm

rpm -Uvh http://dev.mysql.com/get/mysql57-community-release-el7-9.noarch.rpm
</code></pre>
<p>二 、安装 Nginx、MySQL、PHP</p>
<pre><code>yum -y install nginx

yum -y install mysql-community-server

yum -y install php70w-devel php70w.x86_64 php70w-cli.x86_64 php70w-common.x86_64 php70w-gd.x86_64 php70w-ldap.x86_64 php70w-mbstring.x86_64 php70w-mcrypt.x86_64  php70w-pdo.x86_64   php70w-mysqlnd  php70w-fpm php70w-opcache php70w-pecl-redis php70w-pecl-mongo
</code></pre>
<p>三 、配置 MySQL</p>
<pre><code>systemctl start mysqld #启动MySQL

grep 'temporary password' /var/log/mysqld.log #查找默认密码
#输出：2018-07-12T09:04:58.532626Z 1 [Note] A temporary password is generated for root@localhost: qrky5irl-y5O

mysql -uroot -p'qrky5irl-y5O' #登录MySQL

set password for 'root'@'localhost'=password('!@#123qwe'); #修改密码 包含字符，英文，数字

vim /etc/my.cnf #配置默认编码
[mysqld]
character_set_server=utf8
init_connect='SET NAMES utf8'

systemctl restart mysqld #重启MySQL

systemctl enable mysqld #设置开机启动

配置文件：/etc/my.cnf
日志文件：/var/log/mysqld.log
服务启动脚本：/usr/lib/systemd/system/mysqld.service
socket 文件：/var/run/mysqld/mysqld.pid
</code></pre>
<p>四 、配置 Nginx</p>
<pre><code>systemctl status firewalld #查看防火墙配置，如果显示active(running)，则需要调整防火墙规则的配置。

vim /etc/firewalld/zones/public.xml #配置防火墙规则
&lt;zone&gt;
&lt;service name="nginx"/&gt;
&lt;zone&gt;

systemctl reload firewalld #重启firewalld

vim /etc/nginx/nginx.conf #修改Nginx配置
#在server{}里添加：
location / {
    #定义首页索引文件的名称
    index index.php index.html index.htm;   
}
# PHP 脚本请求全部转发到 FastCGI处理. 使用FastCGI默认配置.
location ~ .php$ {
    fastcgi_pass 127.0.0.1:9000;
    fastcgi_index index.php;
    fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
    include fastcgi_params;
}

systemctl start nginx #启动Nginx

systemctl enable nginx #设置开机启动
</code></pre>
<p>五 、配置PHP</p>
<pre><code>systemctl enable php-fpm #设置开机启动

systemctl start php-fpm #启动php-fpm
</code></pre>
<p>六 、测试</p>
<pre><code>vim /usr/share/nginx/html/phpinfo.php #创建测试页
&lt;?php
phpinfo();
?&gt;

#页面访问http://IP地址/phpinfo.php，成功输入，配置成功
</code></pre>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=65</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Memcached和Memcache</title>
		<link>http://www.waicai.org/?p=56</link>
		<comments>http://www.waicai.org/?p=56#comments</comments>
		<pubDate>Thu, 13 Oct 2016 13:37:59 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[PHP]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=56</guid>
		<description><![CDATA[Memcache::set bool Memcache::set ( string $key , mixed  [&#8230;]]]></description>
				<content:encoded><![CDATA[<p><a href="http://php.net/manual/zh/memcache.set.php">Memcache::set</a></p>
<p>bool Memcache::set ( string <code>$key</code> , mixed <code>$var</code> [, int <code>$flag</code> [, int <code>$expire</code> ]] )</p>
<p>参数<br />
key<br />
要设置值的key。<br />
var<br />
要存储的值，字符串和数值直接存储，其他类型序列化后存储。<br />
flag<br />
使用MEMCACHE_COMPRESSED指定对值进行压缩(使用zlib)。<br />
expire<br />
当前写入缓存的数据的失效时间。如果此值设置为0表明此数据永不过期。你可以设置一个UNIX时间戳或 以秒为单位的整数（从当前算起的时间差）来说明此数据的过期时间，但是在后一种设置方式中，不能超过 2592000秒（30天）。</p>
<p>&nbsp;</p>
<p><a href="http://php.net/manual/zh/memcached.set.php">Memcached::set</a></p>
<p>public bool Memcached::set ( string <code>$key</code> , mixed <code>$value</code> [, int <code>$expiration</code> ] )</p>
<p>参数<br />
key<br />
用于存储值的键名。<br />
value<br />
存储的值。<br />
expiration<br />
到期时间，默认为 0。 更多信息请参见到期时间。</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=56</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Windows+GO+Sublime Text 3</title>
		<link>http://www.waicai.org/?p=52</link>
		<comments>http://www.waicai.org/?p=52#comments</comments>
		<pubDate>Fri, 01 Jul 2016 09:15:46 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[GO]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=52</guid>
		<description><![CDATA[1.安装Sublime Text 3 略，自行百度 2.下载并安装Go语言开发包 http://www.gol [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>1.安装Sublime Text 3</p>
<p>略，自行百度</p>
<p>2.下载并安装Go语言开发包</p>
<p>http://www.golangtc.com/download</p>
<p>3.安装Package Control</p>
<p>调出命令行view-&gt;show console</p>
<p>import urllib.request,os; pf = &#8216;Package Control.sublime-package&#8217;; ipp = sublime.installed_packages_path(); urllib.request.install_opener( urllib.request.build_opener( urllib.request.ProxyHandler()) ); open(os.path.join(ipp, pf), &#8216;wb&#8217;).write(urllib.request.urlopen( &#8216;http://sublime.wbond.net/&#8217; + pf.replace(&#8216; &#8216;,&#8217;%20&#8242;)).read())</p>
<p>4.安装Gosublime</p>
<p>调出搜索栏 Ctrl+Shift+p</p>
<p>分别搜索并安装Install Package 和Gosublime</p>
<p>5.编写hello.go文件</p>
<p>package main</p>
<p>import (<br />
&#8220;fmt&#8221;<br />
)</p>
<p>func main() {<br />
fmt.Println(&#8220;Hello World!&#8221;)<br />
}</p>
<p>6.Ctrl+b编译运行</p>
<p>go run hello.go</p>
<p>7.备注</p>
<p>如果报错，查看环境变量的设置GOPATH增加go\bin以及代码所在路径</p>
<p>如果报错，查看环境变量的设置GOROOT增加go\bin</p>
<p>如果报错，查看环境变量的设置Path增加%GOROOT%\bin</p>
<p>编译可执行文件命令<em>go build -o C:\test.exe C:\test.go</em></p>
<p>&nbsp;</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=52</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>GIT的简单命令</title>
		<link>http://www.waicai.org/?p=49</link>
		<comments>http://www.waicai.org/?p=49#comments</comments>
		<pubDate>Thu, 23 Jun 2016 08:48:17 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[工具]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=49</guid>
		<description><![CDATA[拉取master git pull origin master 新增版本/文件 git add -A 提交本地 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>拉取master<br />
git pull origin master</p>
<p>新增版本/文件<br />
git add -A</p>
<p>提交本地<br />
git commit -am &#8220;注释&#8221;</p>
<p>提交分支<br />
git push origin 分支</p>
<p>切换分支<br />
git checkout 分支</p>
<p>把我的代码在本地master合并<br />
git merge &#8211;squash 我的分支</p>
<p>保留当前分支修改<br />
git stash</p>
<p>切换分支后合并修改<br />
git stash pop</p>
<p>创建分支<br />
git branch 新分支<br />
git checkout 新分支</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=49</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>WINDOWS环境安装PHP的RABBITMQ扩展</title>
		<link>http://www.waicai.org/?p=45</link>
		<comments>http://www.waicai.org/?p=45#comments</comments>
		<pubDate>Mon, 23 Nov 2015 12:13:25 +0000</pubDate>
		<dc:creator><![CDATA[waicai]]></dc:creator>
				<category><![CDATA[PHP]]></category>

		<guid isPermaLink="false">http://www.waicai.org/?p=45</guid>
		<description><![CDATA[1.下载amqp扩展模块php_amqp.dll，放在\php\ext\目录下；rabbitmq扩展模块rab [&#8230;]]]></description>
				<content:encoded><![CDATA[<p>1.下载amqp扩展模块php_amqp.dll，放在\php\ext\目录下；rabbitmq扩展模块rabbitmq.1.dll，放在\php目录下</p>
<p>下载地址：http://pecl.php.net/package/amqp/1.4.0/windows</p>
<p>根据PHP版本（5.3,5.4,5.5,5.6），是否线程安全（TS,NTS），PHP服务的版本（X84，X64）</p>
<p>2.在PHP.INI中写入</p>
<p>[amqp]</p>
<p>extension=php_amqp.dll</p>
<p>3.在APACHE的HTTPD.CONF中写入</p>
<p>LoadFile  &#8221;C:\xampp\php\rabbitmq.1.dll&#8221;</p>
<p>路径根据环境自行修改</p>
<p>4.重启服务</p>
]]></content:encoded>
			<wfw:commentRss>http://www.waicai.org/?feed=rss2&#038;p=45</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
