Преобразование имен элементов XML в формат camelCase с помощью XSLT

Мне нужно преобразовать все имена элементов в XML в стратегию именования camelCase с помощью XSLT.

Вот ответ от грядущего сервера:

<ROOT_ELEMENT>
        <ELEMENT_1>2595959584</ELEMENT_1>
        <Element_Two>Lorem</Element_Two>
        <eLement_Three>Ipsum</eLement_Three>

        <BIRTH_date>
           <Day>21</Day>
           <Month>09</Month>
           <Year>1955</Year>
        </BIRTH_date>
 </ROOT_ELEMENT>

Вот что я хочу преобразовать:

<rootElement>
        <element1>2595959584</element1>
        <elementTwo>Lorem</elementTwo>
        <elementThree>Ipsum</elementThree>      
        <birthDate>
           <day>21</day>
           <month>09</month>
           <year>1955</year>
        </birthDate>
</rootElement>

Любая помощь приветствуется.

Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
1
0
968
4
Перейти к ответу Данный вопрос помечен как решенный

Ответы 4

В XSLT 3.0 это

string-join(tokenize(name(), '_') ! 
  (upper-case(substring(., 1, 1)) || lower-case(substring(., 2))))

В 1.0 это намного сложнее, но все же возможно (по крайней мере, если ваши имена элементов используют только буквы ASCII). Вам понадобится рекурсивный именованный шаблон для выполнения токенизации (вы можете найти его в http://www.exslt.org, ищите str:tokenize в модуле строк), а в именованном шаблоне, где вы обрабатываете один токен, вам нужно будет использовать translate($x, 'abc...z', 'ABC...Z') для выполнения конверсия случая.

Вы можете избежать рекурсии и упростить логику, если знаете, что в имени будет только одно подчеркивание; в этом случае вы можете использовать substring-before() и substring-after(), чтобы получить две части имени.

Спасибо за ответ. Но я только сегодня познакомился с XSLT. Я еще ничего не знал о XSLT. Есть ли полное решение этой проблемы.

Mevlüt Sungur 19.02.2019 13:37

Я не готов писать код XSLT 1.0 для такой задачи, это абсолютно неинтересно. Перейдите к более поздней версии, и такие люди, как я, с большей вероятностью предложат помощь.

Michael Kay 19.02.2019 13:43

На самом деле, в XSLT 3.0 тоже сложнее: приведенное выше вернет PascalCase, а требование для camelCase (первый символ в нижнем регистре).

michael.hor257k 20.02.2019 13:50
Ответ принят как подходящий

Предполагая, что в имени элемента будет не более одного разделителя _ и если имена не содержат символов, отличных от ASCII, вы можете использовать:

XSLT 1.0

<xsl:stylesheet version = "1.0" 
xmlns:xsl = "http://www.w3.org/1999/XSL/Transform">
<xsl:output method = "xml" version = "1.0" encoding = "UTF-8" indent = "yes"/>
<xsl:strip-space elements = "*"/>

<xsl:template match = "*">
    <xsl:param name = "upper-case" select = "'ABCDEFGHIJKLMNOPQRSTUVWXYZ'"/>
    <xsl:param name = "lower-case" select = "'abcdefghijklmnopqrstuvwxyz'"/>
    <xsl:variable name = "name" select = "name()" />
    <xsl:variable name = "word2" select = "substring-after($name, '_')" />
    <xsl:variable name = "new-name">
        <xsl:value-of select = "translate(substring-before(concat($name, '_'), '_'), $upper-case, $lower-case)"/> 
        <xsl:value-of select = "translate(substring($word2, 1, 1), $lower-case, $upper-case)"/>   
        <xsl:value-of select = "translate(substring($word2, 2), $upper-case, $lower-case)"/>  
    </xsl:variable>
    <xsl:element name = "{$new-name}">
        <xsl:apply-templates/>
    </xsl:element>
</xsl:template>

</xsl:stylesheet>

Добавлен:

Если (как кажется) вы используете процессор, поддерживающий функцию расширения XSLT str:tokenize(), вы можете сделать:

XSLT 1.0 + EXSLT

<xsl:stylesheet version = "1.0" 
xmlns:xsl = "http://www.w3.org/1999/XSL/Transform"
xmlns:str = "http://exslt.org/strings"
extension-element-prefixes = "str">
<xsl:output method = "xml" version = "1.0" encoding = "UTF-8" indent = "yes"/>
<xsl:strip-space elements = "*"/>

<xsl:template match = "*">
    <xsl:param name = "upper-case" select = "'ABCDEFGHIJKLMNOPQRSTUVWXYZ'"/>
    <xsl:param name = "lower-case" select = "'abcdefghijklmnopqrstuvwxyz'"/>
    <xsl:variable name = "new-name">
        <xsl:for-each select = "str:tokenize(name(), '_')">
            <xsl:choose>
                <xsl:when test = "position() = 1">
                    <xsl:value-of select = "translate(., $upper-case, $lower-case)"/>   
                </xsl:when>
                <xsl:otherwise>
                    <xsl:value-of select = "translate(substring(., 1, 1), $lower-case, $upper-case)"/>   
                    <xsl:value-of select = "translate(substring(., 2), $upper-case, $lower-case)"/>
                </xsl:otherwise>
            </xsl:choose>
        </xsl:for-each> 
    </xsl:variable>
    <xsl:element name = "{$new-name}">
        <xsl:apply-templates/>
    </xsl:element>
</xsl:template>

</xsl:stylesheet>

Это снимает ограничение на количество разделителей _, которые может иметь имя элемента. Остается другое ограничение: в параметрах $upper-case и $lower-case должен быть указан каждый возможный символ, который может встречаться в имени элемента и иметь варианты в верхнем и нижнем регистре.

Еще один способ сделать первую букву в XSLT 1.0 заглавной:

<?xml version = "1.0" encoding = "UTF-8"?>
<xsl:stylesheet version = "1.0"
xmlns:xsl = "http://www.w3.org/1999/XSL/Transform">

<xsl:output method = "xml" indent = "yes" />

<xsl:variable name = "smallcase" select = "'abcdefghijklmnopqrstuvwxyz'" />
<xsl:variable name = "uppercase" select = "'ABCDEFGHIJKLMNOPQRSTUVWXYZ'" />

<xsl:template match = "*">
    <xsl:variable name = "convertElementName">
        <xsl:choose>
            <xsl:when test = "contains(local-name(.),'_')">
                <xsl:value-of
                    select = "concat(translate(substring-before(local-name(.),'_'), $uppercase, $smallcase),
                            translate(substring(substring-after(local-name(.),'_'),1,1), $smallcase, $uppercase),
                            translate(substring(substring-after(local-name(.),'_'),2,string-length(local-name(.))-1), $uppercase, $smallcase))" />
            </xsl:when>
            <xsl:otherwise>
                <xsl:value-of select = "translate(local-name(.), $uppercase, $smallcase)" />
            </xsl:otherwise>
        </xsl:choose>
    </xsl:variable>

    <xsl:element name = "{$convertElementName}">
        <xsl:apply-templates select = "@* | node()" />
    </xsl:element>
</xsl:template>

<xsl:template match = "@*">
    <xsl:attribute name = "{local-name(.)}">
        <xsl:value-of select = "." />
    </xsl:attribute>
</xsl:template>
</xsl:stylesheet>
<xsl:template match = "*">
    <xsl:choose>
        <xsl:when test = "contains(name(), '_')">
            <xsl:element name = "{concat(substring-before(lower-case(name()), '_'), substring(substring-after(upper-case(name()), '_'), 1,1), substring(substring-after(lower-case(name()), '_'), 2))}">
                <xsl:apply-templates/>
            </xsl:element>
        </xsl:when>
        <xsl:otherwise>
                <xsl:element name = "{lower-case(name())}">
                    <xsl:apply-templates/>
                </xsl:element>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>

Еще один ответ, если возможно, используйте этот код.

Этот вопрос помечен как XSLT 1.0. Ваш ответ требует XSLT 2.0.

michael.hor257k 20.02.2019 10:38

Откуда взялась строчная функция? Я получил эту ошибку. javax.xml.transform.TransformerException: не удалось найти функцию: нижний регистр

Mevlüt Sungur 20.02.2019 12:10

Этот ответ был XSLT 2.0

Sam 20.02.2019 12:36

Другие вопросы по теме