工作,学习,生活,这里将会有一些记录. 备用域名:http://meisw.wdlinux.cn 注册 | 登陆
浏览模式: 标准 | 列表全部文章

编译Android源代码过程中的一些问题解决方法

必要的工具

Git
git版本需在1.5.4之上

sudo pacman -S git             ;archlinux
sudo apt-get install git-core ;debian

Repo
Repo是git命令的Python封装

curl http://android.git.kernel.org/repo > ~/works/bin/repo
chmod a+x ~/works/bin/repo

Python
Python版本应在2.4之上

sudo pacman -S python
sudo apt-get install python

JDK 1.6

sudo pacman -S jdk
sudo apt-get install sun-java5-jdk

然后在~/.bashrc中加入

export ANDROID_JAVA_HOME=$JAVA_HOME

否则在编译过程中可能会出现com.sun.javadoc找不到等错误。

获取Android源代码

1. 初始化要下载的文件列表:

mkdir -p ~/works/android
cd ~/works/android
repo init -u git://android.git.kernel.org/platform/manifest.git

如果想检出除master外其他分支上的代码可以用-b选项:

mkdir -p ~/works/android
cd ~/works/android
repo init -u git://android.git.kernel.org/platform/manifest.git -b cupcake

2. 配置git帐户

git config --global user.email "xxxxx@xxxxxxx"
git config --global user.name "xxxxxx"

3. 同步文件列表:

repo sync

在第一次下载全部代码完成后,可以按模块更新子项目的代码:

repo sync project-path

其中的project-path可以在src/.repo/manifests/default.xml中找到:

<?xml version="1.0" encoding="UTF-8"?>
<manifest>
  <remote  name="korg"
           fetch="git://android.git.kernel.org/"
           review="review.source.android.com" />
  <default revision="master"
           remote="korg" />
 
  <project path="build" name="platform/build">
    <copyfile src="core/root.mk" dest="Makefile" />
  </project>
 
  <project path="kernel"
           name="kernel/common"
           revision="refs/heads/android-2.6.27" />
 
  <project path="bionic" name="platform/bionic" />
 
  <project path="bootable/bootloader/legacy" name="platform/bootable/bootloader/legacy" />
  <project path="bootable/diskinstaller" name="platform/bootable/diskinstaller" />
  <project path="bootable/recovery" name="platform/bootable/recovery" />
  ... ...oject path="build" name="platform/build">
    <copyfile src="core/root.mk" dest="Makefile" />
  </project>
 
  <project path="kernel"
           name="kernel/common"
           revision="refs/heads/android-2.6.27" />
 
  <project path="bionic" name="platform/bionic" />
 
  <project path="bootable/bootloader/legacy" name="platform/bootable/bootloader/legacy" />
  <project path="bootable/diskinstaller" name="platform/bootable/diskinstaller" />
  <project path="bootable/recovery" name="platform/bootable/recovery" />
  ... ...

编译Android源代码

编译映像

cd ~/works/android
make

映像编译成功后会在目录 ~/works/android/out/target/product/generic 下产生一些image文件

ramdisk.img system.img userdata.img android-info.txt

验证,运行这些模块:

export ANDROID_PRODUCT_OUT=/home/yunt/works/android/out/target/product/generic
cd out/host/linux-x86/bin
./emulator

错误处理

java.util.zip.ZipException: duplicate entry: hyts_Foo.c 错误

java.util.zip.ZipException: duplicate entry: hyts_Foo.c
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

解决办法:
这个问题是hyts_Foo.c文件重复,在根目录查找哪里重复,然后将其删除保留一个

$ find . -name hyts_Foo.c
./libcore/luni/src/test/resources/hyts_Foo.c
./libcore/support/src/test/resources/hyts_Foo.c
$ rm libcore/luni/src/test/resources/hyts_Foo.c

再次 make 即可

java.util.zip.ZipException: duplicate entry: staffNS.xml 错误

