Wordpress 加入小组

43个成员 1293个话题 创建时间:2013-08-10

WordPress Initialization(Wordperss工作流程.) 置顶

发表于 2013-08-24 21899 次查看

I began learning about WordPress by building themes and reading through the multitude of awesome tutorials posted across the net. A lot was learned through trial and error, and a lot was learned from copying and pasting. However, there is one topic for which I have yet to find comprehensive information: the WordPress initialization process. I mean from start to finish, how does the WordPress core take the initial page request and give back such beautiful web pages? It’s time to take the plunge and actually step through the WordPress core, file by file, so I can get a better understanding of how our favorite personal publishing platform initializes itself and then displays my content, all of our content, for the world to see. I hope this post will serve not only as a place that I can return to as a reference but also as a place for all of you to turn to when you need a refresher on the inner workings of the WordPress core structure, or if you’re just starting out and need a better understanding of what WordPress is doing with your theme and your code. Let’s get started…

 

This post is based on the WordPress trunk code base. All of the links to Core Trac source code point to the trunk version of that code. I’ve done this because the WordPress core will change faster than I can keep this post updated, so following the links will always take you to the latest version of the relevant source, regardless of whether or not the content of this article has been updated. The downside to this decision is that you may follow a link that does not accurately reflect the code base for your version of WordPress. If you need to reference the core code base of a previous version of WordPress, the information in this post will at the very least get you close.

Update: On January 29, 2013 I reviewed the links and content in this post and updated it to reflect changes to the current version of WordPress; which at the time of the update was v3.5.1.

All of the magic flowing out of your server starts with a single file: index.php. This simple file handles the initial request each and every time a page is loaded from your site. It serves two main functions. First it defines a constant named 'WP_USE_THEMES'and sets it’s value to true. This tells the WordPress core that your site should use the active theme to display the pages it’s serving. Next it requires a file named wp-blog-header.php. This file sits in the root directory of your site with index.php.

wp-blog-header is singularly responsible for three fundamental processes during the initialization of WordPress. First it requires a file named wp-load.php, then it executes the function wp() and finally it requires a file named template-loader.phpwp-load.php is also located in the root directory with index.php and wp-blog-header.php. The wp() function is defined in /wp-includes/functions.phptemplate-loader.phpalso resides in the /wp-includes directory with functions.php. We will get to the wp-includes directory in a little bit. We first have to step through the responsibilities of wp-load and the files that it requires.

wp-load has a simple yet extremely important role in WordPress’ initialization. It starts by setting a constant named 'ABSPATH' to the root directory of your WordPress install. This constant is used extensively throughout the remainder of the initialization process to easily and consistently refer to config files relative to your root directory.wp-load completes its work by loading wp-config.php. Most of us should be at least a little familiar with this file. It wasn’t too long ago that manual editing of your wp-configwas still a requirement during the install process. Today we all enjoy an automatedwp-config setup as long as our server’s user permissions are configured correctly.wp-load actually runs through some checks to determine where your wp-config file is located. This is because you can keep your wp-config file outside of your root install directory. I’m not sure why you would but I assume it has a lot to do with security. If any one does know, please explain in the comments. If wp-load fails to find a config file then it throws you a friendly error message with a button to create one.

If wp-load successfully located your wp-config file then this step uses that information to set the last bit of configuration settings before continuing with loading up the WordPress core. The comments at the top of wp-config sum this step up quite nicely “This file has the following configurations: MySQL settings, Table Prefix, Secret Keys, WordPress Language, and ABSPATH“. I can’t put it any more succinctly than that. After wp-config is parsed WordPress has all the information it needs to securely access your database and load your content. I’m not sure why ABSPATH is checked again but it is, it’s the last line of code before ALL the fun begins. The final line of wp-config requires wp-settings.php which is responsible for loading the lion’s share of WordPress.

Finally some action! Up until now we’ve been wandering through mundane information collection. wp-settings.php starts off by defining a new constant named'WPINC'. This constant points to the /wp-includes directory that was mentioned above. The majority of the predefined functions that WordPress needs for initialization are defined in this directory’s files. After defining WPINC it requires three files from that directory, load.phpdefault-constants.php and version.php. It’s important to remember that WordPress is not actually executing these functions. They are being included using the PHP function require(), which makes their contents available to WordPress’ execution environment so the functions they contain can be called when they’re needed. load.php includes the following functions that are required to properly setup WordPress:

wp_unregister_GLOBALS()
Turns register globals off
wp_fix_server_vars()
Fixes $_SERVER variables for various setups
wp_check_php_mysql_versions()
Checks for the required PHP version, and the MySQL extension or a database drop-in
wp_favicon_request()
This function sends a zero-length favicon header and bails so that the entire WordPress core isn’t loaded when just a favicon is requested
wp_maintenance()
This function dies with a maintenance message when special “maintenance” conditions are met
timer_start()
PHP 4 standard microtime start capture
timer_stop()
Returns and/or displays the time from the page start to when function is called
wp_debug_mode()
Sets PHP error handling and handles WordPress debug mode
wp_set_lang_dir()
Sets the location of the language directory
require_wp_db()
Loads the correct database class file
wp_set_wpdb_vars()
Sets the database table prefix and the format specifiers for database table columns
wp_start_object_cache()
Starts the WordPress object cache
wp_not_installed()
Redirects to the installer if WordPress is not installed
wp_get_mu_plugins()
Returns array of must-use plugin files to be included in global scope
wp_get_active_and_valid_plugins()
Returns array of plugin files to be included in global scope
wp_set_internal_encoding()
Sets internal encoding using mb_internal_encoding()
wp_magic_quotes()
Adds magic quotes to $_GET, $_POST, $_COOKIE, and $_SERVER
shutdown_action_hook()
Runs just before PHP shuts down execution
wp_clone()
Copies an object
is_admin()
Checks whether the current request is for a network or blog admin page
is_blog_admin()
Checks whether the current request is for a blog admin screen /wp-admin/
is_network_admin()
Checks whether the current request is for a network admin screen /wp-admin/network/
is_user_admin()
Checks whether the current request is for a user admin screen /wp-admin/user/
is_multisite()
Checks whether Multisite support is enabled
get_current_blog_id()
Retrieve the current blog id
wp_load_translations_early()
Attempts an early load of translations

Once those essential functions have been loaded wp-settings loads six more functions from default-constants.php that will be used to ensure that the global environment for further initialization can be configured properly. The list below outlines these six functions and the constants they define.

wp_initial_constants()
Defines initial WordPress constants
WP_MEMORY_LIMIT
WP_MAX_MEMORY_LIMIT
$blog_id
WP_CONTENT_DIR
WP_DEBUG
WP_DEBUG_DISPLAY
WP_DEBUG_LOG
WP_CACHE
MEDIA_TRASH
SHORTINIT
MINUTE_IN_SECONDS
HOUR_IN_SECONDS
DAY_IN_SECONDS
WEEK_IN_SECONDS
YEAR_IN_SECONDS
wp_plugin_directory_constants()
Defines plugin directory WordPress constants
WP_CONTENT_URL
WP_PLUGIN_DIR
WP_PLUGIN_URL
PLUGINDIR
WPMU_PLUGIN_DIR
WPMU_PLUGIN_URL
MUPLUGINDIR
wp_cookie_constants()
Defines cookie related WordPress constants
COOKIEHASH
USER_COOKIE
PASS_COOKIE
AUTH_COOKIE
SECURE_AUTH_COOKIE
LOGGED_IN_COOKIE
TEST_COOKIE
COOKIEPATH
SITECOOKIEPATH
ADMIN_COOKIE_PATH
PLUGINS_COOKIE_PATH
COOKIE_DOMAIN
wp_ssl_constants()
Defines cookie related WordPress constants
FORCE_SSL_ADMIN
FORCE_SSL_LOGIN
wp_functionality_constants()
Defines functionality related WordPress constants
AUTOSAVE_INTERVAL
EMPTY_TRASH_DAYS
WP_POST_REVISIONS
WP_CRON_LOCK_TIMEOUT
wp_templating_constants()
Defines templating related WordPress constants
TEMPLATEPATH
STYLESHEETPATH
WP_DEFAULT_THEME

Finally version.php is loaded which provides the following six variables:

$wp_version
The WordPress version string
$wp_db_version
Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema
$tinymce_version
Holds the TinyMCE version
$manifest_version
Holds the cache manifest version
$required_php_version
Holds the required PHP version
$required_mysql_version
Holds the required MySQL version

Now we’re cookin’!! The next 40 lines of code in wp-settings call many of the functions included above. wp_initial_constants() is run to setup memory, debug and caching constants; wp_check_php_mysql_versions() makes sure that we’re running the minimum required versions of both PHP and MySQL; magic quotes are disabled because they will be added later; the default timezone is set; thewp_unregister_GLOBALS() function is executed to disable PHP’s register globals setting; wp_fix_server_vars() fixes various $_SERVER related problems across different hosts; wp_favicon_request() ensures that the entire WordPress core isn’t loaded when only a favicon is being requested; wp_maintenance() (a very cool function) checks to see if there is a .maintenance file in your ABSPATH that was created less than 10 minutes ago, if there is it displays a default maintenance message that can be overridden by placing a custom maintenance.php file in your /wp-contentdirectory; timer_start() loads the timer; wp_debug_mode() checks to see if debugging is enabled and turns on PHP’s error reporting if it is; the WP_CACHE constant is checked and if it returns true then /wp-content/advanced-cache.php is included; and finallywp_set_lang_dir() sets the path to the language files.

The WP_CACHE constant is used by 3rd party caching solutions like W3TCWP Super Cache or Batcache. If WP_CACHE returns true then advanced-cache.php acts as the “bootstrapper” to load the caching engine you’re using (thanks to Mark Jaquith for that info).

At this point we have a pretty solid foundation for WordPress to build the rest of it’s initialization on top of. Now it’s time to load some the core files that provide WordPress’ true functionality. Next up wp-settings requires five more files that it classifies as “early WordPress files”. They are compat.phpfunctions.phpclass-wp.phpclass-wp-error.php and plugin.php. All of these are located in the /wp-includes directory. While these may be early WordPress files they shouldn’t be underestimated. A tremendous amount of the functionality we expect from WordPress is provided in these five files. They are long and complex, functions.php alone is over 4,500 lines! I am not going to go into the details of each file’s functions the same way I did above. I will, in future posts, address each of these files individually to give them the attention they deserve. I am after all writing this post (and future posts) to learn more about the core myself. The descriptions below have been taken directly from the source code comments on Trac. It’s more important for this post to focus on the overall initialization of WordPress so I’ll just outline these functions here:

