PHP 8.1 では、多くのコア関数に引数として null
を渡すことが非推奨になりました。私の主な問題は、htmlspecialchars(php)
や trim(php)
のような関数に関するもので、null
は silently で null に変換されなくなります。文字列。
多くのコードを使用せずにこの問題を解決するために、元の組み込み関数の名前を変更し、入力を null
から (空の) 文字列に変換するラッパーに置き換えてみました。
このアプローチの主な問題は、関数 rename_function(PECL apd)
が機能しなくなり、最後に更新されたのが 2004 1 であることです。
関数が呼び出されるたびに null チェックを記述してすべてのコードが 2 倍大きくなるのを避けるために、組み込み関数を何らかの形で書き直す必要があります。
他に思いつく唯一の解決策は、カスタム関数を使用することですが、それでも、私が持っているすべてのコードとサードパーティのライブラリを調べる必要があります。
PHP 8.1 では、組み込み関数に渡されたときに、null が暗黙的に空の文字列に変換されなくなりました。
私は、そのような「問題」がどのように見られ、解決されるかについて、別の絵を描いていると思います (補足として、既存の回答は私の支持を得ています)。これは、概説したアプローチの正誤を減じるものではなく、相互に有益であることが期待される追加の視点にすぎません。すべてのプロジェクトは異なります。
与えられた前提:
さて、これは (最初は) レポートの問題であるように私には思えます。 #E_DEPRECATED
これの利点は、(コードだけでなく) コードに非推奨の通知が付いていることを認識できることです。レポートを報告しないことにより。
は確かに有効です。
一方、非推奨通知を抑制すると、非推奨通知が表示されなくなる可能性があります。非推奨通知によってコード ベースを失った場合、技術的にはまだ簡単に損失から回復できる可能性があります (再度、非推奨通知を報告します)。しかし、変更が延長されると、圧倒的なノイズ (E_TOO_MUCH_NOISE) になる可能性があります。このケース では、私の考えは、通常、非推奨通知を抑制せず、関数呼び出しを「沈黙」させることです。簡単ですが、良い意味でも悪い意味でも愚かです: リーリー
これはもちろん、標準のテキスト ツールを使用してコード ベースに適用できる操作です。また、@
あなたが PHP 開発者で、このようなコードをエディターで見ている場合 (ケース #2 ~ #4)、すぐに抑制演算子が過去に使用された場所も表示されます:
リーリー叫び声を上げます 4 つのケースすべてにおいて、少なくとも、あなたの興味を引くでしょう。注意してください ($混合
黙っていてくれてありがとう。実行時ではなく、)。
1 だけで、これらの場所に 悲鳴を上げさせます。
E_DEPRECATEDを報告しないことで沈黙を保つ最初の方法とは異なり、この方法では情報が簡単に失われる可能性がありますが、情報はすべての
それは騒音問題の解決に役立ちますか?ここで働くことをやめたら、それはまったくうまくいきません。次に、コードを
#@@
シンボルを使用することで保持されます。- 記号でペイントし、それ以上のアクションは行わないことにします。これにより、最初の解決策 (非推奨メッセージが報告されない) を使用して、コードに触れることなく解決を完了できます。
それでは、その利点は何でしょうか?コードは p> サイレントに実行されますが、PHP は依然として診断メッセージを提供します。つまり、(コードの実行中に) PHP エラー ハンドラーをリスナーとして登録できるようになりました。
コード レベルでのみ、@
シンボルは (通常は) コード内でも簡単に見つけられるため、これらの場所を確認するのは簡単です。パート 2
はたとえば、テンプレート コード (出力) では、具象型は問題ではないため、文字列に変換することが推奨される解決策である可能性が最も高くなります。 リーリー テンプレート (出力) は安定したままです。
しかし、実際の入力処理に関しては、非推奨の通知によって、デフォルト値の欠落 (物事が過度に複雑になる)、不明瞭な値の処理 (null と null、文字列、ブール値の区別など) など、修正する価値のある実際の潜在的な欠陥が明らかになる可能性があります。数値)、PHP のオブジェクトを含む配列)、または一般的な
$mixed
を使用します。このような
はもう必要ありません - 削除できます (trim($mixed)
は、長年にわたって忘れられ、アップグレードされていないセキュリティである可能性があります (より優れたセキュリティが利用可能です)。このようなコードの場合、$mixed
が実際には$string
であることをすでに望んでおり、要求していると確信しています。 trim ()# を使用します。 ##。理由は非常に単純で、少なくとも次の 2 つのことが直接思い浮かびます。a)
trim()お気に入りの- 修正の 1 つ: コードを削除してください!) - または -
b) は 文字列の処理を行っていますが、文字列以外のものを存在させたくないので問題が発生します。問題は、一般的にショットガンのアプローチでは機能しないことです (ギースカンネ、誰か?)。 李>
-
$mixed を使用したパッチ適用は完全に有効ですか? ''
元の使用法が文字列または
null
の場合は のみ。リーリー
しかし、そうでない場合、たとえば 42 のような数値は、非推奨メッセージの代わりに TypeError をスローします。これにより、実行中のコードと実行されていないコードが区別されます。つまり、場所の確認、可能であればさらなるクラスタリング、より特殊な修正の適用など、ここで維持する必要があることはさらにあります。欠落しているテストやアサーションが明らかになったり、アプリケーション フロー全体を安定させるのに時間がかかる場合などが考えられます。
この場合、コードの移行を完了し、クラスタ化して、null マージ演算子を処理し、実際の修正に必要な適切な事務処理を行います。 Null 合体演算子を使用して非明白なエラーの抑制を完了し、
@抑制演算子を削除した後、修復計画でこの情報が取得されないと、この情報が失われる可能性があります。
これらの分野でより知識があるように見えるときに、頭をかいたり目をこすったりすることに私は驚きません。そして、これらのエラーは PHP 8.1 バージョンが原因ではなく、バージョンの変更によって (再び) エラーが発生するだけであり、場合によっては、PHP のバージョンを維持することによって完全なエラーのクラスターが副次的に発生することさえある、と自分に言い聞かせます。
######カンニングペーパー######
(string)$mixed
- 以前の動作$mixed ?? ''
-null- でのみ
を抑制します
- #@
- 空 ($mixed)? '' : xxx($mixed)
TypeError
エラー- 完全なエラー抑制。該当する場合は、コードベースを文書化する必要があります。
@@
- これが起こった場合、これは調べてみるのが興味深い場所かもしれません。
- 典型的な空の麻痺/混合混乱であるガベージを取り除き、コード ベースを大幅に簡素化する機会を利用してクラスターを探します。スカラー型 (PHP 7) に移行し、必要に応じて PHP の「クラシック」および「厳密な」型付けを使用して、徹底的な厳密な型付けを導入します。 PHP 7.0 アサーションと PHP 8.1 非推奨メッセージは、これを適切にサポートしています。
エラーハンドラ
エラー処理には魔法はありません。エラー処理は PHP.net で文書化された標準 (Example #1) であり、エラー イベントのオブザーバーとして機能し、エラー イベントを区別できます。抑制されたエラーと抑制されていないエラーは、
リーリーerror_reporting(php)
/error_reporting(php-ini)
を介して、少なくとも通常必要なレベルまで処理されます。 , 区別する必要がある場合 (運用環境では、通常、E_DEPRECATED
は report の一部ではありません)。このサンプル ハンドラーは、非推奨イベントおよびE_ALL
についても含め、報告されたすべてのエラーをスローするため、@
抑制演算子はスローしないようにする必要があります。同様のエラー ハンドラーは、報告される非推奨のコードを含む、3v4l.org の拡張例 にあります。
E_USER_DEPRECATED
技術的には、エラー抑制演算子は、上で概説した
E_DEPRECATED
と同じ方法でE_USER_DEPRECATED
と組み合わせて使用できます。ただし、これを制御する機能はあまりなく、 および すでにプロジェクトの依存関係にあるサードパーティのコードによってすでに使用されている可能性があります。次のようなコードは珍しいことではありません:
リーリーこれはまったく同じことを行います。非推奨イベントを発行しますが、それらを PHP レポートから除外します。これらを購読すると、ノイズに溺れてしまう可能性があります。
E_DEPRECATED
を使用すると、いつでも PHP から直接「優れたオリジナル」を取得できます。@
アプローチを検討し、それについてコメントすると、すぐに赤/黒のフラグを立てます (正しく!)。赤ちゃんをお風呂の水と一緒に捨てるのは簡単です@
抑制演算子。私の答えでは、その目的は非推奨通知を抑制することだけです しかし、 これを使用した結果、 すべての 診断メッセージとエラー (一部の PHP バージョンでは致命的なメッセージやエラーも) が抑制されます。そのため、PHP はそれ以上の診断を行わずに 255 で終了します。慎重に handle を処理してください。この演算子は強力です。コードベースでの使用状況を追跡し、ベースライン/期待を満たしているかどうかを常に確認します。法的な状況に備えて、サイレンサーの使用を検討してください。コードを移植/保守する場合は、最初にコードにマークを付けます。バッチの編集が完了したら、再度削除します。まず、覚えておくべき 2 つのこと:
htmlspecialchars($something)
は、htmlspecialchars($something ?? '')
次に、いくつかのオプションがあります:
?? ''
を追加したり、論理エラーを修正したりできる場合がありますが、とにかく null は必要ありません。などのカスタム関数を作成し、コード内で直接検索して置換します。
use function nullableoverride\htmlspecialchars;
を追加すると、その関数がビルドされた関数の代わりに使用されます。 -in関数。ただし、これはすべてのファイルに追加する必要があるため、自動的に追加するツールが必要になる場合があります。スキルによっては、正規表現の検索と置換を使用して、単純なケースに
?? ''