java.util.zip.ZipException: duplicate entry: staffNS.xml
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

解决办法同上

java.util.zip.ZipException: duplicate entry: staff2.dtd 错误

java.util.zip.ZipException: duplicate entry: staff2.dtd
at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
at sun.tools.jar.Main.addFile(Main.java:713)
at sun.tools.jar.Main.update(Main.java:567)
at sun.tools.jar.Main.run(Main.java:202)
at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

同上

java.util.zip.ZipException: duplicate entry: xhtml1-strict.dtd错误

java.util.zip.ZipException: duplicate entry: xhtml1-strict.dtd
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

java.util.zip.ZipException: duplicate entry: staff2.xml 错误

java.util.zip.ZipException: duplicate entry: staff2.xml
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

java.util.zip.ZipException: duplicate entry: hc_staff.xml 错误

java.util.zip.ZipException: duplicate entry: hc_staff.xml
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

java.util.zip.ZipException: duplicate entry: staff.xml 错误

java.util.zip.ZipException: duplicate entry: staff.xml
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

java.util.zip.ZipException: duplicate entry: staff.dtd 错误

java.util.zip.ZipException: duplicate entry: staff.dtd
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

java.util.zip.ZipException: duplicate entry: staffNS.dtd 错误

java.util.zip.ZipException: duplicate entry: staffNS.dtd
        at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
        at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
        at sun.tools.jar.Main.addFile(Main.java:713)
        at sun.tools.jar.Main.update(Main.java:567)
        at sun.tools.jar.Main.run(Main.java:202)
        at sun.tools.jar.Main.main(Main.java:1149)
