使用Java给微信公众号创建自定义的菜单

atool 发表了文章 • 0 个评论 • 1541 次浏览 • 2016-04-22 12:53 • 来自相关话题

最近一段时间,微信公众平台做了很多的改进啊~

微信目前将公众帐号分成服务号和订阅号两种,之前的微信公众号默认为订阅号,不过有一次机会可以自由选择,两者之间各有优劣:

服务号每月只能推送一条消息,不过消息是显示在消息列表中,并有消息提示;订阅号每天有一条消息推送,不过消息不显示在消息列表,没有消息提示。

除此之外,目前微信公众号可以自定义菜单,下面将介绍如果使用Java作一个微信公众号的菜单,参考API可到官网查看:http://mp.weixin.qq.com/wiki/index.php?title=%E8%87%AA%E5%AE%9A%E4%B9%89%E8%8F%9C%E5%8D%95%E6%8E%A5%E5%8F%A3
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;

import org.json.JSONObject;

public class MenuUtil {
/**
* 获得ACCESS_TOKEN
* @Title: getAccess_token
* @Description: 获得ACCESS_TOKEN
* @param @return 设定文件
* @return String 返回类型
* @throws
*/
private static String getAccess_token(){

String APPID="";
String APPSECRET="";

String url = "https://api.weixin.qq.com/cgi- ... 3B%2B APPID + "&secret=" +APPSECRET;
String accessToken = null;
try {
URL urlGet = new URL(url);
HttpURLConnection http = (HttpURLConnection) urlGet.openConnection();

http.setRequestMethod("GET"); //必须是get方式请求
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");

JSONObject demoJson = new JSONObject(message);
accessToken = demoJson.getString("access_token");

System.out.println(message);
} catch (Exception e) {
e.printStackTrace();
}
return accessToken;
}

/**
* 创建Menu
* @Title: createMenu
* @Description: 创建Menu
* @param @return
* @param @throws IOException 设定文件
* @return int 返回类型
* @throws
*/
public static String createMenu() {
String menu = "{\"button\":[{\"type\":\"click\",\"name\":\"MENU01\",\"key\":\"1\"},{\"type\":\"click\",\"name\":\"天气查询\",\"key\":\"西安\"},{\"name\":\"日常工作\",\"sub_button\":[{\"type\":\"click\",\"name\":\"待办工单\",\"key\":\"01_WAITING\"},{\"type\":\"click\",\"name\":\"已办工单\",\"key\":\"02_FINISH\"},{\"type\":\"click\",\"name\":\"我的工单\",\"key\":\"03_MYJOB\"},{\"type\":\"click\",\"name\":\"公告消息箱\",\"key\":\"04_MESSAGEBOX\"},{\"type\":\"click\",\"name\":\"签到\",\"key\":\"05_SIGN\"}]}]}";

//此处改为自己想要的结构体,替换即可
String access_token= getAccess_token();

String action = "https://api.weixin.qq.com/cgi- ... en%3B
try {
URL url = new URL(action);
HttpURLConnection http = (HttpURLConnection) url.openConnection();

http.setRequestMethod("POST");
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();
OutputStream os= http.getOutputStream();
os.write(menu.getBytes("UTF-8"));//传入参数
os.flush();
os.close();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");
return "返回信息"+message;
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return "createMenu 失败";
}

/**
* 删除当前Menu
* @Title: deleteMenu
* @Description: 删除当前Menu
* @param @return 设定文件
* @return String 返回类型
* @throws
*/
public static String deleteMenu()
{
String access_token= getAccess_token();
String action = "https://api.weixin.qq.com/cgi-bin/menu/delete? access_token="+access_token;
try {
URL url = new URL(action);
HttpURLConnection http = (HttpURLConnection) url.openConnection();

http.setRequestMethod("GET");
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();
OutputStream os= http.getOutputStream();
os.flush();
os.close();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");
return "deleteMenu返回信息:"+message;
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return "deleteMenu 失败";
}

public static void main(String[] args) {
System.out.println(createMenu());
}
}代码来源于:http://www.oschina.net/code/snippet_945711_24355 ,后续会做PHP版,并加入到自己的微信公众号上面,我的微信公众号二维码见右侧。Enjoy~ 查看全部
最近一段时间,微信公众平台做了很多的改进啊~

微信目前将公众帐号分成服务号和订阅号两种,之前的微信公众号默认为订阅号,不过有一次机会可以自由选择,两者之间各有优劣:

服务号每月只能推送一条消息,不过消息是显示在消息列表中,并有消息提示;订阅号每天有一条消息推送,不过消息不显示在消息列表,没有消息提示。

除此之外,目前微信公众号可以自定义菜单,下面将介绍如果使用Java作一个微信公众号的菜单,参考API可到官网查看:http://mp.weixin.qq.com/wiki/index.php?title=%E8%87%AA%E5%AE%9A%E4%B9%89%E8%8F%9C%E5%8D%95%E6%8E%A5%E5%8F%A3
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;

import org.json.JSONObject;

public class MenuUtil {
/**
* 获得ACCESS_TOKEN
* @Title: getAccess_token
* @Description: 获得ACCESS_TOKEN
* @param @return 设定文件
* @return String 返回类型
* @throws
*/
private static String getAccess_token(){

String APPID="";
String APPSECRET="";

String url = "https://api.weixin.qq.com/cgi- ... 3B%2B APPID + "&secret=" +APPSECRET;
String accessToken = null;
try {
URL urlGet = new URL(url);
HttpURLConnection http = (HttpURLConnection) urlGet.openConnection();

http.setRequestMethod("GET"); //必须是get方式请求
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");

JSONObject demoJson = new JSONObject(message);
accessToken = demoJson.getString("access_token");

System.out.println(message);
} catch (Exception e) {
e.printStackTrace();
}
return accessToken;
}

/**
* 创建Menu
* @Title: createMenu
* @Description: 创建Menu
* @param @return
* @param @throws IOException 设定文件
* @return int 返回类型
* @throws
*/
public static String createMenu() {
String menu = "{\"button\":[{\"type\":\"click\",\"name\":\"MENU01\",\"key\":\"1\"},{\"type\":\"click\",\"name\":\"天气查询\",\"key\":\"西安\"},{\"name\":\"日常工作\",\"sub_button\":[{\"type\":\"click\",\"name\":\"待办工单\",\"key\":\"01_WAITING\"},{\"type\":\"click\",\"name\":\"已办工单\",\"key\":\"02_FINISH\"},{\"type\":\"click\",\"name\":\"我的工单\",\"key\":\"03_MYJOB\"},{\"type\":\"click\",\"name\":\"公告消息箱\",\"key\":\"04_MESSAGEBOX\"},{\"type\":\"click\",\"name\":\"签到\",\"key\":\"05_SIGN\"}]}]}";

//此处改为自己想要的结构体,替换即可
String access_token= getAccess_token();

String action = "https://api.weixin.qq.com/cgi- ... en%3B
try {
URL url = new URL(action);
HttpURLConnection http = (HttpURLConnection) url.openConnection();

http.setRequestMethod("POST");
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();
OutputStream os= http.getOutputStream();
os.write(menu.getBytes("UTF-8"));//传入参数
os.flush();
os.close();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");
return "返回信息"+message;
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return "createMenu 失败";
}

/**
* 删除当前Menu
* @Title: deleteMenu
* @Description: 删除当前Menu
* @param @return 设定文件
* @return String 返回类型
* @throws
*/
public static String deleteMenu()
{
String access_token= getAccess_token();
String action = "https://api.weixin.qq.com/cgi-bin/menu/delete? access_token="+access_token;
try {
URL url = new URL(action);
HttpURLConnection http = (HttpURLConnection) url.openConnection();

http.setRequestMethod("GET");
http.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
http.setDoOutput(true);
http.setDoInput(true);
System.setProperty("sun.net.client.defaultConnectTimeout", "30000");//连接超时30秒
System.setProperty("sun.net.client.defaultReadTimeout", "30000"); //读取超时30秒

http.connect();
OutputStream os= http.getOutputStream();
os.flush();
os.close();

InputStream is =http.getInputStream();
int size =is.available();
byte[] jsonBytes =new byte[size];
is.read(jsonBytes);
String message=new String(jsonBytes,"UTF-8");
return "deleteMenu返回信息:"+message;
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return "deleteMenu 失败";
}

public static void main(String[] args) {
System.out.println(createMenu());
}
}
代码来源于:http://www.oschina.net/code/snippet_945711_24355 ,后续会做PHP版,并加入到自己的微信公众号上面,我的微信公众号二维码见右侧。Enjoy~

Python minidom去除xml文本中空行、空格、tab,美化xml显示样式

atool 发表了文章 • 0 个评论 • 2595 次浏览 • 2016-04-22 12:37 • 来自相关话题

使用Python minidom的库函数进行xml文件或者文本的增加借点、删除借点、增加属性、删除属性、合并xml文件或者文本等这些操作的时候,往往多多出很多的空行,使用toxml()方法返回字符串保存到文件之后,空行依然存在,非常不美观,下面使用python介绍怎么去除xml文本中空行、空格、tab,美化xml显示样式,代码如下:
@classmethod
def _beautifulFormat(self, xmlDomObject):
'''美化xml格式
'''
if xmlDomObject:
#优化格式显示
xmlStr = xmlDomObject.toprettyxml(indent = '', newl = '', encoding = 'utf-8')
xmlStr = xmlStr.replace('\t', '').replace('\n', '')
xmlDomObject = minidom.parseString(xmlStr)
xmlStr = xmlDomObject.toprettyxml(indent = '\t', newl = '\n', encoding = 'utf-8')

return xmlStr
else:
return False这个是写到一个类中的静态方法,传入参数为minidom中的xml Object对象,返回的是格式化美化之后的xml文本。

大致思路是首先去掉xml中的所有空行、tab,然后在用minidom读入没有空行,tab的xml文本,然后再使用toprettyxml()方法格式化输出字符串。

自己在程序中使用,没有任何问题,如有问题请留言,如果帮助到你,请点点广告,你懂的~ 查看全部
使用Python minidom的库函数进行xml文件或者文本的增加借点、删除借点、增加属性、删除属性、合并xml文件或者文本等这些操作的时候,往往多多出很多的空行,使用toxml()方法返回字符串保存到文件之后,空行依然存在,非常不美观,下面使用python介绍怎么去除xml文本中空行、空格、tab,美化xml显示样式,代码如下:
 @classmethod
def _beautifulFormat(self, xmlDomObject):
'''美化xml格式
'''
if xmlDomObject:
#优化格式显示
xmlStr = xmlDomObject.toprettyxml(indent = '', newl = '', encoding = 'utf-8')
xmlStr = xmlStr.replace('\t', '').replace('\n', '')
xmlDomObject = minidom.parseString(xmlStr)
xmlStr = xmlDomObject.toprettyxml(indent = '\t', newl = '\n', encoding = 'utf-8')

return xmlStr
else:
return False
这个是写到一个类中的静态方法,传入参数为minidom中的xml Object对象,返回的是格式化美化之后的xml文本。

大致思路是首先去掉xml中的所有空行、tab,然后在用minidom读入没有空行,tab的xml文本,然后再使用toprettyxml()方法格式化输出字符串。

自己在程序中使用,没有任何问题,如有问题请留言,如果帮助到你,请点点广告,你懂的~

dom4j解析创建Xml:org.dom4j.DocumentException: 2字节的UTF-8序列的2无效

atool 发表了文章 • 0 个评论 • 2595 次浏览 • 2016-04-22 11:26 • 来自相关话题

最近刚刚做完Java版Dota改键软件KeyHelper的界面配置操作,配置信息保存到xml中,采用dom4j进行解析和创建,基本做的差不多,并导出为jar自己在使用测试。

在Eclipse中运行时没有任何异常和错误,但是导出jar包之后使用start java -jar keyHelper.jar运行的时候,报异常:org.dom4j.DocumentException: 2字节的UTF-8序列的2无效...后面是一大串错误stack信息。

搜索了一下,网上关于dom4j解析xml汉字乱码问题一大堆,其中解决办法基本没有,大都是在网上发发问题,但是没有人解决,只有一个提到说将xml的编码改成为GBK或者GB2312,这个测试了,基本上是可行的,但是一般的软件开发都是采用UTF-8,将配置换成其他编码,会导致软件界面上是乱码,所以这个解决办法其实有一点鸡肋。自己尝试过在创建保存xml的时候,利用构造的domcument,调用asXml()方法得到document的string字符串,然后采用java io利用utf-8编码写入文件中,但是到读取的时候依然会报错。

再三折腾之下,发现了一个既不用修改xml编码,又可以解决这个中文解析的异常,贴出我的解析的创建xml的代码。

一、解析Config.xml配置信息
private void loadFromXml() {
try {
AXReader reader = new SAXReader();
File file = new File(CONFIG_FILE);
Document document = reader.read(new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8")));// 读取XML文件
Element xmlRoot = document.getRootElement();// 得到根节点
lastKeySet = xmlRoot.elementText("lastKeySet");
lastLocationX = Integer.valueOf(xmlRoot.elementText("lastLocationX"));
lastLocationY = Integer.valueOf(xmlRoot.elementText("lastLocationY"));
} catch (Exception e) {
e.printStackTrace();
lastKeySet = "";//上次选择的改键方案
lastLocationX = 500;//上次窗口位置
lastLocationY = 100;//上次窗口位置
}
}其中代码:Document document = reader.read(newBufferedReader(newInputStreamReader(newFileInputStream(file), "UTF-8")));是关键。

二、创建Config.xml配置信息
public void saveToXml() {
/** 建立document对象 */
Document document = DocumentHelper.createDocument();
/** 建立config根节点 */
Element configElement = document.addElement("config");
configElement.addElement("lastKeySet").setText(lastKeySet);
configElement.addElement("lastLocationX").setText(lastLocationX + "");
configElement.addElement("lastLocationY").setText(lastLocationY + "");
try{
/* 将document中的内容写入文件中 */
OutputFormat format = new OutputFormat();
format.setEncoding("UTF-8");
XMLWriter writer = new XMLWriter(new FileOutputStream(new File(CONFIG_FILE)), format);
writer.write(document);
writer.close();
} catch(Exception ex) {
ex.printStackTrace();
}
}其中代码:OutputFormat format = newOutputFormat();
format.setEncoding("UTF-8");
XMLWriter writer = newXMLWriter(newFileOutputStream(newFile(CONFIG_FILE)), format);是关键,请务必按照这个写法进行。具体为什么,我自己也不知道,唉~
Enjoy~ 查看全部
最近刚刚做完Java版Dota改键软件KeyHelper的界面配置操作,配置信息保存到xml中,采用dom4j进行解析和创建,基本做的差不多,并导出为jar自己在使用测试。

在Eclipse中运行时没有任何异常和错误,但是导出jar包之后使用start java -jar keyHelper.jar运行的时候,报异常:org.dom4j.DocumentException: 2字节的UTF-8序列的2无效...后面是一大串错误stack信息。

搜索了一下,网上关于dom4j解析xml汉字乱码问题一大堆,其中解决办法基本没有,大都是在网上发发问题,但是没有人解决,只有一个提到说将xml的编码改成为GBK或者GB2312,这个测试了,基本上是可行的,但是一般的软件开发都是采用UTF-8,将配置换成其他编码,会导致软件界面上是乱码,所以这个解决办法其实有一点鸡肋。自己尝试过在创建保存xml的时候,利用构造的domcument,调用asXml()方法得到document的string字符串,然后采用java io利用utf-8编码写入文件中,但是到读取的时候依然会报错。

再三折腾之下,发现了一个既不用修改xml编码,又可以解决这个中文解析的异常,贴出我的解析的创建xml的代码。

一、解析Config.xml配置信息
private void loadFromXml() {
try {
AXReader reader = new SAXReader();
File file = new File(CONFIG_FILE);
Document document = reader.read(new BufferedReader(new InputStreamReader(new FileInputStream(file), "UTF-8")));// 读取XML文件
Element xmlRoot = document.getRootElement();// 得到根节点
lastKeySet = xmlRoot.elementText("lastKeySet");
lastLocationX = Integer.valueOf(xmlRoot.elementText("lastLocationX"));
lastLocationY = Integer.valueOf(xmlRoot.elementText("lastLocationY"));
} catch (Exception e) {
e.printStackTrace();
lastKeySet = "";//上次选择的改键方案
lastLocationX = 500;//上次窗口位置
lastLocationY = 100;//上次窗口位置
}
}
其中代码:Document document = reader.read(newBufferedReader(newInputStreamReader(newFileInputStream(file), "UTF-8")));是关键。

二、创建Config.xml配置信息
public void saveToXml() {
/** 建立document对象 */
Document document = DocumentHelper.createDocument();
/** 建立config根节点 */
Element configElement = document.addElement("config");
configElement.addElement("lastKeySet").setText(lastKeySet);
configElement.addElement("lastLocationX").setText(lastLocationX + "");
configElement.addElement("lastLocationY").setText(lastLocationY + "");
try{
/* 将document中的内容写入文件中 */
OutputFormat format = new OutputFormat();
format.setEncoding("UTF-8");
XMLWriter writer = new XMLWriter(new FileOutputStream(new File(CONFIG_FILE)), format);
writer.write(document);
writer.close();
} catch(Exception ex) {
ex.printStackTrace();
}
}
其中代码:
OutputFormat format = newOutputFormat();
format.setEncoding("UTF-8");
XMLWriter writer = newXMLWriter(newFileOutputStream(newFile(CONFIG_FILE)), format);
是关键,请务必按照这个写法进行。具体为什么,我自己也不知道,唉~
Enjoy~

PHP读ini文件内容到二维数组,写二维数组到ini文件

atool 发表了文章 • 0 个评论 • 1008 次浏览 • 2016-04-22 11:24 • 来自相关话题

本文主要是说明PHP中二维数据和ini配置文件之间的转化和持久化问题,使用ini存储二维数组的目地是为了持久化保存一些很小的数据,使用数据库又会显得杀鸡使用了牛刀...

在Java语言中,对于一些经常使用的小量数据,一般会使用static类型的集合类来存储在内存中,然后,在php中,即使申明了static,页面一刷新,数据依然没有了,这样对于一些少量状态数据,要么使用数据库,要么使用memcache(需要另行配置软件,一般的php空间应该不支持吧),再就是使用shmop_open函数操作内存(似乎一般的主机也不会开发这个扩展,安全问题吧),或者使用其封装类SimpleSHM。

既然对于一般的主机来说,都不支持上述操作,那么上面的方法就不实用了,我想就只能使用文件进行存储了,php读写文件有四种方式,这个内容大家可以google“php读写文件有四种方式”,博文很多,而且内容都是统一的。我要讲的是其中的一个ini文件和多维数组的转化——php读写ini文件。

1.php读ini文件到二维数组
这个比较简单,使用函数:$userArray = parse_ini_file ( "user_state.ini", true );即可。
2.php写一/二维数组到ini文件
php之所以能够从ini文件中读出二维数组,是因为它有一个约定的数据格式,只要按照这个格式来进行写入内容即可,这个函数是php类库不提供的,但是下面的代码是本博客的公众微信号代码中一部分,用于保存用户的操作状态的代码,代码也不复杂。其中方法write_ini_file可以完成二维数组写入到ini文件中。
<?php
/**
* 记录用户状态
* @author http://50vip.com/
*
*/
class UserState {
public static function getUserState($userId) {
$userArray = parse_ini_file ( "user_state.ini", true );
if (count ( $userArray ) == 0) {
$userArray = array ();
$userArray ["$userId"] = array (
"state" => '0',
"info" => ''
);
UserState::write_ini_file($userArray, 'user_state.ini', true);
return array ("state" => '0', "info" => '');
}
else {
return $userArray["$userId"];
}
}
public static function setUserState($userId, $state, $info) {
$userArray = parse_ini_file ( "user_state.ini", true );
$userArray ["$userId"] = array (
"state" => $state,
"info" => $info
);
UserState::write_ini_file($userArray, 'user_state.ini', true);
}
/**
* 写ini配置文件
* @author http://50vip.com/
*/
private static function write_ini_file($assoc_arr, $path, $has_sections = true) {
$content = "";
if ($has_sections) {
foreach ( $assoc_arr as $key => $elem ) {
$content .= "[" . $key . "]\n";
foreach ( $elem as $key2 => $elem2 ) {
if (is_array ( $elem2 )) {
for($i = 0; $i < count ( $elem2 ); $i ++) {
$content .= $key2 . "[] = " . $elem2 [$i] . "\n";
}
} else if ($elem2 == "")
$content .= $key2 . " = \n";
else
$content .= $key2 . " = " . $elem2 . "\n";
}
}
} else {
foreach ( $assoc_arr as $key => $elem ) {
if (is_array ( $elem )) {
for($i = 0; $i < count ( $elem ); $i ++) {
$content .= $key2 . "[] = " . $elem [$i] . "\n";
}
} else if ($elem == "")
$content .= $key2 . " = \n";
else
$content .= $key2 . " = " . $elem . "\n";
}
}
if (! $handle = fopen ( $path, 'w' )) {
return false;
}
if (! fwrite ( $handle, $content )) {
return false;
}
fclose ( $handle );
return true;
}
}
?>上面的代码既包含的数组到ini的方法,也包含了它的使用方式。下面是ini配置文件的大致内容:
[oFCbmjmZD2pqUBrBB25XNjhyv6qQ]
state = 1
info =Enjoy~ 查看全部
本文主要是说明PHP中二维数据和ini配置文件之间的转化和持久化问题,使用ini存储二维数组的目地是为了持久化保存一些很小的数据,使用数据库又会显得杀鸡使用了牛刀...

在Java语言中,对于一些经常使用的小量数据,一般会使用static类型的集合类来存储在内存中,然后,在php中,即使申明了static,页面一刷新,数据依然没有了,这样对于一些少量状态数据,要么使用数据库,要么使用memcache(需要另行配置软件,一般的php空间应该不支持吧),再就是使用shmop_open函数操作内存(似乎一般的主机也不会开发这个扩展,安全问题吧),或者使用其封装类SimpleSHM。

既然对于一般的主机来说,都不支持上述操作,那么上面的方法就不实用了,我想就只能使用文件进行存储了,php读写文件有四种方式,这个内容大家可以google“php读写文件有四种方式”,博文很多,而且内容都是统一的。我要讲的是其中的一个ini文件和多维数组的转化——php读写ini文件。

1.php读ini文件到二维数组
这个比较简单,使用函数:
$userArray = parse_ini_file ( "user_state.ini", true );
即可。
2.php写一/二维数组到ini文件
php之所以能够从ini文件中读出二维数组,是因为它有一个约定的数据格式,只要按照这个格式来进行写入内容即可,这个函数是php类库不提供的,但是下面的代码是本博客的公众微信号代码中一部分,用于保存用户的操作状态的代码,代码也不复杂。其中方法write_ini_file可以完成二维数组写入到ini文件中。
<?php
/**
* 记录用户状态
* @author http://50vip.com/
*
*/
class UserState {
public static function getUserState($userId) {
$userArray = parse_ini_file ( "user_state.ini", true );
if (count ( $userArray ) == 0) {
$userArray = array ();
$userArray ["$userId"] = array (
"state" => '0',
"info" => ''
);
UserState::write_ini_file($userArray, 'user_state.ini', true);
return array ("state" => '0', "info" => '');
}
else {
return $userArray["$userId"];
}
}
public static function setUserState($userId, $state, $info) {
$userArray = parse_ini_file ( "user_state.ini", true );
$userArray ["$userId"] = array (
"state" => $state,
"info" => $info
);
UserState::write_ini_file($userArray, 'user_state.ini', true);
}
/**
* 写ini配置文件
* @author http://50vip.com/
*/
private static function write_ini_file($assoc_arr, $path, $has_sections = true) {
$content = "";
if ($has_sections) {
foreach ( $assoc_arr as $key => $elem ) {
$content .= "[" . $key . "]\n";
foreach ( $elem as $key2 => $elem2 ) {
if (is_array ( $elem2 )) {
for($i = 0; $i < count ( $elem2 ); $i ++) {
$content .= $key2 . "[] = " . $elem2 [$i] . "\n";
}
} else if ($elem2 == "")
$content .= $key2 . " = \n";
else
$content .= $key2 . " = " . $elem2 . "\n";
}
}
} else {
foreach ( $assoc_arr as $key => $elem ) {
if (is_array ( $elem )) {
for($i = 0; $i < count ( $elem ); $i ++) {
$content .= $key2 . "[] = " . $elem [$i] . "\n";
}
} else if ($elem == "")
$content .= $key2 . " = \n";
else
$content .= $key2 . " = " . $elem . "\n";
}
}
if (! $handle = fopen ( $path, 'w' )) {
return false;
}
if (! fwrite ( $handle, $content )) {
return false;
}
fclose ( $handle );
return true;
}
}
?>
上面的代码既包含的数组到ini的方法,也包含了它的使用方式。下面是ini配置文件的大致内容:
[oFCbmjmZD2pqUBrBB25XNjhyv6qQ]
state = 1
info =
Enjoy~

使用in语句查询数据,并按照in里面的顺序排序

atool 发表了文章 • 0 个评论 • 383 次浏览 • 2016-04-22 11:19 • 来自相关话题

经常会用到用select * from tab where id in (1,2,3,4)语句。如果需要按照查询的id列表排列查询出来的数据记录的顺序该怎么做呢?
如下:
一、Accessselect * From 表 Where id in(1,5,3) order by instr(',1,5,3,',','&id&',')二、MSSQLselect * From 表 Where id in(1,5,3) order by charindex(','+rtrim(cast(id as varchar(10)))+',',',1,5,3,')三、MySql数据库
1.数字型in查询,db_node_id为intSELECT db_node_id, title FROM da_elements WHERE db_node_id IN (1342,1344,1343) ORDER BY FIELD( db_node_id, 1342,1344,1343 );2.字符型in查询,db_node_id为varcharSELECT db_node_id, title FROM da_elements WHERE db_node_id IN ('1342','1344','1343') ORDER BY FIELD( db_node_id, '1342','1344','1343' );Enjoy~ 查看全部
经常会用到用select * from tab where id in (1,2,3,4)语句。如果需要按照查询的id列表排列查询出来的数据记录的顺序该怎么做呢?
如下:
一、Access
select * From 表 Where id in(1,5,3) order by instr(',1,5,3,',','&id&',')
二、MSSQL
select * From 表 Where id in(1,5,3) order by charindex(','+rtrim(cast(id as varchar(10)))+',',',1,5,3,')
三、MySql数据库
1.数字型in查询,db_node_id为int
SELECT db_node_id, title FROM da_elements WHERE db_node_id IN (1342,1344,1343)  ORDER BY FIELD( db_node_id, 1342,1344,1343 );
2.字符型in查询,db_node_id为varchar
SELECT db_node_id, title FROM da_elements WHERE db_node_id IN ('1342','1344','1343')  ORDER BY FIELD( db_node_id, '1342','1344','1343' );
Enjoy~

Java中java.util.PriorityQueue优先级队列使用方法

atool 发表了文章 • 0 个评论 • 687 次浏览 • 2016-04-22 11:10 • 来自相关话题

PriorityQueue是个基于优先级堆的极大优先级队列。
此队列按照在构造时所指定的顺序对元素排序,既可以根据元素的自然顺序来指定排序(参阅 Comparable),也可以根据 Comparator 来指定,这取决于使用哪种构造方法。优先级队列不允许 null 元素。依靠自然排序的优先级队列还不允许插入不可比较的对象(这样做可能导致 ClassCastException)。比如队列 1 3 5 10 2 自动会被排列 1 2 3 5 10。测试代码如下:
import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.Queue;

public class PriorityQueueExample {
public static void main(String[] args) {
Queue<Integer> qi = new PriorityQueue<Integer>();

qi.add(5);
qi.add(2);
qi.add(1);
qi.add(10);
qi.add(3);

while (!qi.isEmpty()){
System.out.print(qi.poll() + ",");
}
System.out.println();
System.out.println("-----------------------------");
//自定义比较器
Comparator<Integer> cmp;
cmp = new Comparator<Integer>() {
public int compare(Integer e1, Integer e2) {
return e2 - e1;
}
};
Queue<Integer> q2 = new PriorityQueue<Integer>(5,cmp);
q2.add(2);
q2.add(8);
q2.add(9);
q2.add(1);
while (!q2.isEmpty()){
System.out.print(q2.poll() + ",");
}
}
}输出结果如下:
output

1,2,3,5,10,
-----------------------------
9,8,2,1,

代码中的自定义的比较器,可以让我们自由定义比较的顺序:
//自定义比较器
Comparator<Integer> cmp;
cmp = new Comparator<Integer>() {
public int compare(Integer e1, Integer e2) {
return e2 - e1;
}
};

此队列的头是按指定排序方式的最小元素。如果多个元素都是最小值,则头是其中一个元素——选择方法是任意的。队列检索操作 poll、remove、peek 和 element 访问处于队列头的元素。优先级队列是无界的,但是有一个内部容量,控制着用于存储队列元素的数组的大小。它总是至少与队列的大小相同。随着不断向优先级队列添加元素,其容量会自动增加。无需指定容量增加策略的细节。

注意1:该队列是用数组实现,但是数组大小可以动态增加,容量无限。

注意2:此实现不是同步的。不是线程安全的。如果多个线程中的任意线程从结构上修改了列表, 则这些线程不应同时访问 PriorityQueue。实例,这时请使用线程安全的PriorityBlockingQueue 类。

注意3:不允许使用 null 元素。

注意4:此实现为插入方法(offer、poll、remove() 和 add 方法)提供 O(log(n)) 时间;为 remove(Object) 和 contains(Object)方法提供线性时间;为检索方法(peek、element 和 size)提供固定时间。

注意5:方法iterator()中提供的迭代器并不保证以有序的方式遍历优先级队列中的元素。至于原因可参考下面关于PriorityQueue的内部实现如果需要按顺序遍历,请考虑使用 Arrays.sort(pq.toArray())。

注意6:可以在构造函数中指定如何排序。如:PriorityQueue()使用默认的初始容量(11)创建一个 PriorityQueue,并根据其自然顺序来排序其元素(使用 Comparable)。PriorityQueue(int initialCapacity)使用指定的初始容量创建一个 PriorityQueue,并根据其自然顺序来排序其元素(使用 Comparable)。PriorityQueue(int initialCapacity, Comparator comparator)。使用指定的初始容量创建一个PriorityQueue,并根据指定的比较器comparator来排序其元素。

注意7:此类及其迭代器实现了 Collection 和 Iterator 接口的所有可选 方法。PriorityQueue的内部实现。PriorityQueue对元素采用的是堆排序,头是按指定排序方式的最小元素。堆排序只能保证根是最大(最小),整个堆并不是有序的。方法iterator()中提供的迭代器可能只是对整个数组的依次遍历。也就只能保证数组的第一个元素是最小的。实例1的结果也正好与此相符。 查看全部
PriorityQueue是个基于优先级堆的极大优先级队列。
此队列按照在构造时所指定的顺序对元素排序,既可以根据元素的自然顺序来指定排序(参阅 Comparable),也可以根据 Comparator 来指定,这取决于使用哪种构造方法。优先级队列不允许 null 元素。依靠自然排序的优先级队列还不允许插入不可比较的对象(这样做可能导致 ClassCastException)。比如队列 1 3 5 10 2 自动会被排列 1 2 3 5 10。测试代码如下:
import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.Queue;

public class PriorityQueueExample {
public static void main(String[] args) {
Queue<Integer> qi = new PriorityQueue<Integer>();

qi.add(5);
qi.add(2);
qi.add(1);
qi.add(10);
qi.add(3);

while (!qi.isEmpty()){
System.out.print(qi.poll() + ",");
}
System.out.println();
System.out.println("-----------------------------");
//自定义比较器
Comparator<Integer> cmp;
cmp = new Comparator<Integer>() {
public int compare(Integer e1, Integer e2) {
return e2 - e1;
}
};
Queue<Integer> q2 = new PriorityQueue<Integer>(5,cmp);
q2.add(2);
q2.add(8);
q2.add(9);
q2.add(1);
while (!q2.isEmpty()){
System.out.print(q2.poll() + ",");
}
}
}
输出结果如下:
output

1,2,3,5,10,
-----------------------------
9,8,2,1,

代码中的自定义的比较器,可以让我们自由定义比较的顺序:
//自定义比较器
Comparator<Integer> cmp;
cmp = new Comparator<Integer>() {
public int compare(Integer e1, Integer e2) {
return e2 - e1;
}
};

此队列的头是按指定排序方式的最小元素。如果多个元素都是最小值,则头是其中一个元素——选择方法是任意的。队列检索操作 poll、remove、peek 和 element 访问处于队列头的元素。优先级队列是无界的,但是有一个内部容量,控制着用于存储队列元素的数组的大小。它总是至少与队列的大小相同。随着不断向优先级队列添加元素,其容量会自动增加。无需指定容量增加策略的细节。

注意1:该队列是用数组实现,但是数组大小可以动态增加,容量无限。

注意2:此实现不是同步的。不是线程安全的。如果多个线程中的任意线程从结构上修改了列表, 则这些线程不应同时访问 PriorityQueue。实例,这时请使用线程安全的PriorityBlockingQueue 类。

注意3:不允许使用 null 元素。

注意4:此实现为插入方法(offer、poll、remove() 和 add 方法)提供 O(log(n)) 时间;为 remove(Object) 和 contains(Object)方法提供线性时间;为检索方法(peek、element 和 size)提供固定时间。

注意5:方法iterator()中提供的迭代器并不保证以有序的方式遍历优先级队列中的元素。至于原因可参考下面关于PriorityQueue的内部实现如果需要按顺序遍历,请考虑使用 Arrays.sort(pq.toArray())。

注意6:可以在构造函数中指定如何排序。如:PriorityQueue()使用默认的初始容量(11)创建一个 PriorityQueue,并根据其自然顺序来排序其元素(使用 Comparable)。PriorityQueue(int initialCapacity)使用指定的初始容量创建一个 PriorityQueue,并根据其自然顺序来排序其元素(使用 Comparable)。PriorityQueue(int initialCapacity, Comparator comparator)。使用指定的初始容量创建一个PriorityQueue,并根据指定的比较器comparator来排序其元素。

注意7:此类及其迭代器实现了 Collection 和 Iterator 接口的所有可选 方法。PriorityQueue的内部实现。PriorityQueue对元素采用的是堆排序,头是按指定排序方式的最小元素。堆排序只能保证根是最大(最小),整个堆并不是有序的。方法iterator()中提供的迭代器可能只是对整个数组的依次遍历。也就只能保证数组的第一个元素是最小的。实例1的结果也正好与此相符。

Java元组Tuple使用实例

atool 发表了文章 • 0 个评论 • 3341 次浏览 • 2016-04-22 11:07 • 来自相关话题

一、为什么使用元组tuple?
元组和列表list一样,都可能用于数据存储,包含多个数据;但是和列表不同的是:列表只能存储相同的数据类型,而元组不一样,它可以存储不同的数据类型,比如同时存储int、string、list等,并且可以根据需求无限扩展。
比如说在web应用中,经常会遇到一个问题就是数据分页问题,查询分页需要包含几点信息:当前页数、页大小;查询结果返回数据为:当前页的数据记录,但是如果需要在前台显示当前页、页大小、总页数等信息的时候,就必须有另外一个信息就是:数据记录总数,然后根据上面的信息进行计算得到总页数等信息。这个时候查询某一页信息的时候需要返回两个数据类型,一个是list(当前也的数据记录),一个是int(记录总数)。当然,完全可以在两个方法、两次数据库连接中得到这两个值。事实上在查询list的时候,已经通过sql查询得到总计录数,如果再开一个方法,再做一次数据库连接来查询总计录数,不免有点多此一举、浪费时间、浪费代码、浪费生命。言重了~在这种情况下,我们就可以利用二元组,在一次数据库连接中,得到总计录数、当前页记录,并存储到其中,简单明了!

二、源码实例
二元组:
/** <p>Title: TwoTuple</p>
* <p>Description: 两个元素的元组,用于在一个方法里返回两种类型的值</p>
* @author Xewee.Zhiwei.Wang@gmail.com
* @site http://wzwahl36.net
* @version 2012-3-21 上午11:15:03
* @param <A>
* @param <B>
*/
public class TwoTuple<A, B> {
public final A first;
public final B second;

public TwoTuple(A a, B b) {
this.first = a;
this.second = b;
}
}扩展为三元组(按此可以任意扩展)
/**
* <p>Title: ThreeTuple</p>
* <p>Description: 三个元素的元组,用于在一个方法里返回三种类型的值</p>
* @author Xewee.Zhiwei.Wang@gmail.com
* @site http://wzwahl36.net
* @version 2012-3-21 上午11:15:50
* @param <A>
* @param <B>
* @param <C>
*/
public class ThreeTuple<A, B, C> extends TwoTuple<A, B> {
public final C third;

public ThreeTuple(A a, B b, C c) {
super(a, b);
this.third = c;
}
}元组操作工具类、测试类(可按需自定义)

import java.util.ArrayList;
import java.util.List;
 
import com.bluesea.bean.GoodsBean;
 
/**
 * <p>Title: TupleUtil</p>
 * <p>Description:
 * 元组辅助类,用于多种类型值的返回,如在分页的时候,后台存储过程既返回了查询得到的
 * 当页的数据(List类型),又得到了数据表中总共的数据总数(Integer类型),然后将这
 * 两个参数封装到该类中返回到action中使用
 * 使用泛型方法实现,利用参数类型推断,编译器可以找出具体的类型
 * </p>
 * @author Xewee.Zhiwei.Wang@gmail.com
 * @site http://wzwahl36.net
 * @version 2012-3-21 上午09:59:39
 * @param <A>
 * @param <B>
 */
public class TupleUtil {
     
    public static <A, B> TwoTuple<A, B> tuple(A a, B b) {
        return new TwoTuple<A, B>(a, b);
    }
     
    public static <A, B, C> ThreeTuple<A, B, C> tuple(A a, B b, C c) {
        return new ThreeTuple<A, B, C>(a, b, c);
    }
 
    // 测试
    public static void main(String[] args) {
        List<GoodsBean> goodsBeans = new ArrayList<GoodsBean>();
        for(int i = 1; i < 26; i++) {
            GoodsBean goodsBean = new GoodsBean();
            goodsBean.setGoodsId(i);
            goodsBeans.add(goodsBean);
        }
        Integer totalProperty = 47;
//      TupleUtil<List<GoodsBean>, Integer> tupleUtil = new TupleUtil<List<GoodsBean>, Integer>(goodsBeans, totalProperty);
        TwoTuple<List<GoodsBean>, Integer> twoTuple = TupleUtil.tuple(goodsBeans, totalProperty);
        List<GoodsBean> list = twoTuple.first;
        System.out.println(list);
        System.out.println(twoTuple.second);
    }
}
三、Demo下载
搬到SAE上之后,链接丢失,Demo为以上几个文件打包,请自行处理。 查看全部
一、为什么使用元组tuple?
元组和列表list一样,都可能用于数据存储,包含多个数据;但是和列表不同的是:列表只能存储相同的数据类型,而元组不一样,它可以存储不同的数据类型,比如同时存储int、string、list等,并且可以根据需求无限扩展。
比如说在web应用中,经常会遇到一个问题就是数据分页问题,查询分页需要包含几点信息:当前页数、页大小;查询结果返回数据为:当前页的数据记录,但是如果需要在前台显示当前页、页大小、总页数等信息的时候,就必须有另外一个信息就是:数据记录总数,然后根据上面的信息进行计算得到总页数等信息。这个时候查询某一页信息的时候需要返回两个数据类型,一个是list(当前也的数据记录),一个是int(记录总数)。当然,完全可以在两个方法、两次数据库连接中得到这两个值。事实上在查询list的时候,已经通过sql查询得到总计录数,如果再开一个方法,再做一次数据库连接来查询总计录数,不免有点多此一举、浪费时间、浪费代码、浪费生命。言重了~在这种情况下,我们就可以利用二元组,在一次数据库连接中,得到总计录数、当前页记录,并存储到其中,简单明了!

二、源码实例
二元组:
/** <p>Title: TwoTuple</p>
* <p>Description: 两个元素的元组,用于在一个方法里返回两种类型的值</p>
* @author Xewee.Zhiwei.Wang@gmail.com
* @site http://wzwahl36.net
* @version 2012-3-21 上午11:15:03
* @param <A>
* @param <B>
*/
public class TwoTuple<A, B> {
public final A first;
public final B second;

public TwoTuple(A a, B b) {
this.first = a;
this.second = b;
}
}
扩展为三元组(按此可以任意扩展)
/**
* <p>Title: ThreeTuple</p>
* <p>Description: 三个元素的元组,用于在一个方法里返回三种类型的值</p>
* @author Xewee.Zhiwei.Wang@gmail.com
* @site http://wzwahl36.net
* @version 2012-3-21 上午11:15:50
* @param <A>
* @param <B>
* @param <C>
*/
public class ThreeTuple<A, B, C> extends TwoTuple<A, B> {
public final C third;

public ThreeTuple(A a, B b, C c) {
super(a, b);
this.third = c;
}
}
元组操作工具类、测试类(可按需自定义)

import java.util.ArrayList;
import java.util.List;
 
import com.bluesea.bean.GoodsBean;
 
/**
 * <p>Title: TupleUtil</p>
 * <p>Description:
 * 元组辅助类,用于多种类型值的返回,如在分页的时候,后台存储过程既返回了查询得到的
 * 当页的数据(List类型),又得到了数据表中总共的数据总数(Integer类型),然后将这
 * 两个参数封装到该类中返回到action中使用
 * 使用泛型方法实现,利用参数类型推断,编译器可以找出具体的类型
 * </p>
 * @author Xewee.Zhiwei.Wang@gmail.com
 * @site http://wzwahl36.net
 * @version 2012-3-21 上午09:59:39
 * @param <A>
 * @param <B>
 */
public class TupleUtil {
     
    public static <A, B> TwoTuple<A, B> tuple(A a, B b) {
        return new TwoTuple<A, B>(a, b);
    }
     
    public static <A, B, C> ThreeTuple<A, B, C> tuple(A a, B b, C c) {
        return new ThreeTuple<A, B, C>(a, b, c);
    }
 
    // 测试
    public static void main(String[] args) {
        List<GoodsBean> goodsBeans = new ArrayList<GoodsBean>();
        for(int i = 1; i < 26; i++) {
            GoodsBean goodsBean = new GoodsBean();
            goodsBean.setGoodsId(i);
            goodsBeans.add(goodsBean);
        }
        Integer totalProperty = 47;
//      TupleUtil<List<GoodsBean>, Integer> tupleUtil = new TupleUtil<List<GoodsBean>, Integer>(goodsBeans, totalProperty);
        TwoTuple<List<GoodsBean>, Integer> twoTuple = TupleUtil.tuple(goodsBeans, totalProperty);
        List<GoodsBean> list = twoTuple.first;
        System.out.println(list);
        System.out.println(twoTuple.second);
    }
}
三、Demo下载
搬到SAE上之后,链接丢失,Demo为以上几个文件打包,请自行处理。

微信公众号消息接口开发——Java Servlet开发

atool 发表了文章 • 0 个评论 • 1469 次浏览 • 2016-04-22 11:02 • 来自相关话题

张小龙搞出来的微信真的很牛逼啊,短短两三年就2亿用户量,这个数据似乎在警告QQ部门,IM软件上,你们QQ的老大位置并不稳固,仍需努力。当然QQ和微信不是一个平台上的产品,并不存在很大的竞争问题。

微信公众号消息接口API只有相当于只有一个简单调用链接,但是却完全改变了最初微信公众号的概念。不过说实话,这个接口真的设计的不怎么的...验证过程个字符处理啊,加密啊,拼接啊毫无用处,在申请页面填写的token也毫无用处,这些都是题外话了。不管这个接口腾讯设计得好不好,我们需要关心的是我们利用这个接口能够做什么,怎么做?

废话少说,言归正传,微信官网给我们提供了PHP的例子,所以做PHP的同学只需要画瓢即可,对于Java平台,网上还没有人说起,欲知Java平台怎么开发微信公众号接口,请看下文:

下面列出使用Java Servlet开发的一个简单的微信公众号接口调用的例子,如果有疑问,可以参见我本博的微信公众号:红色石头。

另外,我想大家一定很想知道《模拟登录微信公众平台,主动推送图文消息给用户》的有关信息。点开即可看到了~~~




 
package com.wzwcxl.pubwx.hustac.pubwx;

import java.io.IOException;
import java.io.InputStream;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;

import com.wzwcxl.pubwx.hustac.util.StringUtil;

/**
* 微信公众号接口调用Java版
* @author Xewee.Zhiwei.Wang
* @version 2012-12-17 上午9:16:02
* @Contract wzwahl36@QQ.com or http://wzwahl36.net/
*/
public class Test extends HttpServlet {
private static final long serialVersionUID = 1L;

@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------get---------------");
// response.getWriter().write(request.getParameter("echostr"));
doPost(request, response);
}

@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------post---------------");
try {
InputStream is = request.getInputStream();
// 取HTTP请求流长度
int size = request.getContentLength();
// 用于缓存每次读取的数据
byte[] buffer = new byte[size];
// 用于存放结果的数组
byte[] xmldataByte = new byte[size];
int count = 0;
int rbyte = 0;
// 循环读取
while (count < size) {
// 每次实际读取长度存于rbyte中
rbyte = is.read(buffer);
for(int i=0;i<rbyte;i++) {
xmldataByte[count + i] = buffer[i];
}
count += rbyte;
}
is.close();
String requestStr = new String(xmldataByte, "UTF-8");
Document doc = DocumentHelper.parseText(requestStr);
Element rootElt = doc.getRootElement();
String content = rootElt.elementText("Content");
String toUserName = rootElt.elementText("ToUserName");
String fromUserName = rootElt.elementText("FromUserName");
//得到所有的有用数据
System.out.println(content+ ":" + toUserName + ":" + fromUserName);
//文本消息
if (! StringUtil.isBlank(content) && "text".equals(content)) {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[text]]></MsgType>";
responseStr += "<Content>输入text或者news返回相应类型的消息,另外推荐你关注 '红色石头'(完全采用Java完成),反馈和建议请到http://wzwahl36.net</Content>";
responseStr += "<FuncFlag>0</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}
//图文消息
else if (! StringUtil.isBlank(content) && "news".equals(content)) {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[news]]></MsgType>";
responseStr += "<Content><![CDATA[]]></Content>";

responseStr += "<ArticleCount>2</ArticleCount>";

responseStr += "<Articles>";
responseStr += "<item>";
responseStr += "<Title><![CDATA[图文消息——红色石头]]></Title>";
responseStr += "<Discription><![CDATA[图文消息正文——红色石头]]></Discription>";
responseStr += "<PicUrl><![CDATA[http://redstones.sinaapp.com/res/images/redstones_wx_258.jpg]]></PicUrl>";
responseStr += "<Url><![CDATA[http://redstones.sinaapp.com/]]></Url>";
responseStr += "</item>";

responseStr += "<item>";
responseStr += "<Title><![CDATA[图文消息——红色石头]]></Title>";
responseStr += "<Discription><![CDATA[图文消息正文——红色石头]]></Discription>";
responseStr += "<PicUrl><![CDATA[http://redstones.sinaapp.com/res/images/redstones_wx_258.jpg]]></PicUrl>";
responseStr += "<Url><![CDATA[http://redstones.sinaapp.com/]]></Url>";
responseStr += "</item>";

responseStr += "</Articles>";
responseStr += "<FuncFlag>1</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}
//不能识别
else {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[text]]></MsgType>";
responseStr += "<Content>输入text或者news返回相应类型的消息,另外推荐你关注 '红色石头'(完全采用Java完成),反馈和建议请到http://wzwahl36.net</Content>";
responseStr += "<FuncFlag>0</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}

} catch (Exception e) {
e.printStackTrace();
}
}
}需要注意是一点是:在验证链接时,将get方法写为以下:
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------get---------------");
response.getWriter().write(request.getParameter("echostr"));
// doPost(request, response);
}
验证通过之后,将get方法和post方法写成一样的。
另外注意servlet的编码问题,我这个servlet里面没有写编码是因为我设置了编码过滤器:
package com.wzwcxl.pubwx.hustac.filter;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;

/**
* Servlet Filter implementation class EncodingFilter
*/
public class EncodingFilter implements Filter {

/**
* Default constructor.
*/
public EncodingFilter() {
// TODO Auto-generated constructor stub
}

/**
* @see Filter#destroy()
*/
public void destroy() {
// TODO Auto-generated method stub
}

/**
* @see Filter#doFilter(ServletRequest, ServletResponse, FilterChain)
*/
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
response.setContentType("text/html;charset=utf-8");
response.setCharacterEncoding("utf-8");
request.setCharacterEncoding("utf-8");
System.out.println("EncodingFilter: UTF-8");

chain.doFilter(request, response);
}

/**
* @see Filter#init(FilterConfig)
*/
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}附加:很多人问到代码中的StringUtil类中的isBlank方法是什么?然后要我发给他,其实StringUtil只有isBlank方法,这个方法就是用来判断字符串是不是空,所谓字符串为空,就是等于null或者trim之后为"",每次问到这个我都不知道怎么回答,所以在这里附加说明一下~

Enjoy~有空点点广告,感谢~~~ 查看全部
张小龙搞出来的微信真的很牛逼啊,短短两三年就2亿用户量,这个数据似乎在警告QQ部门,IM软件上,你们QQ的老大位置并不稳固,仍需努力。当然QQ和微信不是一个平台上的产品,并不存在很大的竞争问题。

微信公众号消息接口API只有相当于只有一个简单调用链接,但是却完全改变了最初微信公众号的概念。不过说实话,这个接口真的设计的不怎么的...验证过程个字符处理啊,加密啊,拼接啊毫无用处,在申请页面填写的token也毫无用处,这些都是题外话了。不管这个接口腾讯设计得好不好,我们需要关心的是我们利用这个接口能够做什么,怎么做?

废话少说,言归正传,微信官网给我们提供了PHP的例子,所以做PHP的同学只需要画瓢即可,对于Java平台,网上还没有人说起,欲知Java平台怎么开发微信公众号接口,请看下文:

下面列出使用Java Servlet开发的一个简单的微信公众号接口调用的例子,如果有疑问,可以参见我本博的微信公众号:红色石头。

另外,我想大家一定很想知道《模拟登录微信公众平台,主动推送图文消息给用户》的有关信息。点开即可看到了~~~
redstones_wx_430.jpg

 
package com.wzwcxl.pubwx.hustac.pubwx;

import java.io.IOException;
import java.io.InputStream;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;

import com.wzwcxl.pubwx.hustac.util.StringUtil;

/**
* 微信公众号接口调用Java版
* @author Xewee.Zhiwei.Wang
* @version 2012-12-17 上午9:16:02
* @Contract wzwahl36@QQ.com or http://wzwahl36.net/
*/
public class Test extends HttpServlet {
private static final long serialVersionUID = 1L;

@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------get---------------");
// response.getWriter().write(request.getParameter("echostr"));
doPost(request, response);
}

@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------post---------------");
try {
InputStream is = request.getInputStream();
// 取HTTP请求流长度
int size = request.getContentLength();
// 用于缓存每次读取的数据
byte[] buffer = new byte[size];
// 用于存放结果的数组
byte[] xmldataByte = new byte[size];
int count = 0;
int rbyte = 0;
// 循环读取
while (count < size) {
// 每次实际读取长度存于rbyte中
rbyte = is.read(buffer);
for(int i=0;i<rbyte;i++) {
xmldataByte[count + i] = buffer[i];
}
count += rbyte;
}
is.close();
String requestStr = new String(xmldataByte, "UTF-8");
Document doc = DocumentHelper.parseText(requestStr);
Element rootElt = doc.getRootElement();
String content = rootElt.elementText("Content");
String toUserName = rootElt.elementText("ToUserName");
String fromUserName = rootElt.elementText("FromUserName");
//得到所有的有用数据
System.out.println(content+ ":" + toUserName + ":" + fromUserName);
//文本消息
if (! StringUtil.isBlank(content) && "text".equals(content)) {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[text]]></MsgType>";
responseStr += "<Content>输入text或者news返回相应类型的消息,另外推荐你关注 '红色石头'(完全采用Java完成),反馈和建议请到http://wzwahl36.net</Content>";
responseStr += "<FuncFlag>0</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}
//图文消息
else if (! StringUtil.isBlank(content) && "news".equals(content)) {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[news]]></MsgType>";
responseStr += "<Content><![CDATA[]]></Content>";

responseStr += "<ArticleCount>2</ArticleCount>";

responseStr += "<Articles>";
responseStr += "<item>";
responseStr += "<Title><![CDATA[图文消息——红色石头]]></Title>";
responseStr += "<Discription><![CDATA[图文消息正文——红色石头]]></Discription>";
responseStr += "<PicUrl><![CDATA[http://redstones.sinaapp.com/res/images/redstones_wx_258.jpg]]></PicUrl>";
responseStr += "<Url><![CDATA[http://redstones.sinaapp.com/]]></Url>";
responseStr += "</item>";

responseStr += "<item>";
responseStr += "<Title><![CDATA[图文消息——红色石头]]></Title>";
responseStr += "<Discription><![CDATA[图文消息正文——红色石头]]></Discription>";
responseStr += "<PicUrl><![CDATA[http://redstones.sinaapp.com/res/images/redstones_wx_258.jpg]]></PicUrl>";
responseStr += "<Url><![CDATA[http://redstones.sinaapp.com/]]></Url>";
responseStr += "</item>";

responseStr += "</Articles>";
responseStr += "<FuncFlag>1</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}
//不能识别
else {
String responseStr = "<xml>";
responseStr += "<ToUserName><![CDATA[" + fromUserName
+ "]]></ToUserName>";
responseStr += "<FromUserName><![CDATA[" + toUserName
+ "]]></FromUserName>";
responseStr += "<CreateTime>" + System.currentTimeMillis()
+ "</CreateTime>";
responseStr += "<MsgType><![CDATA[text]]></MsgType>";
responseStr += "<Content>输入text或者news返回相应类型的消息,另外推荐你关注 '红色石头'(完全采用Java完成),反馈和建议请到http://wzwahl36.net</Content>";
responseStr += "<FuncFlag>0</FuncFlag>";
responseStr += "</xml>";
response.getWriter().write(responseStr);
}

} catch (Exception e) {
e.printStackTrace();
}
}
}
需要注意是一点是:在验证链接时,将get方法写为以下:
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("----------get---------------");
response.getWriter().write(request.getParameter("echostr"));
// doPost(request, response);
}
验证通过之后,将get方法和post方法写成一样的。
另外注意servlet的编码问题,我这个servlet里面没有写编码是因为我设置了编码过滤器:
package com.wzwcxl.pubwx.hustac.filter;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;

