RangeTokenMap.cpp
上传用户:huihehuasu
上传日期:2007-01-10
资源大小:6948k
文件大小:10k
源码类别:

xml/soap/webservice

开发平台:

C/C++

  1. /*
  2.  * The Apache Software License, Version 1.1
  3.  *
  4.  * Copyright (c) 2001 The Apache Software Foundation.  All rights
  5.  * reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  *
  11.  * 1. Redistributions of source code must retain the above copyright
  12.  *    notice, this list of conditions and the following disclaimer.
  13.  *
  14.  * 2. Redistributions in binary form must reproduce the above copyright
  15.  *    notice, this list of conditions and the following disclaimer in
  16.  *    the documentation and/or other materials provided with the
  17.  *    distribution.
  18.  *
  19.  * 3. The end-user documentation included with the redistribution,
  20.  *    if any, must include the following acknowledgment:
  21.  *       "This product includes software developed by the
  22.  *        Apache Software Foundation (http://www.apache.org/)."
  23.  *    Alternately, this acknowledgment may appear in the software itself,
  24.  *    if and wherever such third-party acknowledgments normally appear.
  25.  *
  26.  * 4. The names "Xerces" and "Apache Software Foundation" must
  27.  *    not be used to endorse or promote products derived from this
  28.  *    software without prior written permission. For written
  29.  *    permission, please contact apache@apache.org.
  30.  *
  31.  * 5. Products derived from this software may not be called "Apache",
  32.  *    nor may "Apache" appear in their name, without prior written
  33.  *    permission of the Apache Software Foundation.
  34.  *
  35.  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36.  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37.  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38.  * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39.  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40.  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41.  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42.  * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43.  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44.  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45.  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46.  * SUCH DAMAGE.
  47.  * ====================================================================
  48.  *
  49.  * This software consists of voluntary contributions made by many
  50.  * individuals on behalf of the Apache Software Foundation, and was
  51.  * originally based on software copyright (c) 2001, International
  52.  * Business Machines, Inc., http://www.ibm.com .  For more information
  53.  * on the Apache Software Foundation, please see
  54.  * <http://www.apache.org/>.
  55.  */
  56. /*
  57.  * $Log: RangeTokenMap.cpp,v $
  58.  * Revision 1.5  2001/10/25 15:06:26  tng
  59.  * Thread safe the static instance.
  60.  *
  61.  * Revision 1.4  2001/10/23 23:13:41  peiyongz
  62.  * [Bug#880] patch to PlatformUtils:init()/term() and related. from Mark Weaver
  63.  *
  64.  * Revision 1.3  2001/07/16 21:28:25  knoaman
  65.  * fix bug - no delete for the static instance in destructor.
  66.  *
  67.  * Revision 1.2  2001/05/11 13:26:45  tng
  68.  * Copyright update.
  69.  *
  70.  * Revision 1.1  2001/05/03 18:17:40  knoaman
  71.  * Some design changes:
  72.  * o Changed the TokenFactory from a single static instance, to a
  73.  *    normal class. Each RegularExpression object will have its own
  74.  *    instance of TokenFactory, and that instance will be passed to
  75.  *    other classes that need to use a TokenFactory to create Token
  76.  *    objects (with the exception of RangeTokenMap).
  77.  * o Added a new class RangeTokenMap to map a the different ranges
  78.  *    in a given category to a specific RangeFactory object. In the old
  79.  *    design RangeFactory had dual functionality (act as a Map, and as
  80.  *    a factory for creating RangeToken(s)). The RangeTokenMap will
  81.  *    have its own copy of the TokenFactory. There will be only one
  82.  *    instance of the RangeTokenMap class, and that instance will be
  83.  *    lazily deleted when XPlatformUtils::Terminate is called.
  84.  *
  85.  */
  86. // ---------------------------------------------------------------------------
  87. //  Includes
  88. // ---------------------------------------------------------------------------
  89. #include <util/regx/RangeTokenMap.hpp>
  90. #include <util/regx/RangeToken.hpp>
  91. #include <util/regx/RegxDefs.hpp>
  92. #include <util/regx/TokenFactory.hpp>
  93. #include <util/regx/RangeFactory.hpp>
  94. #include <util/PlatformUtils.hpp>
  95. #include <util/XMLExceptMsgs.hpp>
  96. #include <util/XMLRegisterCleanup.hpp>
  97. // ---------------------------------------------------------------------------
  98. //  Static member data initialization
  99. // ---------------------------------------------------------------------------
  100. RangeTokenMap* RangeTokenMap::fInstance = 0;
  101. // ---------------------------------------------------------------------------
  102. //  RangeTokenElemMap: Constructors and Destructor
  103. // ---------------------------------------------------------------------------
  104. RangeTokenElemMap::RangeTokenElemMap(unsigned int categoryId) :
  105.     fCategoryId(categoryId)
  106.     , fRange(0)
  107.     , fNRange(0)
  108. {
  109. }
  110. RangeTokenElemMap::~RangeTokenElemMap()
  111. {
  112. }
  113. // ---------------------------------------------------------------------------
  114. //  RangeTokenMap: Constructors and Destructor
  115. // ---------------------------------------------------------------------------
  116. RangeTokenMap::RangeTokenMap() :
  117.     fRegistryInitialized(0)
  118.     , fTokenRegistry(0)
  119.     , fRangeMap(0)
  120.     , fCategories(0)
  121.     , fTokenFactory(0) {
  122. }
  123. RangeTokenMap::~RangeTokenMap() {
  124.     delete fTokenRegistry;
  125.     fTokenRegistry = 0;
  126.     delete fRangeMap;
  127.     fRangeMap = 0;
  128.     delete fCategories;
  129.     fCategories = 0;
  130.     delete fTokenFactory;
  131.     fTokenFactory = 0;
  132.     fInstance = 0;
  133. }
  134. // ---------------------------------------------------------------------------
  135. //  RangeTokenMap: Getter methods
  136. // ---------------------------------------------------------------------------
  137. RangeToken* RangeTokenMap::getRange(const XMLCh* const keyword,
  138.    const bool complement) {
  139. if (fTokenRegistry == 0 || fRangeMap == 0 || fCategories == 0)
  140. return 0;
  141.     if (!fTokenRegistry->containsKey(keyword))
  142. return 0;
  143. RangeTokenElemMap* elemMap = 0;
  144. // Use a faux scope to synchronize while we do this
  145.     {
  146.         XMLMutexLock lockInit(&fMutex);
  147. elemMap = fTokenRegistry->get(keyword);
  148. RangeToken* rangeTok = 0;
  149. if (elemMap->getRangeToken() == 0) {
  150. unsigned int categId = elemMap->getCategoryId();
  151. const XMLCh* categName = fCategories->getValueForId(categId);
  152. RangeFactory* rangeFactory = fRangeMap->get(categName);
  153. if (rangeFactory == 0)
  154. return 0;
  155. rangeFactory->buildRanges();
  156. }
  157. if (complement && ((rangeTok = elemMap->getRangeToken()) != 0)) {
  158. elemMap->setRangeToken((RangeToken*)
  159. RangeToken::complementRanges(rangeTok, fTokenFactory),
  160. complement);
  161. }
  162.     }
  163. return (elemMap == 0) ? 0 : elemMap->getRangeToken(complement);
  164. }
  165. // ---------------------------------------------------------------------------
  166. //  RangeTokenMap: Putter methods
  167. // ---------------------------------------------------------------------------
  168. void RangeTokenMap::addCategory(const XMLCh* const categoryName) {
  169.     if (fCategories)
  170.     fCategories->addOrFind(categoryName);
  171. }
  172. void RangeTokenMap::addRangeMap(const XMLCh* const categoryName,
  173.                                 RangeFactory* const rangeFactory) {
  174.     if (fRangeMap)
  175.     fRangeMap->put((void*)categoryName, rangeFactory);
  176. }
  177. void RangeTokenMap::addKeywordMap(const XMLCh* const keyword,
  178.                                  const XMLCh* const categoryName) {
  179.     if (fCategories == 0 || fTokenRegistry == 0)
  180.         return;
  181. unsigned int categId = fCategories->getId(categoryName);
  182. if (categId == 0) {
  183. ThrowXML1(RuntimeException, XMLExcepts::Regex_InvalidCategoryName, categoryName);
  184. }
  185.     if (fTokenRegistry->containsKey(keyword)) {
  186.         RangeTokenElemMap* elemMap = fTokenRegistry->get(keyword);
  187. if (elemMap->getCategoryId() != categId)
  188. elemMap->setCategoryId(categId);
  189. return;
  190. }
  191. fTokenRegistry->put((void*) keyword, new RangeTokenElemMap(categId));
  192. }
  193. // ---------------------------------------------------------------------------
  194. //  RangeTokenMap: Setter methods
  195. // ---------------------------------------------------------------------------
  196. void RangeTokenMap::setRangeToken(const XMLCh* const keyword,
  197.                                   RangeToken* const tok,const bool complement) {
  198.     if (fTokenRegistry == 0)
  199. return;
  200. if (fTokenRegistry->containsKey(keyword)) {
  201.         fTokenRegistry->get(keyword)->setRangeToken(tok, complement);
  202.     }
  203.     else {
  204. ThrowXML1(RuntimeException, XMLExcepts::Regex_KeywordNotFound, keyword);
  205. }
  206. }
  207. // ---------------------------------------------------------------------------
  208. //  RangeTokenMap: Initialization methods
  209. // ---------------------------------------------------------------------------
  210. void RangeTokenMap::initializeRegistry() {
  211. XMLMutexLock lockInit(&fMutex);
  212. if (fRegistryInitialized)
  213. return;
  214.     fTokenFactory = new TokenFactory();
  215.     fTokenRegistry = new RefHashTableOf<RangeTokenElemMap>(109);
  216.     fRangeMap = new RefHashTableOf<RangeFactory>(29);
  217. fCategories = new XMLStringPool();
  218. fRegistryInitialized = true;
  219. }
  220. // ---------------------------------------------------------------------------
  221. //  RangeTokenMap: Instance methods
  222. // ---------------------------------------------------------------------------
  223. RangeTokenMap* RangeTokenMap::instance() {
  224.     static XMLRegisterCleanup instanceCleanup;
  225.     if (!fInstance) {
  226.         RangeTokenMap* t = new RangeTokenMap();
  227.         if (XMLPlatformUtils::compareAndSwap((void **)&fInstance, t, 0) != 0)
  228.         {
  229.             delete t;
  230.         }
  231.         else
  232.         {
  233.             instanceCleanup.registerCleanup(reinitInstance);
  234.         }
  235.     }
  236.     return (fInstance);
  237. }
  238. // -----------------------------------------------------------------------
  239. //  Notification that lazy data has been deleted
  240. // -----------------------------------------------------------------------
  241. void RangeTokenMap::reinitInstance() {
  242. delete fInstance;
  243. fInstance = 0;
  244. }
  245. /**
  246.   * End of file RangeTokenMap.cpp
  247.   */