Como faço para dividir uma seqüência de caracteres para que eu possa acessar o item x?
-
08-06-2019 - |
Pergunta
Usando o SQL Server, como faço para dividir uma seqüência de caracteres para que eu possa acessar o item x?
Dê uma seqüência de caracteres "Olá John Smith".Como posso dividir a seqüência de caracteres de espaço e acessar o item no índice 1, que deve retornar "João"?
Solução
Você pode encontrar a solução no SQL Função Definida pelo Utilizador para Analisar uma Cadeia de caracteres Delimitados por úteis (a partir de O Projeto De Código De).
Você pode usar essa lógica simples:
Declare @products varchar(200) = '1|20|3|343|44|6|8765'
Declare @individual varchar(20) = null
WHILE LEN(@products) > 0
BEGIN
IF PATINDEX('%|%', @products) > 0
BEGIN
SET @individual = SUBSTRING(@products,
0,
PATINDEX('%|%', @products))
SELECT @individual
SET @products = SUBSTRING(@products,
LEN(@individual + '|') + 1,
LEN(@products))
END
ELSE
BEGIN
SET @individual = @products
SET @products = NULL
SELECT @individual
END
END
Outras dicas
Eu não acredito que o SQL Server tem um built-in função de divisão, de modo que não seja uma UDF, a única resposta que eu sei é seqüestrar o PARSENAME função:
SELECT PARSENAME(REPLACE('Hello John Smith', ' ', '.'), 2)
PARSENAME leva uma seqüência de caracteres e divide-no período de caracteres.É preciso um número como seu segundo argumento, e que o número especifica de qual segmento da string para retorno (trabalhando de trás para a frente).
SELECT PARSENAME(REPLACE('Hello John Smith', ' ', '.'), 3) --return Hello
Óbvio problema é quando a cadeia já contém um período.Eu ainda acho que usar uma UDF é a melhor maneira...alguma outra sugestão?
Primeiro, crie uma função (usando CTE, expressão de tabela comum não acabar com a necessidade de uma tabela temporária)
create function dbo.SplitString
(
@str nvarchar(4000),
@separator char(1)
)
returns table
AS
return (
with tokens(p, a, b) AS (
select
1,
1,
charindex(@separator, @str)
union all
select
p + 1,
b + 1,
charindex(@separator, @str, b + 1)
from tokens
where b > 0
)
select
p-1 zeroBasedOccurance,
substring(
@str,
a,
case when b > 0 then b-a ELSE 4000 end)
AS s
from tokens
)
GO
Em seguida, usá-lo como qualquer tabela (ou de modificá-lo para caber dentro de seu armazenados existentes proc) como este.
select s
from dbo.SplitString('Hello John Smith', ' ')
where zeroBasedOccurance=1
Atualização
Versão anterior poderia falhar para entrada de seqüência de caracteres mais de 4000 caracteres.Esta versão tem o cuidado de limitar a:
create function dbo.SplitString
(
@str nvarchar(max),
@separator char(1)
)
returns table
AS
return (
with tokens(p, a, b) AS (
select
cast(1 as bigint),
cast(1 as bigint),
charindex(@separator, @str)
union all
select
p + 1,
b + 1,
charindex(@separator, @str, b + 1)
from tokens
where b > 0
)
select
p-1 ItemIndex,
substring(
@str,
a,
case when b > 0 then b-a ELSE LEN(@str) end)
AS s
from tokens
);
GO
Uso permanece a mesma.
A maioria das soluções aqui usar loops while ou CTEs recursivas.Um conjunto abordagem baseada vai ser superior, eu prometo:
CREATE FUNCTION [dbo].[SplitString]
(
@List NVARCHAR(MAX),
@Delim VARCHAR(255)
)
RETURNS TABLE
AS
RETURN ( SELECT [Value] FROM
(
SELECT
[Value] = LTRIM(RTRIM(SUBSTRING(@List, [Number],
CHARINDEX(@Delim, @List + @Delim, [Number]) - [Number])))
FROM (SELECT Number = ROW_NUMBER() OVER (ORDER BY name)
FROM sys.all_objects) AS x
WHERE Number <= LEN(@List)
AND SUBSTRING(@Delim + @List, [Number], LEN(@Delim)) = @Delim
) AS y
);
Mais em dividir as funções, porque (e a prova de que), enquanto os ciclos e as CTEs recursivas não escala, e a melhor alternativa, se a divisão de seqüências provenientes da camada de aplicação:
- Dividir as cadeias da maneira certa – ou a melhor forma
- A Divisão De Cadeias De Caracteres :Um Follow-Up
- A Divisão De Cadeias De Caracteres :Agora com menos T-SQL
- A comparação de seqüência de caracteres de divisão / concatenação métodos
- O processamento de uma lista de inteiros :a minha abordagem
- A divisão de uma lista de inteiros :outro roundup
- Mais na divisão de listas :personalizado delimitadores, impedindo duplicatas, e a manutenção da ordem
- A remoção de Duplicatas a partir de Seqüências de caracteres no SQL Server
No SQL Server 2016, ou acima, porém, você deve olhar para STRING_SPLIT()
e STRING_AGG()
:
Você pode utilizar uma tabela de Número para fazer a análise de seqüência de caracteres.
Criar uma física tabela de números:
create table dbo.Numbers (N int primary key);
insert into dbo.Numbers
select top 1000 row_number() over(order by number) from master..spt_values
go
Criar a tabela de teste com 1000000 linhas
create table #yak (i int identity(1,1) primary key, array varchar(50))
insert into #yak(array)
select 'a,b,c' from dbo.Numbers n cross join dbo.Numbers nn
go
Criar a função
create function [dbo].[ufn_ParseArray]
( @Input nvarchar(4000),
@Delimiter char(1) = ',',
@BaseIdent int
)
returns table as
return
( select row_number() over (order by n asc) + (@BaseIdent - 1) [i],
substring(@Input, n, charindex(@Delimiter, @Input + @Delimiter, n) - n) s
from dbo.Numbers
where n <= convert(int, len(@Input)) and
substring(@Delimiter + @Input, n, 1) = @Delimiter
)
go
De utilização (saídas de 3mil linhas na década de 40 no meu laptop)
select *
from #yak
cross apply dbo.ufn_ParseArray(array, ',', 1)
limpeza
drop table dbo.Numbers;
drop function [dbo].[ufn_ParseArray]
O desempenho aqui não é incrível, mas a chamada de uma função de mais de um milhão de linha de tabela não é a melhor ideia.Se executar uma seqüência de caracteres dividida em várias linhas evitar a função.
Esta questão é não se trata de uma string split abordagem, mas sobre como obter o n-ésimo elemento.
Todas as respostas aqui estão fazendo algum tipo de cadeia de divisão usando recursão, CTE
s, de várias CHARINDEX
, REVERSE
e PATINDEX
, inventando funções, ligue para o CLR métodos, o número de tabelas, CROSS APPLY
s ...A maioria das respostas capa de muitas linhas de código.
Mas, se você realmente não quero nada mais do que uma abordagem para obter o n-ésimo elemento - isso pode ser feito como real one-liner, não UDF, nem mesmo um sub-select...E como um benefício extra: tipo de seguro
Obter parte 2 delimitado por um espaço:
DECLARE @input NVARCHAR(100)=N'part1 part2 part3';
SELECT CAST(N'<x>' + REPLACE(@input,N' ',N'</x><x>') + N'</x>' AS XML).value('/x[2]','nvarchar(max)')
Claro você pode usar variáveis de para o delimitador e a posição (uso sql:column
para recuperar a posição diretamente a partir de uma consulta do valor):
DECLARE @dlmt NVARCHAR(10)=N' ';
DECLARE @pos INT = 2;
SELECT CAST(N'<x>' + REPLACE(@input,@dlmt,N'</x><x>') + N'</x>' AS XML).value('/x[sql:variable("@pos")][1]','nvarchar(max)')
Se a seqüência de caracteres pode incluir proibida caracteres (especialmente entre &><
), você ainda pode fazê-lo desta maneira.Basta usar FOR XML PATH
em sua primeira seqüência de caracteres para substituir todos os caracteres proibidos com a montagem da seqüência de escape implicitamente.
É um caso muito especial se - além disso - o delimitador é o ponto-e-vírgula.Neste caso, substitua o delimitador primeiro para '#DLMT#', e substituí-lo para as tags XML finalmente:
SET @input=N'Some <, > and &;Other äöü@€;One more';
SET @dlmt=N';';
SELECT CAST(N'<x>' + REPLACE((SELECT REPLACE(@input,@dlmt,'#DLMT#') AS [*] FOR XML PATH('')),N'#DLMT#',N'</x><x>') + N'</x>' AS XML).value('/x[sql:variable("@pos")][1]','nvarchar(max)');
ATUALIZAÇÃO para o SQL-Server 2016+
Lamentavelmente os desenvolvedores esqueceu de devolver a parte do índice com STRING_SPLIT
.Mas, usando o SQL-Server 2016+, há OPENJSON
.
O documentação afirma claramente:
Quando OPENJSON analisa um JSON matriz, a função retorna os índices dos elementos no texto JSON como chaves.
Uma seqüência de caracteres como 1,2,3
não precisa de nada mais do que suportes: [1,2,3]
.
Uma seqüência de palavras como this is an example
precisa ser ["this","is","an"," example"]
.
Estes são muito fáceis operações de cadeia de caracteres.Basta experimentá-lo:
DECLARE @str VARCHAR(100)='Hello John Smith';
SELECT [value]
FROM OPENJSON('["' + REPLACE(@str,' ','","') + '"]')
WHERE [key]=1 --zero-based!
Aqui é um UDF, que irá fazê-lo.Ele irá retornar uma tabela de valores delimitados por, ainda não tentei todos os cenários, mas o seu exemplo funciona bem.
CREATE FUNCTION SplitString
(
-- Add the parameters for the function here
@myString varchar(500),
@deliminator varchar(10)
)
RETURNS
@ReturnTable TABLE
(
-- Add the column definitions for the TABLE variable here
[id] [int] IDENTITY(1,1) NOT NULL,
[part] [varchar](50) NULL
)
AS
BEGIN
Declare @iSpaces int
Declare @part varchar(50)
--initialize spaces
Select @iSpaces = charindex(@deliminator,@myString,0)
While @iSpaces > 0
Begin
Select @part = substring(@myString,0,charindex(@deliminator,@myString,0))
Insert Into @ReturnTable(part)
Select @part
Select @myString = substring(@mystring,charindex(@deliminator,@myString,0)+ len(@deliminator),len(@myString) - charindex(' ',@myString,0))
Select @iSpaces = charindex(@deliminator,@myString,0)
end
If len(@myString) > 0
Insert Into @ReturnTable
Select @myString
RETURN
END
GO
Você poderia chamá-lo assim:
Select * From SplitString('Hello John Smith',' ')
Editar:Atualizado solução para lidar com delimters com um len>1, como em :
select * From SplitString('Hello**John**Smith','**')
Aqui eu posto uma maneira simples de solução
CREATE FUNCTION [dbo].[split](
@delimited NVARCHAR(MAX),
@delimiter NVARCHAR(100)
) RETURNS @t TABLE (id INT IDENTITY(1,1), val NVARCHAR(MAX))
AS
BEGIN
DECLARE @xml XML
SET @xml = N'<t>' + REPLACE(@delimited,@delimiter,'</t><t>') + '</t>'
INSERT INTO @t(val)
SELECT r.value('.','varchar(MAX)') as item
FROM @xml.nodes('/t') as records(r)
RETURN
END
Executar a função como esta
select * from dbo.split('Hello John Smith',' ')
Na minha opinião vcs estão fazendo é muito complicado.Basta criar um CLR UDF e ser feito com ele.
using System;
using System.Data;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using Microsoft.SqlServer.Server;
using System.Collections.Generic;
public partial class UserDefinedFunctions {
[SqlFunction]
public static SqlString SearchString(string Search) {
List<string> SearchWords = new List<string>();
foreach (string s in Search.Split(new char[] { ' ' })) {
if (!s.ToLower().Equals("or") && !s.ToLower().Equals("and")) {
SearchWords.Add(s);
}
}
return new SqlString(string.Join(" OR ", SearchWords.ToArray()));
}
};
Que tal usar string
e values()
declaração?
DECLARE @str varchar(max)
SET @str = 'Hello John Smith'
DECLARE @separator varchar(max)
SET @separator = ' '
DECLARE @Splited TABLE(id int IDENTITY(1,1), item varchar(max))
SET @str = REPLACE(@str, @separator, '''),(''')
SET @str = 'SELECT * FROM (VALUES(''' + @str + ''')) AS V(A)'
INSERT INTO @Splited
EXEC(@str)
SELECT * FROM @Splited
Conjunto de resultados alcançados.
id item
1 Hello
2 John
3 Smith
Eu uso a resposta de frederic, mas isso não funciona no SQL Server 2005
Eu modifiquei e eu estou usando select
com union all
e ele funciona
DECLARE @str varchar(max)
SET @str = 'Hello John Smith how are you'
DECLARE @separator varchar(max)
SET @separator = ' '
DECLARE @Splited table(id int IDENTITY(1,1), item varchar(max))
SET @str = REPLACE(@str, @separator, ''' UNION ALL SELECT ''')
SET @str = ' SELECT ''' + @str + ''' '
INSERT INTO @Splited
EXEC(@str)
SELECT * FROM @Splited
E o resultado é:
id item
1 Hello
2 John
3 Smith
4 how
5 are
6 you
Este padrão funciona bem e você pode generalizar
Convert(xml,'<n>'+Replace(FIELD,'.','</n><n>')+'</n>').value('(/n[INDEX])','TYPE')
^^^^^ ^^^^^ ^^^^
nota CAMPO, ÍNDICE de e TIPO.
Deixe algum tabela com identificadores como
sys.message.1234.warning.A45
sys.message.1235.error.O98
....
Em seguida, você pode escrever
SELECT Source = q.value('(/n[1])', 'varchar(10)'),
RecordType = q.value('(/n[2])', 'varchar(20)'),
RecordNumber = q.value('(/n[3])', 'int'),
Status = q.value('(/n[4])', 'varchar(5)')
FROM (
SELECT q = Convert(xml,'<n>'+Replace(fieldName,'.','</n><n>')+'</n>')
FROM some_TABLE
) Q
a divisão de fundição e de todas as partes.
Se seu banco de dados de compatibilidade de nível de 130 ou mais, em seguida, você pode usar o STRING_SPLIT a função de, junto com o DESLOCAMENTO DE BUSCA cláusulas para obter o item específico pelo índice.
Para obter o item no o índice N (baseado em zero), você pode usar o seguinte código
SELECT value
FROM STRING_SPLIT('Hello John Smith',' ')
ORDER BY (SELECT NULL)
OFFSET N ROWS
FETCH NEXT 1 ROWS ONLY
Para verificar a o nível de compatibilidade do banco de dados, execute este código:
SELECT compatibility_level
FROM sys.databases WHERE name = 'YourDBName';
Eu estava procurando a solução de líquido e a seguir funciona para mim.Ref.
E você chamar a função como esta :
SELECT * FROM dbo.split('ram shyam hari gopal',' ')
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE FUNCTION [dbo].[Split](@String VARCHAR(8000), @Delimiter CHAR(1))
RETURNS @temptable TABLE (items VARCHAR(8000))
AS
BEGIN
DECLARE @idx INT
DECLARE @slice VARCHAR(8000)
SELECT @idx = 1
IF len(@String)<1 OR @String IS NULL RETURN
WHILE @idx!= 0
BEGIN
SET @idx = charindex(@Delimiter,@String)
IF @idx!=0
SET @slice = LEFT(@String,@idx - 1)
ELSE
SET @slice = @String
IF(len(@slice)>0)
INSERT INTO @temptable(Items) VALUES(@slice)
SET @String = RIGHT(@String,len(@String) - @idx)
IF len(@String) = 0 break
END
RETURN
END
Outra ainda, obter n ī esima parte da cadeia de delimitador de função:
create function GetStringPartByDelimeter (
@value as nvarchar(max),
@delimeter as nvarchar(max),
@position as int
) returns NVARCHAR(MAX)
AS BEGIN
declare @startPos as int
declare @endPos as int
set @endPos = -1
while (@position > 0 and @endPos != 0) begin
set @startPos = @endPos + 1
set @endPos = charindex(@delimeter, @value, @startPos)
if(@position = 1) begin
if(@endPos = 0)
set @endPos = len(@value) + 1
return substring(@value, @startPos, @endPos - @startPos)
end
set @position = @position - 1
end
return null
end
e o uso de:
select dbo.GetStringPartByDelimeter ('a;b;c;d;e', ';', 3)
o que retorna:
c
Tente isso:
CREATE function [SplitWordList]
(
@list varchar(8000)
)
returns @t table
(
Word varchar(50) not null,
Position int identity(1,1) not null
)
as begin
declare
@pos int,
@lpos int,
@item varchar(100),
@ignore varchar(100),
@dl int,
@a1 int,
@a2 int,
@z1 int,
@z2 int,
@n1 int,
@n2 int,
@c varchar(1),
@a smallint
select
@a1 = ascii('a'),
@a2 = ascii('A'),
@z1 = ascii('z'),
@z2 = ascii('Z'),
@n1 = ascii('0'),
@n2 = ascii('9')
set @ignore = '''"'
set @pos = 1
set @dl = datalength(@list)
set @lpos = 1
set @item = ''
while (@pos <= @dl) begin
set @c = substring(@list, @pos, 1)
if (@ignore not like '%' + @c + '%') begin
set @a = ascii(@c)
if ((@a >= @a1) and (@a <= @z1))
or ((@a >= @a2) and (@a <= @z2))
or ((@a >= @n1) and (@a <= @n2))
begin
set @item = @item + @c
end else if (@item > '') begin
insert into @t values (@item)
set @item = ''
end
end
set @pos = @pos + 1
end
if (@item > '') begin
insert into @t values (@item)
end
return
end
Teste assim:
select * from SplitWordList('Hello John Smith')
O exemplo a seguir usa uma CTE recursiva
Atualização 18.09.2013
CREATE FUNCTION dbo.SplitStrings_CTE(@List nvarchar(max), @Delimiter nvarchar(1))
RETURNS @returns TABLE (val nvarchar(max), [level] int, PRIMARY KEY CLUSTERED([level]))
AS
BEGIN
;WITH cte AS
(
SELECT SUBSTRING(@List, 0, CHARINDEX(@Delimiter, @List + @Delimiter)) AS val,
CAST(STUFF(@List + @Delimiter, 1, CHARINDEX(@Delimiter, @List + @Delimiter), '') AS nvarchar(max)) AS stval,
1 AS [level]
UNION ALL
SELECT SUBSTRING(stval, 0, CHARINDEX(@Delimiter, stval)),
CAST(STUFF(stval, 1, CHARINDEX(@Delimiter, stval), '') AS nvarchar(max)),
[level] + 1
FROM cte
WHERE stval != ''
)
INSERT @returns
SELECT REPLACE(val, ' ','' ) AS val, [level]
FROM cte
WHERE val > ''
RETURN
END
Demonstração SQLFiddle
Alter Function dbo.fn_Split
(
@Expression nvarchar(max),
@Delimiter nvarchar(20) = ',',
@Qualifier char(1) = Null
)
RETURNS @Results TABLE (id int IDENTITY(1,1), value nvarchar(max))
AS
BEGIN
/* USAGE
Select * From dbo.fn_Split('apple pear grape banana orange honeydew cantalope 3 2 1 4', ' ', Null)
Select * From dbo.fn_Split('1,abc,"Doe, John",4', ',', '"')
Select * From dbo.fn_Split('Hello 0,"&""&&&&', ',', '"')
*/
-- Declare Variables
DECLARE
@X xml,
@Temp nvarchar(max),
@Temp2 nvarchar(max),
@Start int,
@End int
-- HTML Encode @Expression
Select @Expression = (Select @Expression For XML Path(''))
-- Find all occurences of @Delimiter within @Qualifier and replace with |||***|||
While PATINDEX('%' + @Qualifier + '%', @Expression) > 0 AND Len(IsNull(@Qualifier, '')) > 0
BEGIN
Select
-- Starting character position of @Qualifier
@Start = PATINDEX('%' + @Qualifier + '%', @Expression),
-- @Expression starting at the @Start position
@Temp = SubString(@Expression, @Start + 1, LEN(@Expression)-@Start+1),
-- Next position of @Qualifier within @Expression
@End = PATINDEX('%' + @Qualifier + '%', @Temp) - 1,
-- The part of Expression found between the @Qualifiers
@Temp2 = Case When @End < 0 Then @Temp Else Left(@Temp, @End) End,
-- New @Expression
@Expression = REPLACE(@Expression,
@Qualifier + @Temp2 + Case When @End < 0 Then '' Else @Qualifier End,
Replace(@Temp2, @Delimiter, '|||***|||')
)
END
-- Replace all occurences of @Delimiter within @Expression with '</fn_Split><fn_Split>'
-- And convert it to XML so we can select from it
SET
@X = Cast('<fn_Split>' +
Replace(@Expression, @Delimiter, '</fn_Split><fn_Split>') +
'</fn_Split>' as xml)
-- Insert into our returnable table replacing '|||***|||' back to @Delimiter
INSERT @Results
SELECT
"Value" = LTRIM(RTrim(Replace(C.value('.', 'nvarchar(max)'), '|||***|||', @Delimiter)))
FROM
@X.nodes('fn_Split') as X(C)
-- Return our temp table
RETURN
END
Eu sei que é uma velha Pergunta, mas eu acho que alguém pode beneficiar a minha solução.
select
SUBSTRING(column_name,1,CHARINDEX(' ',column_name,1)-1)
,SUBSTRING(SUBSTRING(column_name,CHARINDEX(' ',column_name,1)+1,LEN(column_name))
,1
,CHARINDEX(' ',SUBSTRING(column_name,CHARINDEX(' ',column_name,1)+1,LEN(column_name)),1)-1)
,SUBSTRING(SUBSTRING(column_name,CHARINDEX(' ',column_name,1)+1,LEN(column_name))
,CHARINDEX(' ',SUBSTRING(column_name,CHARINDEX(' ',column_name,1)+1,LEN(column_name)),1)+1
,LEN(column_name))
from table_name
Vantagens:
- Separa todas as 3 sub-cadeias deliminator por ' '.
- Não se deve usar o loop while, como diminui o desempenho.
- Não há necessidade de Pivô como todos resultante sub-string que irá ser exibida no uma Linha
Limitações:
- Deve-se saber o total não existe.de espaços (sub-cadeia de caracteres).
Nota:a solução pode dar a sub-cadeia de caracteres até a N.
Para superou a limitação, podemos usar o seguinte ref.
Mas, novamente acima solução não pode ser usado em uma tabela (Actaully eu não era capaz de usá-lo).
Novamente, eu espero que esta solução pode ajudar alguém.
Atualização: No caso de Registros > 50000 ele não é aconselhável para usar LOOPS
como isso poderá prejudicar o Desempenho
Quase todas as outras respostas divisão de código está a substituir a seqüência de caracteres que está sendo dividida que resíduos de ciclos de CPU e executa desnecessários alocações de memória.
Eu capa de uma maneira muito melhor de fazer uma seqüência de dividir aqui: http://www.digitalruby.com/split-string-sql-server/
Aqui está o código:
SET NOCOUNT ON
-- You will want to change nvarchar(MAX) to nvarchar(50), varchar(50) or whatever matches exactly with the string column you will be searching against
DECLARE @SplitStringTable TABLE (Value nvarchar(MAX) NOT NULL)
DECLARE @StringToSplit nvarchar(MAX) = 'your|string|to|split|here'
DECLARE @SplitEndPos int
DECLARE @SplitValue nvarchar(MAX)
DECLARE @SplitDelim nvarchar(1) = '|'
DECLARE @SplitStartPos int = 1
SET @SplitEndPos = CHARINDEX(@SplitDelim, @StringToSplit, @SplitStartPos)
WHILE @SplitEndPos > 0
BEGIN
SET @SplitValue = SUBSTRING(@StringToSplit, @SplitStartPos, (@SplitEndPos - @SplitStartPos))
INSERT @SplitStringTable (Value) VALUES (@SplitValue)
SET @SplitStartPos = @SplitEndPos + 1
SET @SplitEndPos = CHARINDEX(@SplitDelim, @StringToSplit, @SplitStartPos)
END
SET @SplitValue = SUBSTRING(@StringToSplit, @SplitStartPos, 2147483647)
INSERT @SplitStringTable (Value) VALUES(@SplitValue)
SET NOCOUNT OFF
-- You can select or join with the values in @SplitStringTable at this point.
Você pode dividir uma seqüência de caracteres no SQL sem a necessidade de uma função:
DECLARE @bla varchar(MAX)
SET @bla = 'BED40DFC-F468-46DD-8017-00EF2FA3E4A4,64B59FC5-3F4D-4B0E-9A48-01F3D4F220B0,A611A108-97CA-42F3-A2E1-057165339719,E72D95EA-578F-45FC-88E5-075F66FD726C'
-- http://stackoverflow.com/questions/14712864/how-to-query-values-from-xml-nodes
SELECT
x.XmlCol.value('.', 'varchar(36)') AS val
FROM
(
SELECT
CAST('<e>' + REPLACE(@bla, ',', '</e><e>') + '</e>' AS xml) AS RawXml
) AS b
CROSS APPLY b.RawXml.nodes('e') x(XmlCol);
Se você precisa de suporte arbitrário seqüências de caracteres (com xml caracteres especiais)
DECLARE @bla NVARCHAR(MAX)
SET @bla = '<html>unsafe & safe Utf8CharsDon''tGetEncoded ÄöÜ - "Conex"<html>,Barnes & Noble,abc,def,ghi'
-- http://stackoverflow.com/questions/14712864/how-to-query-values-from-xml-nodes
SELECT
x.XmlCol.value('.', 'nvarchar(MAX)') AS val
FROM
(
SELECT
CAST('<e>' + REPLACE((SELECT @bla FOR XML PATH('')), ',', '</e><e>') + '</e>' AS xml) AS RawXml
) AS b
CROSS APPLY b.RawXml.nodes('e') x(XmlCol);
Pura conjunto-solução usando TVF
com recursiva CTE
.Você pode JOIN
e APPLY
essa função para qualquer conjunto de dados.
create function [dbo].[SplitStringToResultSet] (@value varchar(max), @separator char(1))
returns table
as return
with r as (
select value, cast(null as varchar(max)) [x], -1 [no] from (select rtrim(cast(@value as varchar(max))) [value]) as j
union all
select right(value, len(value)-case charindex(@separator, value) when 0 then len(value) else charindex(@separator, value) end) [value]
, left(r.[value], case charindex(@separator, r.value) when 0 then len(r.value) else abs(charindex(@separator, r.[value])-1) end ) [x]
, [no] + 1 [no]
from r where value > '')
select ltrim(x) [value], [no] [index] from r where x is not null;
go
Uso:
select *
from [dbo].[SplitStringToResultSet]('Hello John Smith', ' ')
where [index] = 1;
Resultado:
value index
-------------
John 1
Começando com O SQL Server de 2016 nós string_split
DECLARE @string varchar(100) = 'Richard, Mike, Mark'
SELECT value FROM string_split(@string, ',')
Uma abordagem moderna usando STRING_SPLIT, requer o SQL Server 2016 e acima.
DECLARE @string varchar(100) = 'Hello John Smith'
SELECT
ROW_NUMBER() OVER (ORDER BY value) AS RowNr,
value
FROM string_split(@string, ' ')
Resultado:
RowNr value
1 Hello
2 John
3 Smith
Agora é possível obter th n-ésimo elemento de número de linha.
Arão Bertrand, a resposta é grande, mas falho.Ele não precisão lidar com um espaço como um delimitador (como foi o exemplo em questão original), pois a função de comprimento de tiras de espaços à direita.
A seguir está o seu código, com um pequeno ajuste para permitir um espaço delimitador:
CREATE FUNCTION [dbo].[SplitString]
(
@List NVARCHAR(MAX),
@Delim VARCHAR(255)
)
RETURNS TABLE
AS
RETURN ( SELECT [Value] FROM
(
SELECT
[Value] = LTRIM(RTRIM(SUBSTRING(@List, [Number],
CHARINDEX(@Delim, @List + @Delim, [Number]) - [Number])))
FROM (SELECT Number = ROW_NUMBER() OVER (ORDER BY name)
FROM sys.all_objects) AS x
WHERE Number <= LEN(@List)
AND SUBSTRING(@Delim + @List, [Number], LEN(@Delim+'x')-1) = @Delim
) AS y
);
Aqui está uma função que irá realizar a questão do objetivo de dividir uma seqüência de caracteres e acessando o item X:
CREATE FUNCTION [dbo].[SplitString]
(
@List VARCHAR(MAX),
@Delimiter VARCHAR(255),
@ElementNumber INT
)
RETURNS VARCHAR(MAX)
AS
BEGIN
DECLARE @inp VARCHAR(MAX)
SET @inp = (SELECT REPLACE(@List,@Delimiter,'_DELMTR_') FOR XML PATH(''))
DECLARE @xml XML
SET @xml = '<split><el>' + REPLACE(@inp,'_DELMTR_','</el><el>') + '</el></split>'
DECLARE @ret VARCHAR(MAX)
SET @ret = (SELECT
el = split.el.value('.','varchar(max)')
FROM @xml.nodes('/split/el[string-length(.)>0][position() = sql:variable("@elementnumber")]') split(el))
RETURN @ret
END
Uso:
SELECT dbo.SplitString('Hello John Smith', ' ', 2)
Resultado:
John
SOLUÇÃO SIMPLES PARA A ANÁLISE DO PRIMEIRO E ÚLTIMO NOME
DECLARE @Name varchar(10) = 'John Smith'
-- Get First Name
SELECT SUBSTRING(@Name, 0, (SELECT CHARINDEX(' ', @Name)))
-- Get Last Name
SELECT SUBSTRING(@Name, (SELECT CHARINDEX(' ', @Name)) + 1, LEN(@Name))
No meu caso (e em muitos outros parece...), eu tenho uma lista do primeiro e último nomes separados por um único espaço.Isso pode ser utilizado diretamente dentro de uma instrução select para analisar primeiro e último nome.
-- i.e. Get First and Last Name from a table of Full Names
SELECT SUBSTRING(FullName, 0, (SELECT CHARINDEX(' ', FullName))) as FirstName,
SUBSTRING(FullName, (SELECT CHARINDEX(' ', FullName)) + 1, LEN(FullName)) as LastName,
From FullNameTable
Eu sei que é tarde, mas eu recentemente tive esse requisito e veio com o código abaixo.Eu não tenho uma escolha para utilizar a função definida pelo Usuário.Espero que isso ajude.
SELECT
SUBSTRING(
SUBSTRING('Hello John Smith' ,0,CHARINDEX(' ','Hello John Smith',CHARINDEX(' ','Hello John Smith')+1)
),CHARINDEX(' ','Hello John Smith'),LEN('Hello John Smith')
)
Bem, a minha não é de todo simples, mas aqui está o código que eu uso para dividir um delimitado por vírgulas variável de entrada em valores individuais, e colocá-lo em uma variável de tabela.Tenho certeza de que você pode modificar um pouco para dividir com base em um espaço e, em seguida, para fazer o básico de uma consulta seleção contra a variável de tabela para obter seus resultados.
-- Create temporary table to parse the list of accounting cycles.
DECLARE @tblAccountingCycles table
(
AccountingCycle varchar(10)
)
DECLARE @vchAccountingCycle varchar(10)
DECLARE @intPosition int
SET @vchAccountingCycleIDs = LTRIM(RTRIM(@vchAccountingCycleIDs)) + ','
SET @intPosition = CHARINDEX(',', @vchAccountingCycleIDs, 1)
IF REPLACE(@vchAccountingCycleIDs, ',', '') <> ''
BEGIN
WHILE @intPosition > 0
BEGIN
SET @vchAccountingCycle = LTRIM(RTRIM(LEFT(@vchAccountingCycleIDs, @intPosition - 1)))
IF @vchAccountingCycle <> ''
BEGIN
INSERT INTO @tblAccountingCycles (AccountingCycle) VALUES (@vchAccountingCycle)
END
SET @vchAccountingCycleIDs = RIGHT(@vchAccountingCycleIDs, LEN(@vchAccountingCycleIDs) - @intPosition)
SET @intPosition = CHARINDEX(',', @vchAccountingCycleIDs, 1)
END
END
O conceito é praticamente o mesmo.Uma outra alternativa é aproveitar o .NET compatibilidade dentro do SQL Server 2005.Você pode escrever-se, essencialmente, de um simples método .LÍQUIDO que iria dividir a seqüência de caracteres e, em seguida, expor que, como um procedimento armazenado/função.
Isso é algo que eu fiz, a fim de obter um determinado token em uma seqüência de caracteres.(Testado no MSSQL 2008)
Em primeiro lugar, criar as seguintes funções:(encontrado em: aqui
CREATE FUNCTION dbo.SplitStrings_Moden
(
@List NVARCHAR(MAX),
@Delimiter NVARCHAR(255)
)
RETURNS TABLE
WITH SCHEMABINDING AS
RETURN
WITH E1(N) AS ( SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1
UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1
UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1),
E2(N) AS (SELECT 1 FROM E1 a, E1 b),
E4(N) AS (SELECT 1 FROM E2 a, E2 b),
E42(N) AS (SELECT 1 FROM E4 a, E2 b),
cteTally(N) AS (SELECT 0 UNION ALL SELECT TOP (DATALENGTH(ISNULL(@List,1)))
ROW_NUMBER() OVER (ORDER BY (SELECT NULL)) FROM E42),
cteStart(N1) AS (SELECT t.N+1 FROM cteTally t
WHERE (SUBSTRING(@List,t.N,1) = @Delimiter OR t.N = 0))
SELECT Item = SUBSTRING(@List, s.N1, ISNULL(NULLIF(CHARINDEX(@Delimiter,@List,s.N1),0)-s.N1,8000))
FROM cteStart s;
e
create FUNCTION dbo.getToken
(
@List NVARCHAR(MAX),
@Delimiter NVARCHAR(255),
@Pos int
)
RETURNS varchar(max)
as
begin
declare @returnValue varchar(max);
select @returnValue = tbl.Item from (
select ROW_NUMBER() over (order by (select null)) as id, * from dbo.SplitStrings_Moden(@List, @Delimiter)
) as tbl
where tbl.id = @Pos
return @returnValue
end
em seguida, você pode usá-lo assim:
select dbo.getToken('1111_2222_3333_', '_', 1)
que retorno 1111