/**
* Servlet Filter implementation class EncodingFilter
*/
public class EncodingFilter implements Filter {

/**
* Default constructor.
*/
public EncodingFilter() {
// TODO Auto-generated constructor stub
}

/**
* @see Filter#destroy()
*/
public void destroy() {
// TODO Auto-generated method stub
}

/**
* @see Filter#doFilter(ServletRequest, ServletResponse, FilterChain)
*/
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
response.setContentType("text/html;charset=utf-8");
response.setCharacterEncoding("utf-8");
request.setCharacterEncoding("utf-8");
System.out.println("EncodingFilter: UTF-8");

chain.doFilter(request, response);
}

/**
* @see Filter#init(FilterConfig)
*/
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
附加:很多人问到代码中的StringUtil类中的isBlank方法是什么?然后要我发给他,其实StringUtil只有isBlank方法,这个方法就是用来判断字符串是不是空,所谓字符串为空,就是等于null或者trim之后为"",每次问到这个我都不知道怎么回答,所以在这里附加说明一下~

Enjoy~有空点点广告,感谢~~~

记住这53个要点提高PHP编程效率

atool 发表了文章 • 0 个评论 • 647 次浏览 • 2016-04-21 16:27 • 来自相关话题

用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这么做,它是一种可以把多个字符串当作参数的“函数”(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加上了双引号)。

