Android发展的一个重要方面Makefile分析

时间:2024-01-03 08:25:56

Android发展的一个重要方面Makefile分析

随着移动互联网的发展,移动开发也越来越吃香了。眼下最火的莫过于android。android是什么就不用说了,android自从开源以来,就受到非常多人的追捧。当然。一部人追捧它是由于它是Google开发的。对一个程序

员来说,一个系统值不值得追捧得要拿代码来说话。我这里并不打算分析android的代码。而是android的makefile。或许大家已经知道了在android源代码里,我们能够看见非常多makefile文件,起初我也不明确。经过一段时间的研究,后来慢慢明确了。我想通过分析

andorid的makefile来告诉大家怎样写makefile。

对于一个程序新手而言,好的IDE是他们追捧的对象。但当他接触的代码多了之后,就会逐渐发现IDE不够用了,由于有好多东西用IDE是不好做的,

比如自己主动编译。測试,版本号控制,编译定制等。这跟政治课上的一句话有点像:资本主义開始的时候是促进生产力发展的,但到了后来又成了阻碍生产力发展的因素

了。假设一个程序不能摆脱IDE的限制(不是不用。而是要有选择的用),那么他就非常难提高。

要知道。IDE和makefile代表了两种不同的思

想:IDE依据强调的是简化计算机与用户的交互。而makefile体现的是自己主动化。

对于一个一開始就接触linux的人来说,makefile可能是比較easy学的(熟能生巧),对于一个一開始就接触Windows的人来

说,makefile就不太好学,这主要是应该非常多时候会不自觉地去用Visual Studio(Visual

Studio是个好东西。特别是它的调试)。不知道大叫有没有这个的感觉:一个人假设先接触c,再接触java会比較easy点;假设一个人先接触java,

再接触c,就会比較反感c。

这个先引用一下百度百科对makefile的一些描写叙述:

一个project中的源文件不计数。其按类型、功能、模块分别放在若干个文件夹中。makefile定义了一系列的规则来指定,哪些文件须要先编译,哪些文件

须要后编译,哪些文件须要又一次编译,甚至于进行更复杂的功能操作,由于 makefile就像一个Shell脚本一样。当中也能够运行操作系统的命令。

makefile带来的优点就是——“自己主动化编译”,一旦写好。仅仅须要一个make命令,整个project全然自己主动编译,极大的提高了软件开发的效率。make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说。大多数的IDE都有这个命令,比方:Delphi的make,Visual
C++的nmake,Linux下GNU的make。可见。makefile都成为了一种在project方面的编译方法。 

Make工具最主要也是最主要的功能就是通过makefile文件来描写叙述源程序之间的相互关系并自己主动维护编译工作。

而makefile

文件须要依照某种语法进行编写。文件里须要说明怎样编译各个源文件并连接生成可运行文件,并要求定义源文件之间的依赖关系。makefile

文件是很多编译器--包含 Windows
NT 下的编译器--维护编译信息的经常用法,仅仅是在集成开发环境中,用户通过友好的界面改动

makefile 文件而已。

对于android而言。android使用的是GNU的make,因此它的makefile格式也是GNU的makefile格式。

如今网络上关

于makefile最好的文档就是陈皓的《跟我一起写makefile》。这份文档对makefile进行了具体的介绍,因此推荐大家先看这份文档(电子

版能够

首先我们来看看android里makefile的写法

(1)Android.mk文件首先须要指定LOCAL_PATH变量,用于查找源文件。因为普通情况下

Android.mk和须要编译的源文件在同一文件夹下。所以定义成例如以下形式:

LOCAL_PATH:=$(call my-dir)

上面的语句的意思是将LOCAL_PATH变量定义成本文件所在文件夹路径。

(2)Android.mk中能够定义多个编译模块,每一个编译模块都是以include $(CLEAR_VARS)開始

以include $(BUILD_XXX)结束。

include $(CLEAR_VARS)

CLEAR_VARS由编译系统提供,指定让GNU MAKEFILE为你清除除LOCAL_PATH以外的全部LOCAL_XXX变量,

如LOCAL_MODULE。LOCAL_SRC_FILES,LOCAL_SHARED_LIBRARIES,LOCAL_STATIC_LIBRARIES等。

include $(BUILD_STATIC_LIBRARY)表示编译成静态库

include $(BUILD_SHARED_LIBRARY)表示编译成动态库。

include $(BUILD_EXECUTABLE)表示编译成可运行程序

(3)举比例如以下(frameworks/base/libs/audioflinger/Android.mk):

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS) 模块一

