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为以上几个文件打包,请自行处理。

华中科技大学 - 关于财务处全面推行不等待报账的通知及报账单下载

其他类型atool 发表了文章 • 0 个评论 • 713 次浏览 • 2016-04-22 11:05 • 来自相关话题

报过账都知道报账多么蛋疼,多么浪费时间,下文是报账同学的福音啊,原文链接:http://fiscal.hust.edu.cn/showcwdt.asp?xh=M7Uo

为了缓解教职员工办理财务报账业务排长队的现象,经学校批准,决定从即日起在财务处全面推行不等待报账,现将相关事项通知如下:

1.不等待报账,是指报账经办人将原始单据交至财务处专柜后,由财务处工作人员在规定时间办理报账,不需要在财务处现场排队等待的方式。

其流程:首先由报账经办人将原始单据按类别整理、归类、粘贴、汇总,按要求填写报账信息单,将原始单据及报账信息表交财务专柜收取,装入财务处预置的专用袋,并领取回执单。

2.票据整理、归类、粘贴是否规范,报销信息单填写是否准确,手续是否齐全是关系到这项工作顺利进行的前提条件,为了提高工作效率,减少多次往返补办手续的现象,最好请经费项目负责人指定专人负责。

3.原始单据及报销信息单交财务专柜后,正常情况下(以单据符合报账要求为基础)五个工作日内完成报销。

4.不等待报账方式区别于传统报销形式,需要不断摸索、完善,在刚开始推行阶段,如有不足,敬请谅解。同时,请各位师生提出宝贵意见与建议。

财务处

 2013年1月18日
 
