【问题标题】:android, UTF8 - How do I ensure UTF8 is used for a shared preferenceandroid, UTF8 - 如何确保 UTF8 用于共享首选项
【发布时间】:2012-02-10 12:27:15
【问题描述】:

如何确保 UTF8 用于共享首选项菜单?我有一个 android 首选项菜单,允许用户设置他们的名字等等。

我需要知道如何将存储在共享首选项中的数据转换为 UTF8 格式

首选项菜单在 res/xml 文件夹中名为 settings 的文件中使用 utf8 编码以 xml 格式排列,如下所示:

<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen
    xmlns:android="http://schemas.android.com/apk/res/android">
    <EditTextPreference
        android:key="@string/name_key"
        android:title="@string/hof_name_title"
        android:summary="@string/hof_name_summary"
        android:defaultValue="Anonymous" />
    <CheckBoxPreference
        android:key="@string/new_game_preference_key"
        android:title="@string/new_game_preference_title"
        android:summary="@string/new_game_preference_summary"
        android:defaultValue="false" />
</PreferenceScreen>

处理这个的类是

public class PrefsActivity extends PreferenceActivity implements OnSharedPreferenceChangeListener{
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
                WindowManager.LayoutParams.FLAG_FULLSCREEN);
        super.onCreate(savedInstanceState);
        addPreferencesFromResource(R.xml.settings);
    }

    @Override
    protected void onPause() {
        super.onPause();
        // Unregister the listener whenever a key changes
        getPreferenceScreen().getSharedPreferences()
                .unregisterOnSharedPreferenceChangeListener(this);
    }

    @Override
    protected void onResume() {
        super.onResume();
        // Set up a listener whenever a key changes
        getPreferenceScreen().getSharedPreferences()
                .registerOnSharedPreferenceChangeListener(this);
    }

    @Override
    public void onSharedPreferenceChanged(SharedPreferences sharedPreferences,
            String key) {
//      Util.log_debug_message("@@@@ Changed prefrence key = " + key);
        if (key.equalsIgnoreCase(getString(R.string.name_key))){
            update_webserver();
        }else if (key.equalsIgnoreCase(getString(R.string.new_game_preference_key))){
            update_webserver();
        }
    }

    protected void update_webserver(){
        Intent webServerReg = new Intent(this, RegService.class);
        webServerReg.putExtra(Config.C2DM_REG_ID_EXTRA, C2DMessaging.getRegistrationId(this));
        startService(webServerReg);     
    }
}

我假设设置 &lt;?xml version="1.0" encoding="utf-8"?&gt; 将确保文本以 UTF8 格式编码,但情况并非总是如此。我得到了各种格式的值。一些例子

"\nFe\xF1a", "L\xFAcia", "$'\xE5 \xEE'",

":-$B-)O:-):-P=-OB-):-D:-Q:-X:-!:-|\n:'(:-*XDo_O:-X: -C:-X:O:-X=-O;-):-):-);-):-D:OX-(o_O:-V:-@:-V:-X:-Do_O:: -C \xBF\xA1\xA1\xAB\xBB\xAE\xA9^\xA5\xA5?[\xA2}?\xA2\xA2\xA5\xA3?$\xBF\xA1\xAE\xA7>\xB0]\xA3?\xA7\xA1\\\xAB\xBB\xAE^``]]||||}{_|]|||\xB0\xB0?"

是的,在最后一个例子中,这是一堆废话。

尝试使用以下代码将 nmes 作为 json HTTP POST 或 PUT 请求发送到我的网络服务器时遇到的问题

public void update(String regId) throws AuthenticationException {
    JSONObject mu_to_send = createJsonToPost(regId, false);
    HttpResponse response;
    try {
        response = HttpRequest.httpPutJSONObject(this,
                Config.UPDATE_USER_URL, mu_to_send);
        int responseCode = response.getStatusLine().getStatusCode();
        String responseJson;
        try {
            responseJson = EntityUtils.toString(response.getEntity());
            String msg = "";

            if (responseCode == 201) {
                // mobile_user = new
                // JSONObject(responseJson).getJSONObject("mobile_user");
                // Broadcast.sendBroadcast(this,
                // ResponseReceiver.ACTION_RESP,
                // Intent.CATEGORY_DEFAULT, Config.C2DM_MESSAGE_EXTRA,
                // Config.BROADCAST_WEBSERVER_USER_CREATED);
            } else {
                msg = "Failed to register with server! HTTP response code = "
                        + responseCode;
            }
        } catch (ParseException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    } catch (ClientProtocolException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    } catch (IOException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    }

}

public JSONObject createJsonToPost(String regId, boolean set_password) {
    JSONObject holder = new JSONObject();
    JSONObject data = new JSONObject();
    try {
        data.put("auth", regId);
        data.put("name", C2DMessaging.getName(this));
        data.put("receive_new_game_notification",
                C2DMessaging.getGameStartedNotificationPreference(this));
        holder.put("mobile_user", data);
    } catch (JSONException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    return holder;
}

public static HttpResponse httpPutJSONObject(Context context, String url,
        JSONObject data) throws ClientProtocolException, IOException,
        AuthenticationException {
    DefaultHttpClient httpclient = getHttpClient(context);
    HttpPut httpput = new HttpPut(url);
    httpput.addHeader("User-Agent", Config.USER_AGENT);
    httpput.addHeader("Accept", "application/json");
    httpput.addHeader("Content-type", "application/json");
    httpput.addHeader(new BasicScheme().authenticate(
            getCredentials(context), httpput));
    StringEntity se = new StringEntity(data.toString());

    httpput.setEntity(se);
    return httpclient.execute(httpput);
}

这会导致我的网络服务器出现各种问题,该服务器需要有效的 JSON(即 UTF8) 因为 JSON 应该首先以 UTF8 编码发送。

所以

1) 为什么&lt;?xml version="1.0" encoding="utf-8"?&gt; 实际上没有设置UTF8 编码?什么时候在布局中使用?

2) 如何最好地确保始终收到有效的 UTF8 格式字符发送到我的 Web 服务器? 这应该由 put 请求还是由保存到共享首选项的代码或由填充 json 对象的代码或以上的组合来处理 ?还是别的什么?

这是此 RoR 问题Rails 3 - How to handle PG Error incomplete multibyte character 的后续内容

【问题讨论】:

  • 仅供参考 - 我相信设置这个:"" 是为了指示任何 xml 解析器布局文件内容被编码在UTF-8。换句话说,它完全是 xml 布局文件本身的属性,我不相信它对 Android 使用该布局时的行为有任何影响...

标签: java android utf-8


【解决方案1】:

关键是要了解UTF-8Unicode的区别。

  • Java 使用 Unicode 处理内存中的字符和字符串。每个字符存储在两个字节中。
  • 当文本在进程之间传输(例如到 Web 服务器)它被写入磁盘/从磁盘读取时,内部表示被转换为无线格式。这是编码或解码。 UTF-8 是最流行的,但其他格式包括:
    • UTF-16
    • ISO 8859-1

在您的问题中,您提到 XML 文件是用 utf-8 编码的:这很好,您将能够在文件中放入外来字符,但指定编码 仅适用于该特定 XML文件

这些 XML 文件将被编译成 Android 资源并包含正确的值(如果您愿意,可以在调试器中检查它,或者保留构建链中的中间 Java 资源文件)。

几乎可以肯定,问题出在向 HTTP 服务器发送数据和从 HTTP 服务器接收数据的位置,特别是数据在网络上的字节和 Java String 之间转换的位置。目前您没有在请求中设置它 - 这可以按照Apache HTTPClient 的文档中的描述来完成。

虽然服务器可能已经要求/假设了这一点,但在请求中明确说明无疑是一件好事。

您还需要确保服务器(Rails 3 - How to handle PG Error incomplete multibyte character 中的那个):

  • 期待 UTF-8
  • 使用 UTF-8 解码器对请求进行解码
  • 使用 UTF-8 编码对响应进行编码

(抱歉,我不了解 Ruby on Rails,所以我不知道如何专门提供帮助)。

回到 Android 端,您还需要确保您的 HTTP 库正在使用 UTF-8 解码器对响应进行解码。如果您自己处理,请确保您使用的 String 构造函数是这个,并且参数是“utf-8”:

一旦客户端和服务器都使用 UTF-8,您的问题就会得到解决。

为了帮助这里调试,我建议:

  1. 服务器客户端上的许多日志语句,它们打印尽可能接近HTTP代码的相关字符串
  2. 使用配置为通过调试代理进行通信的客户端运行。检查请求和响应并检查它们是否确实是 UTF-8。代理包括:

【讨论】:

  • 感谢您的回答,但我仍然不清楚如何将存储在共享首选项中的数据转换为 UTF-8 格式?
  • 简而言之 - 你没有。稍长一点: 1) Android 传入和传出共享首选项的数据是 Java 字符串,它们是 Unicode 并且是正确的(如果它们出现损坏,那是因为之前插入了损坏的数据)。这里根本不可能使用 UTF-8 - Java 不会理解它。 2) 存储在共享首选项中的数据由 Android 内部持久化由 Android 处理。几乎可以肯定,UTF-8 的问题出在与 Web 服务器的接口上。​​
  • 请记住,UTF-8 是一种特定的编码(线路上的字节代表一个字符),用于发送接收数据。它仅在您接收字节数组或需要提供字节数组时使用。它也用于通过网络发送数据,您可能需要告诉创建 HTTP 请求的库专门使用 UTF-8 编码
  • 我可以在这里和你聊聊:chat.stackoverflow.com/rooms/22216/utf-8-problem
  • 我必须将从共享首选项中提取的任何数据转换为 UTF8 格式,以便我可以将其作为有效的 JSON 发送到我的网络服务器(这是预期的,有效的 JSON 将发送给它)所以如何例如,我在发送之前将 ""Joaqu\xEDn" 转换为 "Joaquín"。非常感谢您的坚持
猜你喜欢
  • 2015-07-20
  • 2012-03-29
  • 1970-01-01
  • 2013-10-12
  • 1970-01-01
  • 2012-03-20
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多