ifeq ($(AUDIO_POLICY_TEST),true)

ENABLE_AUDIO_DUMP := true

endif

LOCAL_SRC_FILES:= \

AudioHardwareGeneric.cpp \

AudioHardwareStub.cpp \

AudioHardwareInterface.cpp

ifeq ($(ENABLE_AUDIO_DUMP),true)

LOCAL_SRC_FILES += AudioDumpInterface.cpp

LOCAL_CFLAGS += -DENABLE_AUDIO_DUMP

endif

LOCAL_SHARED_LIBRARIES := \

libcutils \

libutils \

libbinder \

libmedia \

libhardware_legacy

ifeq ($(strip $(BOARD_USES_GENERIC_AUDIO)),true)

LOCAL_CFLAGS += -DGENERIC_AUDIO

endif

LOCAL_MODULE:= libaudiointerface

ifeq ($(BOARD_HAVE_BLUETOOTH),true)

LOCAL_SRC_FILES += A2dpAudioInterface.cpp

LOCAL_SHARED_LIBRARIES += liba2dp

LOCAL_CFLAGS += -DWITH_BLUETOOTH -DWITH_A2DP

LOCAL_C_INCLUDES += $(call include-path-for, bluez)

endif

include $(BUILD_STATIC_LIBRARY) 模块一编译成静态库

include $(CLEAR_VARS) 模块二

LOCAL_SRC_FILES:= \

AudioPolicyManagerBase.cpp

LOCAL_SHARED_LIBRARIES := \

libcutils \

libutils \

libmedia

ifeq ($(TARGET_SIMULATOR),true)

LOCAL_LDLIBS += -ldl

else

LOCAL_SHARED_LIBRARIES += libdl

endif

LOCAL_MODULE:= libaudiopolicybase

ifeq ($(BOARD_HAVE_BLUETOOTH),true)

LOCAL_CFLAGS += -DWITH_A2DP

endif

ifeq ($(AUDIO_POLICY_TEST),true)

LOCAL_CFLAGS += -DAUDIO_POLICY_TEST

endif

include $(BUILD_STATIC_LIBRARY) 模块二编译成静态库

include $(CLEAR_VARS) 模块三

LOCAL_SRC_FILES:= \

AudioFlinger.cpp \

AudioMixer.cpp.arm \

AudioResampler.cpp.arm \

AudioResamplerSinc.cpp.arm \

AudioResamplerCubic.cpp.arm \

AudioPolicyService.cpp

LOCAL_SHARED_LIBRARIES := \

libcutils \

libutils \

libbinder \

libmedia \

libhardware_legacy

ifeq ($(strip $(BOARD_USES_GENERIC_AUDIO)),true)

LOCAL_STATIC_LIBRARIES += libaudiointerface libaudiopolicybase

LOCAL_CFLAGS += -DGENERIC_AUDIO

else

LOCAL_SHARED_LIBRARIES += libaudio libaudiopolicy

endif

ifeq ($(TARGET_SIMULATOR),true)

LOCAL_LDLIBS += -ldl

else

LOCAL_SHARED_LIBRARIES += libdl

endif

LOCAL_MODULE:= libaudioflinger

ifeq ($(BOARD_HAVE_BLUETOOTH),true)

LOCAL_CFLAGS += -DWITH_BLUETOOTH -DWITH_A2DP

LOCAL_SHARED_LIBRARIES += liba2dp

endif

ifeq ($(AUDIO_POLICY_TEST),true)

LOCAL_CFLAGS += -DAUDIO_POLICY_TEST

endif

ifeq ($(TARGET_SIMULATOR),true)

ifeq ($(HOST_OS),linux)

LOCAL_LDLIBS += -lrt -lpthread

endif

endif

ifeq ($(BOARD_USE_LVMX),true)

LOCAL_CFLAGS += -DLVMX

LOCAL_C_INCLUDES += vendor/nxp