compat.php
WordPress implementation for PHP functions either missing from older PHP versions or not included by default.
functions.php
Main WordPress API
class-wp.php
WordPress environment setup class.
class-wp-error.php
WordPress Error API. Contains the WP_Error class and the is_wp_error() function.
plugin.php
The plugin API is located in this file, which allows for creating actions and filters and hooking functions, and methods. The functions or methods will then be run when the action or filter is called.
mo.php
Part of the WordPress translation and localization sub-system. I don’t know a whole lot about localization and translation so I will link to this page I found on the Codex and leave further investigation to the reader.

Now that the baseline functionality of the above files has been loaded wp-settingsmoves on to call three more of the functions that were included in wp-load above. Firstrequire_wp_db() is called to load the DB class and set up the global $wpdb object. It accomplishes this by requiring /wp-includes/wp-db.php and then /wp-content/db.php if it exists. wp-db.php is another one of those files with over 1,000 lines of code. I will in the future dedicate a post to it alone. For the purposes of this article it’s only important to understand that wp-db.php is an abstract class file for accessing a MySQL database. The reason that require_wp_db() tries to require /wp-content/db.php is because you can define your own database class to access your WordPress database by extending wp-db.php and placing that code into the /wp-content/db.php file. require_wp_db() finishes up by creating the $wpdb object using the database credentials defined in your wp-config file. Next wp_set_wpdb_vars()furthers the setup of the $wpdb object by setting the field types for the table columns and the table prefix. The $table_prefix variable defined in wp-config is added to the$GLOBALS[] array just prior to this step so WordPress knows exactly which tables it can access. Finally, wp_start_object_cache() is called. It’s a little similar torequire_wp_db() in that it’s responsible for loading an abstract class file. In this case its the WordPress Object Cache API. This class is responsible for caching database objects in memory to speed up frequent queries sent to the database.wp_start_object_cache() first checks to see if there is a file named /wp-content/object-cache.php because (like /wp-content/db.php) you can define your own object cache file or class. If wp_start_object_cache() finds /wp-content/object-cache.php then WordPress’ internal object cache WILL NOT be loaded. If there is no /wp-content/object-cache.php then the default cache filecache.php is loaded.

Next up the default-filters.php is required from the /wp-includes directory. Thedefault-filters.php file adds many (but not all) of the default filters and actions to the WordPress Hooks system. This file goes hand-in-hand with plugin.php that was loaded above. Together these two files prepare the WordPress Plugin API. Anyone interested in plugin development should read through the Plugin API on the codex and have at least these two files open while reading it.

I’m going to group the next 17 lines of code together because after they’re executed we’ll be about halfway through wp-settings.php and moving into loading a huge chunk of the WordPress core files and functionality. After default-filters.php is loaded wp-settings checks to see if we are running multisite. If we are then it loadsms-blogs.php and ms-settings.php from /wp-includes. These two files act like wp-settings.php but handle setup specific to sites that have MultiSite enabled in wp-config.php. I don’t have any experience with MultiSite at this time but I plan on researching it in the future. At that time I will post an article similar to this one that concentrates on how multisite is initialized. Next, the value of SHORTINIT is evaluated and if it’s true then initialization is halted at this point. This constant was defined in thewp_initial_constants() function declared in default-constants.php above. In that file, it’s default value is set set to false. There may be times that you need access to WordPress without loading all of it’s functionality, e.g. using it as a file serving proxy in MultiSite mode. Setting this constant to true allows you to halt initialization and only use the limited functionality loaded up to this point (thanks again to Mark Jaquith for this info). An additional file related to localization and translation: /wp-includes/l10n.php, is required next and finally wp_not_installed() is called to run the installer if we’ve reached this point and WordPress hasn’t been installed. This function was defined in load.php and runs a few checks to see if certain variables and functions have been defined. If they haven’t, that means WordPress isn’t installed and the initialization process dies; at which point you are redirected to the installer /wp-admin/install.php.

With those 17 lines of code out of the way wp-settings moves on to loading the majority of the remaining core files. The following list is long and each file listed is even longer. Just as above, I am going to list them with brief descriptions and link to each file on Trac. Many of the following files deserve their own posts and I will tackle them as time permits. These 40 files constitute the majority of WordPress’ core functionality.