1、如果能将类的方法定义成static,就尽量定义成static,它的速度会提升将近4倍。

2、$row[’id’] 的速度是$row[id]的7倍。

3、echo 比 print 快,并且使用echo的多重参数(译注:指用逗号而不是句点)代替字符串连接,比如echo $str1,$str2。

4、在执行for循环之前确定最大循环数,不要每循环一次都计算最大值,最好运用foreach代替。

5、注销那些不用的变量尤其是大数组,以便释放内存。

6、尽量避免使用__get,__set,__autoload。

7、require_once()代价昂贵。

8、include文件时尽量使用绝对路径,因为它避免了PHP去include_path里查找文件的速度,解析操作系统路径所需的时间会更少。

9、如果你想知道脚本开始执行(译注:即服务器端收到客户端请求)的时刻,使用$_SERVER[‘REQUEST_TIME’]要好于time()。

10、函数代替正则表达式完成相同功能。

11、str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。

12、如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。

13、使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。

14、用@屏蔽错误消息的做法非常低效,极其低效。

15、打开apache的mod_deflate模块,可以提高网页的浏览速度。

16、数据库连接当使用完毕时应关掉,不要用长连接。

17、错误消息代价昂贵。

18、在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。

19、递增一个全局变量要比递增一个局部变量慢2倍。

