次のように数字を表示したい
コード内の各番号の正しい順序の接尾辞(st、nd、rd、またはth)を見つけるにはどうすればよいですか?
$ends = array('th','st','nd','rd','th','th','th','th','th','th');
if (($number %100) >= 11 && ($number%100) <= 13)
$abbreviation = $number. 'th';
else
$abbreviation = $number. $ends[$number % 10];
どこ $number
は書きたい数字です。任意の自然数で動作します。
機能として:
function ordinal($number) {
$ends = array('th','st','nd','rd','th','th','th','th','th','th');
if ((($number % 100) >= 11) && (($number%100) <= 13))
return $number. 'th';
else
return $number. $ends[$number % 10];
}
//Example Usage
echo ordinal(100);
PHPにはこのための機能が組み込まれています 。国際化も処理します!
$locale = 'en_US';
$nf = new NumberFormatter($locale, NumberFormatter::ORDINAL);
echo $nf->format($number);
この機能はPHP 5.3.0以降でのみ利用可能です。
これは、PHPの組み込みの日付/時刻関数で同様の機能を活用することにより、1行で実現できます。私は謙虚に提出します:
解決策:
_function ordinalSuffix( $n )
{
return date('S',mktime(1,1,1,1,( (($n>=10)+($n>=20)+($n==0))*10 + $n%10) ));
}
_
詳細な説明:
組み込みの date()
関数には、n日目の計算を処理するための接尾辞ロジックがあります。 S
がフォーマット文字列で指定されると、サフィックスが返されます。
_date( 'S' , ? );
_
date()
にはタイムスタンプが必要なので(上記の_?
_の場合)、整数_$n
_をday
パラメーターとして mktime()
そして、hour
、minute
、second
、およびmonth
に_1
_のダミー値を使用します。
_date( 'S' , mktime( 1 , 1 , 1 , 1 , $n ) );
_
これは実際に月の範囲外の値(つまり_$n > 31
_)で正常に失敗しますが、29で_$n
_を制限するためにいくつかの簡単なインラインロジックを追加できます。
_date( 'S', mktime( 1, 1, 1, 1, ( (($n>=10)+($n>=20))*10 + $n%10) ));
_
唯一の正の値(2017年5月)これは_$n == 0
_で失敗しますが、その特別な場合に10を追加することで簡単に修正できます:
_date( 'S', mktime( 1, 1, 1, 1, ( (($n>=10)+($n>=20)+($n==0))*10 + $n%10) ));
_
2017年5月更新
@donatJで観察されたように、_>=20
_チェックは常にtrueを返すため、上記は100を超えると失敗します(例: "111st")。これらの世紀ごとにリセットするには、比較にフィルターを追加します。
_date( 'S', mktime( 1, 1, 1, 1, ( (($n>=10)+($n%100>=20)+($n==0))*10 + $n%10) ));
_
便利なように関数にラップするだけです。
ワンライナーは次のとおりです。
$a = <yournumber>;
echo $a.substr(date('jS', mktime(0,0,0,1,($a%10==0?9:($a%100>20?$a%10:$a%100)),2000)),-2);
おそらく最短のソリューション。もちろん、関数でラップできます:
function ordinal($a) {
// return English ordinal number
return $a.substr(date('jS', mktime(0,0,0,1,($a%10==0?9:($a%100>20?$a%10:$a%100)),2000)),-2);
}
よろしく、ポール
EDIT1:11〜13のコードの修正。
EDIT2:111、211、...のコードの修正.
EDIT3:10の倍数でも正しく動作するようになりました。
from http://www.phpro.org/examples/Ordinal-Suffix.html
<?php
/**
*
* @return number with ordinal suffix
*
* @param int $number
*
* @param int $ss Turn super script on/off
*
* @return string
*
*/
function ordinalSuffix($number, $ss=0)
{
/*** check for 11, 12, 13 ***/
if ($number % 100 > 10 && $number %100 < 14)
{
$os = 'th';
}
/*** check if number is zero ***/
elseif($number == 0)
{
$os = '';
}
else
{
/*** get the last digit ***/
$last = substr($number, -1, 1);
switch($last)
{
case "1":
$os = 'st';
break;
case "2":
$os = 'nd';
break;
case "3":
$os = 'rd';
break;
default:
$os = 'th';
}
}
/*** add super script ***/
$os = $ss==0 ? $os : '<sup>'.$os.'</sup>';
/*** return ***/
return $number.$os;
}
?>
これをPHP4向けに書きました。それはうまく機能していて、かなり経済的です。
function getOrdinalSuffix($number) {
$number = abs($number) % 100;
$lastChar = substr($number, -1, 1);
switch ($lastChar) {
case '1' : return ($number == '11') ? 'th' : 'st';
case '2' : return ($number == '12') ? 'th' : 'nd';
case '3' : return ($number == '13') ? 'th' : 'rd';
}
return 'th';
}
シンプルで簡単な回答は次のとおりです。
$Day = 3;
echo date("S", mktime(0, 0, 0, 0, $Day, 0));
//OUTPUT - rd
特定の機能を適用するだけです。
function addOrdinalNumberSuffix($num) {
if (!in_array(($num % 100),array(11,12,13))){
switch ($num % 10) {
// Handle 1st, 2nd, 3rd
case 1: return $num.'st';
case 2: return $num.'nd';
case 3: return $num.'rd';
}
}
return $num.'th';
}
一般的には、それを使用してecho get_placing_string(100);を呼び出すことができます
<?php
function get_placing_string($placing){
$i=intval($placing%10);
$place=substr($placing,-2); //For 11,12,13 places
if($i==1 && $place!='11'){
return $placing.'st';
}
else if($i==2 && $place!='12'){
return $placing.'nd';
}
else if($i==3 && $place!='13'){
return $placing.'rd';
}
return $placing.'th';
}
?>
PHPのdate();
関数に依存しない関数を作成しましたが、必要ではありませんが、現在可能な限りコンパクトで短いものにしました。
コード:(合計121バイト)
_function ordinal($i) { // PHP 5.2 and later
return($i.(($j=abs($i)%100)>10&&$j<14?'th':(($j%=10)>0&&$j<4?['st', 'nd', 'rd'][$j-1]:'th')));
}
_
以下のよりコンパクトなコード。
次のように動作します:
_printf("The %s hour.\n", ordinal(0)); // The 0th hour.
printf("The %s ossicle.\n", ordinal(1)); // The 1st ossicle.
printf("The %s cat.\n", ordinal(12)); // The 12th cat.
printf("The %s item.\n", ordinal(-23)); // The -23rd item.
_
この機能について知っておくべきこと:
floor($i)
、round($i)
、またはceil($i)
を使用します)。format_number($i)
を追加して、コンマ区切りの整数を取得することもできます(数千、数百万などを表示する場合)。$i
_を削除するだけで済みます。この関数は、PHP 5.2は純粋に短い配列構文のために2006年11月にリリースされました。これより前のバージョンがある場合は、10年近く古いのでアップグレードしてください! 、インライン_['st', 'nd', 'rd']
_をarray('st', 'nd', 'rd');
を含む一時変数に置き換えるだけです。
同じ関数(入力を返すことなく)、ただし、理解を深めるための短い関数の分解図:
_function ordinal($i) {
$j = abs($i); // make negatives into positives
$j = $j%100; // modulo 100; deal only with ones and tens; 0 through 99
if($j>10 && $j<14) // if $j is over 10, but below 14 (so we deal with 11 to 13)
return('th'); // always return 'th' for 11th, 13th, 62912th, etc.
$j = $j%10; // modulo 10; deal only with ones; 0 through 9
if($j==1) // 1st, 21st, 31st, 971st
return('st');
if($j==2) // 2nd, 22nd, 32nd, 582nd
return('nd'); //
if($j==3) // 3rd, 23rd, 33rd, 253rd
return('rd');
return('th'); // everything else will suffixed with 'th' including 0th
}
_
コードの更新:
これは、全体で14バイト短い(合計107バイト)変更されたバージョンです。
_function ordinal($i) {
return $i.(($j=abs($i)%100)>10&&$j<14?'th':@['th','st','nd','rd'][$j%10]?:'th');
}
_
または、可能な限り短くして25バイト短くします(合計96バイト):
_function o($i){return $i.(($j=abs($i)%100)>10&&$j<14?'th':@['th','st','nd','rd'][$j%10]?:'th');}
_
この最後の関数では、単にo(121);
を呼び出すだけで、リストした他の関数とまったく同じように実行できます。
コード更新#2:
Ben そして一緒に作業して、38バイト(合計83バイト)削減しました。
_function o($i){return$i.@(($j=abs($i)%100)>10&&$j<14?th:[th,st,nd,rd][$j%10]?:th);}
_
これよりも短くなるとは思わない!しかし、間違っていると証明されることをいとわない。 :)
皆さんお楽しみください。
function ordinal($number){
$last=substr($number,-1);
if( $last>3 || $last==0 || ( $number >= 11 && $number <= 19 ) ){
$ext='th';
}else if( $last==3 ){
$ext='rd';
}else if( $last==2 ){
$ext='nd';
}else{
$ext='st';
}
return $number.$ext;
}
Mktime()を使用せず、pecl intlを必要としない、月の日付(31まで)のさらに短いバージョン:
function ordinal($n) {
return (new DateTime('Jan '.$n))->format('jS');
}
または手続き的に:
echo date_format(date_create('Jan '.$n), 'jS');
私が選んだデフォルトの月(1月)には31日あるため、これはもちろん機能します。
興味深いことに、2月(または31日のない別の月)で試してみると、終了前に再起動します。
...clip...
31st
1st
2nd
3rd
ループ内の日付指定子t
を使用して、今月の日までカウントできます:月の日数。
Lacopoの答えをさらに単純化するために、substr()関数を使用して数値の最後の桁を取得し、それを$ ends配列のインデックスとして使用する次の関数を使用できます。
function get_ordinal($number)
{
$ends=array('th','st','nd','rd','th','th','th','th','th','th');
$last_digit=substr($number, -1, 1);
return $number.$ends[$last_digit];
}
編集
11、12、13で終わる番号については、上記の機能が失敗しているようですが、回答の投稿中に気付きませんでした。だからラコポの答えを使ってください。
日付関数を使用した別の非常に短いバージョンがあります。任意の数(月の日付に制約されない)で機能し、* 11th * 12th * 13thが* 1st * 2nd * 3rd形式に従っていないことを考慮します。
function getOrdinal($n)
{
return $n . date_format(date_create('Jan ' . ($n % 100 < 20 ? $n % 20 : $n % 10)), 'S');
}
PHP.net で答えを見つけました
<?php
function ordinal($num)
{
// Special case "teenth"
if ( ($num / 10) % 10 != 1 )
{
// Handle 1st, 2nd, 3rd
switch( $num % 10 )
{
case 1: return $num . 'st';
case 2: return $num . 'nd';
case 3: return $num . 'rd';
}
}
// Everything else is "nth"
return $num . 'th';
}
?>