📄 cbregexp.cpp
字号:
**
** @param[in] szPattern 検索パターン
** @param[in] szPattern2 置換パターン(NULLなら検索)
** @param[in] bOption 検索オプション
**
** @retval ライブラリに渡す検索パターンへのポインタを返す
** @note 返すポインタは、呼び出し側で delete すること
**
** @date 2003.05.03 かろと 関数に切り出し
*/
char* CBregexp::MakePattern( const char* szPattern, const char* szPattern2, int bOption )
{
static const char CRLF[] = "\r\n"; //!< 復帰?改行
static const char CR[] = "\r"; //!< 復帰
static const char LF[] = "\n"; //!< 改行
static const char LFCR[] = "\n\r"; //!< 改行?復帰
static const char BOT_SUBST[] = "s/\\$(\\)*)$/([\\\\r\\\\n]+)\\$$1/k"; //!< 行末パターンの置換用パターン
int nLen; //!< szPatternの長さ
BREGEXP* sReg = NULL; //!< コンパイル構造体
char szMsg[80] = ""; //!< エラーメッセージ
char szAdd2[5] = ""; //!< 行末あり置換の $数字 格納用
int nParens = 0; //!< 検索パターン(szPattern)中の括弧の数(行末時に使用)
char *szNPattern; //!< 検索パターン
nLen = CheckPattern( szPattern );
if( (m_ePatType & PAT_BOTTOM) != 0 ) {
bool bJustDollar = false; // 行末指定の$のみであるフラグ($の前に \r\nが指定されていない)
szNPattern = MakePatternSub(szPattern, NULL, NULL, bOption);
int matched = BMatch( szNPattern, CRLF, CRLF+sizeof(CRLF)-1, &sReg, szMsg );
if( matched >= 0 ) {
// szNPatternが不正なパターン等のエラーでなかった
// エラー時には sRegがNULLのままなので、sReg->nparensへのアクセスは不正
nParens = sReg->nparens; // オリジナルの検索文字列中の()の数を記憶
if( matched > 0 ) {
if( sReg->startp[0] == &CRLF[1] && sReg->endp[0] == &CRLF[1] ) {
if( BMatch( NULL, CR, CR+sizeof(CR)-1, &sReg, szMsg ) > 0 && sReg->startp[0] == &CR[1] && sReg->endp[0] == &CR[1] ) {
if( BMatch( NULL, LF, LF+sizeof(LF)-1, &sReg, szMsg ) > 0 && sReg->startp[0] == &LF[0] && sReg->endp[0] == &LF[0] ) {
if( BMatch( NULL, LFCR, LFCR+sizeof(LFCR)-1, &sReg, szMsg ) > 0 && sReg->startp[0] == &LFCR[0] && sReg->endp[0] == &LFCR[0] ) {
// 検索文字列は 行末($)のみだった
bJustDollar = true;
}
}
}
}
} else {
if( BMatch( NULL, CR, CR+sizeof(CR)-1, &sReg, szMsg ) <= 0 ) {
if( BMatch( NULL, LF, LF+sizeof(LF)-1, &sReg, szMsg ) <= 0 ) {
if( BMatch( NULL, LFCR, LFCR+sizeof(LFCR)-1, &sReg, szMsg ) <= 0 ) {
// 検索文字列は、文字+行末($)だった
bJustDollar = true;
}
}
}
}
BRegfree(sReg);
sReg = NULL;
}
delete [] szNPattern;
if( bJustDollar == true || (m_ePatType & PAT_TAB) != 0 ) {
// 行末指定の$ or 行頭行末指定 なので、検索文字列を置換
if( BSubst( BOT_SUBST, szPattern, szPattern + nLen, &sReg, szMsg ) > 0 ) {
szPattern = sReg->outp;
if( szPattern2 != NULL ) {
// 置換パターンもあるので、置換パターンの最後に $(nParens+1)を追加
wsprintf( szAdd2, "$%d", nParens + 1 );
}
}
// sReg->outp のポインタを参照しているので、sRegを解放するのは最後に
}
}
szNPattern = MakePatternSub( szPattern, szPattern2, szAdd2, bOption );
if( sReg != NULL ) {
BRegfree(sReg);
}
return szNPattern;
}
/*!
JRE32のエミュレーション関数.空の文字列に対して検索?置換を行うことにより
BREGEXP構造体の生成のみを行う.
@param[in] szPattern0 検索or置換パターン
@param[in] szPattern1 置換後文字列パターン(検索時はNULL)
@param[in] bOption 0x01:大文字小文字の区別をする。
@retval true 成功
@retval false 失敗
*/
bool CBregexp::Compile( const char *szPattern0, const char *szPattern1, int bOption )
{
// DLLが利用可能でないときはエラー終了
if( !IsAvailable() )
return false;
// BREGEXP構造体の解放
ReleaseCompileBuffer();
// ライブラリに渡す検索パターンを作成
// 別関数で共通処理に変更 2003.05.03 by かろと
char *szNPattern = MakePattern( szPattern0, szPattern1, bOption );
m_szMsg[0] = '\0'; //!< エラー解除
if (szPattern1 == NULL) {
// 検索実行
BMatch( szNPattern, m_tmpBuf, m_tmpBuf+1, &m_pRegExp, m_szMsg );
} else {
// 置換実行
BSubst( szNPattern, m_tmpBuf, m_tmpBuf+1, &m_pRegExp, m_szMsg );
}
delete [] szNPattern;
// メッセージが空文字列でなければ何らかのエラー発生。
// サンプルソース参照
if( m_szMsg[0] ){
ReleaseCompileBuffer();
return false;
}
// 行頭条件チェックは、MakePatternに取り込み 2003.05.03 by かろと
return true;
}
/*!
JRE32のエミュレーション関数.既にあるコンパイル構造体を利用して検索(1行)を
行う.
@param[in] target 検索対象領域先頭アドレス
@param[in] len 検索対象領域サイズ
@param[in] nStart 検索開始位置.(先頭は0)
@retval true Match
@retval false No Match または エラー。エラーは GetLastMessage()により判定可能。
*/
bool CBregexp::Match( const char* target, int len, int nStart )
{
int matched; //!< 検索一致したか? >0:Match, 0:NoMatch, <0:Error
// DLLが利用可能でないとき、または構造体が未設定の時はエラー終了
if( (!IsAvailable()) || m_pRegExp == NULL ){
return false;
}
m_szMsg[0] = '\0'; //!< エラー解除
// 拡張関数がない場合は、行の先頭("^")の検索時の特別処理 by かろと
if (BMatchEx == NULL) {
/*
** 行頭(^)とマッチするのは、nStart=0の時だけなので、それ以外は false
*/
if( (m_ePatType & PAT_TOP) != 0 && nStart != 0 ) {
// nStart!=0でも、BMatch()にとっては行頭になるので、ここでfalseにする必要がある
return false;
}
// 検索文字列=NULLを指定すると前回と同一の文字列と見なされる
matched = BMatch( NULL, target + nStart, target + len, &m_pRegExp, m_szMsg );
} else {
// 検索文字列=NULLを指定すると前回と同一の文字列と見なされる
matched = BMatchEx( NULL, target, target + nStart, target + len, &m_pRegExp, m_szMsg );
}
m_szTarget = target;
if ( matched < 0 || m_szMsg[0] ) {
// BMatchエラー
// エラー処理をしていなかったので、nStart>=lenのような場合に、マッチ扱いになり
// 無限置換等の不具合になっていた 2003.05.03 by かろと
return false;
} else if ( matched == 0 ) {
// 一致しなかった
return false;
}
return true;
}
//<< 2002/03/27 Azumaiya
/*!
正規表現による文字列置換
既にあるコンパイル構造体を利用して置換(1行)を
行う.
@param[in] szTarget 置換対象データ
@param[in] nLen 置換対象データ長
@param[in] nStart 置換開始位置(0からnLen未満)
@retval true 成功
@retval false 失敗
*/
bool CBregexp::Replace(const char *szTarget, int nLen, int nStart)
{
int result;
// DLLが利用可能でないとき、または構造体が未設定の時はエラー終了
if( !IsAvailable() || m_pRegExp == NULL )
{
return false;
}
// From Here 2003.05.03 かろと
// nLenが0だと、BSubst()が置換に失敗してしまうので、代用データ(m_tmpBuf)を使う
if( nLen == 0 ) {
szTarget = m_tmpBuf;
nLen = 1;
}
// To Here 2003.05.03 かろと
m_szMsg[0] = '\0'; //!< エラー解除
if (BSubstEx == NULL) {
result = BSubst( NULL, szTarget + nStart, szTarget + nLen, &m_pRegExp, m_szMsg );
} else {
result = BSubstEx( NULL, szTarget, szTarget + nStart, szTarget + nLen, &m_pRegExp, m_szMsg );
}
m_szTarget = szTarget;
// メッセージが空文字列でなければ何らかのエラー発生。
// サンプルソース参照
if( m_szMsg[0] ) {
return false;
}
if( !result ) {
// 置換するものがなかった
return false;
}
return true;
}
//>> 2002/03/27 Azumaiya
/*[EOF]*/
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -