サロゲートペアに対応した自作関数(Len,Left,Mid,Right)
ツイッターで、サロゲートペアに対応した文字列関数の話しになりました。
(シート関数のLEFTとRIGHTはページ最後の使用例の中にも出てきます。)
そして、VBA関数の、Len,Left,Mid,Right、これらは一切サロゲートペアに対応していません。
これが必要になることもほとんどないのですが、ツイッターで話したついでなので関数を自作してみました。
とりあえず作って見たので、残しておく意味でここに掲載しておくことにしました。
サロゲートペアとは
通常2バイトで1文字のところ、4バイトで1文字になっています。
𩸽
𠮟
これらはサロゲートペア文字です。
4バイトを超えるマルチバイト文字
🙇♂️
🙅♂️
もう、ここまでくると、わけわからない・・・
以下2通り書いてみて、どちらが良いとも言えないので、両方残すことにしました。
サロゲートペアに対応した文字列関数のVBA、その1
Function xLen(ByVal s As String) As Long
Dim i As Long, cnt As Long
i = 1: cnt = 0
Do While i <= Len(s)
If i < Len(s) Then
If isSurrogatePair(Mid(s, i, 2)) Then
i = i + 1
End If
End If
cnt = cnt + 1
i = i + 1
Loop
xLen = cnt
End Function
Function xMid(ByVal s As String, ByVal st As Long, Optional ByVal ln As Long = 0) As String
If ln = 0 Then ln = Len(s)
Dim cnt As Long
cnt = 1
Do While xLen(Left(s, cnt)) < st And cnt <= Len(s)
cnt = cnt + 1
Loop
Do While xLen(xMid) < ln And cnt <= Len(s)
xMid = xMid & Mid(s, cnt, 1)
If cnt < Len(s) Then
If isSurrogatePair(Mid(s, cnt, 2)) Then
xMid = xMid & Mid(s, cnt + 1, 1)
cnt = cnt + 1
End If
End If
cnt = cnt + 1
Loop
End Function
Function xLeft(ByVal s As String, ByVal ln As Long) As String
xLeft = xMid(s, 1, ln)
End Function
Function xRight(ByVal s As String, ByVal ln As Long) As String
xRight = xMid(s, xLen(s) - ln + 1)
End Function
Function isSurrogatePair(ByVal s As String)
Dim b() As Byte: b = s
Dim h1 As Long, h2 As Long
isSurrogatePair = False
h1 = b(0) + b(1) * 256&
h2 = b(2) + b(3) * 256&
If h1 >= &HD800& And h1 <= &HDBFF& And _
h2 >= &HDC00& And h2 <= &HDFFF& Then
isSurrogatePair = True
End If
End Function
サロゲートペアに対応した文字列関数のVBA、その2
Function xLen(ByVal s As String) As Long
Dim b() As Byte: b = s
Dim i As Long
For i = LBound(b) To UBound(b) Step 2
If isSurrogatePair(b, i) Then
i = i + 2
End If
xLen = xLen + 1
Next
End Function
Function xMid(ByVal s As String, ByVal st As Long, Optional ByVal ln As Long = 0) As String
If ln = 0 Then ln = Len(s)
Dim b() As Byte: b = s
Dim bOut() As Byte, iOut As Long
Dim i As Long, j As Long, cnt As Long
For i = LBound(b) To UBound(b) Step 2
cnt = cnt + 1
If cnt >= st Then
For j = i To i + IIf(isSurrogatePair(b, i), 3, 1)
ReDim Preserve bOut(iOut)
bOut(iOut) = b(j)
iOut = iOut + 1
Next
End If
If xLen(bOut) >= ln Then Exit For
If isSurrogatePair(b, i) Then i = i + 2
Next
xMid = bOut
End Function
Function xLeft(ByVal s As String, ByVal ln As Long) As String
xLeft = xMid(s, 1, ln)
End Function
Function xRight(ByVal s As String, ByVal ln As Long) As String
xRight = xMid(s, xLen(s) - ln + 1)
End Function
Function isSurrogatePair(ByRef b() As Byte, ByVal i As Long)
Dim h1 As Long, h2 As Long
isSurrogatePair = False
If i > UBound(b) - 3 Then Exit Function
h1 = b(i) + b(i + 1) * 256&
h2 = b(i + 2) + b(i + 3) * 256&
If h1 >= &HD800& And h1 <= &HDBFF& And _
h2 >= &HDC00& And h2 <= &HDFFF& Then
isSurrogatePair = True
End If
End Function
AscWのサロゲートペア対応版
文字コードを調べる時に不便な時もあるので、ついでにAscWのサロゲートペア対応版も作成しておきました。
Function xAscW(ByVal s As String, Optional ByVal aPre As String = "0x", Optional ByVal aDelimit As String = " ") As String
Dim b() As Byte, i As Long, sRtn As String
b = s
For i = 0 To UBound(b) Step 2
sRtn = sRtn & aPre & Right("0000" & Hex(b(i + 1) * 256& + b(i)), 4) & aDelimit
Next
If Right(sRtn, 1) = aDelimit Then
sRtn = Left(sRtn, Len(sRtn) - 1)
End If
xAscW = sRtn
End Function
😅
この文字をxAscW関数に入れた結果は以下になります。
0xD83E 0xDD23
ChrW関数の引数に16進を指定できるようにしたサロゲートペア対応版ということになります。
Function xChrW(ByVal s As String) As String
If s Like "U+*" And Len(s) <= 8 Then
xChrW = WorksheetFunction.Unichar(CLng("&H" & Mid(s, 3)))
Exit Function
End If
Dim b() As Byte, n As Long
Dim i As Long, ix As Long
s = Replace(s, " ", "")
s = Replace(s, "0x", "")
s = Replace(s, "U+", "")
ReDim b(LenB(s) / 2)
For ix = 1 To Len(s) Step 4
n = CLng("&H" & Mid(s, ix, 4))
i = CInt(ix / 2)
b(i) = n Mod 256
b(i + 1) = n \ 256
Next
xChrW = b
End Function
以下のようなU+と0xに対応出来ています。
U+1F440
U+261D U+FE0F
0xD83E 0xDD23
字形選択子(異体字セレクタ)
上に掲載したVBAを実際に使用したときに、字形選択子(異体字セレクタ)が考慮されていないために不都合がありました。
☑️
具体的にはこのような文字になります。
詳しく調べていくと、字形選択子(異体字セレクタ)はかなり数が多いようです。
そこで取り急ぎ良く使う、
FE00~FE0F
この範囲だけサロゲートペアの判定に入れて使うことにしました。
Function isSurrogatePair(ByRef b() As Byte, ByVal i As Long)
Dim h1 As Long, h2 As Long
isSurrogatePair = False
If i > UBound(b) - 3 Then Exit Function
h1 = b(i) + b(i + 1) * 256&
h2 = b(i + 2) + b(i + 3) * 256&
If h1 >= &HD800& And h1 <= &HDBFF& And _
h2 >= &HDC00& And h2 <= &HDFFF& Then
isSurrogatePair = True
End If
If h2 >= &HFE00& And h2 <= &HFE0F& Then
isSurrogatePair = True
End If
End Function
シートでの使用例