20、递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。

21、递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。

22、仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。PHP大概会检查看是否存在全局变量。

23、方法调用看来与类中定义的方法的数量无关,因为我(在测试方法之前和之后都)添加了10个方法,但性能上没有变化。

24、派生类中的方法运行起来要快于在基类中定义的同样的方法。

25、调用带有一个参数的空函数,其花费的时间相当于执行7至8次的局部变量递增操作。类似的方法调用所花费的时间接近于15次的局部变量递增操作。

26、Apache解析一个PHP脚本的时间要比解析一个静态HTML页面慢2至10倍。尽量多用静态HTML页面,少用脚本。

27、除非脚本可以缓存,否则每次调用时都会重新编译一次。引入一套PHP缓存机制通常可以提升25%至100%的性能,以免除编译开销。

28、尽量做缓存,可使用memcached。memcached是一款高性能的内存对象缓存系统,可用来加速动态Web应用程序,减轻数据库负载。对运算码 (OP code)的缓存很有用,使得脚本不必为每个请求做重新编译。

29、当操作字符串并需要检验其长度是否满足某种要求时,你想当然地会使用strlen()函数。此函数执行起来相当快,因为它不做任何计算,只返回在zval 结构(C的内置数据结构,用于存储PHP变量)中存储的已知字符串长度。但是,由于strlen()是函数,多多少少会有些慢,因为函数调用会经过诸多步骤,如字母小写化(译注:指函数名小写化,PHP不区分函数名大小写)、哈希查找,会跟随被调用的函数一起执行。在某些情况下,你可以使用isset() 技巧加速执行你的代码。

(举例如下)

if (strlen($foo) < 5) { echo “Foo is too short”$$ }

(与下面的技巧做比较)

if (!isset($foo{5})) { echo “Foo is too short”$$ }

调用isset()恰巧比strlen()快,因为与后者不同的是,isset()作为一种语言结构,意味着它的执行不需要函数查找和字母小写化。也就是说,实际上在检验字符串长度的顶层代码中你没有花太多开销。

34、当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言,所以请不要修改你的C或Java代码并指望它们能立即变快,没用的。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。牢记这个优化处理不失为一个好主意,因为并不是所有的指令优化器都会做同样的优化处理,并且存在大量没有装配指令优化器的互联网服务提供商(ISPs)和服务器。

35、并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。

36、并非要用类实现所有的数据结构,数组也很有用。

37、不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码?

38、当你需要时,你总能把代码分解成方法。

39、尽量采用大量的PHP内置函数。

40、如果在代码中存在大量耗时的函数,你可以考虑用C扩展的方式实现它们。

41、评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。

42、mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低80%。

43、在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题;

44、尽量的少进行文件操作,虽然PHP的文件操作效率也不低的;

45、优化Select SQL语句,在可能的情况下尽量少的进行Insert、Update操作(在update上,我被恶批过);

46、尽可能的使用PHP内部函数(但是我却为了找个PHP里面不存在的函数,浪费了本可以写出一个自定义函数的时间,经验问题啊!);

