Existe uma maneira fácil no .NET de obter terminações “st”, “nd”, “rd” e “th” para números?[duplicado]
-
09-06-2019 - |
Pergunta
Essa pergunta já tem resposta aqui:
- Existe uma maneira fácil de criar ordinais em C#? 18 respostas
Gostaria de saber se há um método ou string de formato que estou faltando no .NET para converter o seguinte:
1 to 1st
2 to 2nd
3 to 3rd
4 to 4th
11 to 11th
101 to 101st
111 to 111th
Esse link tem um mau exemplo do princípio básico envolvido na escrita de sua própria função, mas estou mais curioso para saber se há uma capacidade embutida que estou faltando.
Solução
A resposta de Scott Hanselman é aceita porque responde diretamente à pergunta.
Para uma solução, no entanto, consulte esta ótima resposta.
Solução
Não, não há capacidade incorporada na biblioteca de classes base do .NET.
Outras dicas
É uma função muito mais simples do que você pensa.Embora possa já existir uma função .NET para isso, a função a seguir (escrita em PHP) faz o trabalho.Não deve ser muito difícil transferi-lo.
function ordinal($num) {
$ones = $num % 10;
$tens = floor($num / 10) % 10;
if ($tens == 1) {
$suff = "th";
} else {
switch ($ones) {
case 1 : $suff = "st"; break;
case 2 : $suff = "nd"; break;
case 3 : $suff = "rd"; break;
default : $suff = "th";
}
}
return $num . $suff;
}
@nickf:Aqui está a função PHP em C#:
public static string Ordinal(int number)
{
string suffix = String.Empty;
int ones = number % 10;
int tens = (int)Math.Floor(number / 10M) % 10;
if (tens == 1)
{
suffix = "th";
}
else
{
switch (ones)
{
case 1:
suffix = "st";
break;
case 2:
suffix = "nd";
break;
case 3:
suffix = "rd";
break;
default:
suffix = "th";
break;
}
}
return String.Format("{0}{1}", number, suffix);
}
Simples, limpo, rápido
private static string GetOrdinalSuffix(int num)
{
if (num.ToString().EndsWith("11")) return "th";
if (num.ToString().EndsWith("12")) return "th";
if (num.ToString().EndsWith("13")) return "th";
if (num.ToString().EndsWith("1")) return "st";
if (num.ToString().EndsWith("2")) return "nd";
if (num.ToString().EndsWith("3")) return "rd";
return "th";
}
Ou melhor ainda, como um método de extensão
public static class IntegerExtensions
{
public static string DisplayWithSuffix(this int num)
{
if (num.ToString().EndsWith("11")) return num.ToString() + "th";
if (num.ToString().EndsWith("12")) return num.ToString() + "th";
if (num.ToString().EndsWith("13")) return num.ToString() + "th";
if (num.ToString().EndsWith("1")) return num.ToString() + "st";
if (num.ToString().EndsWith("2")) return num.ToString() + "nd";
if (num.ToString().EndsWith("3")) return num.ToString() + "rd";
return num.ToString() + "th";
}
}
Agora você pode simplesmente ligar
int a = 1;
a.DisplayWithSuffix();
ou mesmo tão direto quanto
1.DisplayWithSuffix();
Isso já foi abordado, mas não tenho certeza de como vincular a ele.Aqui está o trecho de código:
public static string Ordinal(this int number)
{
var ones = number % 10;
var tens = Math.Floor (number / 10f) % 10;
if (tens == 1)
{
return number + "th";
}
switch (ones)
{
case 1: return number + "st";
case 2: return number + "nd";
case 3: return number + "rd";
default: return number + "th";
}
}
PARA SUA INFORMAÇÃO:Este é um método de extensão.Se a sua versão .NET for inferior a 3.5, basta remover a palavra-chave this
[EDITAR]:Obrigado por apontar que estava incorreto, é isso que você ganha ao copiar/colar o código :)
Aqui está uma versão da função do Microsoft SQL Server:
CREATE FUNCTION [Internal].[GetNumberAsOrdinalString]
(
@num int
)
RETURNS nvarchar(max)
AS
BEGIN
DECLARE @Suffix nvarchar(2);
DECLARE @Ones int;
DECLARE @Tens int;
SET @Ones = @num % 10;
SET @Tens = FLOOR(@num / 10) % 10;
IF @Tens = 1
BEGIN
SET @Suffix = 'th';
END
ELSE
BEGIN
SET @Suffix =
CASE @Ones
WHEN 1 THEN 'st'
WHEN 2 THEN 'nd'
WHEN 3 THEN 'rd'
ELSE 'th'
END
END
RETURN CONVERT(nvarchar(max), @num) + @Suffix;
END
Eu sei que isso não é uma resposta à pergunta do OP, mas como achei útil retirar a função do SQL Server deste thread, aqui está um equivalente Delphi (Pascal):
function OrdinalNumberSuffix(const ANumber: integer): string;
begin
Result := IntToStr(ANumber);
if(((Abs(ANumber) div 10) mod 10) = 1) then // Tens = 1
Result := Result + 'th'
else
case(Abs(ANumber) mod 10) of
1: Result := Result + 'st';
2: Result := Result + 'nd';
3: Result := Result + 'rd';
else
Result := Result + 'th';
end;
end;
..., -1º, 0º faz sentido?
public static string OrdinalSuffix(int ordinal)
{
//Because negatives won't work with modular division as expected:
var abs = Math.Abs(ordinal);
var lastdigit = abs % 10;
return
//Catch 60% of cases (to infinity) in the first conditional:
lastdigit > 3 || lastdigit == 0 || (abs % 100) - lastdigit == 10 ? "th"
: lastdigit == 1 ? "st"
: lastdigit == 2 ? "nd"
: "rd";
}
Outro sabor:
/// <summary>
/// Extension methods for numbers
/// </summary>
public static class NumericExtensions
{
/// <summary>
/// Adds the ordinal indicator to an integer
/// </summary>
/// <param name="number">The number</param>
/// <returns>The formatted number</returns>
public static string ToOrdinalString(this int number)
{
// Numbers in the teens always end with "th"
if((number % 100 > 10 && number % 100 < 20))
return number + "th";
else
{
// Check remainder
switch(number % 10)
{
case 1:
return number + "st";
case 2:
return number + "nd";
case 3:
return number + "rd";
default:
return number + "th";
}
}
}
}
else if (choice=='q')
{
qtr++;
switch (qtr)
{
case(2): strcpy(qtrs,"nd");break;
case(3):
{
strcpy(qtrs,"rd");
cout<<"End of First Half!!!";
cout<<" hteam "<<"["<<hteam<<"] "<<hs;
cout<<" vteam "<<" ["<<vteam;
cout<<"] ";
cout<<vs;dwn=1;yd=10;
if (beginp=='H') team='V';
else team='H';
break;
}
case(4): strcpy(qtrs,"th");break;
Acho que o sufixo ordinal é difícil de conseguir ...você basicamente precisa escrever uma função que use uma opção para testar os números e adicionar o sufixo.
Não há razão para um idioma fornecer isso internamente, especialmente quando é específico do local.
Você pode fazer um pouco melhor do que esse link no que diz respeito à quantidade de código a ser escrito, mas é necessário codificar uma função para isso ...