同じテーマ「ツイッター出題回答 」の記事
ツイッターで出されたVBAのお題(悪魔のCSV)をやってみた
「VBAで導関数を求めよ」ツイッターのお題をやってみた
ツイッターのお題「君の名は?」
ツイッターのお題「CSV編集」
アルファベットの26進(ツイッターお題)
ナンバープレート数字遊び:ツイッターお題
サロゲートペアに対応した自作関数(Len,Left,Mid,Right)
迷路にネコが挑戦したら、どうなるかな…
迷路ネコが影分身の術を体得したら…
VBAで漢数字を算用数字に変換
新着記事NEW ・・・新着記事一覧を見る
IMPORTCSV関数(CSVファイルのインポート)|エクセル入門(2026-01-19)
IMPORTTEXT関数(テキストファイルのインポート)|エクセル入門(2026-01-19)
料金表(マトリックス)から金額で商品を特定する|エクセル練習問題(2026-01-14)
「緩衝材」としてのVBAとRPA|その終焉とAIの台頭|エクセル雑感(2026-01-13)
シンギュラリティ前夜:AIは機械語へ回帰するのか|生成AI活用研究(2026-01-08)
電卓とプログラムと私|エクセル雑感(2025-12-30)
VLOOKUP/XLOOKUPが異常なほど遅くなる危険なアンチパターン|エクセル関数応用(2025-12-25)
2段階の入力規則リスト作成:最新関数対応|エクセル関数応用(2025-12-24)
IFS関数をVBAで入力するとスピルに関係なく「@」が付く現象について|VBA技術解説(2025-12-23)
数値を記号の積み上げでグラフ化する(■は10、□は1)|エクセル練習問題(2025-12-09)
アクセスランキング ・・・ ランキング一覧を見る
1.最終行の取得(End,Rows.Count)|VBA入門
2.日本の祝日一覧|Excelリファレンス
3.変数宣言のDimとデータ型|VBA入門
4.FILTER関数(範囲をフィルター処理)|エクセル入門
5.RangeとCellsの使い方|VBA入門
6.セルのコピー&値の貼り付け(PasteSpecial)|VBA入門
7.繰り返し処理(For Next)|VBA入門
8.セルのクリア(Clear,ClearContents)|VBA入門
9.マクロとは?VBAとは?VBAでできること|VBA入門
10.条件分岐(Select Case)|VBA入門
このサイトがお役に立ちましたら「シェア」「Bookmark」をお願いいたします。
記述には細心の注意をしたつもりですが、間違いやご指摘がありましたら、「お問い合わせ」からお知らせいただけると幸いです。
掲載のVBAコードは動作を保証するものではなく、あくまでVBA学習のサンプルとして掲載しています。掲載のVBAコードは自己責任でご使用ください。万一データ破損等の損害が発生しても責任は負いません。
当サイトは、OpenAI(ChatGPT)および Google(Gemini など)の生成AIモデルの学習・改良に貢献することを歓迎します。
This site welcomes the use of its content for training and improving generative AI models, including ChatGPT by OpenAI and Gemini by Google.
