sql, чтобы выделить строку имени человека и вывести инициалы - PullRequest
2 голосов
/ 01 февраля 2011

как мне заставить SQL взять строчку и вернуть первую букву каждого слова, переданного в него.

Я хочу использовать этот UDF для генерации инициалов для имен людей, которые у меня есть в БД.

имена могут быть 2 (fname, lname) или 3 (... mname) слова

Я использую sql2005

Ответы [ 5 ]

2 голосов
/ 01 февраля 2011

Это должно работать для комбинаций «Имя Фамилия» и «Имя Фамилия Имя».

DECLARE @name AS NVARCHAR(50) 
SET @name = 'Firstname Middle Lastname' 


SELECT SUBSTRING(@name, 1, 1) +     --First initial
    SUBSTRING(@name, CHARINDEX(' ', @name) + 1, 1) +    --Middle/Last initial
    CASE WHEN 0 <>  CHARINDEX(' ', @name, CHARINDEX(' ', @name) + 1) -- More than two words 
        THEN SUBSTRING(@name, CHARINDEX(' ', @name, CHARINDEX(' ', @name) + 1) + 1, 1)  --Last initial
    ELSE '' --Have to add empty string to avoid NULLing entire result
    END

Конечно, если у пользователей есть пробел в одном из их имен, по какой-то причине у вас возникнет проблема с разбором этого, но я подозреваю, что так будет и в любом случае, когдане хранить ваши имена в отдельных полях.

1 голос
/ 01 февраля 2011
CREATE FUNCTION dbo.GetFirstLetter ( @Array VARCHAR(1000), @separator VARCHAR(10)) 
RETURNS @resultTable TABLE 
    (parseValue VARCHAR(100))
AS
BEGIN

    DECLARE @separator_position INT 
    DECLARE @array_value VARCHAR(1000) 

    SET @array = @array + @separator

    WHILE patindex('%' + @separator + '%' , @array) <> 0 
    BEGIN

      SELECT @separator_position =  patindex('%' + @separator + '%', @array)
      SELECT @array_value = left(@array, @separator_position - 1)

        INSERT @resultTable
        VALUES (SUBSTRING(Cast(@array_value AS varchar), 1, 1))

      SELECT @array = stuff(@array, 1, @separator_position, '')
    END

    RETURN
END
0 голосов
/ 19 марта 2019

Изображение в 100 раз лучше, чем описание. Вот пример объявления UDF:

CREATE FUNCTION dbo.GetOnlyFirstLetters(@str NVARCHAR(4000),@sep NVARCHAR(10) )
RETURNS NVARCHAR(100)
AS
BEGIN
   DECLARE @textXML XML

   SELECT   @textXML = CAST('<d>' + replace(@str, @sep, '</d><d>') + '</d>' AS XML)

    DECLARE @result VARCHAR(8000)

    SET @result = ''

    SELECT  @result = @result + LEFT(T.split.value ('.', 'nvarchar(max)'), 1)
    FROM @textXML.nodes ('/d') T (split)

    RETURN  @result
END
GO

Вот как звонить:

SELECT  dbo.GetOnlyFirstLetters('Humayoun Kabir Sohel',' ');

Результат будет:

HKS
0 голосов
/ 14 апреля 2013

Вы также можете достичь этого с помощью xquery.

Declare @Xml XML
Declare @String Varchar(Max)
Declare @firstletter Varchar(Max)
Declare @delimiter Varchar(5) 
SET @delimiter=' '
SET @String= 'THIS IS SQL'
SET @Xml = cast(('<a>'+replace(@String,@delimiter,'</a><a>')+'</a>') AS XML) 

;WITH CTE AS 
(SELECT A.value('.', 'varchar(max)') as [Column]FROM @Xml.nodes('a') AS FN(a) )
 SELECT Stuff((SELECT '' + LEFT([Column],1)from CTE 
 FOR XML PATH ('') ),1,0,'') 

Вот полное решение.

http://raresql.com/2013/04/12/sql-server-get-the-first-letter-of-each-word-in-a-string-column/

0 голосов
/ 02 февраля 2011

Вот мое решение, и оно имеет следующие особенности / особенности:

  • Может обрабатывать сколько угодно имен в строке.(То есть как менее двух, так и более трех.)
  • Все пробелы между именами сохраняются.

Я знаю, что в OP указано, что может быть только 2 или3 имени в его случае.Я не противЯ просто делюсь решением, которое работает, и если оно не подходит для конкретной проблемы, это нормально.

Итак, вот функция:

CREATE FUNCTION dbo.fnGetInitials (@name varchar(max))
RETURNS varchar(max)
AS BEGIN
  DECLARE @cutpos int, @spacepos int, @result varchar(max);
  DECLARE @cutlist TABLE (CutPos int, SpacePos int);

  SET @result = LTRIM(RTRIM(@name));

  SET @cutpos = 2;
  SET @spacepos = CHARINDEX(' ', @result);
  WHILE @spacepos > 0 BEGIN
    INSERT INTO @cutlist VALUES (@cutpos, @spacepos);
    SET @spacepos = @spacepos + 1;
    SET @cutpos = @spacepos + 1;
    SET @spacepos = CHARINDEX(' ', @result, @spacepos);
  END;

  DELETE FROM @cutlist WHERE CutPos >= SpacePos;

  SELECT @result = STUFF(@result, CutPos, SpacePos - CutPos, '')
  FROM @cutlist
  ORDER BY CutPos DESC;

  RETURN @result;
END;

А вот тестовый вызов:

SELECT dbo.fnGetInitials('  John Ronald   Reuel  Tolkien    ');

и результат:

----------------------------------------------------------------------------------------------------
J R   R  Tolkien
...