class-wp-walker.php
A class for displaying various tree-like structures.
class-wp-ajax-response.php
Send XML response back to AJAX request.
formatting.php
Main WordPress Formatting API. If you’ve ever wondered how WordPress replaces your double-line breaks with paragraph elements or inserts the smiley faces in your blog posts, this is the file that does it. This file is also responsible for changing all references to ‘WordPress’ to the proper capitalization.
capabilities.php
WordPress Roles and Capabilities.
query.php
WordPress Query API. Responsible for creating the WP_Query object that allows your front end to talk to the database on the backend. Think: The Loop
theme.php
Theme, template, and stylesheet functions. Defines the functions necessary to control your theme(s) as well as the functions that make the Template Hierarchypossible.
class-wp-theme.php
This file defines the WP_Theme class, which helps developers interact with a theme.
template.php
Functions designed to easily retrieve the path to a template are defined here.
user.php
WordPress User API – Responsible for user authentication.
meta.php
Metadata API. Allows you to add information to your database to describe objects that are stored there.
general-template.php
General template tags that can go anywhere in a template. All of your template tags like get_header() and get_footer() are defined here.
link-template.php
WordPress link template functions. Defines the template tags that are dedicated to navigation between posts. It defines functions like next_posts(), previous_posts() and the_permalink().
author-template.php
Author template functions for use in themes. Defines the functions that get info about the post’s author. These functions must be used inside The Loop.
post.php
Post functions and post utility functions. Defines all the functions necessary for creating, editing and controlling posts.
post-template.php
WordPress Post Template Functions. The functions needed to retrieve post information while in The Loop are defined here.
post-thumbnail-template.php
WordPress Post Thumbnail Functions. The functions needed to retrieve post thumbnail information A theme’s functions.php must call add_theme_support( 'post-thumbnails' ) to use these.
category.php
WordPress Category API. Defines functions for managing category details.
category-template.php
Category Template Tags and API. Defines functions for accessing category details through template tags.
comment.php
Defines functions for managing comments.
comment-template.php
Defines functions for accessing comment details through template tags. These are the functions used in The Loop.
rewrite.php
WordPress Rewrite API. This library of functions handles the creation of “pretty URLs” like /%category%/%postname%/.
feed.php
WordPress Feed API. These functions are designed to be used in The Loop to access and manipulate your site’s RSS/ATOM info.
bookmark.php
Link/Bookmark API. Functions for creating and manipulating bookmarks (blogroll links) in the database.
bookmark-template.php
Defines bookmark template tags for usage in themes.
kses.php
HTML/XHTML filter that only allows some elements and attributes.
cron.php
WordPress CRON API. These functions allow you to create either one-time or scheduled recurring processes that are triggered and executed when someone visits your site.
deprecated.php
Deprecated functions from past WordPress versions (functions come here to die).
script-loader.php
WordPress scripts and styles default loader. These functions load all the JS andCSS that WordPress needs to create and run the admin area.
taxonomy.php
WordPress Taxonomy API.
update.php
A set of functions to check the update service.
canonical.php
Canonical API to handle WordPress Redirecting. These functions make sure that proper SEO practices are enforced in your site’s URLs and even attempts to guess the correct post/page to load if a visitor types in an incorrect URL.
shortcodes.php
WordPress API for creating bbcode like tags or what WordPress calls “shortcodes.”
class-wp-embed.php
API for easily embedding rich media such as videos and images into content.
media.php
WordPress API for media display. Contains the functions WordPress uses to access, edit and display images throughout your site.
http.php
Simple and uniform HTTP request API.
class-http.php
Simple and uniform HTTP request class.
widgets.php
API for creating dynamic sidebar without hardcoding functionality into themes. This class and the functions it defines are what make creating sidebars and widgets in plugins and your theme’s functions.php file so easy.
nav-menu.php
Navigation Menu functions. All the code necessary to create, control and remove menus in the WordPress admin.
nav-menu-template.php
Navigation Menu template functions. All the code necessary to display menus in a WordPress theme.
admin-bar.php
This code handles the building and rendering of the Admin Bar.

Note: If you’d like to read a little more about the inner workings of the Admin Bar (now officially called the Toolbar), check out my first freelance article over at WP Smashing Magazine: Inside The WordPress Toolbar.

Just from reading through these files one at a time I estimate that roughly 75% of the WordPress core has been loaded and we’re about half-way through the wp-settingsfile. After all the require() statements above wp-settings does another check to see if we’re running multisite. If we are then it requires three additional files: ms-functions.phpms-default-filters.php and ms-deprecated.php. These are the MultiSite (formerly WordPress MU) versions of some of the files loaded above. After this check for MultiSite the function wp_plugin_directory_constants(), which was defined in default-constants.php, is executed. It sets up seven constants for plugins to use. Now that those constants have been created a foreach() loop is run to include what are known as “must use” plugins. Back in load.php a function namedwp_get_mu_plugins() was defined. That function reads the location of theWPMU_PLUGIN_DIR constant (which by default is set to /wp-content/mu-plugins) and loads any plugins that it finds. Once that’s done there is another check to see if we’re running MultiSite. If we are then another foreach() loop is executed to include thewp_get_active_network_plugins() function. This function is defined in ms-load.phpand loads any MultiSite related plugins that are needed now from the WP_PLUGIN_DIRconstant, which by default is set to the wp-content/plugins directory. ms-load.phpwas included by the ms-settings.php file. Now that the majority of the Plugin API has been setup the do_action( 'muplugins_loaded' ) hook is executed.