47、循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?);

48、多维数组尽量不要循环嵌套赋值;

49、在可以用PHP内部字符串操作函数的情况下,不要用正则表达式;

50、foreach效率更高,尽量用foreach代替while和for循环;

51、用单引号替代双引号引用字符串;

52、“用i+=1代替i=i+1。符合c/c++的习惯,效率还高”;

53、对global变量,应该用完就unset()掉; 查看全部
用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这么做,它是一种可以把多个字符串当作参数的“函数”(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加上了双引号)。

1、如果能将类的方法定义成static,就尽量定义成static,它的速度会提升将近4倍。

2、$row[’id’] 的速度是$row[id]的7倍。

3、echo 比 print 快,并且使用echo的多重参数(译注:指用逗号而不是句点)代替字符串连接,比如echo $str1,$str2。

4、在执行for循环之前确定最大循环数,不要每循环一次都计算最大值,最好运用foreach代替。

5、注销那些不用的变量尤其是大数组,以便释放内存。

6、尽量避免使用__get,__set,__autoload。

7、require_once()代价昂贵。

8、include文件时尽量使用绝对路径,因为它避免了PHP去include_path里查找文件的速度,解析操作系统路径所需的时间会更少。

9、如果你想知道脚本开始执行(译注:即服务器端收到客户端请求)的时刻,使用$_SERVER[‘REQUEST_TIME’]要好于time()。

