2012-02-10 69 views
4

如何確保UTF8用於共享首選項菜單?我有一個android偏好菜單,允許用戶設置他們的名字等等。android,UTF8 - 如何確保UTF8用於共享首選項

我需要知道如何存儲在共享偏好將數據轉換爲UTF-8格式

偏好菜單中的XML使用UTF8編碼的文件在res/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);  
    } 
} 

我認爲設置<?xml version="1.0" encoding="utf-8"?>將確保文本將在UTF8編碼FO類rmat,但情況並非總是如此。我以各種格式獲取價值。一些例子

「\ 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?「

是的,這是最後一個例子中的一組漏斗。

的問題,我已經開始嘗試使用下面的代碼

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); 
} 

這會導致我的Web服務器上的種種問題,其預計有效發送NMES作爲JSON HTTP POST或PUT請求到我的網絡服務器時, JSON(即UTF8) 由於JSON應該首先以UTF8編碼發送。

所以

1)爲什麼不<?xml version="1.0" encoding="utf-8"?>實際設置UTF8編碼?什麼時候在佈局中使用?

2)如何最好地確保我總能得到有效的UTF8格式字符發送到我的Web服務器? 這應該由放置請求還是由保存到共享首選項的代碼處理,或由填充json對象的代碼或上述 的組合來處理?或者是其他東西?

這是從這個回報率問題Rails 3 - How to handle PG Error incomplete multibyte character

+3

僅供參考 - 我認爲設置這樣的: 「<?XML版本=」 1.0" 編碼=「UTF-8 '?>「是指示任何xml解析器,佈局文件的內容以UTF-8編碼。換句話說,它完全是xml佈局文件本身的一個屬性,我不相信它在使用該佈局時對Android的行爲有任何影響... – 2013-01-03 22:20:06

回答

2

的關鍵是要了解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。代理包括:

+0

感謝您的回答,但我仍然不清楚如何轉換將共享首選項中存儲的數據轉換爲UTF-8格式? – jamesc 2013-01-05 00:28:09

+0

總之 - 你沒有。略長:1)Android傳遞給共享首選項的數據是Java字符串,它們是Unicode並且是正確的(如果它們看起來損壞了,那是因爲腐敗數據被插入了)。這裏根本不可能使用UTF-8 - Java不會理解它。 2)存儲在共享首選項中的數據由Android內部* persisted *由Android處理。幾乎可以肯定的是,UTF-8的問題在於與Web服務器的接口。 – 2013-01-05 02:31:50

+0

請記住,UTF-8是*發送*或*接收*數據時使用的特定編碼(表示字符的線上字節)。它僅在您接收到一個字節數組或需要提供一個字節時使用。它也用於通過Web發送數據,並且您可能需要告訴創建HTTP請求的庫專門使用UTF-8編碼 – 2013-01-05 02:39:55