LOCAL_STATIC_LIBRARIES += liblifevibes

LOCAL_SHARED_LIBRARIES += liblvmxservice

# LOCAL_SHARED_LIBRARIES += liblvmxipc

endif

include $(BUILD_SHARED_LIBRARY) 模块三编译成动态库

(4)编译一个应用程序(APK)

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# Build all java files in the java subdirectory-->直译(建立在java子文件夹中的全部Java文件)

LOCAL_SRC_FILES := $(call all-subdir-java-files)



# Name of the APK to build-->直译(创建APK的名称)

LOCAL_PACKAGE_NAME := LocalPackage



# Tell it to build an APK-->直译(告诉它来建立一个APK)

include $(BUILD_PACKAGE)

(5)编译一个依赖于静态Java库(static.jar)的应用程序

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# List of static libraries to include in the package

LOCAL_STATIC_JAVA_LIBRARIES := static-library



# Build all java files in the java subdirectory

LOCAL_SRC_FILES := $(call all-subdir-java-files)



# Name of the APK to build

LOCAL_PACKAGE_NAME := LocalPackage



# Tell it to build an APK

include $(BUILD_PACKAGE)

(6)编译一个须要用平台的key签名的应用程序

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# Build all java files in the java subdirectory

LOCAL_SRC_FILES := $(call all-subdir-java-files)



# Name of the APK to build

LOCAL_PACKAGE_NAME := LocalPackage



LOCAL_CERTIFICATE := platform



# Tell it to build an APK

include $(BUILD_PACKAGE)

(7)编译一个须要用特定key前面的应用程序

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# Build all java files in the java subdirectory

LOCAL_SRC_FILES := $(call all-subdir-java-files)



# Name of the APK to build

LOCAL_PACKAGE_NAME := LocalPackage



LOCAL_CERTIFICATE := vendor/example/certs/app



# Tell it to build an APK

include $(BUILD_PACKAGE)

(8)加入一个预编译应用程序

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# Module name should match apk name to be installed.

LOCAL_MODULE := LocalModuleName

LOCAL_SRC_FILES := $(LOCAL_MODULE).apk

LOCAL_MODULE_CLASS := APPS

LOCAL_MODULE_SUFFIX := $(COMMON_ANDROID_PACKAGE_SUFFIX)



include $(BUILD_PREBUILT)

(9)加入一个静态JAVA库

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)



# Build all java files in the java subdirectory

LOCAL_SRC_FILES := $(call all-subdir-java-files)



# Any libraries that this library depends on

LOCAL_JAVA_LIBRARIES := android.test.runner



# The name of the jar file to create

LOCAL_MODULE := sample



# Build a static jar file.

include $(BUILD_STATIC_JAVA_LIBRARY)

(10)Android.mk的编译模块中间能够定义相关的编译内容。也就是指定相关的变量例如以下:

LOCAL_AAPT_FLAGS

LOCAL_ACP_UNAVAILABLE

LOCAL_ADDITIONAL_JAVA_DIR 



LOCAL_AIDL_INCLUDES

LOCAL_ALLOW_UNDEFINED_SYMBOLS

LOCAL_ARM_MODE

LOCAL_ASFLAGS

LOCAL_ASSET_DIR

LOCAL_ASSET_FILES 在Android.mk文件里编译应用程序(BUILD_PACKAGE)时设置此变量,表示资源文件,

一般会定义成LOCAL_ASSET_FILES += $(call find-subdir-assets)



LOCAL_BUILT_MODULE_STEM 

LOCAL_C_INCLUDES 额外的C/C++编译头文件路径,用LOCAL_PATH表示本文件所在文件夹

举比例如以下:

LOCAL_C_INCLUDES += extlibs/zlib-1.2.3

LOCAL_C_INCLUDES += $(LOCAL_PATH)/src 



LOCAL_CC 指定C编译器

LOCAL_CERTIFICATE 签名认证

LOCAL_CFLAGS 为C/C++编译器定义额外的标志(如宏定义),举例:LOCAL_CFLAGS += -DLIBUTILS_NATIVE=1



LOCAL_CLASSPATH

LOCAL_COMPRESS_MODULE_SYMBOLS