10、函数代替正则表达式完成相同功能。

11、str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。

12、如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。

13、使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。

14、用@屏蔽错误消息的做法非常低效,极其低效。

15、打开apache的mod_deflate模块,可以提高网页的浏览速度。

16、数据库连接当使用完毕时应关掉,不要用长连接。

17、错误消息代价昂贵。

18、在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。

19、递增一个全局变量要比递增一个局部变量慢2倍。

20、递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。

21、递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。

22、仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。PHP大概会检查看是否存在全局变量。

23、方法调用看来与类中定义的方法的数量无关,因为我(在测试方法之前和之后都)添加了10个方法,但性能上没有变化。

24、派生类中的方法运行起来要快于在基类中定义的同样的方法。

25、调用带有一个参数的空函数,其花费的时间相当于执行7至8次的局部变量递增操作。类似的方法调用所花费的时间接近于15次的局部变量递增操作。

26、Apache解析一个PHP脚本的时间要比解析一个静态HTML页面慢2至10倍。尽量多用静态HTML页面,少用脚本。

27、除非脚本可以缓存,否则每次调用时都会重新编译一次。引入一套PHP缓存机制通常可以提升25%至100%的性能,以免除编译开销。

28、尽量做缓存,可使用memcached。memcached是一款高性能的内存对象缓存系统,可用来加速动态Web应用程序,减轻数据库负载。对运算码 (OP code)的缓存很有用,使得脚本不必为每个请求做重新编译。