表格下载:财务处不等待报账信息单.xls(http://fiscal.hust.edu.cn/uplo ... 0.xls)

不等待方式报账须知(试行)

1.对票据的要求

票据内容齐全,大小写金额一致(金额无涂改),票据上盖有开票单位财务(发票)专用章。购货品种较多,可另附详细清单,所附清单上的总金额、公章必须与发票一致。大型商场及连锁超市需提供机打发票

2.票据粘贴要求

(1)票据按从左至右呈“鱼鳞”状排列粘贴,粘贴单大小以不超过差旅费报销单为范围,一张不够,多张粘贴。

(2)票据粘贴请用浆糊或胶水,切勿使用透明胶、双面胶、订书钉等。

(3)市内交通费报销单,所附出租车票据一律在金额处裁剪整齐,然后呈鱼鳞状粘贴。

(4)招待餐费,定额票据按金额归并粘贴,非定额票据较多需粘贴。

(5)电话费,定额票据按金额归并粘贴,非定额票据按时间顺序粘贴。

(6)其他票据整理,相同业务、相同单位放在一起,整齐排列。

(7)每张粘贴单最多粘贴单据20张,每张粘贴单上请注明票据张数和票据汇总金额。

3.票据归类的要求

(1)差旅费发生的票据单独归类粘贴,按要求填写差旅费报销单。

(2)票据按现金业务、同城转账业务、外地汇款业务分别归类,并分开填写报账单,从经费本支付的应分别填写经费单。

(3)国库业务与非国库业务的票据分开粘贴,并分别填写报账单。

4.票据审批签字要求

(1)项目负责人对票据的真实性负责,并在报账单及经费单上签字。如票据是项目负责人本人提供的,如差旅费、电话费,报账单与经费单需相关领导审批签字。

(2)如发生特殊情况,需报单位财务一支笔审批签字。

(3)采购物品由项目负责人、经办人,验收人(保管人)三人签字。

(4)按学校经费审批权限的暂行规定(校财【2002】5号文)办理。

5.注意事项

(1)购买设备达到固定资产标准的,需附固定资产验收凭证。达到签订合同金额的,须附有双方负责人签字、盖有单位公章的有效合同(含科研协作)。

(2)请注意查询是否有预算,预算是否有浮动比例。

(3)当年取得的报销票据(以报销票据上的日期为准),原则上在当年报销,当年第四季度取得的报销票据,可延长至第二年3月底。

(4)其他注意事项请登陆财务处主页“报账指南”中“报账相关流程”。

(5)如办理现金业务时,需填写委托转帐申请书提供教职工姓名及人事编号。

(6)报账单附件张数需填写每张粘贴单上的票据汇总数。

(7)报账信息单填写要准确,交报销单据领回执单,报销结束前注意保存查询。 查看全部
报过账都知道报账多么蛋疼,多么浪费时间,下文是报账同学的福音啊,原文链接:http://fiscal.hust.edu.cn/showcwdt.asp?xh=M7Uo

为了缓解教职员工办理财务报账业务排长队的现象,经学校批准,决定从即日起在财务处全面推行不等待报账,现将相关事项通知如下:

1.不等待报账,是指报账经办人将原始单据交至财务处专柜后,由财务处工作人员在规定时间办理报账,不需要在财务处现场排队等待的方式。

其流程:首先由报账经办人将原始单据按类别整理、归类、粘贴、汇总,按要求填写报账信息单,将原始单据及报账信息表交财务专柜收取,装入财务处预置的专用袋,并领取回执单。

2.票据整理、归类、粘贴是否规范,报销信息单填写是否准确,手续是否齐全是关系到这项工作顺利进行的前提条件,为了提高工作效率,减少多次往返补办手续的现象,最好请经费项目负责人指定专人负责。

3.原始单据及报销信息单交财务专柜后,正常情况下(以单据符合报账要求为基础)五个工作日内完成报销。

4.不等待报账方式区别于传统报销形式,需要不断摸索、完善,在刚开始推行阶段,如有不足,敬请谅解。同时,请各位师生提出宝贵意见与建议。

财务处

 2013年1月18日
 
表格下载:财务处不等待报账信息单.xls(http://fiscal.hust.edu.cn/uplo ... 0.xls

不等待方式报账须知(试行)

1.对票据的要求

票据内容齐全,大小写金额一致(金额无涂改),票据上盖有开票单位财务(发票)专用章。购货品种较多,可另附详细清单,所附清单上的总金额、公章必须与发票一致。大型商场及连锁超市需提供机打发票

2.票据粘贴要求

(1)票据按从左至右呈“鱼鳞”状排列粘贴,粘贴单大小以不超过差旅费报销单为范围,一张不够,多张粘贴。

(2)票据粘贴请用浆糊或胶水,切勿使用透明胶、双面胶、订书钉等。

(3)市内交通费报销单,所附出租车票据一律在金额处裁剪整齐,然后呈鱼鳞状粘贴。

(4)招待餐费,定额票据按金额归并粘贴,非定额票据较多需粘贴。

(5)电话费,定额票据按金额归并粘贴,非定额票据按时间顺序粘贴。

(6)其他票据整理,相同业务、相同单位放在一起,整齐排列。

(7)每张粘贴单最多粘贴单据20张,每张粘贴单上请注明票据张数和票据汇总金额。

3.票据归类的要求

(1)差旅费发生的票据单独归类粘贴,按要求填写差旅费报销单。

(2)票据按现金业务、同城转账业务、外地汇款业务分别归类,并分开填写报账单,从经费本支付的应分别填写经费单。

(3)国库业务与非国库业务的票据分开粘贴,并分别填写报账单。

4.票据审批签字要求

(1)项目负责人对票据的真实性负责,并在报账单及经费单上签字。如票据是项目负责人本人提供的,如差旅费、电话费,报账单与经费单需相关领导审批签字。

(2)如发生特殊情况,需报单位财务一支笔审批签字。

(3)采购物品由项目负责人、经办人,验收人(保管人)三人签字。

(4)按学校经费审批权限的暂行规定(校财【2002】5号文)办理。

5.注意事项

(1)购买设备达到固定资产标准的,需附固定资产验收凭证。达到签订合同金额的,须附有双方负责人签字、盖有单位公章的有效合同(含科研协作)。

(2)请注意查询是否有预算,预算是否有浮动比例。

(3)当年取得的报销票据(以报销票据上的日期为准),原则上在当年报销,当年第四季度取得的报销票据,可延长至第二年3月底。

(4)其他注意事项请登陆财务处主页“报账指南”中“报账相关流程”。

(5)如办理现金业务时,需填写委托转帐申请书提供教职工姓名及人事编号。

(6)报账单附件张数需填写每张粘贴单上的票据汇总数。

(7)报账信息单填写要准确,交报销单据领回执单,报销结束前注意保存查询。

收到新浪SinaSAE高级开发者认证的证书了~

云计算atool 发表了文章 • 0 个评论 • 694 次浏览 • 2016-04-22 11:03 • 来自相关话题

刚刚四点钟的时候还在sae群里面问在武汉为啥3天了,还没有收到证书?群里的朋友都说天气不好,交通不好,会收到的~~~我也还在担心1号就回家,如果发到学校我收不到了,过了20天的认证时间,不是要重新申请了...刚想到这里,电话就响了,说是ems,我想大致就是证书了!果然就是sina sae高级开发者认证的证书了,哈皮...

在本博客中,我专门新建了一个文章类型叫SinaSAE,里面记录了申请中级,拿到中级证书,申请高级,拿到高级证书的过程,想想其实也很简单的啊;另外的几篇博客是写的一些sina sae服务的应用技术博文。大家有兴趣申请sina证书的可以好好看看~

和拿到中级证书一样,我也上传一张证书图片得瑟一下,以勉励更多的人加入sina sae。

说实话,sina sae没有以前稳定了,服务质量也没有以前好了,一直在闹绑定域名的解析问题,我都想换其他虚拟空间了,正在准备中...暂时看sina的表现了。
SAE现在是深坑。。。个人不用使用。 查看全部
刚刚四点钟的时候还在sae群里面问在武汉为啥3天了,还没有收到证书?群里的朋友都说天气不好,交通不好,会收到的~~~我也还在担心1号就回家,如果发到学校我收不到了,过了20天的认证时间,不是要重新申请了...刚想到这里,电话就响了,说是ems,我想大致就是证书了!果然就是sina sae高级开发者认证的证书了,哈皮...

在本博客中,我专门新建了一个文章类型叫SinaSAE,里面记录了申请中级,拿到中级证书,申请高级,拿到高级证书的过程,想想其实也很简单的啊;另外的几篇博客是写的一些sina sae服务的应用技术博文。大家有兴趣申请sina证书的可以好好看看~

和拿到中级证书一样,我也上传一张证书图片得瑟一下,以勉励更多的人加入sina sae。

说实话,sina sae没有以前稳定了,服务质量也没有以前好了,一直在闹绑定域名的解析问题,我都想换其他虚拟空间了,正在准备中...暂时看sina的表现了。
SAE现在是深坑。。。个人不用使用。

微信公众号消息接口开发——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()掉;