android,UTF8 - 如何确保 UTF8 用于共享首选项

jam*_*esc 5 java android utf-8

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

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

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

<?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>
Run Code Online (Sandbox Code Playgroud)

处理这个问题的类是

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);     
    }
}
Run Code Online (Sandbox Code Playgroud)

我假设该设置 <?xml version="1.0" encoding="utf-8"?>将确保文本以 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><[\xA3~?= ~~ \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);
}
Run Code Online (Sandbox Code Playgroud)

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

所以

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

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

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

And*_*ock 3

关键是要理解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 - 如何处理 PG 错误不完整的多字节字符中的服务器):

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

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

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

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

为了帮助调试,我建议:

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