29、当操作字符串并需要检验其长度是否满足某种要求时,你想当然地会使用strlen()函数。此函数执行起来相当快,因为它不做任何计算,只返回在zval 结构(C的内置数据结构,用于存储PHP变量)中存储的已知字符串长度。但是,由于strlen()是函数,多多少少会有些慢,因为函数调用会经过诸多步骤,如字母小写化(译注:指函数名小写化,PHP不区分函数名大小写)、哈希查找,会跟随被调用的函数一起执行。在某些情况下,你可以使用isset() 技巧加速执行你的代码。

(举例如下)

if (strlen($foo) < 5) { echo “Foo is too short”$$ }

(与下面的技巧做比较)

if (!isset($foo{5})) { echo “Foo is too short”$$ }

调用isset()恰巧比strlen()快,因为与后者不同的是,isset()作为一种语言结构,意味着它的执行不需要函数查找和字母小写化。也就是说,实际上在检验字符串长度的顶层代码中你没有花太多开销。

34、当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言,所以请不要修改你的C或Java代码并指望它们能立即变快,没用的。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。牢记这个优化处理不失为一个好主意,因为并不是所有的指令优化器都会做同样的优化处理,并且存在大量没有装配指令优化器的互联网服务提供商(ISPs)和服务器。

35、并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。

36、并非要用类实现所有的数据结构,数组也很有用。

37、不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码?

38、当你需要时,你总能把代码分解成方法。

39、尽量采用大量的PHP内置函数。

40、如果在代码中存在大量耗时的函数,你可以考虑用C扩展的方式实现它们。

41、评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。

42、mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低80%。

43、在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题;

44、尽量的少进行文件操作,虽然PHP的文件操作效率也不低的;

45、优化Select SQL语句,在可能的情况下尽量少的进行Insert、Update操作(在update上,我被恶批过);

46、尽可能的使用PHP内部函数(但是我却为了找个PHP里面不存在的函数,浪费了本可以写出一个自定义函数的时间,经验问题啊!);

47、循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?);

48、多维数组尽量不要循环嵌套赋值;

49、在可以用PHP内部字符串操作函数的情况下,不要用正则表达式;

50、foreach效率更高,尽量用foreach代替while和for循环;

51、用单引号替代双引号引用字符串;

52、“用i+=1代替i=i+1。符合c/c++的习惯,效率还高”;

53、对global变量,应该用完就unset()掉;