make: *** [out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar] Error 1
make: *** Deleting file `out/target/common/obj/JAVA_LIBRARIES/core-tests_intermediates/javalib.jar'

分享在Linux下编译Android源代码并修改调试系统自带应用的方法

 

下面的内容是在看过了很多的文章还有经过了很多的尝试之后得出的总结, 也作为自己以后可能重复这个过程的笔记.
从哪里引用到的内容都加上链接了.欢迎大家讨论.

首先下载并编译Android源码是必须的, 这就要求你必须要安装Linux, 我自己用的是Ubuntu 11.04,
在虚拟机装Ubuntu的话, 我给1个G的内存而已, 感觉已经是足够了, 不像网上说的需要至少1.5个G,
因为我的内存只有2G,如果分给虚拟机1.5的话,反而会卡得完全动不了.

而且整个过程我都是在root下执行的, 如果不是root, 建议使用sudo -sH来进行以下所有操作.
虽然官方是要求64位的操作系统才能编译, 但是我用的是32位的, 如何修改我会在下文说明.
还有一个问题是只有用Ubuntu 11.04这个特定版本编译的时候才会出现的,不知道会不会在新版的源码已经被修复了,
如果编译的时候出现出现的"field 'arch' has incomplete type"错误可以看我之前的这篇文章:
( http://www.blogjava.net/jayslong/archive/2011/05/12/fix_the_problem_that_android_source_code_wont_compile_in_ubuntu_11_04.html )


具体如何repo获取源码网上到处都是教程,我就不说了.严格按官方的教程来是没问题的.
另外提醒一点,repo的时候用参数 "-j 数字" 可以使用多线程下载.

需要注意的一点的还有很多教程还在说必须用JDK1.5 编译源码, 1.6编译SDK之类的,
其实是已经不需要了, 最新版的Android 只要直接装sun-java6-jdk就好了. (apt-get找不到就更新你的源或者直接去官网下)

然后如果你用的是32位的Linux, 编译之前必须做如下修改: (方法来自: http://blog.csdn.net/yihui8/archive/2011/03/10/6237433.aspx )


.
/external/clearsilver/cgi/Android.mk 
.
/external/clearsilver/java-jni/Android.mk 
.
/external/clearsilver/util/Android.mk 
.
/external/clearsilver/cs/Android.mk

四个文件中的
LOCAL_CFLAGS 
+= -m64 
LOCAL_LDFLAGS 
+= -m64 
注释掉,或者将“
64”换成“32
LOCAL_CFLAGS 
+= -m32 
LOCAL_LDFLAGS 
+= -m32 

   然后,将
.
/build/core/main.mk 中的
ifneq (
64,$(findstring 64,$(build_arch))) 
改为:
ifneq (i686,$(findstring i686,$(build_arch))) 


做完以上修改之后, 就可以开始编译源码了, cd到源码根目录下, 打"make". 然后去看场电影吧. 回来就编好了..
这一步会在android/out/target/product/generic/下面生产3个后缀为img的文件,建议拷贝一份出来吧,
我就遇到过这几个文件莫名消失结果需要重新make的情况.

然后回来如果你愿意的话,再执行一下make sdk. 具体查看这里的第6和第7点吧.
( http://blog.csdn.net/jackyu613/archive/2010/07/31/5778646.aspx )
如果你跟我一样只需要自己修改并编译和测试系统app, 是不需要make sdk这一步的

然后工作之前需要配置一下环境变量:  

gedit ~/.bashrc 注意修改成你自己放源码的路径, 将以下2行添加到文件末尾:

export PATH
=$PATH:/home/xxx/android/out/host/linux-x86/bin
export ANDROID_PRODUCT_OUT
=/home/xxx/android/out/target/product/generic

最后记得执行一下
source ~
/.bashrc


并且在源码根目录下执行一下:

. build/envsetup.sh

会帮你启用几个新的命令.比如下面我们要用到的"mmm"

然后就可以去修改你想要修改的系统app了, 全部都在android/packages下面.
具体修改的例子我会另外写一个文章来说,这次先说整个流程吧.
参考自( http://www.devp.com/home.php?mod=space&uid=11248&do=blog&id=1040 )

比如我修改的是系统的英文输入法, 位于android/packages
修改完后直接在android目录下执行, 指定到包含有Android.mk文件的目录就对了 :

mmm packages/imputmethods/LatinIME/

很快就会编译好了
(千万不要在根目录不加任何参数直接执行"mmm",不然你会后悔的,这等同于之前花了好长时间的"make")

然后就要把编译好的文件安装到模拟器中, 你可以之前就启动了模拟器, 也可以现在才启动, 都可以的,
安装前后也不需要重启模拟器, 执行了adb install 直接可以看到效果.

启动模拟器可以在android/out/target/product/generic/ 下直接执行这个命令:

emulator -image system.img -data userdata.img -ramdisk ramdisk.img

然后用这个命令把自己编译好的apk文件传入模拟器:  

adb install -/home/xxx/android/out/target/product/generic/system/app/***.apk

到这里基本就可以实现整个修改和调试系统app的流程啦,
其实前面都是一劳永逸的环境配置工作,实际上一直要用到的就是mmm和adb install -r 这两个命令而已

因为有些系统app涉及到jni的部分, 在Eclipse下面查看项目是会出错的, 我也不知道怎么实现在Eclipse正常调试,
所以只能是在文本编辑器里直接修改再install到模拟器里查看效果.

auto-sign下载,以及android签名使用方法

使用方法!
安装并配置好JAVA(这个不说了,网上很多!)
将auto-sign解压到任意目录(自己能找到的!)
将改好的包改名为update.zip
并将update.zip 文件复制到签名工具的目录下!
运行Sign.bat
会需要一点时间,过一会,目录下会多出一个update_signed.zip文件!
这个文件就是已经经过签名的刷机包!
比特网论坛将文件复制出来,改成update.zip 就可以用这个包刷机了!
具体改包中的何种文件会达到何种效果以后慢慢说!(我也只知道几个,呵呵)

sign.bat文件内容如下!

  1. @ECHO OFF
  2. Echo Auto-sign Created By Dave Da illest 1
  3. Echo Update.zip is now being signed and will be renamed to update_signed.zip
  4.  
  5. java -jar signapk.jar testkey.x509.pem testkey.pk8 update.zip update_signed.zip
  6.  
  7. Echo Signing Complete
  8.  
  9. Pause
  10. EXIT
复制代码


看完这些,大家应该明白了!
其实,这个命令完全可以再CMD下自己手动打上去。
就是通过signapk.jar testkey.x509.pemtestkey.pk8
这三个文件,将update.zip包签名,并保存为update_signed.zip
两个包的名字其实自己可以随便弄,反正到最后还是要改成update.zip 才能刷的!

php绘图库jpgraph

 jpgraph这个强大的绘图组件能根据你的需要画出任意图形,甚至"aleax的网站访问量统计图"这种也相当简单好画.只要你能提供数据,它就能画图.  简单的说, 画图就像你刚学数据结构时候编写的一些处理数据算法, 而这里只是多了几个调用绘图函数的过程来把处理的数据填进去自动画图.

  除了简单易用, 它还有一个好处: 支持中文。

  先到官方网站下载相应的库,注意有支持php4.*和php5.*两种版本的, 选择合适你的下载,然后先了解一下文件的组织结构最上层目录有doc 和 src 两个文件夹, doc 里面是文档和类库说明, src 是一些配置文件和绘制某一 "图元" 的php文件,注意,我说的"图元"的意思就是它们是一张图的基本组成部分,你完全可以自由组合画出复杂的图形。 src里还有一个example文件夹, 里面有几百个例子,都是相当简单的,我们完全可以不必看文档只要模仿里面的例子就可以快速画出图来。 运行几个图形,挑几个能抓住你眼睛的图形,看一下代码,组合一下功能,就出来了,画图简单到只要五分钟,中文支持就更简单了,内置支持simsun字附,在要图上写中文时SetFont(FF_SIMSUN,FS_BOLD) 就可以了.

  下面是我组合的一个例子,代码很简单,因为数据我没有处理, 这里只演示画图的功能代码 , 当然,填充的数据是很灵活的,可以来自数据库,可以由你重新写算法处理,X轴显示的数据也一样,至于画曲线图,柱图,饼图,还是混合图,就随你选了

    include ("date/jpgraph.php");
    include ("date/jpgraph_line.php");
    include ("date/jpgraph_error.php");
    $datay = array(1,3,13,5,8,6,17,21,15,7,4,1); //填充的数据  
    $graph = new Graph(400,200,"auto");
    $graph->img->SetMargin(35,35,35,35);  
    $graph->img->SetAntiAliasing();
    $graph->SetScale("textlin");
    $graph->SetShadow();
    $graph->title->Set("**曲线图");
    $graph->xaxis->title->Set("$year");
    $graph->xaxis->title->SetFont(FF_SIMSUN,FS_BOLD);
    $graph->yaxis->title->Set("身高(cm)");
    $graph->SetMarginColor("lightblue");
    $graph->yaxis->title->SetFont(FF_SIMSUN,FS_BOLD);
    $graph->title->SetFont(FF_SIMSUN,FS_BOLD);
    $graph->xaxis->SetPos("min");
    $graph->yaxis->HideZeroLabel();
    $graph->ygrid->SetFill(true,'#EFEFEF@0.5','#BBCCFF@0.5');
    $a=array("1月","2月","3月","4月","5月","6月","7月","8月","9月","10月","11月","12月");//X轴
    $graph->xaxis->SetTickLabels($a);
    $graph->xaxis->SetFont(FF_SIMSUN);
    $graph->yscale->SetGrace(20);    
    $p1 = new LinePlot($datay);
    $p1->mark->SetType(MARK_FILLEDCIRCLE);
    $p1->mark->SetFillColor("red");
    $p1->mark->SetWidth(4);
    $p1->SetColor("blue");
    $p1->SetCenter();
    $graph->Add($p1);
    $graph->Stroke();

http://tech.ddvip.com/2008-10/122449617280389.html

 

-------------------

比如我想让x轴标上time,y轴标上flux

$g   =   new   Graph(500,400);
$g-> xaxis-> title-> Set( "time ");
$g-> yaxis-> title-> Set( "flux ");

 

--------------------

http://www.docin.com/p-113240144.html

http://www.leapsoul.cn/?p=1367

MYSQL子查询的五种形式

mysql从4.1版开始支持子查询功能,在此版本前,可以用join写连表查询来进行替代,但不推荐这么写,相当的麻烦。以下是mysql子查询的几种常见写法:

mysql从4.1版开始支持子查询功能,在此版本前,可以用join写连表查询来进行替代,但不推荐这么写,相当的麻烦。mysql子查询的几种常见写法:

1. select * from xxx where col = [any|all](select * from xxxx); 

该句法可分为加关键词和不加关键词的写法,当不加关键词的时候,子查询语句返回的是一个离散值(注意是一个),查询语句将以子查询语句的结果作为自 己 where子句的条件进行查询,该句法可以在子查询语句前加入any、all、some等关键字,此时子查询语句返回的是一组离散值。any则表示,查询 语句是以子查询返回的值作为一个范围,在此值范围内进行查询,其与in关键字相像;all关键不太好了解,表示起全部匹配啥的。

2. select * from xxx where col in (select * from xxxx); 

该 句法相当的明了,就是查询语句的where子句是以子查询语句的结果作为其范围的,与上一种语法的any相似。

3. select row(value1,value2.....) = [any](selectcol1,col2..); 

该语句的执行实质是:子查询语句的执行所得的结果与查询的结果集匹配,若能找到匹配的返回真,否则返回假,并且两边的结果集都是一组离散值。

4. select  .... where col = [not] exists (select......); 

该语句比较蹩脚,其执行是这样的:当子查询进行操作,有返回结果的时候,该语句才会执行,而且有多少个结果,语句就执行几次。

5. select .... from (select .....) as name where ...... 

该句法,在平时用的比较少,而且并不好理解。其实是这么一回事,通过子查询执行的结果来构造一张新的表(这个表是一张衍生数据表,是一张虚拟的表),其用来作为主句的查询的对象,该句法功能非常强大,在一些复杂的查询中会经常用到。

子查询虽然挺方便,但其有不少缺点,其不支持limit,而且经实验证明其执行效率相当不理想,在一般的情况下,还是不推荐是用子查询。

Cookie与Session结合的一个初始化简例

看到这几天问有关Session为何无法使用的帖子比较多,我把我一直用的一段Session初始化代码发出来给大家参考一下。再每个页面头部引入这段代码,就可以直接使用$_SESSION存储数据

$sessionId session_id();              //尝试获取由PHP自身获得的SessionID(来源途径有URL或Cookie)

if(empty($sessionId) || preg_match('/[^A-Za-z0-9]/i'$sessionId)) {    //检查此ID是否合法

          
$sessionId $_COOKIE['PHPSESSID'];          //如ID不合法尝试从Cookie中获取

}

if(empty(
$sessionId) || preg_match('/[^A-Za-z0-9]/i'$sessionId)) { //再次检查ID是否合法

          
$sessionId md5(microtime());         //如还未有合法ID则生成一个

}

session_id($sessionId);         //使用新生成的ID(或者原先取到的合法ID)

session_start();              //启动Session

setcookie('PHPSESSID'$sessionIdtime() + 180);    //设置Session有效期(180秒)

http://bbs.phpchina.com/viewthread.php?tid=78712

file_get_contents超时问题及解决方案

经常会使用php中的file_get_contents()函数去抓取远程页面, 对方页面文件过大、对方页面反应时间太久或网络传输慢等原因都可能会导致file_get_contents()执行时间超过php的最大执行时间。而一 旦file_get_contents出错即会终止整个程序。

测试用例

 

?
1
2
3
4
5
6
7
8
<?php
  
ini_set("max_execution_time", 2);
$url = "http://aymoo.cn/files/jQuery1.2API.chm";
$html = file_get_contents($url);
var_dump($html);
  
?>

 

上面代码首先将php的最大执行时间设置为2秒,然后去远程读取170KB的文件,按照本人的网络环境下载170KB文件需时超过1s.

 

现象

Fatal error: Maximum execution time of 2 second exceeded in C:\wamp\www\phptest\timeout.php on line 4

这是由网络造成的php执行超时,在file_get_contents超时时即报错并且程序停止执行。而基本上大多数需求是即使发生错误也要求程序继续向下执行。

解决办法

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
<?php
ini_set("max_execution_time", 2);
$url = "http://aymoo.cn/files/jQuery1.2API.chm";
//$html = file_get_contents($url);
   
$ch = curl_init($url);
curl_setopt($ch, CURLOPT_TIMEOUT, 1);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
$html = curl_exec($ch);
curl_close ($ch);
   
var_dump($html);
?>

现象

bool(false)

成功解决了超时导致程序不能继续执行的问题。

这里需要注意的是,设置curl的CURLOPT_TIMEOUT的值应该小于php最大执行时间。

为什么要说应该而不是必须?用两个例子对比来说明,假设抓取页面需4s。

example 1
max_execution_time 为 2
CURLOPT_TIMEOUT 为 3

example 2
max_execution_time 为 10
CURLOPT_TIMEOUT 为 20

我对这个过程的理解是,php监控整个脚本的时间,curl监控抓取页面的时间。对例1, 抓取远程页面的过程中已经达到php最大执行时间,得到的结果和file_get_contents一样。
例2中,整个脚本的执行时间不到5s, 所以不受设定时间的限制.

 

-------------------------------------------

 

一、增加超时的时间限制
这里需要注意:set_time_limit只是设置你的PHP程序的超时时间,而不是file_get_contents函数读取URL的超时时间。
我一开始以为set_time_limit也能影响到file_get_contents,后来经测试,是无效的。真正的修改file_get_contents延时可以用resource $context的timeout参数:

复制代码 代码如下:
$opts = array(
‘http'=>array(
‘method'=>”GET”,
‘timeout'=>60,
)
);
$context = stream_context_create($opts);
$html =file_get_contents('http://www.example.com', false, $context);
fpassthru($fp);

二、一次有延时的话那就多试几次
有时候失败是因为网络等因素造成,没有解决办法,但是可以修改程序,失败时重试几次,仍然失败就放弃,因为file_get_contents()如果失败将返回 FALSE,所以可以下面这样编写代码:

复制代码 代码如下:
$cnt=0;
while($cnt < 3 && ($str=@file_get_contents('http…'))===FALSE) $cnt++;

apache/nginx支持ssi

apache默认不支持ssi的,可以在apache下做如下设置:
修改Apache配置文件httpd.conf
1. 确认加载include.so模块,将注释去掉:
LoadModule include_module libexec/apache2/mod_include.so
2. AddType部分去掉这两段注释:
AddType text/html .shtml
AddOutputFilter INCLUDES .shtml
3. Directory目录权限里面找到
Options Indexes FollowSymLinks
增加Includes修改为:
Options Indexes FollowSymLinks Includes
4. 重新启动Apache

在nginx下做如下设置:
在 http 段添加:
ssi on;
ssi_silent_errors off;
ssi_types text/shtml;


location ~* \.shtml$ {
      ssi on;
      ssi_silent_errors off;
      ssi_value_length 1024;
      ssi_types text/shtml;
}


location / {
        ssi on;
        ssi_silent_errors on;
        ssi_value_length 1024;
        ssi_types text/shtml;
}