PHP 関数の部分文字列の置換
str_replace — 部分文字列置換 [str_replace]
混合 str_replace ( 混合 $search 、 混合 $replace 、 混合 $subject [, int &$count ] )
php functionstr_replace: 文字列または配列を返します。文字列または配列はすべて search であり、すべて に置き換えられます。 。
これで、この関数の使用法をいくつか知ることができます。たとえば、str_replace("#", "-", "dizaz#7#final")、str_replace(array('#', '$'), " - "、"dizaz#7$final") など、これらの呼び出しメソッドについて、PHP の内部ではどのように実装されているのでしょうか? [PHP カーネルの深い理解] の観点から、ここで簡単な分析を行います。
テストコード: コードをコピー
コードは次のとおりです:
$object = "dizaz #7#final"; $res = str_replace("#", "-", $object);
上記と同様の文字で始まる場合、先頭の「#」が文字「-」に置き換えられます。
準備作業:
PHP ソース コードをダウンロードします。http://www.php.net からダウンロードします。
コードを読み取るための独自のツールを作成します [VIM CSCOPE を使用します] : Linux ユーザーは、グラフィカル ソース コード表示ツール kscope [google の]
コンパイル ツール [gcc]、デバッグ ツール [gdb] もお勧めします。また、GDB グラフィカル ポート DDD も非常に優れており、PHP ソース コードをコンパイルするには
分析を開始する:
[PHP カーネルの詳細な理解] を読むと、PHP が提供する標準関数が格納されているディレクトリが PHP-SOURCE-DIR/ext/standard ディレクトリにあることがわかります。これは文字列関数であるため、このディレクトリで str_replace 関数によって実装されたファイル文字列を見つけるのは簡単です。 [もちろん CScope でロックするのは簡単です。cs find s str_replace を使用します]
その定義実装を知るためのクエリ:
コードをコピー
コードは次のとおりです:
/* {{{ protomixed str_replace(mixed search,mixed replace,mixed subject [, int &replace_count])
haystack 内のすべての検索を次のように置き換えます。 replace */
PHP_FUNCTION(str_replace) { php_str_replace_common(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1) }
/* }}} */
;次に、関数 php_str_replace_common 関数を表示する必要があります。
コードをコピー
コードは次のとおりです。
/* {{{ php_str_replace_common
*/
static void php_str_replace_common(INTERNAL_FUNCTION_PARAMETERS, int case_sensitivity)
{
/**
* TODO
* typedef struct _zval_struct zval;
* typedef struct _zend_class_entry zend_class_entry
*
* struct _zval_struct {
* zvalue_value 値;
* zend_uint refcount__gc;
* zend_uchar 型;
* zend_uchar is_ref__gc;
* };
*
* typedef Union _zvalue_value {
* long lval;
* ダブル dval;
* struct {
* char *val;
* int len;
* } str;
* ハッシュテーブル *ht;
* zend_object_value obj;
* } zvalue_value;
*
* typedef struct _zend_object {
* zend_class_entry *ce;
* ハッシュテーブル *プロパティ;
* ハッシュテーブル *ガード;
* } zend_object;
*
*/
zval **件名、**検索、**置換、**件名エントリ、**zcount = NULL;
zval *結果;
char *string_key;
uint string_key_len;
ulong num_key;
int カウント = 0;
int argc = ZEND_NUM_ARGS();
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ZZZ|Z", &search, &replace, &subject, &zcount) == FAILURE) {
return;
}
SEPARATE_ZVAL(検索);
SEPARATE_ZVAL(置換);
SEPARATE_ZVAL(件名);
/* 文字列を扱っていることを確認し、置換を実行します。 */
if (Z_TYPE_PP(search) != IS_ARRAY) {
....//代码省滤
} else { /* サブジェクトが配列でない場合 */
php_str_replace_in_subject(*検索、*置換、件名、戻り値、大文字と小文字の区別、(argc > 3) &count : NULL);
}
if (argc > 3) {
zval_dtor(*zcount);
ZVAL_LONG(*zcount, count);
}
}
/* }}} */
继续跟追跡php_str_replace_in_subject
复制代代コード如下:
/* {{{ php_str_replace_in_subject
*/
static void php_str_replace_in_subject(zval *search, zval *replace, zval **subject, zval *result, int case_sensitivity, int *replace_count)
{
zval **search_entry,
**replace_entry = NULL,
temp_result;
char *replace_value = NULL;
int replace_len = 0;
/* 文字列を扱っていることを確認してください。 */
convert_to_string_ex(件名);
Z_TYPE_P(結果) = IS_STRING;
if (Z_STRLEN_PP(件名) == 0) {
ZVAL_STRINGL(result, "", 0, 1);
戻る;
}
/* 検索が配列の場合 */
if (Z_TYPE_P(search) == IS_ARRAY) {
...//不走这步
} else {
if (Z_STRLEN_P(search) == 1) { // 例中にのみ”#”があるため、この操作を実行します。
php_char_to_str_ex(Z_STRVAL_PP(subject),//subject の値、また dizaz#7#final
Z_STRLEN_PP(subject), //subject の長さを取得します
Z_STRVAL_P(search)[0], //ただ 1 つある”#” ため、最初の 1 文字だけが必要です
Z_STRVAL_P(replace), / //代替希望の文字符,现在是“-”
Z_STRLEN_P(replace), //目标文字の長さ,现在である1
result, //代替换结結果
case_sensitivity, //大小写しが敏感かどうか,默认是1
replace_count); //置換次数
} else if (Z_STRLEN_P(search) > 1) {
Z_STRVAL_P(result) = php_str_to_str_ex(Z_STRVAL_PP(件名), Z_STRLEN_PP(件名),
Z_STRVAL_P(検索), Z_STRLEN_P( search)、
Z_STRVAL_P(置換)、Z_STRLEN_P(置換)、&Z_STRLEN_P(結果)、case_sensitivity、replace_count);
} else {
MAKE_COPY_ZVAL(件名, 結果);
}
}
}
に到達したため、私たちの目標は php_char_to_str_ex 関数を最終的に決定しました。この関数の数を分析する必要があるだけで OK です。その結果は次のとおりです。
复制代码代码如下:
/* {{{ php_char_to_str_ex
*/
PHPAPI int php_char_to_str_ex(char *str, uint len, char from, char *to, int to_len, zval *result, int case_sensitivity, int * replace_count)
{
int char_count = 0;
int selected = 0;
char *source, *target, *tmp, *source_end=str len, *tmp_end = NULL; (case_sensitivity) { //今 case_sensitivity = 1
char *p = str, *e = p len;
//何回置換する必要があるかを計算します
while ((p = memchr(p , from, ( e - p))) {
char_count ;
p ;
}
} else {
for (source = str;source
char_count ;
}
}
}
if (char_count == 0 && case_sensitivity) {
ZVAL_STRINGL(result, str, len, 1);
return 0;
}
//置換後の長さを計算し、結果に格納します。
Z_STRLEN_P(result) = len (char_count * (to_len - 1));
//メモリを適用し、置換されたデータを保存します
Z_STRVAL_P(result) = target = safety_emalloc(char_count, to_len, len 1) );
//設定結果は文字列です
Z_TYPE_P(result) = IS_STRING;
//ターゲットと結果の値は同じメモリブロックを指すため、ターゲットのみが必要です処理される
if ( case_sensitivity) {
char *p = str, *e = p len, *s = str;
while ((p = memchr(p, from, (e - p)) )) { //どの文字が出現するかを判定 #
memcpy(target, s, (p - s)) // # より前のデータをターゲットにコピー
target =
memcpy( target, to, to_len); // ターゲット文字を target にコピーします [もちろん、このときのターゲットは target p-s を開始します]
target =
s = p; (replace_count) {
*replace_count = 1; //置換の数を設定します
}
}
//後でさらにある場合は、ターゲットに追加し続けます。ターゲットが指すブロックのデータが置き換えられました。
if (s < e) {
memcpy(target, s, (e - s));
target = e - s;
} else {
for (source = str;source
replaced = 1;
if (replace_count) {
* replace_count = 1;
}
for (tmp = to, tmp_end = tmp to_len; tmp < tmp_end; tmp ) {
*target = *tmp>} } else {
*ターゲット = *ソース;
}
}
}
リターン
}
/* }}} */
上でコメントしたように、これで文字の文字列への置換が完了します。戻り方や詳しい処理については、PHPの実行処理を相対的に理解している必要があります。
もちろん、gdb を使用して php_char_to_str_ex 関数にブレークポイントを設定し、その詳細な実行プロセスを理解することもできます。
次の記事は文字列を文字列に置き換える解析についてです。
概要:
結果は zval に保存されます。
置換の実装は非常に巧妙であり、学ぶことができます。
引き続きソース コードを表示し、さらに執筆スキルと設計スキルを学ぶ必要があります。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック











JWTは、JSONに基づくオープン標準であり、主にアイデンティティ認証と情報交換のために、当事者間で情報を安全に送信するために使用されます。 1。JWTは、ヘッダー、ペイロード、署名の3つの部分で構成されています。 2。JWTの実用的な原則には、JWTの生成、JWTの検証、ペイロードの解析という3つのステップが含まれます。 3. PHPでの認証にJWTを使用する場合、JWTを生成および検証でき、ユーザーの役割と許可情報を高度な使用に含めることができます。 4.一般的なエラーには、署名検証障害、トークンの有効期限、およびペイロードが大きくなります。デバッグスキルには、デバッグツールの使用とロギングが含まれます。 5.パフォーマンスの最適化とベストプラクティスには、適切な署名アルゴリズムの使用、有効期間を合理的に設定することが含まれます。

PHPの魔法の方法は何ですか? PHPの魔法の方法には次のものが含まれます。1。\ _ \ _コンストラクト、オブジェクトの初期化に使用されます。 2。\ _ \ _リソースのクリーンアップに使用される破壊。 3。\ _ \ _呼び出し、存在しないメソッド呼び出しを処理します。 4。\ _ \ _ get、dynamic属性アクセスを実装します。 5。\ _ \ _セット、動的属性設定を実装します。これらの方法は、特定の状況で自動的に呼び出され、コードの柔軟性と効率を向上させます。

静的結合(静的::) PHPで後期静的結合(LSB)を実装し、クラスを定義するのではなく、静的コンテキストで呼び出しクラスを参照できるようにします。 1)解析プロセスは実行時に実行されます。2)継承関係のコールクラスを検索します。3)パフォーマンスオーバーヘッドをもたらす可能性があります。

PHPとPythonにはそれぞれ独自の利点があり、プロジェクトの要件に従って選択します。 1.PHPは、特にWebサイトの迅速な開発とメンテナンスに適しています。 2。Pythonは、データサイエンス、機械学習、人工知能に適しており、簡潔な構文を備えており、初心者に適しています。

PHPは、電子商取引、コンテンツ管理システム、API開発で広く使用されています。 1)eコマース:ショッピングカート機能と支払い処理に使用。 2)コンテンツ管理システム:動的コンテンツの生成とユーザー管理に使用されます。 3)API開発:RESTFUL API開発とAPIセキュリティに使用されます。パフォーマンスの最適化とベストプラクティスを通じて、PHPアプリケーションの効率と保守性が向上します。

PHPは、サーバー側で広く使用されているスクリプト言語で、特にWeb開発に適しています。 1.PHPは、HTMLを埋め込み、HTTP要求と応答を処理し、さまざまなデータベースをサポートできます。 2.PHPは、ダイナミックWebコンテンツ、プロセスフォームデータ、アクセスデータベースなどを生成するために使用され、強力なコミュニティサポートとオープンソースリソースを備えています。 3。PHPは解釈された言語であり、実行プロセスには語彙分析、文法分析、編集、実行が含まれます。 4.PHPは、ユーザー登録システムなどの高度なアプリケーションについてMySQLと組み合わせることができます。 5。PHPをデバッグするときは、error_reporting()やvar_dump()などの関数を使用できます。 6. PHPコードを最適化して、キャッシュメカニズムを使用し、データベースクエリを最適化し、組み込み関数を使用します。 7

PHPは依然として動的であり、現代のプログラミングの分野で重要な位置を占めています。 1)PHPのシンプルさと強力なコミュニティサポートにより、Web開発で広く使用されています。 2)その柔軟性と安定性により、Webフォーム、データベース操作、ファイル処理の処理において顕著になります。 3)PHPは、初心者や経験豊富な開発者に適した、常に進化し、最適化しています。

PHP and Python each have their own advantages, and the choice should be based on project requirements. 1.PHPは、シンプルな構文と高い実行効率を備えたWeb開発に適しています。 2。Pythonは、簡潔な構文とリッチライブラリを備えたデータサイエンスと機械学習に適しています。