LOCAL_COPY_HEADERS install应用程序时须要复制的头文件,必须同一时候定义LOCAL_COPY_HEADERS_TO



LOCAL_COPY_HEADERS_TO install应用程序时复制头文件的目的路径

LOCAL_CPP_EXTENSION 假设你的C++文件不是以cpp为文件后缀,你能够通过LOCAL_CPP_EXTENSION指定C++文件后缀名 

如:LOCAL_CPP_EXTENSION := .cc

注意统一模块中C++文件后缀必须保持一致。

LOCAL_CPPFLAGS 传递额外的标志给C++编译器,如:LOCAL_CPPFLAGS += -ffriend-injection

LOCAL_CXX 指定C++编译器



LOCAL_DX_FLAGS

LOCAL_EXPORT_PACKAGE_RESOURCES

LOCAL_FORCE_STATIC_EXECUTABLE 假设编译的可运行程序要进行静态链接(运行时不依赖于不论什么动态库)。则设置LOCAL_FORCE_STATIC_EXECUTABLE:=true

眼下仅仅有libc有静态库形式。这个仅仅有文件系统中/sbin文件夹下的应用程序会用到。这个文件夹下的应用程序在执行时通常

文件系统的其他部分还没有载入,所以必须进行静态链接。

LOCAL_GENERATED_SOURCES



LOCAL_INSTRUMENTATION_FOR

LOCAL_INSTRUMENTATION_FOR_PACKAGE_NAME

LOCAL_INTERMEDIATE_SOURCES

LOCAL_INTERMEDIATE_TARGETS

LOCAL_IS_HOST_MODULE

LOCAL_JAR_MANIFEST

LOCAL_JARJAR_RULES

LOCAL_JAVA_LIBRARIES 编译java应用程序和库的时候指定包括的java类库,眼下有core和framework两种

多数情况下定义成:LOCAL_JAVA_LIBRARIES := core framework

注意LOCAL_JAVA_LIBRARIES不是必须的。并且编译APK时不同意定义(系统会自己主动加入)



LOCAL_JAVA_RESOURCE_DIRS

LOCAL_JAVA_RESOURCE_FILES

LOCAL_JNI_SHARED_LIBRARIES

LOCAL_LDFLAGS 传递额外的參数给连接器(务必注意參数的顺序)



LOCAL_LDLIBS 为可运行程序或者库的编译指定额外的库,指定库以"-lxxx"格式,举例:

LOCAL_LDLIBS += -lcurses -lpthread

LOCAL_LDLIBS += -Wl,-z,origin 



LOCAL_MODULE 生成的模块的名称(注意应用程序名称用LOCAL_PACKAGE_NAME而不是LOCAL_MODULE)

LOCAL_MODULE_PATH 生成模块的路径



LOCAL_MODULE_STEM 



LOCAL_MODULE_TAGS 生成模块的标记 



LOCAL_NO_DEFAULT_COMPILER_FLAGS

LOCAL_NO_EMMA_COMPILE

LOCAL_NO_EMMA_INSTRUMENT

LOCAL_NO_STANDARD_LIBRARIES

LOCAL_OVERRIDES_PACKAGES

LOCAL_PACKAGE_NAME APK应用程序的名称

LOCAL_POST_PROCESS_COMMAND



LOCAL_PREBUILT_EXECUTABLES 预编译including $(BUILD_PREBUILT)或者$(BUILD_HOST_PREBUILT)时所用,指定须要复制的可运行文件

LOCAL_PREBUILT_JAVA_LIBRARIES

LOCAL_PREBUILT_LIBS 预编译including $(BUILD_PREBUILT)或者$(BUILD_HOST_PREBUILT)时所用, 指定须要复制的库.

LOCAL_PREBUILT_OBJ_FILES

LOCAL_PREBUILT_STATIC_JAVA_LIBRARIES 



LOCAL_PRELINK_MODULE 是否须要预连接处理(默认须要,用来做动态库优化)

LOCAL_REQUIRED_MODULES 指定模块执行所依赖的模块(模块安装时将会同步安装它所依赖的模块)



LOCAL_RESOURCE_DIR

LOCAL_SDK_VERSION

LOCAL_SHARED_LIBRARIES 可链接动态库



LOCAL_SRC_FILES 编译源文件

LOCAL_STATIC_JAVA_LIBRARIES

LOCAL_STATIC_LIBRARIES 可链接静态库 



LOCAL_UNINSTALLABLE_MODULE

LOCAL_UNSTRIPPED_PATH



LOCAL_WHOLE_STATIC_LIBRARIES 指定模块所须要加载的完整静态库(这些精通库在链接是不同意链接器删除当中没用的代码)



LOCAL_YACCFLAGS



OVERRIDE_BUILT_MODULE_PATH

接下来我们具体看一下android里的makefile文件

android最顶层的文件夹结构例如以下:



|-- Makefile (全局的Makefile) 

|-- bionic (Bionic含义为仿生,这里面是一些基础的库的源码) 

|-- bootloader (引导载入器) 

|-- build (build文件夹中的内容不是目标所用的代码,而是编译和配置所须要的脚本和工具) 

|-- dalvik (JAVA虚拟机) 

|-- development (程序开发所须要的模板和工具) 

|-- external (目标机器使用的一些库) 

|-- frameworks (应用程序的框架层) 

|-- hardware (与硬件相关的库) 

|-- kernel (Linux2.6的源码) 

|-- packages (Android的各种应用程序) 

|-- prebuilt (Android在各种平台下编译的预置脚本) 

|-- recovery (与目标的恢复功能相关) 