wp-settings then moves on to call two more of the functions defined in default-constants.phpwp_cookie_constants() and wp_ssl_constants() are responsible for guaranteeing that we have unique hashes for our cookies and whether or not to force admin and login SSL connections respectively. Once the cookies and SSL settings have been set the file vars.php is required. This is actually a really cool file and I had no idea that this functionality existed in WordPress until I popped it open. vars.phpstarts by setting the $pagenow variable which always holds the location of the page that’s currently being viewed. After that it tests to see what browser is being used to view the page and stores that in one of a series of nine possible variables: $is_lynx,$is_gecko$is_winIE$is_macIE$is_opera$is_NS4$is_safari$is_chrome or$is_iphone. How ’bout them apples!! If you’ve ever wanted to serve custom content to your visitors based on their browser from inside your template files these variables allow you to do just that. After the browser detection, vars.php runs a few checks to determine what server software WordPress is running on, either Apache or IIS. Finally, it finishes up by setting one final variable: $is_mobile. The variable is a boolean value (true|false) which will be set to true if the value of$_SERVER['HTTP_USER_AGENT'] matches any of the most common mobile browsers. Pretty fun stuff from a file of only 97 lines!

Up next wp-settings fires off two more functions. create_initial_taxonomies()(which is defined in taxonomies.php) and create_initial_post_types() (which is defined in post.php). The first sets up and initializes the default WordPresstaxonomies and the second sets up the default post types (post, page, etc.). There is a warning in wp-settings alerting all you developers out there that the defaults in these two files will be run again on the init hook that executes at the end of wp-settings. Taxonomies are how WordPress organizes information internally. We are all familiar with the two most notable taxonomies already: categories and tags. Custom taxonomies were introduced in v2.3 and updated in v2.8 and v3. There is a great article from Justin Tadlock on custom taxonomies in v2.8 here and another from NetTuts on v3 custom taxonomies available here. If you’ve looked at the admin area of any decent theme you know that there are usually options to change the “type” of post (e.g. – aside, quote, gallery, etc.). Those defaults are all created in thecreate_initial_post_types() function. v3 of WordPress added the ability to create custom post types. Here’s another great article from Justin Tadlock on custom post types in v3.

At this point most of the core, the Plugin API, taxonomies and post types are loaded so wp-settings starts loading the active theme. It does this by executingregister_theme_directory() and passing in get_theme_root() as an argument. Both of these functions were defined in theme.php. They are responsible for determining the location of and loading the active theme’s directory. Once the active theme’s directory has been loaded wp-settings loads all active plugins by running aforeach() loop to step through each plugin found by runningwp_get_active_and_valid_plugins(). This function (defined in load.php) looks through /wp-content/plugins and returns any valid plugins that it finds. After the active plugins are loaded two new files are required. pluggable.php and pluggable-deprecated.php include useful functions for working with logged in users, sending mail and getting/setting user info. They are defined and required at this point so they can be overridden by any plugins loaded through the upcoming do_action( 'plugins_loaded' ) hook. The pluggable-deprecated.php file is included to maintain backward compatibility but obviously none of those functions should be used because they will be removed in a future version. The default encoding is set next by running wp_set_internal_encoding() which was defined in load.php, it defaults to UTF-8. Now a quick check is made to see if the constant WP_CACHE is set to true and if the wp_cache_postload() function exists. If it does then it gets executed to setup any remaining caching related functionality that had to wait until more of the core was loaded. At this point the entire Plugin API is up and running so the do_action( 'plugins_loaded' ) hook is fired to enable plugin developers to execute any custom functionality provided in their plugins when they’re loaded.

wp-settings now fires off wp_functionality_constants() and wp_magic_quotes().wp_functionality_constants(), defined in default-constants.php, sets theAUTOSAVE_INTERVAL, EMPTY_TRASH_DAYS and WP_POST_REVISIONS constants to handle the housekeeping for your posts. wp_magic_quotes(), defined in load.php, takes all the variables from $_GET, $_POST, $_COOKIE and $_SERVER and sanitizes them by running stripslashes_deep() and add_magic_quotes() and then adds them to an array that gets assigned to $_REQUEST. Then the do_action( 'sanitize_comment_cookies' ) hook is executed.

The next section of wp-settings handles the creation and setup of some pretty important objects. First it creates a new WP_Query object and assigns it to the variable $wp_the_query. Then it creates another variable that points to a reference to that variable called $wp_query. This is the default loop that you’re accessing when you run if (have_posts()) : while (have_posts()) : the_post();. After that’s done it creates the WP_Rewrite object (defined in rewrite.php for handling permalinks) and assigns that to the global variables array in $GLOBALS['wp_rewrite']. Then it creates a new WP object (which was defined in class-wp.php and is responsible for creating the whole WordPress environment) and assigns it to the variable $wp. Next it creates a new WP_Widget_Factory object and assigns that to the global variables array at $GLOBALS['wp_widget_factory'], and finally creates a new WP_Roles object, assigning it to the global variables array at $GLOBALS['wp_roles']Once all those tasks have competed the hook do_action( 'setup_theme' ) is executed to allow plugins to take action right as a theme is getting ready to be loaded.

