この関数が文字列の最初の出現を取得することを理解しています。
しかし、私が欲しいのは2回目の発生です。
それをどうやってやるの?
オプションの3番目のパラメーターとして検索開始のオフセットを指定し、最初に見つかった場所に検索対象の長さを追加して検索を開始することで、検索を開始して計算する必要があります。
$pos1 = strpos($haystack, $needle);
$pos2 = strpos($haystack, $needle, $pos1 + strlen($needle));
私はこの質問が古いことを知っていますが、サブストリングのX番目の出現を取得するために書いた関数は次のとおりです。
/**
* Find the position of the Xth occurrence of a substring in a string
* @param $haystack
* @param $needle
* @param $number integer > 0
* @return int
*/
function strposX($haystack, $needle, $number){
if($number == '1'){
return strpos($haystack, $needle);
}elseif($number > '1'){
return strpos($haystack, $needle, strposX($haystack, $needle, $number - 1) + strlen($needle));
}else{
return error_log('Error: Value for parameter $number is out of range');
}
}
Smokey_Budの再帰関数により、スクリプトが大幅に遅くなりました。正規表現を使用すると、この場合ははるかに高速になります(発生を検出するため)。
function strposX($haystack, $needle, $number)
{
// decode utf8 because of this behaviour: https://bugs.php.net/bug.php?id=37391
preg_match_all("/$needle/", utf8_decode($haystack), $matches, PREG_OFFSET_CAPTURE);
return $matches[0][$number-1][1];
}
// get position of second 'wide'
$pos = strposX('Hello wide wide world', 'wide', 2);
文字列の2番目の出現を見つけるには、strposに前のオフセットを追加することにより、「offset」パラメーターとともに strpos を使用できます。
$source = "Hello world, how you doing world...world ?";
$find = "world";
$offset = 0;
$findLength = strlen($find);
$occurrence = 0;
while (($offset = strpos($source, $find, $offset))!== false) {
if ($occurrence ++) {
break;
}
$offset += $findLength;
}
echo $offset;
配列にオフセットを保存することにより、すべての出現を見つけることができます
while (($offset = strpos($source, $find, $offset))!== false) {
$occurrences[] = $offset;
$offset += $findLength;
}
var_export($occurrences);
または、$ occurrenceを照合することで特定の発生を取得できます
//find 3rd occurrence
if ($occurrence ++ == 2) {
break;
}
これを試すことができますが、私は試してはいませんが
$pos = strpos($haystack, $needle, strpos($haystack, $needle)+strlen($needle));
シンプルは美しい
function strposX($haystack, $needle, $n = 0)
{
$offset = 0;
for ($i = 0; $i < $n; $i++) {
$pos = strpos($haystack, $needle, $offset);
if ($pos !== false) {
$offset = $pos + strlen($needle);
} else {
return false;
}
}
return $offset;
}
$offset = strposX($result, "\n", $n);
if ($offset === false) {
$offset = strlen($result) - 1;
}
古い質問ですが、誰かが文字列のENDから出現を見つける方法を探している場合(たとえば、最後から3番目のドット)、次の関数が機能します(エンコードを混乱させないためにoncodes関数を使用したくない)
$str = "NooooYesYesNo";
function find_occurence_from_end($haystack, $needle, $num) {
for ($i=1; $i <=$num ; $i++) {
# first loop return position of needle
if($i == 1) {
$pos = strrpos($haystack, $needle);
}
# subsequent loops trim haystack to pos and return needle's new position
if($i != 1) {
$haystack = substr($haystack, 0, $pos);
$pos = strrpos($haystack, $needle);
}
}
return $pos;
}
$pos = find_occurence_from_end($str, "Yes", 2);
// 5
とても簡単です。基本的に、針の出現を検出するたびに、文字列をその位置に「トリミング」します。そのため、毎回最新の位置を返しながら、トリミングを続けます。
_function substr_Index( $str, $nth ){
$str2 = '';
$posTotal = 0;
for($i=0; $i < $nth; $i++){
if($str2 != ''){
$str = $str2;
}
$pos = strpos($str, ':');
$str2 = substr($str, $pos+1);
$posTotal += $pos+1;
}
return $posTotal-1;
}
echo substr($mystring, substr_Index( $mystring , 2) );
_
関数は、n番目の区切り文字の位置を返します。
_substr_Index
_の2番目のパラメーターは0より大きくなければなりません。
2番目のオカレンスを見つけるには、substr_Index( $mystring , 2)
を使用します
文字が2回以上出現するかどうかを見つけるために働いたので、それらをstrlenすることで2回出現するexが存在することがわかりました($ matchesをまったく使用しないでください):
$string = '1234|6|#red';
if(strlen(preg_match_all('/|/', $string,$matches, PREG_OFFSET_CAPTURE)) ==2){
echo 'i have 2 occurence of char: |';
}
簡単に、それをするだけです:
$i = $pos = 0;
do {
$pos = strpos( $string, $needle, $pos+1 );
} while( $i++ < $nth);
あなたの状況の$ nthは2です。
古い質問ですが、explode
の使用はどうですか?
$corpus = "how many words are there in a dictionary? I'm not going to count them Word by Word...";
$looking_for = 'Word';
$instance = 2;
$parts = explode($looking_for, $corpus, $instance + 1);
array_pop($parts);
$position = strlen(implode($looking_for, $parts));