`-- system (Android的底层的一些库)

本文将要分析的是build文件夹下的makefile和shell文件,android的代码是1.5的版本号。

基本的文件夹结构例如以下:

1.makefile入门

1.1 makefile helloworld

1.2 用makefile构建交叉编译环境

1.3 makefile里面的一些技巧

2.android makefile分析

2.1 android shell分析

2.2 android build下的各个makefile分析

3. android其它文件夹的android.mk分析



大家先通过网络的一些文章来了解一下andoroid的makefile。

1.1 makefile helloworld



Makefile的规则例如以下:



target ... : prerequisites ... 



command ... ...



target能够是一个目标文件,也能够是Object File(比如helloworld.obj),也能够是运行文件和标签。



prerequisites就是生成target所须要的文件或是目标。



command

也就是要达到target这个目标所须要运行的命令。这里没有说“使用生成target所须要运行的命令”,是由于target可能是标签。

须要注意的是

command前面必须是TAB键,而不是空格,因此喜欢在编辑器里面将TAB键用空格替换的人须要特别小心了。

我们敲代码一般喜欢写helloworld。当我们写了一个c的helloworld之后。我们该怎样写helloworld来编译helloworld.c呢?



以下就是编译helloworld的makefile。



helloworld : helloworld.o



cc -o helloworld helloworld .o



helloworld.o : helloworld.c 



cc -c main.c



clean:



rm helloworld helloworl.o



之后我们运行make就能够编译helloworld.c了,运行make clean就能够清除编译结果了(事实上就是删除helloworld helloworl.o)。

可能有人问为什么运行make就会生成helloworld呢?这得从make的默认处理说起:make将makefile的第一个target作为作为终于的



target,凡是这个规则依赖的规则都将被运行,否则就不会运行。

所以在运行make的时候,clean这个规则就没有被运行。



面的是最简单的makefile,复杂点makefile就開始使用高级点的技巧了。比如使用变量,使用隐式规则,运行负责点shell命令(常见的是字

符串处理和文件处理等),这里不打算介绍这些规则,后面在分析android的makefile时会结合详细代码进行详细分析,大家能够先看看陈皓的《跟

我一起写makefile》来了解了解。

makefile的大体的结构是程序树形的。例如以下:





Android发展的一个重要方面Makefile分析





这样写起makefile也简单,我们将要达到的目标作为第一个规则,然后将目标分解成子目标,然后一个个写规则,依次类推,直到最以下的规则非常easy实现为止。这事实上和算法里面的分治法非常像,将一个复杂的问题分而治之。





到树,我想到了编译原理里面的语法分析,语法分析里面有自顶而下的分析方法和自底而下的分析方法。当然makefile并非要做语法分析,而是要做与语

法分析分析相反的事。

(语法分析要做的是一个句子是不是依据语法能够推出来。而makefile要做的是依据规则生成一个command

运行队列。

)只是makefile的规则和词法分析还是非常像的。以下出一道编译原理上面的一个样例,大家能够理解一下makefile和词法分析的不同点

和同样点:



-> 

-> |||ε 

-> 

-> |ε 

-> + 

-> - 

-> > 

-> >=







最后,介绍一下autoconfautomake,使用这两个工具能够自己主动生成makefile。

Android发展的一个重要方面Makefile分析





上面的图能够看出。通过autoscan,我们能够依据代码生成一个叫做configure.scan的文件,然后我们编辑这个文件,參数一个

configure.in的文件。接着我们写一个makefile.am的文件,然后就能够用automake生成makefile.in。最后,依据

makefile.in和configure就能够生成makefile了。在非常多开源的project里面,我们都能够看到

makefile.am,configure.in,makefine.in,configure文件,还有可能看到一个十分复杂的makefile文

件,很多人学习makefile的时候想通过看这个文件来学习,终于却发现太复杂了。假设我们知道这个文件是自己主动生成的。就理解这个makefile文件

为什么这个复杂了。

2.在用户的home文件夹(cd ~)建一个文件夹cross-compile



3.在cross-compile创建一个文件cross.env,内容例如以下:



export WORK_DIR=~/cross-compile 

export ROOTFS_DIR=$WORK_DIR/rootfs 

export ARCH=arm 

export PKG_CONFIG_PATH=$ROOTFS_DIR/usr/local/lib/pkgconfig:$ROOTFS_DIR/usr/lib/pkgconfig:$ROOTFS_DIR/usr/X11R6/lib/pkgconfig

if [ ! -e "$ROOTFS_DIR/usr/local/include" ]; then mkdir -p $ROOTFS_DIR/usr/local/include;fi; 

if [ ! -e "$ROOTFS_DIR/usr/local/lib" ]; then mkdir -p $ROOTFS_DIR/usr/local/lib; fi; 

if [ ! -e "$ROOTFS_DIR/usr/local/etc" ]; then mkdir -p $ROOTFS_DIR/usr/local/etc; fi; 

if [ ! -e "$ROOTFS_DIR/usr/local/bin" ]; then mkdir -p $ROOTFS_DIR/usr/local/bin; fi; 

if [ ! -e "$ROOTFS_DIR/usr/local/share" ]; then mkdir -p $ROOTFS_DIR/usr/local/share; fi; 

if [ ! -e "$ROOTFS_DIR/usr/local/man" ]; then mkdir -p $ROOTFS_DIR/usr/local/man; fi; 

if [ ! -e "$ROOTFS_DIR/usr/include" ]; then mkdir -p $ROOTFS_DIR/usr/include; fi; 

if [ ! -e "$ROOTFS_DIR/usr/lib" ]; then mkdir -p $ROOTFS_DIR/usr/lib; fi; 

if [ ! -e "$ROOTFS_DIR/usr/etc" ]; then mkdir -p $ROOTFS_DIR/usr/etc; fi; 

if [ ! -e "$ROOTFS_DIR/usr/bin" ]; then mkdir -p $ROOTFS_DIR/usr/bin; fi; 

if [ ! -e "$ROOTFS_DIR/usr/share" ]; then mkdir -p $ROOTFS_DIR/usr/share; fi; 

if [ ! -e "$ROOTFS_DIR/usr/man" ]; then mkdir -p $ROOTFS_DIR/usr/man; fi;



4.开启命令行,进入cross-compile文件夹下,运行. cross.env



5.将编译linux时生产的头文件。so等复制到cross-compile文件夹下rootfs/usr相应的文件夹(头文件一般能够拷pc的,so一定要拷arm版的)。

5.下载要编译的源码。并放在cross-compile文件夹下

====================================================================================

      谈谈程序猿发展的5个出路   
         程序猿接私活的几个注意事项

      从IT菜鸟变“骨干”的10个建议  
      就业市场最急需的10大类IT人才

     怎样成为软件设计师混合型人才   应届生就业,“IT行业”平均收入最高

      找程序猿做老公的10个优点  
         学Android软件开发的就业钱景分析

      给IT新兵职业发展的15个建议  
      程序猿成美2014收入最高职业

====================================================================================

版权声明:本文博客原创文章,博客,未经同意,不得转载。