Now wp-settings moves on to setup a few environment requirements for getting our theme loaded. The function wp_templating_constants() (defined in default-constants.php) is executed at this point. This function is responsible for creating the constants familiar to theme designers, TEMPLATEPATH and STYLESHEETPATH. In addition, it checks to see if the WP_DEFAULT_THEME constant has been set and if not it sets it to (currently) twentytwelve. This default theme is used for any new install of WordPress as well as if the current theme somehow goes missing. After that function theload_default_textdomain() function is run to ensure that proper language settings are loaded. Once the default text domain has been set the default locale file is determined and required. Once both of those steps are complete a WP_Locale object is created and assigned to the global variables array at $GLOBALS['wp_locale']. Anyone interested in how WordPress handles so many multiple languages or wants to help WordPress support more languages, these are the files you want to study.

We are so close to the end I can taste it! This next chunk of code will be interesting to any parent/child theme designer who wonders why a child’s functions.php file is loaded before the parent’s, but all the parent’s template files are loaded before the child’s. I know I always have. After the locale is setup wp-settings runs through an if statement checking the values of TEMPLATEPATH and STYLESHEETPATH. IfSTYLESHEETPATH is different than TEMPLATEPATH it loads it’s functions.php first then it calls the one in TEMPLATEPATH. I guess if you think it through it makes sense. You want the child to be able to set up any custom functions before the parent can define them. This is why parent themes should check if a function exists before executing it, just incase the child has defined a custom one. Once that code executes wp-settingscalls the do_action( 'after_setup_theme' ) hook for any themes or plugins that need to execute custom code when a theme is done with its setup. The init method of the $wp object is called next to setup the current user’s settings and the hookdo_action( 'init' ) is run. This is just about it folks, there is only one more check for MultiSite and then the hook do_action( 'wp_loaded' ) is run for any theme or plugin that needs to run any code once WordPress’ core has fully loaded. As the curtain sets on wp-settings the WordPress core has completed loading and execution returns to wp-blog-header.php. I know, it feels like we left that file a few days ago! It always amazes me that all of the functionality we just covered loads in mere seconds while the request is being processed.

If you’ve made it this far I would just like to congratulate you. We’re almost done! Lets just do a really quick recap of where we are at this point. When the visitor’s browser sent our server a request index.php set WP_USE_THEMES to true and required wp-blog-header.php. That file first called wp-load.php, then it executes the function wp() and finally requires template-loader.phpwp-load.php defined the ABSPATH constant and grabbed the site’s configuration settings from wp-config.php before calling wp-settings.phpwp-settings.php (the majority of this article, which we just covered) handled the loading of all core functionality, plugins and the active theme. Now thatwp-settings.php is complete and execution has returned to wp-blog-header.php we can move on to the conclusion of our tour through WordPress’ initialization.

wp-blog-header.php now executes a function defined in functions.php named wp(). If you weren’t paying attention, about 3/4 of the way through wp-settings.php a global variable named $wp was defined and instantiated as a WP object (that WP object was defined in class-wp.php). This object is responsible for controlling the entire WordPress environment after it’s been loaded. The wp() function called by wp-blog-header.php finalizes the setup of this object by calling its main() function. This effectively puts the finishing touches on the global WordPress environment and also ensures that the current query has been properly configured so The Loop works the way we all expect it to. I gotta give one final shout-out to Mark Jaquith for pointing me to the info on this function. I’m tellin’ you this guy is a badass! If you don’t follow him on Twitter, do yourself a favor and follow him now. I’ll just say this, that thing on his shoulders ain’t just a hat rack my friends!

We can now finally display a page to our visitor! The template-loader.php file is responsible for determining not only what template should be used for displaying the content but also what kind of visitor our server is responding to. If this request is for a robot, a feed or a trackback then it serves the appropriate content for those requests. If the current request is from a robot then it calls the do_robots hook. This hook is defined in functions.php and serves a robots.txt file when called. If the request is for one of our feeds then the function do_feed() is called (also defined in functions.php) and then serves whichever feed format has been requested. If the request is a trackback or pingback then the wp-trackback.php file is required. This file runs through some fancy checks and then builds the response which is served back based on whether a trackback or pingback request was received. If the request is for anything other than what we just covered then WordPress steps through a 25 line elseif statement to determine the type of template to load. This is how the Template Hierarchy is implemented in WordPress. There’s a great overview of the template hierarchy, written by Chris Coyier, over at his (and Jeff Star’sDigging into WordPresssite. I highly recommend that site in general, loads of great tips and tricks!

WHEW!! Well, if you think about it, everything we just covered is what happened when you clicked whatever link brought you to this post. It’s taken me a lot longer than I expected to write this; my back hurts and I really need to get back to learning some Ruby programming! :-) So I’m just going to thank you if you’ve made it to the end. Thank you for sticking with me. I hope you’ve learned as much reading this as I did writing it. I intend to be back soon with a (shorter) post on a more specific topic.

