首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在VB.Net中使用browscap.ini

在VB.Net中使用browscap.ini
EN

Stack Overflow用户
提问于 2017-02-20 23:55:09
回答 1查看 174关注 0票数 0

从2013年到现在(3年多),我一直在我的主VB.Net项目中使用http://www.useragentstring.com/从用户代理字符串中获取浏览器名称/版本和操作系统名称/版本,以将统计数据添加到我的本地web应用程序中。

但是,最近,在过去的几个月里,这个网站一直不可靠,有很多停机时间。因此,为了避免在我的统计数据中丢失数据,我搜索了本地解决方案,而不是在线解决方案。我发现http://browscap.org/是一个老网站(自1998年以来),至今仍在上传更新的用户代理信息(browscap.ini)。它是为PHP设计的,但我在那里找到了一个C#实现:https://www.gocher.me/C-Sharp-Browscap

但作为一名VB.Net开发人员,我没有找到任何针对它的VB实现。我用谷歌搜索了很多次,但都没有成功。有没有人给VB.NET买了一个?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2017-02-20 23:55:09

我终于可以把C#解决方案转换成VB.NET了。

代码语言:javascript
复制
Public Class CompareByLength
Implements IComparer(Of String)
    Private Function Compare(ByVal x As String, ByVal y As String) as Integer _
        Implements IComparer(Of String).Compare
            If x Is Nothing Then
                If y Is Nothing Then
                    Return 0
                Else
                    Return 1
                End If
            Else
                If y Is Nothing Then
                    Return -1
                Else
                    Dim retval As Integer = x.Length.CompareTo(y.Length)
                    If retval <> 0 Then
                        Return -retval
                    Else
                        return -x.CompareTo(y)
                    End If

                End If
            End If
        End Function
End Class

Public Class BrowsCap
    Private Declare Function GetPrivateProfileSectionNames Lib "kernel32.dll" Alias "GetPrivateProfileSectionNamesA" (ByVal lpReturnedString As Byte(), ByVal nSize As Integer, ByVal lpFileName As String) As Integer
    Private Declare Function GetPrivateProfileSection Lib "kernel32.dll" Alias "GetPrivateProfileSectionA" (ByVal lpAppName As String, ByVal lpReturnedBuffer As Byte(), ByVal nSize As Integer, ByVal lpFileName As String) As Integer
    Private Declare Function GetPrivateProfileString Lib "kernel32.dll" Alias "GetPrivateProfileStringA" (ByVal lpApplicationName As String, ByVal lpKeyName As String, ByVal lpDefault As String, ByVal lpReturnedBuffer As String, ByVal nSize As Integer, ByVal lpFileName As String) As Integer

    Private path As String
    Private sections As String()

    Private Function GetSectionNames() As String()
        Dim maxsize As Integer = 500
        Do
            Dim bytes(maxsize) As Byte
            Dim size As Integer = GetPrivateProfileSectionNames(bytes, maxsize, path)
            If size < maxsize - 2 Then
                Dim Selected As String = Encoding.ASCII.GetString(bytes, 0, size - (IIf(size > 0, 1, 0)))
                Return Selected.Split(New Char() {ControlChars.NullChar})
            End If
            maxsize = maxsize * 2
        Loop
    End Function

    Public Sub IniFileName(ByVal INIPath As String)
        path = INIPath
        sections = GetSectionNames()
        Array.Sort(sections, New CompareByLength())
    End Sub

    public Function IniReadValue(ByVal Section As String, ByVal Key As String) As String
      Dim temp As New StringBuilder(255)
      Dim i As Integer = GetPrivateProfileString(Section, Key, "", temp.ToString(), 255, path)
      Return temp.ToString()
    End Function

    Private Function findMatch(ByVal Agent As String) As String
      If sections IsNot Nothing Then
        For Each SecHead As String In sections
          If (SecHead.IndexOf("*", 0) = -1) And (SecHead.IndexOf("?", 0) = -1) And (SecHead = Agent) Then
            If IniReadValue(SecHead, "parent") <> "DefaultProperties" Then
              Return SecHead
            End If
          End If
        Next
        For Each SecHead As String In sections
          Try
            If (SecHead.IndexOf("*", 0) > -1) Or (SecHead.IndexOf("?", 0) > -1) Then
              if Regex.IsMatch(Agent, "^" + Regex.Escape(SecHead).Replace("\*", ".*").Replace("\?", ".") + "$") Then
                Return SecHead
              End If
            End If
          Catch ex As Exception
            'Console.WriteLine(ex)
          End Try
        Next
        Return "*"
      End If
      Return ""
    End Function

    Public Function getValues(ByVal Agent As String) As NameValueCollection 
      Dim match As String = findMatch(Agent)
      Dim col As NameValueCollection = New NameValueCollection()
      Do
        Dim entries() As string
        Dim goon As Boolean = true
        Dim maxsize As Integer = 500
        While goon
          Dim bytes(maxsize) As Byte
          Dim size As Integer = GetPrivateProfileSection(match, bytes, maxsize, path)
          If size < maxsize - 2
            Dim section As String = Encoding.ASCII.GetString(bytes, 0, size - IIf(size > 0, 1, 0))
            entries = section.Split(New Char() {ControlChars.NullChar})
            goon = False
          End If
          maxsize = maxsize * 2
        End While
        match = ""
        If entries.Length > 0 Then
          For Each entry As String In entries
            Dim ent As String() = entry.Split(New Char() {"="C})
            If ent(0) = "Parent" Then
              match = ent(1)
            else if col(ent(0)) is nothing Then
              col.Add(ent(0), ent(1))
            End If
          Next
        End If
      Loop While match <> ""
      Return col
    End Function
End Class

下面是如何使用它:

代码语言:javascript
复制
Dim dict As Dictionary(Of String, Object) = New Dictionary(Of String, Object)
Dim bc As New BrowsCap
bc.IniFileName(Server.MapPath("/App_Data/lite_asp_browscap.ini"))
Dim Entry As NameValueCollection = bc.getValues(Request.UserAgent)
For Each s As String In Entry.AllKeys
    dict.Add(s, Entry(s))
Next
' dict("Browser") will contains browser name like "IE" or "Chrome".
' dict("Version") will contains browser version like "11.0" or "56.0".
' dict("Platform") will contains OS name and version like "Win7".

剩下的唯一要做的就是偶尔刷新一下我的browscap.ini (或lite_asp_browscap.ini) (比如一周一次)。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/42348988

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档