3回复
  • 2楼 第一个好知用户 2013-08-26

    我有时间的时候,翻译一下。

  • 3楼 第一个好知用户 2013-09-16

    我最初学习Wordpress一方面是通过自己做一些主题,另一方面是在网上阅读一些优秀的教程.在反复的练习和错误中我学习了很多,在只是粘贴和复制中我也学习了很多.但是我还没有遇到一个全面讨论关于Wordpress如何初始化的话题.初始化的概念就是:Wordpress收到最初的客户请求后如何把一个漂亮的网页呈现在用户面前?现在是时候花大量精力去一个文件一个文件的去了解Wordpress了.这样也便于我了解这个我最喜爱的内容发布平台是如何完成自己的初始化的过程,是如何把那些精彩的内容呈现在用户面前.我希望这篇文章一方面自己可做为备份,另一方面也希望想更清晰的了解Wordpress的工作原理和组织架构的你提供参考,我想这对于一个刚刚学会Wordpress,但又想很清楚Wordpress的工作机制的人来说还是会有一些帮助的.言归正传,我们开始吧.

     

    这篇文章是基于Wordpress的主干核心文件(函数).所有的编译文件都是指向该文件的主干核心文件.我这样做的原因是Wordpress更新的速度比我这篇文章更新的速度要快.所以,不管我写的这篇文章的内容是否更新,下面的一些链接会把Wordpress最新的相关文件呈现在你的面前.所以说,下面的一些链接也不一定是你目前在用的Wordpress版本的文件.如果你需要参考Wordpress以前的版本的核心代码的话,这篇文章也是挺接近的.

     

    更新:在2013年1月29号的时候,我检查了这些链接和内容,并把相关信息更新至目前使用的最新版本:V3.5.1.

     

    Index.php是所有文件的入口(是魔术的开始).Wordpress每显示一个页面,index.php都会处理初始请求.index.php主要完成两个任务.首先,他定义了一个常量叫’WP_USE_THEMES’ 并把这个常量设置为true.这是告诉Wordpress应该使用激活的那个主题显示内容,然后呢,这个index.php引入了wp-blog-header.php这个文件.index.php和wp-blog-header.php都位于wordpress的根目录下.

     

    在Wordpress的初始化过程中wp-blog-header.php这个文件主要完成三个重要的工作.首先,

    wp-blog-header.php引入了wp-load.php这个文件,然后执行wp()函数,最后引入template-loader.php文件. wp-load.php也是位于Wordpress的根目录下.wp()函数是在/wp-includes/functions.php这个文件里面定义的,template-loader.php和functions.php文件都是位于/wp-includes目录下.

     

    我们首先来看一下wp-load.php这个文件以及其引用文件. wp-load.php内容很简单,但是在Wordpress的初始化的过程中起着相当重要的作用.这个文件首先定义了’ABSPATH’常量,这个常量是Wordpress的根目录,在Wordpress初始化的进程中,这个常量会极其广泛使用,并且因着这个常量,Wordpress会很方便的调用位于Wordpress根目录下的配置文件和其他文件. wp-load.php引入wp-config.php文件后,他自身的工作就算是完成了,想必大家对wp-config.php文件会很熟悉,在很早的时候安装Wordpress的过程中可能需要手动编辑该文件,现在是省事不少,Wordpress可以自成生成该文件,并且配置好.事实上,wp-load.php文件会检查wp-config.php文件的位置,绝大部分程序员都会把wp-config.php文件放在Wordpress安装的根目录下,但是可能是处于安全的缘故,有些程序员会把wp-config.php文件放在其他的地方(在路径设置正确的情况下,这种情况是允许的),但如果wp-load.php文件找不到wp-config.php的话,wp-load.php文件会生成一个很友好的错误提示,并且有一个生成wp-config.php文件的按钮.

     

    接着Wordpress就开始有一系列的动作了.

     

    如果wp-load.php成功的加载了wp-config.php文件的话,在他处理加载Wordpress核心文件之前,会根据wp-fonig.php文件里的设置来做相应的配置.wp-config.php文件的注释部分有非常好的说明”该文件有以下的相关配置:MySQL的设置,数据库表的前缀,安全码,Wordpress的语言,和ABSPATH.这个文件的注释部分说的很好,我就不作过多的说明.wp-config文件解析完毕后,Wordpress就获取到所需的各种信息,然后连接数据库,接着是加载内容.我不知道为什么这里再一次的检查了ABSPATH文件,wp-config的最后一行是引入了wp-setting.php文件,Wordpress的绝大部分文件都是wp-setting.php文件引入的.

     

    wp-setting.php一开始就定义了一个’WPINC’常量,这个常量是指向/wp-inludes 文件夹(上面有提到过).Wordpress初始化的大部分预定义函数(或作方法,这里出现的function统一翻译成函数).通过定义该常量,接着wp-setting.php很方便的从 /wp-includes 文件夹里引入了load.php,default-constants.php和version.php这三个文件.事实上,此时Wordpress并没有立即执行这些函数,只是用了php的 require()语句去引入包涵这些函数的文件,以备不时之需. load.php(位于 /wp-include.php文件里面)文件里面定义了以下函数.

     

    wp_unregister_GLOBALS()

          这个函数首先判断register_globals是不是打开了,那么这个是什么用途呢,如果这个变量打开了,则会产生如下效果,可以从url中获取变量值,例如$_GET["var"] = “abc”; 那么打开以后默认$var = “abc”; (可以参考官方文档)

     

    wp_fix_server_vars()

         修改$_SERVER 变量.

     

    wp_check_php_mysql_versions()

    检查php和mysql的版本信息

     

    wp_favicon_request()

       这个文件是当用户输入 ..wordpress/favicon.ico的时候就不必加载Wordpress.

     

    timer_start()

       PHP4 中标准的系统时间捕获方式

     

    timer_stop()

      返回时间,并且当 $display = 1 时输出时间


    wp_debug_mode()

      设置PHP的错误处理方式和 WP 的调试模式。

     

    wp_set_lang_dir()

      设置常量 WP_LANG_DIR,即语言文件所在的文件夹。

     

    require_wp_db()

     加载和数据库有关的Class.

     

     

    wp_set_wpdb_vars()

    设置数据库表前缀和数据库表列的格式说明符。

     

    wp_start_object_cache()

    开启对象缓存。如果wp-content文件夹中存在object-cache.php文件,则使用他作为扩展的对象缓存。

     

    wp_not_installed()

    如果还没有安装 WP,则启动安装程序

     

    wp_get_mu_plugins()

    返回 must-use 插件的文件数组,默认的 must-use 插件在 wp-content/mu-plugins 目录,如果要修改默认目录,可配置 wp-config.php 文件中的 WPMU_PLUGIN_DIR 和 WPMU_PLUGIN_URL 常量。

     

    wp_get_active_and_valid_plugins()

    返回激活的合法的插件文件数组,默认的插件在 wp-content/plugins 目录,如果要修改默认目录,可配置 wp-config.php 文件中的 WP_PLUGIN_DIR 和 WP_PLUGIN_URL 常量。

     

    wp_set_internal_encoding()

    利用 mb_internal_encoding() 函数设置内部编码方式,默认设置为UTF-8.

     

    wp_magic_quotes()

    对 $_GET, $_POST, $_COOKIE, and $_SERVER 添加魔术引用

     

    shutdown_action_hook()

    触发挂载点 do_action( ‘shutdown’ ),在PHP关闭执行之前运行。

     

    wp_clone()

    克隆对象。只有在 PHP5 以上的版本中才能克隆成功,否则返回原来的对象。

     

    is_admin()

    返回当前请求是否在 WP 的后台。

     

    is_blog_admin()

    返回当前请求是否在 WP 的管理面板中(network administration).

     

    is_network_admin()

    Checks whether the current request is for a network admin screen /wp-admin/network/

    is_user_admin()

    Checks whether the current request is for a user admin screen /wp-admin/user/

    is_multisite()

    返回是否支持多站点

    get_current_blog_id()

    返回当前Blog的id

    wp_load_translations_early()

    这里是检查加载异常,是否有些文件被重复加载,如果有,则中止程序.

     

    一旦以上的几个必需函数加载成功后,wp-setting.php引入了 default-constants.php文件,这个文件里面定义了6个函数.这些函数主要是定义了一些Wordpress在接下来的初始化进程中需要用到的常量(这些变量通常wp-config.php中改写).

    1)wp_initial_constants( ),定义 WP 初始化常量 WP_MEMORY_LIMIT,WP_CONTENT_DIR,WP_DEBUG,WP_DEBUG_DISPLAY,WP_DEBUG_LOG,WP_CACHE,MEDIA_TRASH,SHORTINIT 和全局变量 $blog_id(用于设置多博客)。

    2)wp_plugin_directory_constants( ),定义 WP 插件文件夹的常量 WP_CONTENT_URL,WP_PLUGIN_DIR,WP_PLUGIN_URL,WPMU_PLUGIN_DIR,WPMU_PLUGIN_URL。

    3)wp_cookie_constants( ),定义 WP 与cookie相关的常量 COOKIEHASH,USER_COOKIE,PASS_COOKIE,AUTH_COOKIE,SECURE_AUTH_COOKIE,LOGGED_IN_COOKIE,TEST_COOKIE,COOKIEPATH,SITECOOKIEPATH,ADMIN_COOKIE_PATH,PLUGINS_COOKIE_PATH,COOKIE_DOMAIN。

    4)wp_ssl_constants( ),定义 WP 与 ssl 相关的常量 FORCE_SSL_ADMIN,FORCE_SSL_LOGIN。

    5)wp_functionality_constants( ),定义 WP 与功能相关的常量 AUTOSAVE_INTERVAL,EMPTY_TRASH_DAYS,WP_POST_REVISIONS。

    6)wp_templating_constants( ),定义 WP 与模板相关的常量 TEMPLATEPATH,STYLESHEETPATH,WP_DEFAULT_THEME。

     

     

    接着wp-setting.php加载了version.php文件.wp-setting.php定义了以下6个变量.

    $wp_version

    Wordpress的版本信息

    $wp_db_version

    Wordpress数据库版本信息

    $tinymce_version

    Wordpress编辑器(我的最爱^^)版本信息.

    $manifest_version

    Wordpress缓存机制相关的版本信息

    $required_php_version

    Wordpress所需的PHP版本信息(安装需求)

    $required_mysql_version

       Wordpress所需mySQL版本信息. (安装需求)

     (未完待续... ...)

     

     

     

     

     

     

     

     

     

     

     

     

  • 4楼 逗你玩 2013-09-16

    果断收藏

发表回复
功能维护升级中,维护完成完后将再次开放,非常抱歉给您学习造成的不便。